Refactor gamepad code to be sync
This commit is contained in:
parent
9a450cfe9c
commit
efcc13bea7
|
@ -4,11 +4,11 @@ use gilrs::Button::LeftTrigger2;
|
||||||
use gilrs::Button::RightTrigger2;
|
use gilrs::Button::RightTrigger2;
|
||||||
use gilrs::EventType::*;
|
use gilrs::EventType::*;
|
||||||
use gilrs::Gilrs;
|
use gilrs::Gilrs;
|
||||||
|
use std::sync::mpsc;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
use std::thread::sleep;
|
use std::thread::sleep;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use tokio::sync::mpsc;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Axis {
|
pub enum Axis {
|
||||||
|
@ -41,73 +41,70 @@ pub enum GamepadEvent {
|
||||||
/// }
|
/// }
|
||||||
/// ```
|
/// ```
|
||||||
pub struct Gamepad {
|
pub struct Gamepad {
|
||||||
speed_setpoint: Mutex<(f32, f32, f32)>,
|
gilrs: Gilrs,
|
||||||
|
// Cached speed setpoint for (X, Y, Positive-Z, Negative-Z)
|
||||||
|
speed_setpoint: (f32, f32, f32, f32),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Gamepad {
|
impl Gamepad {
|
||||||
/// Create a new `Gamepad` and spawn the underlying tasks for updating internal setpoints
|
/// Create a new `Gamepad` and spawn the underlying tasks for updating internal setpoints
|
||||||
///
|
///
|
||||||
/// The tasks are terminated on drop.
|
/// The tasks are terminated on drop.
|
||||||
pub async fn new() -> Result<Arc<Self>, gilrs::Error> {
|
pub fn new() -> Result<Self, gilrs::Error> {
|
||||||
let (gamepad_tx, gamepad_rx) = mpsc::channel(8);
|
let gilrs = Gilrs::new().unwrap();
|
||||||
let res = Arc::new(Gamepad {
|
|
||||||
speed_setpoint: Mutex::new((0.0, 0.0, 0.0)),
|
|
||||||
});
|
|
||||||
tokio::task::spawn_blocking(move || {
|
|
||||||
let mut gilrs = Gilrs::new().unwrap();
|
|
||||||
for (_id, gamepad) in gilrs.gamepads() {
|
for (_id, gamepad) in gilrs.gamepads() {
|
||||||
println!("{} is {:?}", gamepad.name(), gamepad.power_info());
|
println!(
|
||||||
}
|
"Found gamepad {}: {:?}",
|
||||||
loop {
|
gamepad.name(),
|
||||||
sleep(Duration::from_millis(1));
|
gamepad.power_info()
|
||||||
if let Some(event) = gilrs.next_event() {
|
);
|
||||||
if let Some(internal_event) = Self::map_event(event) {
|
|
||||||
if gamepad_tx.blocking_send(internal_event).is_err() {
|
|
||||||
// receiver dropped
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
tokio::spawn(Self::handle_events(res.clone(), gamepad_rx));
|
|
||||||
|
|
||||||
Ok(res)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the current speed that the user is dialing in on the gamepad
|
Ok(Self {
|
||||||
pub fn speed_setpoint(&self) -> (f32, f32, f32) {
|
gilrs,
|
||||||
*self.speed_setpoint.lock().unwrap()
|
speed_setpoint: (0.0, 0.0, 0.0, 0.0),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Update the setpoints in accordance to incoming `GamepadEvent`s
|
fn get_next(&mut self) -> Option<GamepadEvent> {
|
||||||
async fn handle_events(self_arc: Arc<Self>, mut events_rx: mpsc::Receiver<GamepadEvent>) {
|
self.gilrs
|
||||||
// There is two z-axes that counter each other.
|
.next_event()
|
||||||
let mut z_positive = 0.0;
|
.and_then(|event| Self::map_event(event))
|
||||||
let mut z_negative = 0.0;
|
}
|
||||||
while let Some(event) = events_rx.recv().await {
|
|
||||||
|
// Get all `GamepadEvent`s since the last call
|
||||||
|
pub fn get_pending(&mut self) -> Vec<GamepadEvent> {
|
||||||
|
(0..).map_while(|_| self.get_next()).collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Update the current speed via the movements on the controller and return the current setpoint
|
||||||
|
///
|
||||||
|
/// If you don't want to update, just supply an empty `updates` slice
|
||||||
|
pub fn speed_setpoint(&mut self, updates: &[GamepadEvent]) -> (f32, f32, f32) {
|
||||||
|
for event in updates {
|
||||||
match event {
|
match event {
|
||||||
GamepadEvent::TerminatePressed => break,
|
GamepadEvent::TerminatePressed => break,
|
||||||
GamepadEvent::AxisPosition(axis, value) => {
|
GamepadEvent::AxisPosition(axis, value) => {
|
||||||
// I won't panic if you don't panic!
|
// I won't panic if you don't panic!
|
||||||
let mut speed_setpoint = self_arc.speed_setpoint.lock().unwrap();
|
let mut speed_setpoint = self.speed_setpoint;
|
||||||
match axis {
|
match axis {
|
||||||
Axis::X => speed_setpoint.0 = value,
|
Axis::X => speed_setpoint.0 = *value,
|
||||||
Axis::Y => speed_setpoint.1 = value,
|
Axis::Y => speed_setpoint.1 = *value,
|
||||||
Axis::ZPositive => {
|
Axis::ZPositive => {
|
||||||
z_positive = value;
|
speed_setpoint.2 = *value;
|
||||||
speed_setpoint.2 = z_positive - z_negative;
|
|
||||||
}
|
}
|
||||||
Axis::ZNegative => {
|
Axis::ZNegative => {
|
||||||
z_negative = value;
|
speed_setpoint.3 = *value;
|
||||||
speed_setpoint.2 = z_positive - z_negative;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
println!("Game pad quit!")
|
(
|
||||||
|
self.speed_setpoint.0,
|
||||||
|
self.speed_setpoint.1,
|
||||||
|
self.speed_setpoint.2 - self.speed_setpoint.3,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn map_event(event: gilrs::Event) -> Option<GamepadEvent> {
|
fn map_event(event: gilrs::Event) -> Option<GamepadEvent> {
|
||||||
|
|
Loading…
Reference in a new issue