use crate::event::Key;
use crossterm::event::{self, Event as CEvent};
use std::{
sync::mpsc,
thread,
time::{Duration, Instant},
};
#[derive(Debug, Clone, Copy)]
pub struct EventConfig {
pub exit_key: Key,
pub tick_rate: Duration,
}
impl Default for EventConfig {
fn default() -> EventConfig {
EventConfig {
exit_key: Key::Ctrl('c'),
tick_rate: Duration::from_millis(250),
}
}
}
pub enum Event<I> {
Input(I),
Tick,
}
pub struct Events {
rx: mpsc::Receiver<Event<Key>>,
_tx: mpsc::Sender<Event<Key>>,
}
impl Events {
pub fn new(tick_rate: u64) -> Self {
Events::with_config(EventConfig {
tick_rate: Duration::from_millis(tick_rate),
..Default::default()
})
}
pub fn with_config(config: EventConfig) -> Events {
let (tx, rx) = mpsc::channel();
let event_tx = tx.clone();
let tick_rate = config.tick_rate;
thread::spawn(move || {
let mut last_tick = Instant::now();
loop {
let timeout = tick_rate
.checked_sub(last_tick.elapsed())
.unwrap_or_else(|| Duration::from_secs(0));
if event::poll(timeout).unwrap() {
if let CEvent::Key(key_event) = event::read().unwrap() {
let key = Key::from(key_event);
if event_tx.send(Event::Input(key)).is_err() {
break;
}
}
}
if last_tick.elapsed() >= tick_rate {
if event_tx.send(Event::Tick).is_err() {
break;
}
last_tick = Instant::now();
}
}
});
Events { rx, _tx: tx }
}
pub fn next(&self) -> Result<Event<Key>, mpsc::RecvError> {
self.rx.recv()
}
}