Skip to main content

azul_layout/managers/
sensors.rs

1//! Sensor manager — cross-platform state for the motion-sensor surface
2//! (SUPER_PLAN_2 §1 feature 5 + research/03).
3//!
4//! Continuous + push-driven, like geolocation:
5//!
6//! - The **platform backend** (`dll/src/desktop/extra/sensors/<plat>.rs`)
7//!   subscribes to CoreMotion (`CMMotionManager`) / Android `SensorManager`
8//!   and calls [`push_sensor_reading`] on every sample (arbitrary thread).
9//! - The dll **layout pass** drains the channel via
10//!   [`drain_sensor_readings`] and folds each into the manager through
11//!   [`SensorManager::set_reading`].
12//! - **Callbacks** read `reading(kind)` synchronously (via
13//!   `CallbackInfo::get_sensor_reading`) to drive tilt / shake / compass UI.
14//!
15//! One reading slot per [`SensorKind`]. No platform deps
16//! (SUPER_PLAN_2 §0.5); the channel mirrors `geolocation.rs` verbatim.
17
18use alloc::vec::Vec;
19
20use azul_core::dom::DomNodeId;
21use azul_core::events::{
22    EventData, EventProvider, EventSource as CoreEventSource, EventType, SyntheticEvent,
23};
24use azul_core::task::Instant;
25pub use azul_core::sensors::{SensorKind, SensorReading};
26
27/// Cross-platform sensor state. One per `App` — the OS exposes a single
28/// per-process sensor subscription, not per-window.
29#[derive(Debug, Clone, PartialEq, Default)]
30pub struct SensorManager {
31    /// Latest accelerometer reading (m/s²), or `None` until a sample arrives.
32    pub accelerometer: Option<SensorReading>,
33    /// Latest gyroscope reading (rad/s).
34    pub gyroscope: Option<SensorReading>,
35    /// Latest magnetometer reading (µT).
36    pub magnetometer: Option<SensorReading>,
37    /// `true` when a reading advanced since the last event-pass drain. Set by
38    /// [`set_reading`](Self::set_reading), read by the `EventProvider` impl,
39    /// cleared by [`clear_pending_event`](Self::clear_pending_event).
40    pub pending_event: bool,
41}
42
43impl SensorManager {
44    pub fn new() -> Self {
45        Self::default()
46    }
47
48    /// Latest reading for `kind`, or `None` if no backend has delivered one.
49    pub fn reading(&self, kind: SensorKind) -> Option<SensorReading> {
50        match kind {
51            SensorKind::Accelerometer => self.accelerometer,
52            SensorKind::Gyroscope => self.gyroscope,
53            SensorKind::Magnetometer => self.magnetometer,
54        }
55    }
56
57    /// Apply a reading the backend delivered. Returns `true` if it advanced
58    /// (bit-pattern different from the previous, so missing-as-`NaN` axes
59    /// don't make every sample look "changed").
60    pub fn set_reading(&mut self, reading: SensorReading) -> bool {
61        let slot = match reading.kind {
62            SensorKind::Accelerometer => &mut self.accelerometer,
63            SensorKind::Gyroscope => &mut self.gyroscope,
64            SensorKind::Magnetometer => &mut self.magnetometer,
65        };
66        let changed = match slot {
67            Some(prev) => !reading_bitwise_eq(prev, &reading),
68            None => true,
69        };
70        *slot = Some(reading);
71        if changed {
72            self.pending_event = true;
73        }
74        changed
75    }
76
77    /// Clear the pending-event flag. The dll calls this after the event pass
78    /// has collected the `SensorChanged` event (mirrors `clear_changeset`).
79    pub fn clear_pending_event(&mut self) {
80        self.pending_event = false;
81    }
82}
83
84impl EventProvider for SensorManager {
85    /// Yield a window-level `SensorChanged` event when a reading advanced
86    /// since the last drain (target = root; read the value via
87    /// `CallbackInfo::get_sensor_reading` inside the callback).
88    fn get_pending_events(&self, timestamp: Instant) -> Vec<SyntheticEvent> {
89        if self.pending_event {
90            alloc::vec![SyntheticEvent::new(
91                EventType::SensorChanged,
92                CoreEventSource::User,
93                DomNodeId::ROOT,
94                timestamp,
95                EventData::None,
96            )]
97        } else {
98            Vec::new()
99        }
100    }
101}
102
103fn reading_bitwise_eq(a: &SensorReading, b: &SensorReading) -> bool {
104    a.kind == b.kind
105        && a.x.to_bits() == b.x.to_bits()
106        && a.y.to_bits() == b.y.to_bits()
107        && a.z.to_bits() == b.z.to_bits()
108        && a.timestamp_ms == b.timestamp_ms
109}
110
111// ────────── Async reading channel (platform backend → manager) ─────────
112//
113// CoreMotion / Android `SensorManager` deliver on an arbitrary thread with
114// no handle to the live `SensorManager` (inside the window's
115// `LayoutWindow`). The backend parks each reading here; the layout pass
116// drains it and applies the latest per kind. Pure Rust — no platform
117// dependency (SUPER_PLAN_2 §0.5). Mirrors the geolocation fix channel.
118
119static PENDING_READINGS: std::sync::Mutex<Vec<SensorReading>> =
120    std::sync::Mutex::new(Vec::new());
121
122/// Park a sensor reading delivered by a platform backend (in the dll).
123/// Thread-safe; poison-recovering.
124pub fn push_sensor_reading(reading: SensorReading) {
125    let mut q = PENDING_READINGS.lock().unwrap_or_else(|e| e.into_inner());
126    q.push(reading);
127}
128
129/// Drain every reading parked by [`push_sensor_reading`], in arrival order.
130/// Called once per layout pass; the caller applies them through
131/// [`SensorManager::set_reading`] (the last per kind wins).
132pub fn drain_sensor_readings() -> Vec<SensorReading> {
133    let mut q = PENDING_READINGS.lock().unwrap_or_else(|e| e.into_inner());
134    core::mem::take(&mut *q)
135}
136
137#[cfg(test)]
138mod tests {
139    use super::*;
140
141    fn r(kind: SensorKind, x: f32, y: f32, z: f32) -> SensorReading {
142        SensorReading {
143            kind,
144            x,
145            y,
146            z,
147            timestamp_ms: 0,
148        }
149    }
150
151    #[test]
152    fn manager_defaults_to_no_readings() {
153        let mgr = SensorManager::new();
154        assert_eq!(mgr.reading(SensorKind::Accelerometer), None);
155        assert_eq!(mgr.reading(SensorKind::Gyroscope), None);
156        assert_eq!(mgr.reading(SensorKind::Magnetometer), None);
157    }
158
159    #[test]
160    fn set_reading_routes_by_kind_and_flags_change() {
161        let mut mgr = SensorManager::new();
162        assert!(mgr.set_reading(r(SensorKind::Accelerometer, 0.0, 0.0, 9.81)));
163        // Only the accelerometer slot is filled.
164        assert!(mgr.reading(SensorKind::Accelerometer).is_some());
165        assert_eq!(mgr.reading(SensorKind::Gyroscope), None);
166        // Same value again — no change.
167        assert!(!mgr.set_reading(r(SensorKind::Accelerometer, 0.0, 0.0, 9.81)));
168        // Different value — change.
169        assert!(mgr.set_reading(r(SensorKind::Accelerometer, 1.0, 0.0, 9.81)));
170        // A different kind fills its own slot.
171        assert!(mgr.set_reading(r(SensorKind::Gyroscope, 0.1, 0.0, 0.0)));
172        assert_eq!(
173            mgr.reading(SensorKind::Gyroscope).map(|r| r.x),
174            Some(0.1)
175        );
176    }
177
178    #[test]
179    fn magnitude_of_resting_accelerometer() {
180        let g = r(SensorKind::Accelerometer, 0.0, 0.0, 9.81);
181        assert!((g.magnitude() - 9.81).abs() < 1e-4);
182    }
183
184    #[test]
185    fn readings_round_trip_through_manager() {
186        let _ = drain_sensor_readings();
187
188        push_sensor_reading(r(SensorKind::Accelerometer, 1.0, 2.0, 3.0));
189        push_sensor_reading(r(SensorKind::Accelerometer, 4.0, 5.0, 6.0)); // last wins per kind
190        push_sensor_reading(r(SensorKind::Magnetometer, 20.0, 0.0, 40.0));
191        let drained = drain_sensor_readings();
192        assert_eq!(drained.len(), 3, "all parked readings drain in order");
193
194        let mut mgr = SensorManager::new();
195        for reading in &drained {
196            mgr.set_reading(*reading);
197        }
198        assert_eq!(
199            mgr.reading(SensorKind::Accelerometer).map(|r| r.x),
200            Some(4.0),
201            "the last accelerometer reading wins"
202        );
203        assert_eq!(
204            mgr.reading(SensorKind::Magnetometer).map(|r| r.z),
205            Some(40.0)
206        );
207
208        assert!(drain_sensor_readings().is_empty());
209    }
210}