|
|
|
@ -1,10 +1,12 @@
|
|
|
|
|
use crate::app::Task;
|
|
|
|
|
use crate::app::{ExternalMsg, InternalMsg, MsgIn};
|
|
|
|
|
use crate::input::Key;
|
|
|
|
|
use anyhow::Error;
|
|
|
|
|
use crossterm::event::{self, Event, MouseEventKind};
|
|
|
|
|
use std::sync::mpsc;
|
|
|
|
|
use std::sync::mpsc::{Receiver, Sender};
|
|
|
|
|
use std::thread;
|
|
|
|
|
use std::time::Duration;
|
|
|
|
|
|
|
|
|
|
pub(crate) struct EventReader {
|
|
|
|
|
task_sender: Sender<Task>,
|
|
|
|
@ -32,8 +34,8 @@ impl EventReader {
|
|
|
|
|
|
|
|
|
|
pub(crate) fn stop(&self) {
|
|
|
|
|
if let Some((stopper, ack)) = &self.stopper {
|
|
|
|
|
stopper.send(true).unwrap();
|
|
|
|
|
ack.recv().unwrap();
|
|
|
|
|
stopper.send(true).unwrap_or_default(); // Let's not panic when xplr stops.
|
|
|
|
|
ack.recv().unwrap_or_default();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -47,39 +49,44 @@ fn keep_reading(tx_msg_in: Sender<Task>, rx_stopper: Receiver<bool>, tx_ack: Sen
|
|
|
|
|
// NOTE: The poll timeout need to stay low, else spawning sub subshell
|
|
|
|
|
// and start typing immediately will cause panic.
|
|
|
|
|
// To reproduce, press `:`, then press and hold `!`.
|
|
|
|
|
match event::read() {
|
|
|
|
|
let res = match event::read() {
|
|
|
|
|
Ok(Event::Key(key)) => {
|
|
|
|
|
let key = Key::from_event(key);
|
|
|
|
|
let msg = MsgIn::Internal(InternalMsg::HandleKey(key));
|
|
|
|
|
tx_msg_in.send(Task::new(msg, Some(key))).unwrap();
|
|
|
|
|
tx_msg_in
|
|
|
|
|
.send(Task::new(msg, Some(key)))
|
|
|
|
|
.map_err(Error::new)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(Event::Mouse(evt)) => match evt.kind {
|
|
|
|
|
MouseEventKind::ScrollUp => {
|
|
|
|
|
let msg = MsgIn::External(ExternalMsg::FocusPrevious);
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).map_err(Error::new)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MouseEventKind::ScrollDown => {
|
|
|
|
|
let msg = MsgIn::External(ExternalMsg::FocusNext);
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).map_err(Error::new)
|
|
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
_ => Ok(()),
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
Ok(Event::Resize(_, _)) => {
|
|
|
|
|
let msg = MsgIn::External(ExternalMsg::Refresh);
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
|
|
|
|
tx_msg_in.send(Task::new(msg, None)).map_err(Error::new)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Err(e) => {
|
|
|
|
|
tx_msg_in
|
|
|
|
|
.send(Task::new(
|
|
|
|
|
MsgIn::External(ExternalMsg::LogError(e.to_string())),
|
|
|
|
|
None,
|
|
|
|
|
))
|
|
|
|
|
.unwrap();
|
|
|
|
|
}
|
|
|
|
|
Err(e) => Err(Error::new(e)),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Err(e) = res {
|
|
|
|
|
tx_msg_in
|
|
|
|
|
.send(Task::new(
|
|
|
|
|
MsgIn::External(ExternalMsg::LogError(e.to_string())),
|
|
|
|
|
None,
|
|
|
|
|
))
|
|
|
|
|
.unwrap_or_default(); // Let's not panic when xplr stops
|
|
|
|
|
thread::sleep(Duration::from_secs(1));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|