From f1c580f94aa09fc5a0970bfaed621d8b80813ca2 Mon Sep 17 00:00:00 2001 From: Andreas Reich Date: Wed, 3 May 2023 15:22:36 +0200 Subject: [PATCH] 3d to 2d projections (#2008) * transform cache now deals with Affine3 matrices only * use a perspective camera in 2d views that sit at a space camera * clarify/simplify use of focal length * refine camera plane distance heuristic for 2D scenes and make it configurable again * comment wip * merge fixup and comment improvements * line & point builder now work with affine transforms * wip * improved image plane heuristic for 2D * hack for image plane distance for in inverse pinhole transforms. remove setting from ui again for 2d views * add viewport transformation to viewbuilder * better viewport transform * limit zoom, correctly handle ui scale under viewport zoom * 2D points now draw as real 2D circles * better 2D rendering for lines with perspective camera around * disable 3D labels in 2D views * space camera no longer required for correct pinhole camera in ui_2d * consistent canvas rect handling, take principal point into account when displaying 2d canvas * comments on the nature of our interim 3D->2D solution * point out that picking should use same transforms * easier point/line flag building * minor cleanup * doc test fix * clarify what sphere_quad's coverage methods do * fix taking only one axis into account for pixel size approximation * comment explaining how to use FORCE_ORTHO_SPANNING * remove unnecessary affine3a multiply method * impl From for wgpu_buffer_types::Mat4 * rename rect top_left to min * remove unnecessary quaternion on pinhole transform calc * make error swallowing on screenshots more explicit * failure to compute camera now logs error and stops from rendering * note on non-square pixels * better handle different x & y focal length + comment * renaming and tests around RectTransform * yet another workaround for https://github.com/gfx-rs/naga/issues/1743 * remove h word --- .vscode/settings.json | 1 + .../src/component_types/transform.rs | 1 + crates/re_renderer/examples/2d.rs | 1 + crates/re_renderer/examples/depth_cloud.rs | 9 +- crates/re_renderer/examples/multiview.rs | 7 +- crates/re_renderer/examples/outlines.rs | 1 + crates/re_renderer/examples/picking.rs | 7 +- crates/re_renderer/shader/depth_cloud.wgsl | 3 +- crates/re_renderer/shader/lines.wgsl | 8 +- crates/re_renderer/shader/point_cloud.wgsl | 34 +- crates/re_renderer/shader/utils/camera.wgsl | 32 +- .../re_renderer/shader/utils/sphere_quad.wgsl | 27 +- .../src/draw_phases/picking_layer.rs | 41 +- crates/re_renderer/src/lib.rs | 4 +- crates/re_renderer/src/line_strip_builder.rs | 36 +- crates/re_renderer/src/point_cloud_builder.rs | 9 +- crates/re_renderer/src/rect.rs | 60 ++- .../re_renderer/src/renderer/debug_overlay.rs | 6 +- crates/re_renderer/src/renderer/lines.rs | 15 +- .../re_renderer/src/renderer/point_cloud.rs | 29 +- crates/re_renderer/src/transform.rs | 171 ++++++++ crates/re_renderer/src/view_builder.rs | 57 ++- crates/re_renderer/src/wgpu_buffer_types.rs | 7 + .../src/wgpu_resources/shader_module_pool.rs | 10 +- crates/re_viewer/src/gpu_bridge/mod.rs | 1 + crates/re_viewer/src/misc/transform_cache.rs | 104 ++--- crates/re_viewer/src/ui/selection_panel.rs | 7 +- .../src/ui/view_spatial/scene/picking.rs | 2 +- .../src/ui/view_spatial/scene/primitives.rs | 18 +- .../view_spatial/scene/scene_part/arrows3d.rs | 3 +- .../view_spatial/scene/scene_part/boxes2d.rs | 3 +- .../view_spatial/scene/scene_part/boxes3d.rs | 4 +- .../view_spatial/scene/scene_part/cameras.rs | 24 +- .../view_spatial/scene/scene_part/images.rs | 6 +- .../view_spatial/scene/scene_part/lines2d.rs | 4 +- .../view_spatial/scene/scene_part/lines3d.rs | 4 +- .../view_spatial/scene/scene_part/meshes.rs | 7 +- .../view_spatial/scene/scene_part/points2d.rs | 8 +- .../view_spatial/scene/scene_part/points3d.rs | 6 +- .../src/ui/view_spatial/space_camera_3d.rs | 4 +- crates/re_viewer/src/ui/view_spatial/ui.rs | 41 +- crates/re_viewer/src/ui/view_spatial/ui_2d.rs | 391 ++++++++++-------- crates/re_viewer/src/ui/view_spatial/ui_3d.rs | 48 +-- crates/re_web_viewer_server/build.rs | 1 + 44 files changed, 807 insertions(+), 455 deletions(-) create mode 100644 crates/re_renderer/src/transform.rs diff --git a/.vscode/settings.json b/.vscode/settings.json index 6a1d745b2645..8a7d3f644cce 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -40,6 +40,7 @@ "Skybox", "smallvec", "swapchain", + "texcoord", "texcoords", "Tonemapper", "tonemapping", diff --git a/crates/re_log_types/src/component_types/transform.rs b/crates/re_log_types/src/component_types/transform.rs index c57e70d1a9aa..f6b5fc4ece8e 100644 --- a/crates/re_log_types/src/component_types/transform.rs +++ b/crates/re_log_types/src/component_types/transform.rs @@ -163,6 +163,7 @@ impl Pinhole { /// Focal length. #[inline] pub fn focal_length(&self) -> Option { + // Use only the first element of the focal length vector, as we don't support non-square pixels. self.resolution.map(|r| self.image_from_cam[0][0] / r[0]) } diff --git a/crates/re_renderer/examples/2d.rs b/crates/re_renderer/examples/2d.rs index eb17d6955cfd..de356a95eedc 100644 --- a/crates/re_renderer/examples/2d.rs +++ b/crates/re_renderer/examples/2d.rs @@ -284,6 +284,7 @@ impl framework::Example for Render2D { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution[0] as f32 / resolution[1] as f32, }, pixels_from_point, ..Default::default() diff --git a/crates/re_renderer/examples/depth_cloud.rs b/crates/re_renderer/examples/depth_cloud.rs index dddc40d1ef71..e83706f991fc 100644 --- a/crates/re_renderer/examples/depth_cloud.rs +++ b/crates/re_renderer/examples/depth_cloud.rs @@ -123,6 +123,7 @@ impl RenderDepthClouds { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution_in_pixel[0] as f32 / resolution_in_pixel[1] as f32, }, pixels_from_point, ..Default::default() @@ -202,6 +203,7 @@ impl RenderDepthClouds { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution_in_pixel[0] as f32 / resolution_in_pixel[1] as f32, }, pixels_from_point, ..Default::default() @@ -287,9 +289,10 @@ impl framework::Example for RenderDepthClouds { let splits = framework::split_resolution(resolution, 1, 2).collect::>(); let frame_size = albedo.dimensions.as_vec2().extend(0.0) / 15.0; - let scale = glam::Mat4::from_scale(frame_size); - let rotation = glam::Mat4::IDENTITY; - let translation_center = glam::Mat4::from_translation(-glam::Vec3::splat(0.5) * frame_size); + let scale = glam::Affine3A::from_scale(frame_size); + let rotation = glam::Affine3A::IDENTITY; + let translation_center = + glam::Affine3A::from_translation(-glam::Vec3::splat(0.5) * frame_size); let world_from_model = rotation * translation_center * scale; let frame_draw_data = { diff --git a/crates/re_renderer/examples/multiview.rs b/crates/re_renderer/examples/multiview.rs index 48b77e24388f..ddcf793d3d3c 100644 --- a/crates/re_renderer/examples/multiview.rs +++ b/crates/re_renderer/examples/multiview.rs @@ -113,7 +113,9 @@ fn build_lines(re_ctx: &mut RenderContext, seconds_since_startup: f32) -> LineDr // Blue spiral, rotating builder .batch("blue spiral") - .world_from_obj(glam::Mat4::from_rotation_x(seconds_since_startup * 10.0)) + .world_from_obj(glam::Affine3A::from_rotation_x( + seconds_since_startup * 10.0, + )) .add_strip((0..1000).map(|i| { glam::vec3( (i as f32 * 0.01).sin() * 2.0, @@ -318,7 +320,7 @@ impl Example for Multiview { let mut builder = PointCloudBuilder::new(re_ctx); builder .batch("Random Points") - .world_from_obj(glam::Mat4::from_rotation_x(seconds_since_startup)) + .world_from_obj(glam::Affine3A::from_rotation_x(seconds_since_startup)) .add_points( self.random_points_positions.len(), self.random_points_positions.iter().cloned(), @@ -341,6 +343,7 @@ impl Example for Multiview { Projection::Perspective { vertical_fov: 70.0 * TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution[0] as f32 / resolution[1] as f32, } } else { Projection::Orthographic { diff --git a/crates/re_renderer/examples/outlines.rs b/crates/re_renderer/examples/outlines.rs index 2cdc2ed6e87c..f8449ac86351 100644 --- a/crates/re_renderer/examples/outlines.rs +++ b/crates/re_renderer/examples/outlines.rs @@ -61,6 +61,7 @@ impl framework::Example for Outlines { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution[0] as f32 / resolution[1] as f32, }, pixels_from_point, outline_config: Some(OutlineConfig { diff --git a/crates/re_renderer/examples/picking.rs b/crates/re_renderer/examples/picking.rs index afcb442ffbba..b3ea0f8a42d2 100644 --- a/crates/re_renderer/examples/picking.rs +++ b/crates/re_renderer/examples/picking.rs @@ -3,8 +3,8 @@ use rand::Rng; use re_renderer::{ renderer::MeshInstance, view_builder::{Projection, TargetConfiguration, ViewBuilder}, - Color32, GpuReadbackIdentifier, IntRect, PickingLayerId, PickingLayerInstanceId, - PickingLayerProcessor, PointCloudBuilder, Size, + Color32, GpuReadbackIdentifier, PickingLayerId, PickingLayerInstanceId, PickingLayerProcessor, + PointCloudBuilder, RectInt, Size, }; mod framework; @@ -135,6 +135,7 @@ impl framework::Example for Picking { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: resolution[0] as f32 / resolution[1] as f32, }, pixels_from_point, outline_config: None, @@ -145,7 +146,7 @@ impl framework::Example for Picking { // Use an uneven number of pixels for the picking rect so that there is a clearly defined middle-pixel. // (for this sample a size of 1 would be sufficient, but for a real application you'd want to use a larger size to allow snapping) let picking_rect_size = 31; - let picking_rect = IntRect::from_middle_and_extent( + let picking_rect = RectInt::from_middle_and_extent( self.picking_position.as_ivec2(), glam::uvec2(picking_rect_size, picking_rect_size), ); diff --git a/crates/re_renderer/shader/depth_cloud.wgsl b/crates/re_renderer/shader/depth_cloud.wgsl index 0c7050815052..b29ddf97778f 100644 --- a/crates/re_renderer/shader/depth_cloud.wgsl +++ b/crates/re_renderer/shader/depth_cloud.wgsl @@ -133,7 +133,8 @@ fn vs_main(@builtin(vertex_index) vertex_idx: u32) -> VertexOut { if 0.0 < point_data.unresolved_radius { // Span quad - let quad = sphere_quad_span(vertex_idx, point_data.pos_in_world, point_data.unresolved_radius, depth_cloud_info.radius_boost_in_ui_points); + let quad = sphere_or_circle_quad_span(vertex_idx, point_data.pos_in_world, point_data.unresolved_radius, + depth_cloud_info.radius_boost_in_ui_points, false); out.pos_in_clip = frame.projection_from_world * Vec4(quad.pos_in_world, 1.0); out.pos_in_world = quad.pos_in_world; out.point_radius = quad.point_resolved_radius; diff --git a/crates/re_renderer/shader/lines.wgsl b/crates/re_renderer/shader/lines.wgsl index 6a143a8233c8..74be45083851 100644 --- a/crates/re_renderer/shader/lines.wgsl +++ b/crates/re_renderer/shader/lines.wgsl @@ -44,6 +44,7 @@ const CAP_START_TRIANGLE: u32 = 8u; const CAP_START_ROUND: u32 = 16u; const CAP_START_EXTEND_OUTWARDS: u32 = 32u; const NO_COLOR_GRADIENT: u32 = 64u; +const FORCE_ORTHO_SPANNING: u32 = 128u; // A lot of the attributes don't need to be interpolated across triangles. // To document that and safe some time we mark them up with @interpolate(flat) @@ -199,7 +200,12 @@ fn vs_main(@builtin(vertex_index) vertex_idx: u32) -> VertexOut { // Resolve radius. // (slight inaccuracy: End caps are going to adjust their center_position) - let camera_ray = camera_ray_to_world_pos(center_position); + var camera_ray: Ray; + if has_any_flag(strip_data.flags, FORCE_ORTHO_SPANNING) || is_camera_orthographic() { + camera_ray = camera_ray_to_world_pos_orthographic(center_position); + } else { + camera_ray = camera_ray_to_world_pos_perspective(center_position); + } let camera_distance = distance(camera_ray.origin, center_position); var strip_radius = unresolved_size_to_world(strip_data.unresolved_radius, camera_distance, frame.auto_size_lines); diff --git a/crates/re_renderer/shader/point_cloud.wgsl b/crates/re_renderer/shader/point_cloud.wgsl index a55404230692..3fff3e6e61b9 100644 --- a/crates/re_renderer/shader/point_cloud.wgsl +++ b/crates/re_renderer/shader/point_cloud.wgsl @@ -36,6 +36,8 @@ var batch: BatchUniformBuffer; // Flags // See point_cloud.rs#PointCloudBatchFlags const ENABLE_SHADING: u32 = 1u; +const DRAW_AS_CIRCLES: u32 = 2u; + const TEXTURE_SIZE: u32 = 2048u; struct VertexOut { @@ -94,7 +96,8 @@ fn vs_main(@builtin(vertex_index) vertex_idx: u32) -> VertexOut { let point_data = read_data(quad_idx); // Span quad - let quad = sphere_quad_span(vertex_idx, point_data.pos, point_data.unresolved_radius, draw_data.radius_boost_in_ui_points); + let quad = sphere_or_circle_quad_span(vertex_idx, point_data.pos, point_data.unresolved_radius, + draw_data.radius_boost_in_ui_points, has_any_flag(batch.flags, DRAW_AS_CIRCLES)); // Output, transform to projection space and done. var out: VertexOut; @@ -108,9 +111,32 @@ fn vs_main(@builtin(vertex_index) vertex_idx: u32) -> VertexOut { return out; } +// TODO(andreas): move this to sphere_quad.wgsl once https://github.com/gfx-rs/naga/issues/1743 is resolved +// point_cloud.rs has a specific workaround in place so we don't need to split vertex/fragment shader here +// +/// Computes coverage of a 2D sphere placed at `circle_center` in the fragment shader using the currently set camera. +/// +/// 2D primitives are always facing the camera - the difference to sphere_quad_coverage is that +/// perspective projection is not taken into account. +fn circle_quad_coverage(world_position: Vec3, radius: f32, circle_center: Vec3) -> f32 { + let to_center = circle_center - world_position; + let distance = length(to_center); + let distance_pixel_difference = fwidth(distance); + return smoothstep(radius + distance_pixel_difference, radius - distance_pixel_difference, distance); +} + +fn coverage(world_position: Vec3, radius: f32, point_center: Vec3) -> f32 { + if is_camera_orthographic() || has_any_flag(batch.flags, DRAW_AS_CIRCLES) { + return circle_quad_coverage(world_position, radius, point_center); + } else { + return sphere_quad_coverage(world_position, radius, point_center); + } +} + + @fragment fn fs_main(in: VertexOut) -> @location(0) Vec4 { - let coverage = sphere_quad_coverage(in.world_position, in.radius, in.point_center); + let coverage = coverage(in.world_position, in.radius, in.point_center); if coverage < 0.001 { discard; } @@ -127,7 +153,7 @@ fn fs_main(in: VertexOut) -> @location(0) Vec4 { @fragment fn fs_main_picking_layer(in: VertexOut) -> @location(0) UVec4 { - let coverage = sphere_quad_coverage(in.world_position, in.radius, in.point_center); + let coverage = coverage(in.world_position, in.radius, in.point_center); if coverage <= 0.5 { discard; } @@ -139,7 +165,7 @@ fn fs_main_outline_mask(in: VertexOut) -> @location(0) UVec2 { // Output is an integer target, can't use coverage therefore. // But we still want to discard fragments where coverage is low. // Since the outline extends a bit, a very low cut off tends to look better. - let coverage = sphere_quad_coverage(in.world_position, in.radius, in.point_center); + let coverage = coverage(in.world_position, in.radius, in.point_center); if coverage < 1.0 { discard; } diff --git a/crates/re_renderer/shader/utils/camera.wgsl b/crates/re_renderer/shader/utils/camera.wgsl index 7f1c71a3d460..280ff765c5c1 100644 --- a/crates/re_renderer/shader/utils/camera.wgsl +++ b/crates/re_renderer/shader/utils/camera.wgsl @@ -15,22 +15,32 @@ struct Ray { direction: Vec3, } -// Returns the ray from the camera to a given world position. -fn camera_ray_to_world_pos(world_pos: Vec3) -> Ray { +// Returns the ray from the camera to a given world position, assuming the camera is perspective +fn camera_ray_to_world_pos_perspective(world_pos: Vec3) -> Ray { + var ray: Ray; + ray.origin = frame.camera_position; + ray.direction = normalize(world_pos - frame.camera_position); + return ray; +} + +// Returns the ray from the camera to a given world position, assuming the camera is orthographic +fn camera_ray_to_world_pos_orthographic(world_pos: Vec3) -> Ray { var ray: Ray; + // The ray originates on the camera plane, not from the camera position + let to_pos = world_pos - frame.camera_position; + let camera_plane_distance = dot(to_pos, frame.camera_forward); + ray.origin = world_pos - frame.camera_forward * camera_plane_distance; + ray.direction = frame.camera_forward; + return ray; +} +// Returns the ray from the camera to a given world position. +fn camera_ray_to_world_pos(world_pos: Vec3) -> Ray { if is_camera_perspective() { - ray.origin = frame.camera_position; - ray.direction = normalize(world_pos - frame.camera_position); + return camera_ray_to_world_pos_perspective(world_pos); } else { - // The ray originates on the camera plane, not from the camera position - let to_pos = world_pos - frame.camera_position; - let camera_plane_distance = dot(to_pos, frame.camera_forward); - ray.origin = world_pos - frame.camera_forward * camera_plane_distance; - ray.direction = frame.camera_forward; + return camera_ray_to_world_pos_orthographic(world_pos); } - - return ray; } // Returns the camera ray direction through a given screen uv coordinates (ranging from 0 to 1, i.e. NOT ndc coordinates) diff --git a/crates/re_renderer/shader/utils/sphere_quad.wgsl b/crates/re_renderer/shader/utils/sphere_quad.wgsl index 00937d8e7701..83fc190496f8 100644 --- a/crates/re_renderer/shader/utils/sphere_quad.wgsl +++ b/crates/re_renderer/shader/utils/sphere_quad.wgsl @@ -4,7 +4,7 @@ /// Span a quad in a way that guarantees that we'll be able to draw a perspective correct sphere /// on it. -fn sphere_quad_span_perspective( +fn sphere_quad( point_pos: Vec3, point_radius: f32, top_bottom: f32, @@ -42,7 +42,7 @@ fn sphere_quad_span_perspective( /// Span a quad in a way that guarantees that we'll be able to draw an orthographic correct sphere /// on it. -fn sphere_quad_span_orthographic(point_pos: Vec3, point_radius: f32, top_bottom: f32, left_right: f32) -> Vec3 { +fn circle_quad(point_pos: Vec3, point_radius: f32, top_bottom: f32, left_right: f32) -> Vec3 { let quad_normal = frame.camera_forward; let quad_right = normalize(cross(quad_normal, frame.view_from_world[1].xyz)); // It's spheres so any orthogonal vector would do. let quad_up = cross(quad_right, quad_normal); @@ -50,7 +50,8 @@ fn sphere_quad_span_orthographic(point_pos: Vec3, point_radius: f32, top_bottom: // Add half a pixel of margin for the feathering we do for antialiasing the spheres. // It's fairly subtle but if we don't do this our spheres look slightly squarish - let radius = point_radius + 0.5 * frame.pixel_world_size_from_camera_distance; + // TODO(andreas): Computing distance to camera here is a bit excessive, should get distance more easily - keep in mind this code runs for ortho & perspective. + let radius = point_radius + 0.5 * approx_pixel_world_size_at(distance(point_pos, frame.camera_position)); return point_pos + pos_in_quad * radius; } @@ -65,10 +66,11 @@ struct SphereQuadData { point_resolved_radius: f32, } -/// Span a quad onto which perspective correct spheres can be drawn. +/// Span a quad onto which circles or perspective correct spheres can be drawn. /// -/// Spanning is done in perspective or orthographically depending of the state of the global cam. -fn sphere_quad_span(vertex_idx: u32, point_pos: Vec3, point_unresolved_radius: f32, radius_boost_in_ui_points: f32) -> SphereQuadData { +/// Note that in orthographic mode, spheres are always circles. +fn sphere_or_circle_quad_span(vertex_idx: u32, point_pos: Vec3, point_unresolved_radius: f32, + radius_boost_in_ui_points: f32, force_circle: bool) -> SphereQuadData { // Resolve radius to a world size. We need the camera distance for this, which is useful later on. let to_camera = frame.camera_position - point_pos; let camera_distance = length(to_camera); @@ -82,24 +84,23 @@ fn sphere_quad_span(vertex_idx: u32, point_pos: Vec3, point_unresolved_radius: f // Span quad var pos: Vec3; - if is_camera_perspective() { - pos = sphere_quad_span_perspective(point_pos, radius, top_bottom, left_right, to_camera, camera_distance); + if is_camera_orthographic() || force_circle { + pos = circle_quad(point_pos, radius, top_bottom, left_right); } else { - pos = sphere_quad_span_orthographic(point_pos, radius, top_bottom, left_right); + pos = sphere_quad(point_pos, radius, top_bottom, left_right, to_camera, camera_distance); } return SphereQuadData(pos, radius); } -fn sphere_quad_coverage(world_position: Vec3, radius: f32, point_center: Vec3) -> f32 { - // There's easier ways to compute anti-aliasing for when we are in ortho mode since it's just circles. - // But it's very nice to have mostly the same code path and this gives us the sphere world position along the way. +/// Computes coverage of a 3D sphere placed at `sphere_center` in the fragment shader using the currently set camera. +fn sphere_quad_coverage(world_position: Vec3, radius: f32, sphere_center: Vec3) -> f32 { let ray = camera_ray_to_world_pos(world_position); // Sphere intersection with anti-aliasing as described by Iq here // https://www.shadertoy.com/view/MsSSWV // (but rearranged and labeled to it's easier to understand!) - let d = ray_sphere_distance(ray, point_center, radius); + let d = ray_sphere_distance(ray, sphere_center, radius); let distance_to_sphere_surface = d.x; let closest_ray_dist = d.y; let pixel_world_size = approx_pixel_world_size_at(closest_ray_dist); diff --git a/crates/re_renderer/src/draw_phases/picking_layer.rs b/crates/re_renderer/src/draw_phases/picking_layer.rs index cd4d6601ebb0..d2466a1d7329 100644 --- a/crates/re_renderer/src/draw_phases/picking_layer.rs +++ b/crates/re_renderer/src/draw_phases/picking_layer.rs @@ -13,14 +13,16 @@ use crate::{ allocator::create_and_fill_uniform_buffer, global_bindings::FrameUniformBuffer, include_shader_module, + rect::RectF32, texture_info::Texture2DBufferInfo, + transform::{ndc_from_pixel, RectTransform}, view_builder::ViewBuilder, wgpu_resources::{ BindGroupDesc, BindGroupEntry, BindGroupLayoutDesc, GpuBindGroup, GpuRenderPipelineHandle, GpuTexture, GpuTextureHandle, PipelineLayoutDesc, PoolError, RenderPipelineDesc, TextureDesc, WgpuResourcePools, }, - DebugLabel, GpuReadbackBuffer, GpuReadbackIdentifier, IntRect, RenderContext, + DebugLabel, GpuReadbackBuffer, GpuReadbackIdentifier, RectInt, RenderContext, }; use smallvec::smallvec; @@ -32,7 +34,7 @@ pub struct PickingResult { /// Picking rect supplied on picking request. /// Describes the area of the picking layer that was read back. - pub rect: IntRect, + pub rect: RectInt, /// Picking id data for the requested rectangle. /// @@ -64,8 +66,7 @@ impl PickingResult { [(pos_on_picking_rect.y * self.rect.width() + pos_on_picking_rect.x) as usize]; self.world_from_cropped_projection.project_point3( - pixel_coord_to_ndc(pos_on_picking_rect.as_vec2(), self.rect.extent.as_vec2()) - .extend(raw_depth), + ndc_from_pixel(pos_on_picking_rect.as_vec2(), self.rect.extent).extend(raw_depth), ) } @@ -81,7 +82,7 @@ impl PickingResult { /// Type used as user data on the gpu readback belt. struct ReadbackBeltMetadata { - picking_rect: IntRect, + picking_rect: RectInt, world_from_cropped_projection: glam::Mat4, user_data: T, @@ -125,14 +126,6 @@ impl From for [u32; 4] { } } -/// Converts a pixel coordinate to normalized device coordinates. -pub fn pixel_coord_to_ndc(coord: glam::Vec2, target_resolution: glam::Vec2) -> glam::Vec2 { - glam::vec2( - coord.x / target_resolution.x * 2.0 - 1.0, - 1.0 - coord.y / target_resolution.y * 2.0, - ) -} - #[derive(thiserror::Error, Debug)] pub enum PickingLayerError { #[error(transparent)] @@ -184,7 +177,7 @@ impl PickingLayerProcessor { ctx: &mut RenderContext, view_name: &DebugLabel, screen_resolution: glam::UVec2, - picking_rect: IntRect, + picking_rect: RectInt, frame_uniform_buffer_content: &FrameUniformBuffer, enable_picking_target_sampling: bool, readback_identifier: GpuReadbackIdentifier, @@ -234,18 +227,14 @@ impl PickingLayerProcessor { DepthReadbackWorkaround::new(ctx, picking_rect.extent, picking_depth_target.handle) }); - let rect_min = picking_rect.left_top.as_vec2(); - let rect_max = rect_min + picking_rect.extent.as_vec2(); - let screen_resolution = screen_resolution.as_vec2(); - // y axis is flipped in NDC, therefore we need to flip the y axis of the rect. - let rect_min_ndc = - pixel_coord_to_ndc(glam::vec2(rect_min.x, rect_max.y), screen_resolution); - let rect_max_ndc = - pixel_coord_to_ndc(glam::vec2(rect_max.x, rect_min.y), screen_resolution); - let scale = 2.0 / (rect_max_ndc - rect_min_ndc); - let translation = -0.5 * (rect_min_ndc + rect_max_ndc); - let cropped_projection_from_projection = glam::Mat4::from_scale(scale.extend(1.0)) - * glam::Mat4::from_translation(translation.extend(0.0)); + let cropped_projection_from_projection = RectTransform { + region_of_interest: picking_rect.into(), + region: RectF32 { + min: glam::Vec2::ZERO, + extent: screen_resolution.as_vec2(), + }, + } + .to_ndc_scale_and_translation(); // Setup frame uniform buffer let previous_projection_from_world: glam::Mat4 = diff --git a/crates/re_renderer/src/lib.rs b/crates/re_renderer/src/lib.rs index 1932e1aeb9e2..f6e07b5839c1 100644 --- a/crates/re_renderer/src/lib.rs +++ b/crates/re_renderer/src/lib.rs @@ -25,6 +25,7 @@ mod global_bindings; mod line_strip_builder; mod point_cloud_builder; mod size; +mod transform; mod wgpu_buffer_types; mod wgpu_resources; @@ -41,7 +42,9 @@ pub use debug_label::DebugLabel; pub use depth_offset::DepthOffset; pub use line_strip_builder::{LineStripBuilder, LineStripSeriesBuilder}; pub use point_cloud_builder::{PointCloudBatchBuilder, PointCloudBuilder}; +pub use rect::{RectF32, RectInt}; pub use size::Size; +pub use transform::RectTransform; pub use view_builder::{AutoSizeConfig, ViewBuilder}; pub use wgpu_resources::WgpuResourcePoolStatistics; @@ -67,7 +70,6 @@ mod file_server; pub use self::file_server::FileServer; mod rect; -pub use rect::IntRect; #[cfg(not(all(not(target_arch = "wasm32"), debug_assertions)))] // wasm or release builds #[rustfmt::skip] // it's auto-generated diff --git a/crates/re_renderer/src/line_strip_builder.rs b/crates/re_renderer/src/line_strip_builder.rs index 9fee0afdb964..2514f621b24b 100644 --- a/crates/re_renderer/src/line_strip_builder.rs +++ b/crates/re_renderer/src/line_strip_builder.rs @@ -65,7 +65,7 @@ impl LineStripSeriesBuilder { pub fn batch(&mut self, label: impl Into) -> LineBatchBuilder<'_> { self.batches.push(LineBatchInfo { label: label.into(), - world_from_obj: glam::Mat4::IDENTITY, + world_from_obj: glam::Affine3A::IDENTITY, line_vertex_count: 0, overall_outline_mask_ids: OutlineMaskPreference::NONE, additional_outline_mask_ids_vertex_ranges: Vec::new(), @@ -104,6 +104,14 @@ impl LineStripSeriesBuilder { pub fn is_empty(&self) -> bool { self.strips.is_empty() } + + pub fn default_box_flags() -> LineStripFlags { + LineStripFlags::CAP_END_ROUND + | LineStripFlags::CAP_START_ROUND + | LineStripFlags::NO_COLOR_GRADIENT + | LineStripFlags::CAP_END_EXTEND_OUTWARDS + | LineStripFlags::CAP_START_EXTEND_OUTWARDS + } } pub struct LineBatchBuilder<'a>(&'a mut LineStripSeriesBuilder); @@ -138,7 +146,7 @@ impl<'a> LineBatchBuilder<'a> { /// Sets the `world_from_obj` matrix for the *entire* batch. #[inline] - pub fn world_from_obj(mut self, world_from_obj: glam::Mat4) -> Self { + pub fn world_from_obj(mut self, world_from_obj: glam::Affine3A) -> Self { self.batch_mut().world_from_obj = world_from_obj; self } @@ -257,13 +265,7 @@ impl<'a> LineBatchBuilder<'a> { ] .into_iter(), ) - .flags( - LineStripFlags::CAP_END_ROUND - | LineStripFlags::CAP_START_ROUND - | LineStripFlags::NO_COLOR_GRADIENT - | LineStripFlags::CAP_END_EXTEND_OUTWARDS - | LineStripFlags::CAP_START_EXTEND_OUTWARDS, - ) + .flags(LineStripSeriesBuilder::default_box_flags()) } /// Add rectangle outlines. @@ -292,13 +294,7 @@ impl<'a> LineBatchBuilder<'a> { ] .into_iter(), ) - .flags( - LineStripFlags::CAP_END_ROUND - | LineStripFlags::CAP_START_ROUND - | LineStripFlags::NO_COLOR_GRADIENT - | LineStripFlags::CAP_END_EXTEND_OUTWARDS - | LineStripFlags::CAP_START_EXTEND_OUTWARDS, - ) + .flags(LineStripSeriesBuilder::default_box_flags()) } /// Adds a 2D series of line connected points. @@ -310,12 +306,14 @@ impl<'a> LineBatchBuilder<'a> { points: impl Iterator, ) -> LineStripBuilder<'_> { self.add_strip(points.map(|p| p.extend(0.0))) + .flags(LineStripFlags::FORCE_ORTHO_SPANNING) } /// Adds a single 2D line segment connecting two points. Uses autogenerated depth value. #[inline] pub fn add_segment_2d(&mut self, a: glam::Vec2, b: glam::Vec2) -> LineStripBuilder<'_> { self.add_strip_2d([a, b].into_iter()) + .flags(LineStripFlags::FORCE_ORTHO_SPANNING) } /// Adds a series of unconnected 2D line segments. @@ -327,6 +325,7 @@ impl<'a> LineBatchBuilder<'a> { segments: impl Iterator, ) -> LineStripBuilder<'_> { self.add_segments(segments.map(|(a, b)| (a.extend(0.0), b.extend(0.0)))) + .flags(LineStripFlags::FORCE_ORTHO_SPANNING) } /// Add 2D rectangle outlines. @@ -345,6 +344,7 @@ impl<'a> LineBatchBuilder<'a> { extent_u.extend(0.0), extent_v.extend(0.0), ) + .flags(LineStripFlags::FORCE_ORTHO_SPANNING) } /// Add 2D rectangle outlines with axis along X and Y. @@ -362,6 +362,7 @@ impl<'a> LineBatchBuilder<'a> { glam::Vec3::X * (max.x - min.x), glam::Vec3::Y * (max.y - min.y), ) + .flags(LineStripFlags::FORCE_ORTHO_SPANNING) } } @@ -390,10 +391,11 @@ impl<'a> LineStripBuilder<'a> { self } + /// Adds (!) flags to the line strip. #[inline] pub fn flags(self, flags: LineStripFlags) -> Self { for strip in self.builder.strips[self.strip_range.clone()].iter_mut() { - strip.flags = flags; + strip.flags |= flags; } self } diff --git a/crates/re_renderer/src/point_cloud_builder.rs b/crates/re_renderer/src/point_cloud_builder.rs index 63278d5c7e69..4862748996aa 100644 --- a/crates/re_renderer/src/point_cloud_builder.rs +++ b/crates/re_renderer/src/point_cloud_builder.rs @@ -63,7 +63,7 @@ impl PointCloudBuilder { pub fn batch(&mut self, label: impl Into) -> PointCloudBatchBuilder<'_> { self.batches.push(PointCloudBatchInfo { label: label.into(), - world_from_obj: glam::Mat4::IDENTITY, + world_from_obj: glam::Affine3A::IDENTITY, flags: PointCloudBatchFlags::ENABLE_SHADING, point_count: 0, overall_outline_mask_ids: OutlineMaskPreference::NONE, @@ -128,7 +128,7 @@ impl<'a> PointCloudBatchBuilder<'a> { /// Sets the `world_from_obj` matrix for the *entire* batch. #[inline] - pub fn world_from_obj(mut self, world_from_obj: glam::Mat4) -> Self { + pub fn world_from_obj(mut self, world_from_obj: glam::Affine3A) -> Self { self.batch_mut().world_from_obj = world_from_obj; self } @@ -244,11 +244,12 @@ impl<'a> PointCloudBatchBuilder<'a> { colors, picking_instance_ids, ) + .flags(PointCloudBatchFlags::DRAW_AS_CIRCLES) } - /// Set flags for this batch. + /// Adds (!) flags for this batch. pub fn flags(mut self, flags: PointCloudBatchFlags) -> Self { - self.batch_mut().flags = flags; + self.batch_mut().flags |= flags; self } diff --git a/crates/re_renderer/src/rect.rs b/crates/re_renderer/src/rect.rs index 8b70e81ac357..f62d59364a2d 100644 --- a/crates/re_renderer/src/rect.rs +++ b/crates/re_renderer/src/rect.rs @@ -2,19 +2,21 @@ /// /// Typically used for texture cutouts etc. #[derive(Clone, Copy, Debug)] -pub struct IntRect { - /// The top left corner of the rectangle. - pub left_top: glam::IVec2, +pub struct RectInt { + /// The corner with the smallest coordinates. + /// + /// In most coordinate spaces this is the to top left corner of the rectangle. + pub min: glam::IVec2, /// The size of the rectangle. pub extent: glam::UVec2, } -impl IntRect { +impl RectInt { #[inline] pub fn from_middle_and_extent(middle: glam::IVec2, size: glam::UVec2) -> Self { Self { - left_top: middle - size.as_ivec2() / 2, + min: middle - size.as_ivec2() / 2, extent: size, } } @@ -38,3 +40,51 @@ impl IntRect { } } } + +/// A 2D rectangle with float coordinates. +#[derive(Clone, Copy, Debug)] +pub struct RectF32 { + /// The corner with the smallest coordinates. + /// + /// In most coordinate spaces this is the to top left corner of the rectangle. + pub min: glam::Vec2, + + /// The size of the rectangle. Supposed to be positive. + pub extent: glam::Vec2, +} + +impl RectF32 { + /// The unit rectangle, defined as (0, 0) - (1, 1). + pub const UNIT: RectF32 = RectF32 { + min: glam::Vec2::ZERO, + extent: glam::Vec2::ONE, + }; + + #[inline] + pub fn max(self) -> glam::Vec2 { + self.min + self.extent + } + + #[inline] + pub fn center(self) -> glam::Vec2 { + self.min + self.extent / 2.0 + } + + #[inline] + pub fn scale_extent(self, factor: f32) -> RectF32 { + RectF32 { + min: self.min * factor, + extent: self.extent * factor, + } + } +} + +impl From for RectF32 { + #[inline] + fn from(rect: RectInt) -> Self { + Self { + min: rect.min.as_vec2(), + extent: rect.extent.as_vec2(), + } + } +} diff --git a/crates/re_renderer/src/renderer/debug_overlay.rs b/crates/re_renderer/src/renderer/debug_overlay.rs index 5b78ac3e3139..d121957ff739 100644 --- a/crates/re_renderer/src/renderer/debug_overlay.rs +++ b/crates/re_renderer/src/renderer/debug_overlay.rs @@ -8,7 +8,7 @@ use crate::{ GpuRenderPipelineHandle, GpuTexture, PipelineLayoutDesc, RenderPipelineDesc, WgpuResourcePools, }, - IntRect, + RectInt, }; use super::{DrawData, FileResolver, FileSystem, RenderContext, Renderer}; @@ -75,7 +75,7 @@ impl DebugOverlayDrawData { ctx: &mut RenderContext, debug_texture: &GpuTexture, screen_resolution: glam::UVec2, - overlay_rect: IntRect, + overlay_rect: RectInt, ) -> Result { let mut renderers = ctx.renderers.write(); let debug_overlay = renderers.get_or_create::<_, DebugOverlayRenderer>( @@ -108,7 +108,7 @@ impl DebugOverlayDrawData { "DebugOverlayDrawData".into(), gpu_data::DebugOverlayUniformBuffer { screen_resolution: screen_resolution.as_vec2().into(), - position_in_pixel: overlay_rect.left_top.as_vec2().into(), + position_in_pixel: overlay_rect.min.as_vec2().into(), extent_in_pixel: overlay_rect.extent.as_vec2().into(), mode: mode as u32, _padding: 0, diff --git a/crates/re_renderer/src/renderer/lines.rs b/crates/re_renderer/src/renderer/lines.rs index a46571928c63..759f66984e7a 100644 --- a/crates/re_renderer/src/renderer/lines.rs +++ b/crates/re_renderer/src/renderer/lines.rs @@ -227,7 +227,18 @@ bitflags! { const CAP_START_EXTEND_OUTWARDS = 0b0010_0000; /// Disable color gradient which is on by default + /// + /// TODO(andreas): Could be moved to per batch flags. const NO_COLOR_GRADIENT = 0b0100_0000; + + /// Forces spanning the line's quads as-if the camera was orthographic. + /// + /// This is useful for lines that are on a plane that is parallel to the camera: + /// Without this flag, the lines will poke through the camera plane as they orient themselves towards the camera. + /// Note that since distances to the camera are computed differently in orthographic mode, this changes how screen space sizes are computed. + /// + /// TODO(andreas): Could be moved to per batch flags. + const FORCE_ORTHO_SPANNING = 0b1000_0000; } } @@ -239,7 +250,7 @@ pub struct LineBatchInfo { /// /// TODO(andreas): We don't apply scaling to the radius yet. Need to pass a scaling factor like this in /// `let scale = Mat3::from(world_from_obj).determinant().abs().cbrt()` - pub world_from_obj: glam::Mat4, + pub world_from_obj: glam::Affine3A, /// Number of vertices covered by this batch. /// @@ -351,7 +362,7 @@ impl LineDrawData { let batches = if batches.is_empty() { vec![LineBatchInfo { - world_from_obj: glam::Mat4::IDENTITY, + world_from_obj: glam::Affine3A::IDENTITY, label: "LineDrawData::fallback_batch".into(), line_vertex_count: vertices.len() as _, overall_outline_mask_ids: OutlineMaskPreference::NONE, diff --git a/crates/re_renderer/src/renderer/point_cloud.rs b/crates/re_renderer/src/renderer/point_cloud.rs index 03611b80a256..4845541f2619 100644 --- a/crates/re_renderer/src/renderer/point_cloud.rs +++ b/crates/re_renderer/src/renderer/point_cloud.rs @@ -49,6 +49,9 @@ bitflags! { pub struct PointCloudBatchFlags : u32 { /// If true, we shade all points in the batch like spheres. const ENABLE_SHADING = 0b0001; + + /// If true, draw 2D camera facing circles instead of spheres. + const DRAW_AS_CIRCLES = 0b0010; } } @@ -112,10 +115,9 @@ pub struct PointCloudBatchInfo { /// Transformation applies to point positions /// - /// TODO(andreas): Since we blindly apply this to positions only there is no restriction on this matrix. /// TODO(andreas): We don't apply scaling to the radius yet. Need to pass a scaling factor like this in /// `let scale = Mat3::from(world_from_obj).determinant().abs().cbrt()` - pub world_from_obj: glam::Mat4, + pub world_from_obj: glam::Affine3A, /// Additional properties of this point cloud batch. pub flags: PointCloudBatchFlags, @@ -201,7 +203,7 @@ impl PointCloudDrawData { let fallback_batches = [PointCloudBatchInfo { label: "fallback_batches".into(), - world_from_obj: glam::Mat4::IDENTITY, + world_from_obj: glam::Affine3A::IDENTITY, flags: PointCloudBatchFlags::empty(), point_count: vertices.len() as _, overall_outline_mask_ids: OutlineMaskPreference::NONE, @@ -622,17 +624,26 @@ impl Renderer for PointCloudRenderer { &pools.bind_group_layouts, ); - let shader_module = pools.shader_modules.get_or_create( - device, - resolver, - &include_shader_module!("../../shader/point_cloud.wgsl"), - ); + let shader_module_desc = include_shader_module!("../../shader/point_cloud.wgsl"); + let shader_module = + pools + .shader_modules + .get_or_create(device, resolver, &shader_module_desc); + + // WORKAROUND for https://github.com/gfx-rs/naga/issues/1743 + let mut shader_module_desc_vertex = shader_module_desc.clone(); + shader_module_desc_vertex.extra_workaround_replacements = + vec![("fwidth(".to_owned(), "f32(".to_owned())]; + let shader_module_vertex = + pools + .shader_modules + .get_or_create(device, resolver, &shader_module_desc_vertex); let render_pipeline_desc_color = RenderPipelineDesc { label: "PointCloudRenderer::render_pipeline_color".into(), pipeline_layout, vertex_entrypoint: "vs_main".into(), - vertex_handle: shader_module, + vertex_handle: shader_module_vertex, fragment_entrypoint: "fs_main".into(), fragment_handle: shader_module, vertex_buffers: smallvec![], diff --git a/crates/re_renderer/src/transform.rs b/crates/re_renderer/src/transform.rs new file mode 100644 index 000000000000..dfff2f7b93ef --- /dev/null +++ b/crates/re_renderer/src/transform.rs @@ -0,0 +1,171 @@ +//! Transformation utilities. +//! +//! Some space definitions to keep in mind: +//! +//! Texture coordinates: +//! * origin top left +//! * full texture ([left; right], [top; bottom]): +//! ([0; 1], [0; 1]) +//! +//! NDC: +//! * origin center +//! * full screen ([left; right], [top; bottom]): +//! ([-1; 1], [1; -1]) +//! +//! Pixel coordinates: +//! * origin top left +//! * full screen ([left; right], [top; bottom]): +//! ([0; `screen_extent.x`], [0; `screen_extent.y`]) + +use crate::rect::RectF32; + +/// Transforms texture coordinates to normalized device coordinates (NDC). +#[inline] +pub fn ndc_from_texcoord(texcoord: glam::Vec2) -> glam::Vec2 { + glam::vec2(texcoord.x * 2.0 - 1.0, 1.0 - texcoord.y * 2.0) +} + +/// Transforms texture coordinates to normalized device coordinates (NDC). +#[inline] +pub fn ndc_from_pixel(pixel_coord: glam::Vec2, screen_extent: glam::UVec2) -> glam::Vec2 { + glam::vec2( + pixel_coord.x / screen_extent.x as f32 * 2.0 - 1.0, + 1.0 - pixel_coord.y / screen_extent.y as f32 * 2.0, + ) +} + +/// Defines a transformation from a rectangular region of interest into a rectangular target region. +/// +/// Transforms map the range of `region_of_interest` to the range of `region`. +#[derive(Clone, Debug)] +pub struct RectTransform { + pub region_of_interest: RectF32, + pub region: RectF32, +} + +impl RectTransform { + /// No-op rect transform that transforms from a unit rectangle to a unit rectangle. + pub const IDENTITY: RectTransform = RectTransform { + region_of_interest: RectF32::UNIT, + region: RectF32::UNIT, + }; + + /// Computes a transformation matrix that applies the rect transform to the NDC space. + /// + /// This matrix is expected to be the left most transformation in the vertex transformation chain. + /// It causes the area described by `region_of_interest` to be mapped to the area described by `region`. + /// Meaning, that `region` represents the full screen of the NDC space. + /// + /// This means that only the relation of the rectangles in `RectTransform` is important. + /// Scaling or moving both rectangles by the same amount does not change the result. + pub fn to_ndc_scale_and_translation(&self) -> glam::Mat4 { + // It's easier to think in texcoord space, and then transform to NDC. + // This texcoord rect specifies the portion of the screen that should become the entire range of the NDC screen. + let texcoord_rect = RectF32 { + min: (self.region_of_interest.min - self.region.min) / self.region.extent, + extent: self.region_of_interest.extent / self.region.extent, + }; + let texcoord_rect_min = texcoord_rect.min; + let texcoord_rect_max = texcoord_rect.max(); + + // y axis is flipped in NDC, therefore we need to flip the y axis of the rect. + let rect_min_ndc = ndc_from_texcoord(glam::vec2(texcoord_rect_min.x, texcoord_rect_max.y)); + let rect_max_ndc = ndc_from_texcoord(glam::vec2(texcoord_rect_max.x, texcoord_rect_min.y)); + + let scale = 2.0 / (rect_max_ndc - rect_min_ndc); + let translation = -0.5 * (rect_min_ndc + rect_max_ndc); + + glam::Mat4::from_scale(scale.extend(1.0)) + * glam::Mat4::from_translation(translation.extend(0.0)) + } + + pub fn scale(&self) -> glam::Vec2 { + self.region.extent / self.region_of_interest.extent + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + pub fn to_ndc_scale_and_translation() { + let region = RectF32 { + min: glam::vec2(1.0, 1.0), + extent: glam::vec2(2.0, 3.0), + }; + + // Identity + { + let rect_transform = RectTransform { + region_of_interest: region, + region, + }; + let identity = rect_transform.to_ndc_scale_and_translation(); + assert_eq!(identity, glam::Mat4::IDENTITY); + } + + // Scale + { + let scale_factor = glam::vec2(2.0, 0.25); + + let rect_transform = RectTransform { + region_of_interest: RectF32 { + // Move the roi to the middle of the region. + min: region.center() - region.extent * scale_factor * 0.5, + extent: region.extent * scale_factor, + }, + region, + }; + let scale = rect_transform.to_ndc_scale_and_translation(); + assert_eq!( + scale, + glam::Mat4::from_scale(1.0 / scale_factor.extend(1.0)) + ); + } + + // Translation + { + let translation_vec = glam::vec2(1.0, 2.0); + + let rect_transform = RectTransform { + region_of_interest: RectF32 { + min: region.min + translation_vec * region.extent, + extent: region.extent, + }, + region, + }; + let translation = rect_transform.to_ndc_scale_and_translation(); + assert_eq!( + translation, + glam::Mat4::from_translation( + glam::vec3(-translation_vec.x, translation_vec.y, 0.0) * 2.0 + ) + ); + } + + // Scale + translation + { + let scale_factor = glam::vec2(2.0, 0.25); + let translation_vec = glam::vec2(1.0, 2.0); + + let rect_transform = RectTransform { + region_of_interest: RectF32 { + // Move the roi to the middle of the region and then apply translation + min: region.center() - region.extent * scale_factor * 0.5 + + translation_vec * region.extent, + extent: region.extent * scale_factor, + }, + region, + }; + let scale_and_translation = rect_transform.to_ndc_scale_and_translation(); + assert_eq!( + scale_and_translation, + glam::Mat4::from_scale(1.0 / scale_factor.extend(1.0)) + * glam::Mat4::from_translation( + glam::vec3(-translation_vec.x, translation_vec.y, 0.0) * 2.0 + ) + ); + } + } +} diff --git a/crates/re_renderer/src/view_builder.rs b/crates/re_renderer/src/view_builder.rs index 86d8fc817b59..65a606963c75 100644 --- a/crates/re_renderer/src/view_builder.rs +++ b/crates/re_renderer/src/view_builder.rs @@ -11,8 +11,9 @@ use crate::{ }, global_bindings::FrameUniformBuffer, renderer::{CompositorDrawData, DebugOverlayDrawData, DrawData, Renderer}, + transform::RectTransform, wgpu_resources::{GpuBindGroup, GpuTexture, PoolError, TextureDesc}, - DebugLabel, IntRect, Rgba, Size, + DebugLabel, RectInt, Rgba, Size, }; type DrawFn = dyn for<'a, 'b> Fn( @@ -109,6 +110,12 @@ pub enum Projection { /// Distance of the near plane. near_plane_distance: f32, + + /// Aspect ratio of the perspective transformation. + /// + /// This is typically just resolution.y / resolution.x. + /// Setting this to anything else is mostly useful when panning & zooming within a fixed transformation. + aspect_ratio: f32, }, /// Orthographic projection with the camera position at the near plane's center, @@ -152,12 +159,29 @@ impl Default for AutoSizeConfig { #[derive(Debug, Clone)] pub struct TargetConfiguration { pub name: DebugLabel, + pub resolution_in_pixel: [u32; 2], pub view_from_world: macaw::IsoTransform, pub projection_from_view: Projection, + /// Defines a viewport transformation from the projected space to the final image space. + /// + /// This can be used to implement pan & zoom independent of the camera projection. + /// Meaning that this transform allows you to zoom in on a portion of a perspectively projected + /// scene. + /// + /// Note only the relation of the rectangles in `RectTransform` is important. + /// Scaling or moving both rectangles by the same amount does not change the result. + /// + /// Internally, this is used to transform the normalized device coordinates to the given portion. + /// This transform is applied to the projection matrix. + pub viewport_transformation: RectTransform, + /// How many pixels are there per point. + /// /// I.e. the ui scaling factor. + /// Note that this does not affect any of the camera & projection properties and is only used + /// whenever point sizes were explicitly specified. pub pixels_from_point: f32, /// How [`Size::AUTO`] is interpreted. @@ -175,7 +199,9 @@ impl Default for TargetConfiguration { projection_from_view: Projection::Perspective { vertical_fov: 70.0 * std::f32::consts::TAU / 360.0, near_plane_distance: 0.01, + aspect_ratio: 1.0, }, + viewport_transformation: RectTransform::IDENTITY, pixels_from_point: 1.0, auto_size_config: Default::default(), outline_config: None, @@ -292,14 +318,12 @@ impl ViewBuilder { }, ); - let aspect_ratio = - config.resolution_in_pixel[0] as f32 / config.resolution_in_pixel[1] as f32; - let (projection_from_view, tan_half_fov, pixel_world_size_from_camera_distance) = match config.projection_from_view.clone() { Projection::Perspective { vertical_fov, near_plane_distance, + aspect_ratio, } => { // We use infinite reverse-z projection matrix // * great precision both with floating point and integer: https://developer.nvidia.com/content/depth-precision-visualized @@ -341,6 +365,8 @@ impl ViewBuilder { vertical_world_size, far_plane_distance, } => { + let aspect_ratio = + config.resolution_in_pixel[0] as f32 / config.resolution_in_pixel[1] as f32; let horizontal_world_size = vertical_world_size * aspect_ratio; // Note that we inverse z (by swapping near and far plane) to be consistent with our perspective projection. let projection_from_view = match camera_mode { @@ -365,8 +391,8 @@ impl ViewBuilder { }; let tan_half_fov = glam::vec2(f32::MAX, f32::MAX); - let pixel_world_size_from_camera_distance = - vertical_world_size / config.resolution_in_pixel[1] as f32; + let pixel_world_size_from_camera_distance = vertical_world_size + / config.resolution_in_pixel[0].max(config.resolution_in_pixel[1]) as f32; ( projection_from_view, @@ -376,6 +402,18 @@ impl ViewBuilder { } }; + // Finally, apply a viewport transformation to the projection. + let ndc_scale_and_translation = config + .viewport_transformation + .to_ndc_scale_and_translation(); + let projection_from_view = ndc_scale_and_translation * projection_from_view; + // Need to take into account that a smaller portion of the world scale is visible now. + let pixel_world_size_from_camera_distance = pixel_world_size_from_camera_distance + / ndc_scale_and_translation + .col(0) + .x + .max(ndc_scale_and_translation.col(1).y); + let mut view_from_world = config.view_from_world.to_mat4(); // For OrthographicCameraMode::TopLeftCorner, we want Z facing forward. match config.projection_from_view { @@ -387,6 +425,7 @@ impl ViewBuilder { }, Projection::Perspective { .. } => {} }; + let camera_position = config.view_from_world.inverse().translation(); let camera_forward = -view_from_world.row(2).truncate(); let projection_from_world = projection_from_view * view_from_world; @@ -662,11 +701,11 @@ impl ViewBuilder { /// Data from the picking rect needs to be retrieved via [`crate::PickingLayerProcessor::next_readback_result`]. /// To do so, you need to pass the exact same `identifier` and type of user data as you've done here: /// ```no_run - /// use re_renderer::{view_builder::ViewBuilder, IntRect, PickingLayerProcessor, RenderContext}; + /// use re_renderer::{view_builder::ViewBuilder, RectInt, PickingLayerProcessor, RenderContext}; /// fn schedule_picking_readback( /// ctx: &mut RenderContext, /// view_builder: &mut ViewBuilder, - /// picking_rect: IntRect, + /// picking_rect: RectInt, /// ) { /// view_builder.schedule_picking_rect( /// ctx, picking_rect, 42, "My screenshot".to_owned(), false, @@ -683,7 +722,7 @@ impl ViewBuilder { pub fn schedule_picking_rect( &mut self, ctx: &mut RenderContext, - picking_rect: IntRect, + picking_rect: RectInt, readback_identifier: GpuReadbackIdentifier, readback_user_data: T, show_debug_view: bool, diff --git a/crates/re_renderer/src/wgpu_buffer_types.rs b/crates/re_renderer/src/wgpu_buffer_types.rs index ee7276337b55..e4207aaed0fc 100644 --- a/crates/re_renderer/src/wgpu_buffer_types.rs +++ b/crates/re_renderer/src/wgpu_buffer_types.rs @@ -289,6 +289,13 @@ impl From for Mat4 { } } +impl From for Mat4 { + #[inline] + fn from(m: glam::Affine3A) -> Self { + glam::Mat4::from(m).into() + } +} + impl From for glam::Mat4 { #[inline] fn from(val: Mat4) -> Self { diff --git a/crates/re_renderer/src/wgpu_resources/shader_module_pool.rs b/crates/re_renderer/src/wgpu_resources/shader_module_pool.rs index 9b42f87d1788..df0f4de5c715 100644 --- a/crates/re_renderer/src/wgpu_resources/shader_module_pool.rs +++ b/crates/re_renderer/src/wgpu_resources/shader_module_pool.rs @@ -21,6 +21,7 @@ macro_rules! include_shader_module { $crate::wgpu_resources::ShaderModuleDesc { label: $crate::DebugLabel::from(stringify!($path).strip_prefix("../../shader/")), source: $crate::include_file!($path), + extra_workaround_replacements: Vec::new(), } }}; } @@ -33,6 +34,9 @@ pub struct ShaderModuleDesc { /// Path to the source code of this shader module. pub source: PathBuf, + + /// Additional text replacement workarounds that may be added on top of globally known workarounds. + pub extra_workaround_replacements: Vec<(String, String)>, } impl PartialEq for ShaderModuleDesc { @@ -46,6 +50,7 @@ impl Hash for ShaderModuleDesc { // NOTE: for a shader, the only thing that should matter is the source // code since we can have many entrypoints referring to the same file! self.source.hash(state); + self.extra_workaround_replacements.hash(state); } } @@ -62,7 +67,10 @@ impl ShaderModuleDesc { .map_err(|err| re_log::error!(err=%re_error::format(err))) .unwrap_or_default(); - for (from, to) in shader_text_workaround_replacements { + for (from, to) in shader_text_workaround_replacements + .iter() + .chain(self.extra_workaround_replacements.iter()) + { source_interpolated.contents = source_interpolated.contents.replace(from, to); } diff --git a/crates/re_viewer/src/gpu_bridge/mod.rs b/crates/re_viewer/src/gpu_bridge/mod.rs index 9cf35e1d4c01..18e3a8e0dcd4 100644 --- a/crates/re_viewer/src/gpu_bridge/mod.rs +++ b/crates/re_viewer/src/gpu_bridge/mod.rs @@ -198,6 +198,7 @@ pub fn render_image( vertical_world_size: space_from_pixel * resolution_in_pixel[1] as f32, far_plane_distance: 1000.0, }, + viewport_transformation: re_renderer::RectTransform::IDENTITY, pixels_from_point: pixels_from_points, auto_size_config: Default::default(), outline_config: None, diff --git a/crates/re_viewer/src/misc/transform_cache.rs b/crates/re_viewer/src/misc/transform_cache.rs index a909f797b2a4..e5c0504ab25e 100644 --- a/crates/re_viewer/src/misc/transform_cache.rs +++ b/crates/re_viewer/src/misc/transform_cache.rs @@ -16,11 +16,10 @@ use crate::misc::TimeControl; #[derive(Clone)] pub struct TransformCache { /// All transforms provided are relative to this reference path. - #[allow(dead_code)] reference_path: EntityPath, /// All reachable entities. - reference_from_entity_per_entity: IntMap, + reference_from_entity_per_entity: IntMap, /// All unreachable descendant paths of `reference_path`. unreachable_descendants: Vec<(EntityPath, UnreachableTransform)>, @@ -98,13 +97,13 @@ impl TransformCache { entity_db, &query, entity_prop_map, - glam::Mat4::IDENTITY, + glam::Affine3A::IDENTITY, false, ); // Walk up from the reference to the highest reachable parent. let mut encountered_pinhole = false; - let mut reference_from_ancestor = glam::Mat4::IDENTITY; + let mut reference_from_ancestor = glam::Affine3A::IDENTITY; while let Some(parent_path) = current_tree.path.parent() { let Some(parent_tree) = &entity_db.tree.subtree(&parent_path) else { // Unlike not having the space path in the hierarchy, this should be impossible. @@ -117,10 +116,13 @@ impl TransformCache { // Note that the transform at the reference is the first that needs to be inverted to "break out" of its hierarchy. // Generally, the transform _at_ a node isn't relevant to it's children, but only to get to its parent in turn! - match inverse_transform_at( + match transform_at( ¤t_tree.path, entity_db, &query, + // TODO(#1988): See comment in transform_at. This is a workaround for precision issues + // and the fact that there is no meaningful image plane distance for 3D->2D views. + |_| 500.0, &mut encountered_pinhole, ) { Err(unreachable_reason) => { @@ -129,8 +131,8 @@ impl TransformCache { break; } Ok(None) => {} - Ok(Some(child_from_parent)) => { - reference_from_ancestor *= child_from_parent; + Ok(Some(parent_from_child)) => { + reference_from_ancestor = reference_from_ancestor * parent_from_child.inverse(); } } @@ -156,7 +158,7 @@ impl TransformCache { entity_db: &EntityDb, query: &LatestAtQuery, entity_properties: &EntityPropertyMap, - reference_from_entity: glam::Mat4, + reference_from_entity: glam::Affine3A, encountered_pinhole: bool, ) { match self @@ -176,8 +178,8 @@ impl TransformCache { let reference_from_child = match transform_at( &child_tree.path, entity_db, - entity_properties, query, + |p| *entity_properties.get(p).pinhole_image_plane_distance.get(), &mut encountered_pinhole, ) { Err(unreachable_reason) => { @@ -199,10 +201,14 @@ impl TransformCache { } } + pub fn reference_path(&self) -> &EntityPath { + &self.reference_path + } + /// Retrieves the transform of on entity from its local system to the space of the reference. /// /// Returns None if the path is not reachable. - pub fn reference_from_entity(&self, entity_path: &EntityPath) -> Option { + pub fn reference_from_entity(&self, entity_path: &EntityPath) -> Option { self.reference_from_entity_per_entity .get(entity_path) .cloned() @@ -220,13 +226,13 @@ impl TransformCache { fn transform_at( entity_path: &EntityPath, entity_db: &EntityDb, - entity_properties: &EntityPropertyMap, query: &LatestAtQuery, + pinhole_image_plane_distance: impl Fn(&EntityPath) -> f32, encountered_pinhole: &mut bool, -) -> Result, UnreachableTransform> { +) -> Result, UnreachableTransform> { if let Some(transform) = query_latest_single(entity_db, entity_path, query) { match transform { - re_log_types::Transform::Rigid3(rigid) => Ok(Some(rigid.parent_from_child().to_mat4())), + re_log_types::Transform::Rigid3(rigid) => Ok(Some(rigid.parent_from_child().into())), // If we're connected via 'unknown' it's not reachable re_log_types::Transform::Unknown => Err(UnreachableTransform::UnknownTransform), @@ -236,70 +242,34 @@ fn transform_at( } else { *encountered_pinhole = true; - // A pinhole camera means that we're looking at an image. - // Images are spanned in their local x/y space. - // Center it and move it along z, scaling the further we move. - let props = entity_properties.get(entity_path); - let distance = *props.pinhole_image_plane_distance.get(); + // A pinhole camera means that we're looking at some 2D image plane from a single point (the pinhole). + // Center the image plane and move it along z, scaling the further the image plane is. + let distance = pinhole_image_plane_distance(entity_path); let focal_length = pinhole.focal_length_in_pixels(); let focal_length = glam::vec2(focal_length.x(), focal_length.y()); let scale = distance / focal_length; let translation = (-pinhole.principal_point() * scale).extend(distance); - let parent_from_child = glam::Mat4::from_scale_rotation_translation( + let parent_from_child = glam::Affine3A::from_translation(translation) + // We want to preserve any depth that might be on the pinhole image. // Use harmonic mean of x/y scale for those. - scale.extend(1.0 / (1.0 / scale.x + 1.0 / scale.y)), - glam::Quat::IDENTITY, - translation, - ); - - Ok(Some(parent_from_child)) - } - } - } - } else { - Ok(None) - } -} + * glam::Affine3A::from_scale( + scale.extend(2.0 / (1.0 / scale.x + 1.0 / scale.y)), + ); -fn inverse_transform_at( - entity_path: &EntityPath, - entity_db: &EntityDb, - query: &LatestAtQuery, - encountered_pinhole: &mut bool, -) -> Result, UnreachableTransform> { - if let Some(parent_transform) = query_latest_single(entity_db, entity_path, query) { - match parent_transform { - re_log_types::Transform::Rigid3(rigid) => Ok(Some(rigid.child_from_parent().to_mat4())), - // If we're connected via 'unknown', everything except whats under `parent_tree` is unreachable - re_log_types::Transform::Unknown => Err(UnreachableTransform::UnknownTransform), + // Above calculation is nice for a certain kind of visualizing a projected image plane, + // but the image plane distance is arbitrary and there might be other, better visualizations! - re_log_types::Transform::Pinhole(pinhole) => { - if *encountered_pinhole { - Err(UnreachableTransform::NestedPinholeCameras) - } else { - *encountered_pinhole = true; + // TODO(#1988): + // As such we don't ever want to invert this matrix! + // However, currently our 2D views require do to exactly that since we're forced to + // build a relationship between the 2D plane and the 3D world, when actually the 2D plane + // should have infinite depth! + // The inverse of this matrix *is* working for this, but quickly runs into precision issues. + // See also `ui_2d.rs#setup_target_config` - // TODO(andreas): If we don't have a resolution we don't know the FOV ergo we don't know how to project. Unclear what to do. - if let (Some(resolution), Some(fov_y)) = (pinhole.resolution(), pinhole.fov_y()) - { - let translation = pinhole.principal_point().extend(-100.0); // Large Y offset so this is in front of all 2d that came so far. TODO(andreas): Find better solution - Ok(Some( - glam::Mat4::from_scale_rotation_translation( - // Scaled with 0.5 since perspective_infinite_lh uses NDC, i.e. [-1; 1] range. - (resolution * 0.5).extend(1.0), - glam::Quat::IDENTITY, - translation, - ) * glam::Mat4::perspective_infinite_lh( - fov_y, - pinhole.aspect_ratio().unwrap_or(1.0), - 0.0, - ), - )) - } else { - Err(UnreachableTransform::InversePinholeCameraWithoutResolution) - } + Ok(Some(parent_from_child)) } } } diff --git a/crates/re_viewer/src/ui/selection_panel.rs b/crates/re_viewer/src/ui/selection_panel.rs index f84b282b4b6a..841ea082c5f6 100644 --- a/crates/re_viewer/src/ui/selection_panel.rs +++ b/crates/re_viewer/src/ui/selection_panel.rs @@ -7,12 +7,9 @@ use re_log_types::{ TimeType, Transform, }; -use crate::{ - ui::{view_spatial::SpatialNavigationMode, Blueprint}, - Item, UiVerbosity, ViewerContext, -}; +use crate::{ui::Blueprint, Item, UiVerbosity, ViewerContext}; -use super::{data_ui::DataUi, space_view::ViewState}; +use super::{data_ui::DataUi, space_view::ViewState, view_spatial::SpatialNavigationMode}; // --- diff --git a/crates/re_viewer/src/ui/view_spatial/scene/picking.rs b/crates/re_viewer/src/ui/view_spatial/scene/picking.rs index b2230142eb39..cc5ec43ce7a4 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/picking.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/picking.rs @@ -182,7 +182,7 @@ fn picking_gpu( // First, figure out where on the rect the cursor is by now. // (for simplicity, we assume the screen hasn't been resized) let pointer_on_picking_rect = - context.pointer_in_pixel - gpu_picking_result.rect.left_top.as_vec2(); + context.pointer_in_pixel - gpu_picking_result.rect.min.as_vec2(); // The cursor might have moved outside of the rect. Clamp it back in. let pointer_on_picking_rect = pointer_on_picking_rect.clamp( glam::Vec2::ZERO, diff --git a/crates/re_viewer/src/ui/view_spatial/scene/primitives.rs b/crates/re_viewer/src/ui/view_spatial/scene/primitives.rs index 407d1eec47fa..5a2e2afa4955 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/primitives.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/primitives.rs @@ -105,22 +105,12 @@ impl SceneSpatialPrimitives { // we calculate a per batch bounding box for lines and points. // TODO(andreas): We should keep these around to speed up picking! for (batch, vertex_iter) in points.iter_vertices_by_batch() { - // Only use points which are an IsoTransform to update the bounding box - // This prevents crazy bounds-increases when projecting 3d to 2d - // See: https://github.com/rerun-io/rerun/issues/1203 - if let Some(transform) = macaw::IsoTransform::from_mat4(&batch.world_from_obj) { - let batch_bb = macaw::BoundingBox::from_points(vertex_iter.map(|v| v.position)); - *bounding_box = bounding_box.union(batch_bb.transform_affine3(&transform.into())); - } + let batch_bb = macaw::BoundingBox::from_points(vertex_iter.map(|v| v.position)); + *bounding_box = bounding_box.union(batch_bb.transform_affine3(&batch.world_from_obj)); } for (batch, vertex_iter) in line_strips.iter_vertices_by_batch() { - // Only use points which are an IsoTransform to update the bounding box - // This prevents crazy bounds-increases when projecting 3d to 2d - // See: https://github.com/rerun-io/rerun/issues/1203 - if let Some(transform) = macaw::IsoTransform::from_mat4(&batch.world_from_obj) { - let batch_bb = macaw::BoundingBox::from_points(vertex_iter.map(|v| v.position)); - *bounding_box = bounding_box.union(batch_bb.transform_affine3(&transform.into())); - } + let batch_bb = macaw::BoundingBox::from_points(vertex_iter.map(|v| v.position)); + *bounding_box = bounding_box.union(batch_bb.transform_affine3(&batch.world_from_obj)); } for mesh in meshes { diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/arrows3d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/arrows3d.rs index f40ca9025684..60d4e5fccf11 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/arrows3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/arrows3d.rs @@ -1,4 +1,3 @@ -use glam::Mat4; use re_data_store::EntityPath; use re_log_types::{ component_types::{ColorRGBA, InstanceKey, Label, Radius}, @@ -21,7 +20,7 @@ impl Arrows3DPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, highlights: &SpaceViewHighlights, ) -> Result<(), QueryError> { scene.num_logged_3d_objects += 1; diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes2d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes2d.rs index 8849d7a2de7a..3de91ba5ad94 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes2d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes2d.rs @@ -1,4 +1,3 @@ -use glam::Mat4; use re_data_store::EntityPath; use re_log_types::{ component_types::{ClassId, ColorRGBA, InstanceKey, Label, Radius, Rect2D}, @@ -27,7 +26,7 @@ impl Boxes2DPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, highlights: &SpaceViewHighlights, ) -> Result<(), QueryError> { scene.num_logged_2d_objects += 1; diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes3d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes3d.rs index 6f06a36f440f..991a45e18bc0 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/boxes3d.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::EntityPath; use re_log_types::{ component_types::{Box3D, ClassId, ColorRGBA, InstanceKey, Label, Quaternion, Radius, Vec3D}, @@ -26,7 +24,7 @@ impl Boxes3DPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, entity_highlight: &SpaceViewOutlineMasks, ) -> Result<(), QueryError> { scene.num_logged_3d_objects += 1; diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/cameras.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/cameras.rs index c4e967341eb5..7a000c0f9e70 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/cameras.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/cameras.rs @@ -81,20 +81,32 @@ impl CamerasPart { return; }; - // If this transform is not representable as rigid transform, the camera is probably under another camera transform, - // in which case we don't (yet) know how to deal with this! - let Some(world_from_camera) = macaw::IsoTransform::from_mat4(&world_from_parent) else { + let frustum_length = *props.pinhole_image_plane_distance.get(); + + // If the camera is our reference, there is nothing for us to display. + if transforms.reference_path() == ent_path { + scene.space_cameras.push(SpaceCamera3D { + ent_path: ent_path.clone(), + view_coordinates, + world_from_camera: macaw::IsoTransform::IDENTITY, + pinhole: Some(pinhole), + picture_plane_distance: frustum_length, + }); return; - }; + } - let frustum_length = *props.pinhole_image_plane_distance.get(); + // If this transform is not representable an iso transform transform we can't display it yet. + // This would happen if the camera is under another camera or under a transform with non-uniform scale. + let Some(world_from_camera) = macaw::IsoTransform::from_mat4(&world_from_parent.into()) else { + return; + }; scene.space_cameras.push(SpaceCamera3D { ent_path: ent_path.clone(), view_coordinates, world_from_camera, pinhole: Some(pinhole), - picture_plane_distance: Some(frustum_length), + picture_plane_distance: frustum_length, }); // TODO(andreas): FOV fallback doesn't make much sense. What does pinhole without fov mean? diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/images.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/images.rs index 8a40a7957168..d0f852ce9d69 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/images.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/images.rs @@ -28,7 +28,7 @@ use super::ScenePart; fn to_textured_rect( ctx: &mut ViewerContext<'_>, annotations: &Annotations, - world_from_obj: glam::Mat4, + world_from_obj: glam::Affine3A, ent_path: &EntityPath, tensor: &DecodedTensor, multiplicative_tint: egui::Rgba, @@ -169,7 +169,7 @@ impl ImagesPart { transforms: &TransformCache, properties: &EntityProperties, ent_path: &EntityPath, - world_from_obj: glam::Mat4, + world_from_obj: glam::Affine3A, highlights: &SpaceViewHighlights, ) -> Result<(), QueryError> { crate::profile_function!(); @@ -360,7 +360,7 @@ impl ImagesPart { }; scene.primitives.depth_clouds.clouds.push(DepthCloud { - world_from_obj, + world_from_obj: world_from_obj.into(), depth_camera_intrinsics: intrinsics.image_from_cam.into(), world_depth_from_texture_depth, point_radius_from_world_depth, diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines2d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines2d.rs index f07375679530..0a9b47b9b5ed 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines2d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines2d.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::EntityPath; use re_log_types::{ component_types::{ColorRGBA, InstanceKey, LineStrip2D, Radius}, @@ -22,7 +20,7 @@ impl Lines2DPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, entity_highlight: &SpaceViewOutlineMasks, ) -> Result<(), QueryError> { scene.num_logged_2d_objects += 1; diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines3d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines3d.rs index 4a527a291fec..492e45391f15 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/lines3d.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::EntityPath; use re_log_types::{ component_types::{ColorRGBA, InstanceKey, LineStrip3D, Radius}, @@ -22,7 +20,7 @@ impl Lines3DPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, entity_highlight: &SpaceViewOutlineMasks, ) -> Result<(), QueryError> { scene.num_logged_3d_objects += 1; diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/meshes.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/meshes.rs index 4e935a2663f4..ec9f452ca039 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/meshes.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/meshes.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::EntityPath; use re_log_types::{ component_types::{ColorRGBA, InstanceKey}, @@ -25,14 +23,13 @@ impl MeshPart { scene: &mut SceneSpatial, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, ctx: &mut ViewerContext<'_>, highlights: &SpaceViewHighlights, ) -> Result<(), QueryError> { scene.num_logged_3d_objects += 1; let _default_color = DefaultColor::EntityPath(ent_path); - let world_from_obj_affine = glam::Affine3A::from_mat4(world_from_obj); let entity_highlight = highlights.entity_outline_mask(ent_path.hash()); let visitor = @@ -56,7 +53,7 @@ impl MeshPart { ) .map(|cpu_mesh| MeshSource { picking_instance_hash, - world_from_mesh: world_from_obj_affine, + world_from_mesh: world_from_obj, mesh: cpu_mesh, outline_mask_ids, }) diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points2d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points2d.rs index 41bdc4a88534..c91c3d55055c 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points2d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points2d.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::{EntityPath, InstancePathHash}; use re_log_types::{ component_types::{ClassId, ColorRGBA, InstanceKey, KeypointId, Label, Point2D, Radius}, @@ -64,7 +62,7 @@ impl Points2DPart { query: &SceneQuery<'_>, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, entity_highlight: &SpaceViewOutlineMasks, ) -> Result<(), QueryError> { crate::profile_function!(); @@ -112,6 +110,10 @@ impl Points2DPart { .primitives .points .batch("2d points") + .flags( + re_renderer::renderer::PointCloudBatchFlags::DRAW_AS_CIRCLES + | re_renderer::renderer::PointCloudBatchFlags::ENABLE_SHADING, + ) .world_from_obj(world_from_obj) .outline_mask_ids(entity_highlight.overall) .picking_object_id(re_renderer::PickingLayerObjectId(ent_path.hash64())); diff --git a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points3d.rs b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points3d.rs index 1b6ca61f9669..1da202dd38fc 100644 --- a/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/scene/scene_part/points3d.rs @@ -1,5 +1,3 @@ -use glam::Mat4; - use re_data_store::{EntityPath, InstancePathHash}; use re_log_types::{ component_types::{ClassId, ColorRGBA, InstanceKey, KeypointId, Label, Point3D, Radius}, @@ -35,7 +33,7 @@ impl Points3DPart { instance_path_hashes: &'a [InstancePathHash], colors: &'a [egui::Color32], annotation_infos: &'a [ResolvedAnnotationInfo], - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, ) -> Result + 'a, QueryError> { let labels = itertools::izip!( annotation_infos.iter(), @@ -70,7 +68,7 @@ impl Points3DPart { query: &SceneQuery<'_>, entity_view: &EntityView, ent_path: &EntityPath, - world_from_obj: Mat4, + world_from_obj: glam::Affine3A, entity_highlight: &SpaceViewOutlineMasks, ) -> Result<(), QueryError> { crate::profile_function!(); diff --git a/crates/re_viewer/src/ui/view_spatial/space_camera_3d.rs b/crates/re_viewer/src/ui/view_spatial/space_camera_3d.rs index f18d17a504a2..589d0a205707 100644 --- a/crates/re_viewer/src/ui/view_spatial/space_camera_3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/space_camera_3d.rs @@ -22,8 +22,8 @@ pub struct SpaceCamera3D { /// The projection transform of a child-entity. pub pinhole: Option, - /// Optional distance of a picture plane from the camera. - pub picture_plane_distance: Option, + /// Distance of a picture plane from the camera. + pub picture_plane_distance: f32, } impl SpaceCamera3D { diff --git a/crates/re_viewer/src/ui/view_spatial/ui.rs b/crates/re_viewer/src/ui/view_spatial/ui.rs index efe2b655f6d5..1a78d99bdf02 100644 --- a/crates/re_viewer/src/ui/view_spatial/ui.rs +++ b/crates/re_viewer/src/ui/view_spatial/ui.rs @@ -152,29 +152,21 @@ impl ViewSpatialState { ) { crate::profile_function!(); - let scene_size = self.scene_bbox_accum.size().length(); - let query = ctx.current_query(); let entity_paths = data_blueprint.entity_paths().clone(); // TODO(andreas): Workaround borrow checker for entity_path in entity_paths { - Self::update_pinhole_property_heuristics( - ctx, - data_blueprint, - &query, - &entity_path, - scene_size, - ); + self.update_pinhole_property_heuristics(ctx, data_blueprint, &query, &entity_path); self.update_depth_cloud_property_heuristics(ctx, data_blueprint, &query, &entity_path); } } fn update_pinhole_property_heuristics( + &self, ctx: &mut ViewerContext<'_>, data_blueprint: &mut DataBlueprintTree, query: &re_arrow_store::LatestAtQuery, entity_path: &EntityPath, - scene_size: f32, ) { if let Some(re_log_types::Transform::Pinhole(_)) = query_latest_single::( @@ -183,14 +175,14 @@ impl ViewSpatialState { query, ) { - let default_image_plane_distance = if scene_size.is_finite() && scene_size > 0.0 { - scene_size * 0.05 - } else { - 1.0 - }; - let mut properties = data_blueprint.data_blueprints_individual().get(entity_path); if properties.pinhole_image_plane_distance.is_auto() { + let scene_size = self.scene_bbox_accum.size().length(); + let default_image_plane_distance = if scene_size.is_finite() && scene_size > 0.0 { + scene_size * 0.05 + } else { + 1.0 + }; properties.pinhole_image_plane_distance = EditableAutoValue::Auto(default_image_plane_distance); data_blueprint @@ -537,8 +529,7 @@ fn axis_name(axis: Option) -> String { pub fn create_labels( scene_ui: &mut SceneSpatialUiData, - ui_from_space2d: egui::emath::RectTransform, - space2d_from_ui: egui::emath::RectTransform, + ui_from_canvas: egui::emath::RectTransform, eye3d: &Eye, parent_ui: &mut egui::Ui, highlights: &SpaceViewHighlights, @@ -548,7 +539,7 @@ pub fn create_labels( let mut label_shapes = Vec::with_capacity(scene_ui.labels.len() * 2); - let ui_from_world_3d = eye3d.ui_from_world(*ui_from_space2d.to()); + let ui_from_world_3d = eye3d.ui_from_world(*ui_from_canvas.to()); for label in &scene_ui.labels { let (wrap_width, text_anchor_pos) = match label.target { @@ -557,7 +548,7 @@ pub fn create_labels( if nav_mode == SpatialNavigationMode::ThreeD { continue; } - let rect_in_ui = ui_from_space2d.transform_rect(rect); + let rect_in_ui = ui_from_canvas.transform_rect(rect); ( // Place the text centered below the rect (rect_in_ui.width() - 4.0).at_least(60.0), @@ -569,10 +560,14 @@ pub fn create_labels( if nav_mode == SpatialNavigationMode::ThreeD { continue; } - let pos_in_ui = ui_from_space2d.transform_pos(pos); + let pos_in_ui = ui_from_canvas.transform_pos(pos); (f32::INFINITY, pos_in_ui + egui::vec2(0.0, 3.0)) } UiLabelTarget::Position3D(pos) => { + // TODO(#1640): 3D labels are not visible in 2D for now. + if nav_mode == SpatialNavigationMode::TwoD { + continue; + } let pos_in_ui = ui_from_world_3d * pos.extend(1.0); if pos_in_ui.w <= 0.0 { continue; // behind camera @@ -627,7 +622,7 @@ pub fn create_labels( label_shapes.push(egui::Shape::galley(text_rect.center_top(), galley)); scene_ui.pickable_ui_rects.push(( - space2d_from_ui.transform_rect(bg_rect), + ui_from_canvas.inverse().transform_rect(bg_rect), label.labeled_instance, )); } @@ -718,7 +713,7 @@ pub fn picking( let _ = view_builder.schedule_picking_rect( ctx.render_ctx, - re_renderer::IntRect::from_middle_and_extent( + re_renderer::RectInt::from_middle_and_extent( picking_context.pointer_in_pixel.as_ivec2(), glam::uvec2(picking_rect_size, picking_rect_size), ), diff --git a/crates/re_viewer/src/ui/view_spatial/ui_2d.rs b/crates/re_viewer/src/ui/view_spatial/ui_2d.rs index 23dfeaf7a882..e2123b513994 100644 --- a/crates/re_viewer/src/ui/view_spatial/ui_2d.rs +++ b/crates/re_viewer/src/ui/view_spatial/ui_2d.rs @@ -1,7 +1,8 @@ use eframe::emath::RectTransform; use egui::{pos2, vec2, Align2, Color32, NumExt as _, Pos2, Rect, ScrollArea, Shape, Vec2}; use macaw::IsoTransform; -use re_data_store::{EntityPath, EntityPropertyMap}; +use re_data_store::{query_latest_single, EntityPath, EntityPropertyMap}; +use re_log_types::Pinhole; use re_renderer::view_builder::{TargetConfiguration, ViewBuilder}; use super::{ @@ -59,26 +60,21 @@ impl View2DState { /// Returns `(desired_size, scroll_offset)` where: /// - `desired_size` is the size of the painter necessary to capture the zoomed view in ui points /// - `scroll_offset` is the position of the `ScrollArea` offset in ui points - fn desired_size_and_offset( - &self, - available_size: Vec2, - scene_rect_accum: Rect, - ) -> (Vec2, Vec2) { + fn desired_size_and_offset(&self, available_size: Vec2, canvas_rect: Rect) -> (Vec2, Vec2) { match self.zoom { ZoomState2D::Scaled { scale, center, .. } => { - let desired_size = scene_rect_accum.size() * scale; + let desired_size = canvas_rect.size() * scale; // Try to keep the center of the scene in the middle of the available size - let scroll_offset = (center.to_vec2() - scene_rect_accum.left_top().to_vec2()) - * scale + let scroll_offset = (center.to_vec2() - canvas_rect.left_top().to_vec2()) * scale - available_size / 2.0; (desired_size, scroll_offset) } ZoomState2D::Auto => { // Otherwise, we autoscale the space to fit available area while maintaining aspect ratio - let scene_bbox = if scene_rect_accum.is_positive() { - scene_rect_accum + let scene_bbox = if canvas_rect.is_positive() { + canvas_rect } else { Rect::from_min_max(Pos2::ZERO, Pos2::new(1.0, 1.0)) }; @@ -101,7 +97,7 @@ impl View2DState { &mut self, response: &egui::Response, ui_to_space: egui::emath::RectTransform, - scene_rect_accum: Rect, + canvas_rect: Rect, available_size: Vec2, ) { // Determine if we are zooming @@ -117,14 +113,14 @@ impl View2DState { if let Some(input_zoom) = hovered_zoom { if input_zoom > 1.0 { let scale = response.rect.height() / ui_to_space.to().height(); - let center = scene_rect_accum.center(); + let center = canvas_rect.center(); self.zoom = ZoomState2D::Scaled { scale, center, accepting_scroll: false, }; // Recursively update now that we have initialized `ZoomState` to `Scaled` - self.update(response, ui_to_space, scene_rect_accum, available_size); + self.update(response, ui_to_space, canvas_rect, available_size); } } } @@ -153,7 +149,8 @@ impl View2DState { // Moving the center in the direction of the desired shift center += shift_in_space; } - scale = new_scale; + // Don't show less than one horizontal scene unit in the entire screen. + scale = new_scale.at_most(available_size.x); accepting_scroll = false; } @@ -181,8 +178,8 @@ impl View2DState { } // If our zoomed region is smaller than the available size - if scene_rect_accum.size().x * scale < available_size.x - && scene_rect_accum.size().y * scale < available_size.y + if canvas_rect.size().x * scale < available_size.x + && canvas_rect.size().y * scale < available_size.y { self.zoom = ZoomState2D::Auto; } @@ -191,7 +188,7 @@ impl View2DState { /// Take the offset from the `ScrollArea` and apply it back to center so that other /// scroll interfaces work as expected. - fn capture_scroll(&mut self, offset: Vec2, available_size: Vec2, scene_rect_accum: Rect) { + fn capture_scroll(&mut self, offset: Vec2, available_size: Vec2, canvas_rect: Rect) { if let ZoomState2D::Scaled { scale, accepting_scroll, @@ -199,7 +196,7 @@ impl View2DState { } = self.zoom { if accepting_scroll { - let center = scene_rect_accum.left_top() + (available_size / 2.0 + offset) / scale; + let center = canvas_rect.left_top() + (available_size / 2.0 + offset) / scale; self.zoom = ZoomState2D::Scaled { scale, center, @@ -222,7 +219,7 @@ pub fn view_2d( ui: &mut egui::Ui, state: &mut ViewSpatialState, space: &EntityPath, - scene: SceneSpatial, + mut scene: SceneSpatial, scene_rect_accum: Rect, space_view_id: SpaceViewId, highlights: &SpaceViewHighlights, @@ -233,9 +230,34 @@ pub fn view_2d( // Save off the available_size since this is used for some of the layout updates later let available_size = ui.available_size(); + // Determine the canvas which determines the extent of the explorable scene coordinates, + // and thus the size of the scroll area. + // + // TODO(andreas): We want to move away from the scroll area and instead work with open ended 2D scene coordinates! + // The term canvas might then refer to the area in scene coordinates visible at a given moment. + // Orthogonally, we'll want to visualize the resolution rectangle of the pinhole camera. + // + // For that we need to check if this is defined by a pinhole camera. + // Note that we can't rely on the camera being part of scene.space_cameras since that requires + // the camera to be added to the scene! + let pinhole = query_latest_single( + &ctx.log_db.entity_db, + space, + &ctx.rec_cfg.time_ctrl.current_query(), + ) + .and_then(|transform| match transform { + re_log_types::Transform::Pinhole(pinhole) => Some(pinhole), + _ => None, + }); + let canvas_rect = pinhole + .and_then(|p| p.resolution()) + .map_or(scene_rect_accum, |res| { + Rect::from_min_max(egui::Pos2::ZERO, egui::pos2(res.x, res.y)) + }); + let (desired_size, offset) = state .state_2d - .desired_size_and_offset(available_size, scene_rect_accum); + .desired_size_and_offset(available_size, canvas_rect); // Bound the offset based on sizes // TODO(jleibs): can we derive this from the ScrollArea shape? @@ -247,184 +269,212 @@ pub fn view_2d( .auto_shrink([false, false]); let scroll_out = scroll_area.show(ui, |ui| { - view_2d_scrollable( - desired_size, - available_size, - ctx, - ui, - state, - space, - scene, - scene_rect_accum, - space_view_id, - highlights, - entity_properties, - ) - }); + let (mut response, painter) = + ui.allocate_painter(desired_size, egui::Sense::click_and_drag()); - // Update the scroll area based on the computed offset - // This handles cases of dragging/zooming the space - state - .state_2d - .capture_scroll(scroll_out.state.offset, available_size, scene_rect_accum); - scroll_out.inner -} - -/// Create the real 2D view inside the scrollable area -#[allow(clippy::too_many_arguments)] -fn view_2d_scrollable( - desired_size: Vec2, - available_size: Vec2, - ctx: &mut ViewerContext<'_>, - parent_ui: &mut egui::Ui, - state: &mut ViewSpatialState, - space: &EntityPath, - mut scene: SceneSpatial, - scene_rect_accum: Rect, - space_view_id: SpaceViewId, - highlights: &SpaceViewHighlights, - entity_properties: &EntityPropertyMap, -) -> egui::Response { - let (mut response, painter) = - parent_ui.allocate_painter(desired_size, egui::Sense::click_and_drag()); + if !response.rect.is_positive() { + return response; // protect against problems with zero-sized views + } - if !response.rect.is_positive() { - return response; // protect against problems with zero-sized views - } + let ui_from_canvas = egui::emath::RectTransform::from_to(canvas_rect, response.rect); + let canvas_from_ui = ui_from_canvas.inverse(); - // Create our transforms. - let ui_from_space = egui::emath::RectTransform::from_to(scene_rect_accum, response.rect); - let space_from_ui = ui_from_space.inverse(); - let space_from_points = space_from_ui.scale().y; - let points_from_pixels = 1.0 / painter.ctx().pixels_per_point(); - let space_from_pixel = space_from_points * points_from_pixels; + state + .state_2d + .update(&response, canvas_from_ui, canvas_rect, available_size); - state - .state_2d - .update(&response, space_from_ui, scene_rect_accum, available_size); + // TODO(andreas): Use the same eye & transformations as in `setup_target_config`. + let eye = Eye { + world_from_view: IsoTransform::IDENTITY, + fov_y: None, + }; - let eye = Eye { - world_from_view: IsoTransform::IDENTITY, - fov_y: None, - }; + let Ok(target_config) = setup_target_config( + &painter, + canvas_from_ui, + &space.to_string(), + state.auto_size_config(), + scene + .primitives + .any_outlines, + pinhole, + ) else { + return response; + }; - let Ok(target_config) = setup_target_config( - &painter, - space_from_ui, - space_from_pixel, - &space.to_string(), - state.auto_size_config(), - scene - .primitives - .any_outlines, - ) else { - return response; - }; + let mut view_builder = ViewBuilder::new(ctx.render_ctx, target_config); - let mut view_builder = ViewBuilder::new(ctx.render_ctx, target_config); - - // Create labels now since their shapes participate are added to scene.ui for picking. - let label_shapes = create_labels( - &mut scene.ui, - ui_from_space, - space_from_ui, - &eye, - parent_ui, - highlights, - SpatialNavigationMode::TwoD, - ); - - if !re_ui::egui_helpers::is_anything_being_dragged(parent_ui.ctx()) { - response = picking( - ctx, - response, - space_from_ui, - painter.clip_rect(), - parent_ui, - eye, - &mut view_builder, - space_view_id, - state, - &scene, - space, - entity_properties, + // Create labels now since their shapes participate are added to scene.ui for picking. + let label_shapes = create_labels( + &mut scene.ui, + ui_from_canvas, + &eye, + ui, + highlights, + SpatialNavigationMode::TwoD, ); - } - // ------------------------------------------------------------------------ + if !re_ui::egui_helpers::is_anything_being_dragged(ui.ctx()) { + response = picking( + ctx, + response, + canvas_from_ui, + painter.clip_rect(), + ui, + eye, + &mut view_builder, + space_view_id, + state, + &scene, + space, + entity_properties, + ); + } - // Screenshot context menu. - let (response, screenshot_mode) = screenshot_context_menu(ctx, response); - if let Some(mode) = screenshot_mode { - let _ = - view_builder.schedule_screenshot(ctx.render_ctx, space_view_id.gpu_readback_id(), mode); - } + // ------------------------------------------------------------------------ - // Draw a re_renderer driven view. - // Camera & projection are configured to ingest space coordinates directly. - { - let command_buffer = match fill_view_builder( - ctx.render_ctx, - &mut view_builder, - scene.primitives, - &ScreenBackground::ClearColor(parent_ui.visuals().extreme_bg_color.into()), - ) { - Ok(command_buffer) => command_buffer, - Err(err) => { - re_log::error!("Failed to fill view builder: {}", err); - return response; - } - }; - painter.add(gpu_bridge::renderer_paint_callback( - ctx.render_ctx, - command_buffer, - view_builder, - painter.clip_rect(), - painter.ctx().pixels_per_point(), + // Screenshot context menu. + let (response, screenshot_mode) = screenshot_context_menu(ctx, response); + if let Some(mode) = screenshot_mode { + view_builder + .schedule_screenshot(ctx.render_ctx, space_view_id.gpu_readback_id(), mode) + .ok(); + } + + // Draw a re_renderer driven view. + // Camera & projection are configured to ingest space coordinates directly. + { + let command_buffer = match fill_view_builder( + ctx.render_ctx, + &mut view_builder, + scene.primitives, + &ScreenBackground::ClearColor(ui.visuals().extreme_bg_color.into()), + ) { + Ok(command_buffer) => command_buffer, + Err(err) => { + re_log::error!("Failed to fill view builder: {}", err); + return response; + } + }; + painter.add(gpu_bridge::renderer_paint_callback( + ctx.render_ctx, + command_buffer, + view_builder, + painter.clip_rect(), + painter.ctx().pixels_per_point(), + )); + } + + painter.extend(show_projections_from_3d_space( + ctx, + ui, + space, + &ui_from_canvas, )); - } - painter.extend(show_projections_from_3d_space( - ctx, - parent_ui, - space, - &ui_from_space, - )); + // Add egui driven labels on top of re_renderer content. + painter.extend(label_shapes); - // Add egui driven labels on top of re_renderer content. - painter.extend(label_shapes); + response + }); - response + // Update the scroll area based on the computed offset + // This handles cases of dragging/zooming the space + state + .state_2d + .capture_scroll(scroll_out.state.offset, available_size, scene_rect_accum); + scroll_out.inner } fn setup_target_config( painter: &egui::Painter, - space_from_ui: RectTransform, - space_from_pixel: f32, + canvas_from_ui: RectTransform, space_name: &str, auto_size_config: re_renderer::AutoSizeConfig, any_outlines: bool, + pinhole: Option, ) -> anyhow::Result { let pixels_from_points = painter.ctx().pixels_per_point(); let resolution_in_pixel = gpu_bridge::viewport_resolution_in_pixels(painter.clip_rect(), pixels_from_points); anyhow::ensure!(resolution_in_pixel[0] > 0 && resolution_in_pixel[1] > 0); - let camera_position_space = space_from_ui.transform_pos(painter.clip_rect().min); + // TODO(#1988): + // The camera setup is done in a way that works well with the way we inverse pinhole camera transformations right now. + // This has a lot of issues though, mainly because we pretend that the 2D plane has a defined depth. + // * very bad depth precision as we limit the depth range from 0 to focal_length_in_pixels + // * depth values in depth buffer are almost non-sensical and can't be used easily for picking + // * 2D rendering can use depth buffer for layering only in a very limited way + // + // Instead we should treat 2D objects as pre-projected with their depth information already lost. + // + // We would define two cameras then: + // * an orthographic camera for handling 2D rendering + // * a perspective camera *at the origin* for 3D rendering + // Both share the same view-builder and the same viewport transformation but are independent otherwise. + + // For simplicity (and to reduce surprises!) we always render with a pinhole camera. + // Make up a default pinhole camera if we don't have one placing it in a way to look at the entire space. + let canvas_size = glam::vec2(canvas_from_ui.to().width(), canvas_from_ui.to().height()); + let default_principal_point = canvas_size * 0.5; + let pinhole = pinhole.unwrap_or_else(|| { + let focal_length_in_pixels = canvas_size.x; + + re_log_types::Pinhole { + image_from_cam: glam::Mat3::from_cols( + glam::vec3(focal_length_in_pixels, 0.0, 0.0), + glam::vec3(0.0, focal_length_in_pixels, 0.0), + default_principal_point.extend(1.0), + ) + .into(), + resolution: Some(canvas_size.into()), + } + }); + + let projection_from_view = re_renderer::view_builder::Projection::Perspective { + vertical_fov: pinhole.fov_y().unwrap_or(Eye::DEFAULT_FOV_Y), + near_plane_distance: 0.01, + aspect_ratio: pinhole + .aspect_ratio() + .unwrap_or(canvas_size.x / canvas_size.y), + }; + + // Put the camera at the position where it sees the entire image plane as defined + // by the pinhole camera. + // TODO(andreas): Support anamorphic pinhole cameras properly. + let focal_length = pinhole.focal_length_in_pixels(); + let focal_length = 2.0 / (1.0 / focal_length.x() + 1.0 / focal_length.y()); // harmonic mean + let Some(view_from_world) = macaw::IsoTransform::look_at_rh( + pinhole + .principal_point() + .extend(-focal_length), + pinhole.principal_point().extend(0.0), + -glam::Vec3::Y, + ) else { + anyhow::bail!("Failed to compute camera transform for 2D view."); + }; + + // Cut to the portion of the currently visible ui area. + let mut viewport_transformation = re_renderer::RectTransform { + region_of_interest: egui_rect_to_re_renderer(painter.clip_rect()), + region: egui_rect_to_re_renderer(*canvas_from_ui.from()), + }; + + // The principal point might not be quite centered. + // We need to account for this translation in the viewport transformation. + let principal_point_offset = default_principal_point - pinhole.principal_point(); + let ui_from_canvas_scale = canvas_from_ui.inverse().scale(); + viewport_transformation.region_of_interest.min += + principal_point_offset * glam::vec2(ui_from_canvas_scale.x, ui_from_canvas_scale.y); Ok({ let name = space_name.into(); - let top_left_position = glam::vec2(camera_position_space.x, camera_position_space.y); TargetConfiguration { name, resolution_in_pixel, - view_from_world: macaw::IsoTransform::from_translation(-top_left_position.extend(0.0)), - projection_from_view: re_renderer::view_builder::Projection::Orthographic { - camera_mode: - re_renderer::view_builder::OrthographicCameraMode::TopLeftCornerAndExtendZ, - vertical_world_size: space_from_pixel * resolution_in_pixel[1] as f32, - far_plane_distance: 1000.0, - }, + view_from_world, + projection_from_view, + viewport_transformation, pixels_from_point: pixels_from_points, auto_size_config, outline_config: any_outlines.then(|| outline_config(painter.ctx())), @@ -432,13 +482,20 @@ fn setup_target_config( }) } +fn egui_rect_to_re_renderer(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( ctx: &ViewerContext<'_>, ui: &egui::Ui, space: &EntityPath, - ui_from_space: &RectTransform, + ui_from_canvas: &RectTransform, ) -> Vec { let mut shapes = Vec::new(); if let HoveredSpace::ThreeD { @@ -450,7 +507,7 @@ fn show_projections_from_3d_space( if space_2d == space { if let Some(pos_2d) = pos_2d { // User is hovering a 2D point inside a 3D view. - let pos_in_ui = ui_from_space.transform_pos(pos2(pos_2d.x, pos_2d.y)); + let pos_in_ui = ui_from_canvas.transform_pos(pos2(pos_2d.x, pos_2d.y)); let radius = 4.0; shapes.push(Shape::circle_filled( pos_in_ui, diff --git a/crates/re_viewer/src/ui/view_spatial/ui_3d.rs b/crates/re_viewer/src/ui/view_spatial/ui_3d.rs index 8797c73e46ae..e1c60ca72e09 100644 --- a/crates/re_viewer/src/ui/view_spatial/ui_3d.rs +++ b/crates/re_viewer/src/ui/view_spatial/ui_3d.rs @@ -329,7 +329,9 @@ pub fn view_3d( projection_from_view: Projection::Perspective { vertical_fov: eye.fov_y.unwrap_or(Eye::DEFAULT_FOV_Y), near_plane_distance: eye.near(), + aspect_ratio: resolution_in_pixel[0] as f32 / resolution_in_pixel[1] as f32, }, + viewport_transformation: re_renderer::RectTransform::IDENTITY, pixels_from_point: ui.ctx().pixels_per_point(), auto_size_config: state.auto_size_config(), @@ -346,7 +348,6 @@ pub fn view_3d( let label_shapes = create_labels( &mut scene.ui, RectTransform::from_to(rect, rect), - RectTransform::from_to(rect, rect), &eye, ui, highlights, @@ -415,8 +416,9 @@ pub fn view_3d( // Screenshot context menu. let (_, screenshot_mode) = screenshot_context_menu(ctx, response); if let Some(mode) = screenshot_mode { - let _ = - view_builder.schedule_screenshot(ctx.render_ctx, space_view_id.gpu_readback_id(), mode); + view_builder + .schedule_screenshot(ctx.render_ctx, space_view_id.gpu_readback_id(), mode) + .ok(); } show_projections_from_2d_space( @@ -531,7 +533,7 @@ fn show_projections_from_2d_space( // Render a thick line to the actual z value if any and a weaker one as an extension // If we don't have a z value, we only render the thick one. let thick_ray_length = if pos.z.is_finite() && pos.z > 0.0 { - Some(pos.z) + pos.z } else { cam.picture_plane_distance }; @@ -562,7 +564,7 @@ fn show_projections_from_2d_space( let cam_to_pos = *pos - cam.position(); let distance = cam_to_pos.length(); let ray = macaw::Ray3::from_origin_dir(cam.position(), cam_to_pos / distance); - add_picking_ray(&mut scene.primitives, ray, scene_bbox_accum, Some(distance)); + add_picking_ray(&mut scene.primitives, ray, scene_bbox_accum, distance); } } } @@ -574,34 +576,26 @@ fn add_picking_ray( primitives: &mut SceneSpatialPrimitives, ray: macaw::Ray3, scene_bbox_accum: &BoundingBox, - thick_ray_length: Option, + thick_ray_length: f32, ) { let mut line_batch = primitives.line_strips.batch("picking ray"); let origin = ray.point_along(0.0); // No harm in making this ray _very_ long. (Infinite messes with things though!) let fallback_ray_end = ray.point_along(scene_bbox_accum.size().length() * 10.0); - - if let Some(line_length) = thick_ray_length { - let main_ray_end = ray.point_along(line_length); - line_batch - .add_segment(origin, main_ray_end) - .color(egui::Color32::WHITE) - .flags(re_renderer::renderer::LineStripFlags::NO_COLOR_GRADIENT) - .radius(Size::new_points(1.0)); - line_batch - .add_segment(main_ray_end, fallback_ray_end) - .color(egui::Color32::DARK_GRAY) - // TODO(andreas): Make this dashed. - .flags(re_renderer::renderer::LineStripFlags::NO_COLOR_GRADIENT) - .radius(Size::new_points(0.5)); - } else { - line_batch - .add_segment(origin, fallback_ray_end) - .color(egui::Color32::WHITE) - .flags(re_renderer::renderer::LineStripFlags::NO_COLOR_GRADIENT) - .radius(Size::new_points(1.0)); - } + let main_ray_end = ray.point_along(thick_ray_length); + + line_batch + .add_segment(origin, main_ray_end) + .color(egui::Color32::WHITE) + .flags(re_renderer::renderer::LineStripFlags::NO_COLOR_GRADIENT) + .radius(Size::new_points(1.0)); + line_batch + .add_segment(main_ray_end, fallback_ray_end) + .color(egui::Color32::DARK_GRAY) + // TODO(andreas): Make this dashed. + .flags(re_renderer::renderer::LineStripFlags::NO_COLOR_GRADIENT) + .radius(Size::new_points(0.5)); } fn default_eye(scene_bbox: &macaw::BoundingBox, space_specs: &SpaceSpecs) -> OrbitEye { diff --git a/crates/re_web_viewer_server/build.rs b/crates/re_web_viewer_server/build.rs index 3837410ac6a2..945cab819778 100644 --- a/crates/re_web_viewer_server/build.rs +++ b/crates/re_web_viewer_server/build.rs @@ -57,6 +57,7 @@ impl<'a> Packages<'a> { // account for locally patched dependencies. rerun_if_changed(path.join("Cargo.toml").as_ref()); rerun_if_changed(path.join("**/*.rs").as_ref()); + rerun_if_changed(path.join("**/*.wgsl").as_ref()); } // Track all direct and indirect dependencies of that root package