use std::sync::mpsc;
use std::sync::{
atomic::{
AtomicBool,
Ordering,
},
Arc,
};
use std::thread;
use std::time::{
Instant,
Duration,
};
use crossterm::event::{
KeyCode,
KeyEvent,
Event as CEvent,
poll as term_poll,
read as term_read,
};
pub enum Event<I> {
Input(I),
Tick,
}
pub struct Events {
rx: mpsc::Receiver<Event<KeyEvent>>,
input_handle: thread::JoinHandle<()>,
ignore_exit_key: Arc<AtomicBool>,
}
#[derive(Debug, Clone, Copy)]
pub struct Config {
pub exit_key: KeyCode,
pub tick_rate: Duration,
}
impl Default for Config {
fn default() -> Config {
Config {
exit_key: KeyCode::Char('q'),
tick_rate: Duration::from_millis(250),
}
}
}
impl Events {
pub fn new() -> Events {
Events::with_config(Config::default())
}
pub fn with_config(config: Config) -> Events {
let (tx, rx) = mpsc::channel();
let ignore_exit_key = Arc::new(AtomicBool::new(false));
let tick_rate = config.tick_rate;
let input_handle = {
let tx = tx.clone();
let ignore_exit_key = Arc::new(AtomicBool::new(false));
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 term_poll(timeout).unwrap() {
if let CEvent::Key(key) = term_read().unwrap() {
if let Err(err) = tx.send(Event::Input(key)) {
eprintln!("{}", err);
return;
}
if !ignore_exit_key.load(Ordering::Relaxed) && key.code == config.exit_key {
return;
}
}
}
if last_tick.elapsed() >= tick_rate {
if tx.send(Event::Tick).is_err() {
break;
}
last_tick = Instant::now();
}
}
})
};
Events {
rx,
ignore_exit_key,
input_handle,
}
}
pub fn next(&self) -> Result<Event<KeyEvent>, mpsc::RecvError> {
self.rx.recv()
}
pub fn disable_exit_key(&mut self) {
self.ignore_exit_key.store(true, Ordering::Relaxed);
}
pub fn enable_exit_key(&mut self) {
self.ignore_exit_key.store(false, Ordering::Relaxed);
}
}