1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
use re_types::{
    archetypes::Pinhole,
    blueprint::{
        archetypes::Background,
        components::{BackgroundKind, VisualBounds2D},
    },
    components::Color,
    Archetype,
};
use re_viewer_context::{TypedComponentFallbackProvider, ViewStateExt};

use crate::{ui::SpatialViewState, SpatialView2D};

impl TypedComponentFallbackProvider<Color> for SpatialView2D {
    fn fallback_for(&self, ctx: &re_viewer_context::QueryContext<'_>) -> Color {
        // Color is a fairly common component, make sure this is the right context.
        if ctx.archetype_name == Some(Background::name()) {
            Color::BLACK
        } else {
            Color::default()
        }
    }
}

impl TypedComponentFallbackProvider<BackgroundKind> for SpatialView2D {
    fn fallback_for(&self, _ctx: &re_viewer_context::QueryContext<'_>) -> BackgroundKind {
        BackgroundKind::SolidColor
    }
}

fn valid_bound(rect: &egui::Rect) -> bool {
    rect.is_finite() && rect.is_positive()
}

/// The pinhole sensor rectangle: [0, 0] - [width, height],
/// ignoring principal point.
fn pinhole_resolution_rect(pinhole: &Pinhole) -> Option<egui::Rect> {
    pinhole
        .resolution()
        .map(|res| egui::Rect::from_min_max(egui::Pos2::ZERO, egui::pos2(res.x, res.y)))
}

impl TypedComponentFallbackProvider<VisualBounds2D> for SpatialView2D {
    fn fallback_for(&self, ctx: &re_viewer_context::QueryContext<'_>) -> VisualBounds2D {
        let Ok(view_state) = ctx.view_state.downcast_ref::<SpatialViewState>() else {
            return VisualBounds2D::default();
        };

        // TODO(andreas): It makes sense that we query the bounding box from the view_state,
        // but the pinhole should be an ad-hoc query instead. For this we need a little bit more state information on the QueryContext.
        let default_scene_rect = view_state
            .pinhole_at_origin
            .as_ref()
            .and_then(pinhole_resolution_rect)
            .unwrap_or_else(|| {
                // TODO(emilk): if there is a single image in this view, use that as the default bounds
                let scene_rect_smoothed = view_state.bounding_boxes.smoothed;
                egui::Rect::from_min_max(
                    scene_rect_smoothed.min.truncate().to_array().into(),
                    scene_rect_smoothed.max.truncate().to_array().into(),
                )
            });

        if valid_bound(&default_scene_rect) {
            default_scene_rect.into()
        } else {
            // Nothing in scene, probably.
            VisualBounds2D::default()
        }
    }
}

re_viewer_context::impl_component_fallback_provider!(SpatialView2D => [BackgroundKind, Color, VisualBounds2D]);