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
//! HiTechnic EV3 / NXT Compass Sensor. (<https://www.generationrobots.com/en/401186-hitechnic-compass-sensor-for-lego-mindstorms-nxt-and-ev3.html>)

use super::{Sensor, SensorPort};
use crate::{Attribute, Device, Driver, Ev3Error, Ev3Result};

/// HiTechnic EV3 / NXT Compass Sensor.
#[derive(Debug, Clone, Device, Sensor)]
pub struct CompassSensor {
    driver: Driver,
    origin: i32, // zero point
}

impl CompassSensor {
    fn new(driver: Driver) -> Self {
        Self { driver, origin: 0 }
    }

    findable!("lego-sensor", "ht-nxt-compass", SensorPort, "Compass", "in");

    /// Command for starting the calibration
    pub const COMMAND_START_CALIBRATION: &'static str = "BEGIN-CAL";

    /// Command for stopping the calibration
    pub const COMMAND_STOP_CALIBRATION: &'static str = "END-CAL";

    // Sensor only have one mode (COMPASS), so setting the mode is not necessary

    /// gets rotation (in degree) from the compass sensor
    pub fn get_rotation(&self) -> Ev3Result<i32> {
        self.get_value0()
    }

    /// sets the origin
    pub fn set_zero(&mut self) -> Ev3Result<()> {
        self.origin = self.get_rotation()?;
        Ok(())
    }

    /// calculates the rotation to the origin / the zero point
    pub fn get_relative_rotation(&self) -> Ev3Result<i32> {
        let pos = self.get_rotation()?;
        let mut rel_rot = pos - self.origin;
        if rel_rot < 0 {
            rel_rot += 360;
        }
        Ok(rel_rot)
    }

    /// calibration:
    /// start the calibration by start_calibration()
    /// turn the robot 360 degrees
    /// end the calibration by stop_calibration()
    /// attention: if calibration has not finished, the get_rotation method always returns -258

    /// starts the calibration
    pub fn start_calibration(&self) -> Ev3Result<()> {
        self.set_command(Self::COMMAND_START_CALIBRATION)
    }

    /// stops the calibration
    pub fn stop_calibration(&self) -> Ev3Result<()> {
        self.set_command(Self::COMMAND_STOP_CALIBRATION)
    }
}