mirror of https://github.com/sayanarijit/xplr
Improve input reading by eliminating pause
Stop the thread instead of pausing it. This improves the CPU usage and responsiveness a slightly.pull/350/head
parent
ac1b40799a
commit
bec80e98df
@ -1,77 +1,86 @@
|
|||||||
use crate::app::Task;
|
use crate::app::Task;
|
||||||
use crate::app::{ExternalMsg, InternalMsg, MsgIn};
|
use crate::app::{ExternalMsg, InternalMsg, MsgIn};
|
||||||
use crate::input::Key;
|
use crate::input::Key;
|
||||||
use anyhow::Result;
|
|
||||||
use crossterm::event::{self, Event, MouseEventKind};
|
use crossterm::event::{self, Event, MouseEventKind};
|
||||||
|
use std::sync::mpsc;
|
||||||
use std::sync::mpsc::{Receiver, Sender};
|
use std::sync::mpsc::{Receiver, Sender};
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
|
||||||
pub fn pause_reading(tx_event_reader: &Sender<bool>, rx_loop_waiter: &Receiver<()>) -> Result<()> {
|
pub(crate) struct EventReader {
|
||||||
tx_event_reader.send(true)?;
|
task_sender: Sender<Task>,
|
||||||
rx_loop_waiter.recv()?;
|
stopper: Option<(Sender<bool>, Receiver<()>)>,
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn resume_reading(tx_event_reader: &Sender<bool>, rx_loop_waiter: &Receiver<()>) -> Result<()> {
|
impl EventReader {
|
||||||
tx_event_reader.send(false)?;
|
pub(crate) fn new(task_sender: Sender<Task>) -> Self {
|
||||||
rx_loop_waiter.recv()?;
|
Self {
|
||||||
Ok(())
|
task_sender,
|
||||||
}
|
stopper: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn keep_reading(
|
pub(crate) fn start(&mut self) {
|
||||||
tx_msg_in: Sender<Task>,
|
let sender = self.task_sender.clone();
|
||||||
rx_event_reader: Receiver<bool>,
|
let (tx_stopper, rx_stopper) = mpsc::channel();
|
||||||
tx_loop_waiter: Sender<()>,
|
let (tx_ack, rx_ack) = mpsc::channel();
|
||||||
) {
|
self.stopper = Some((tx_stopper, rx_ack));
|
||||||
thread::spawn(move || {
|
|
||||||
let mut is_paused = false;
|
|
||||||
loop {
|
|
||||||
if let Ok(paused) = rx_event_reader.try_recv() {
|
|
||||||
is_paused = paused;
|
|
||||||
tx_loop_waiter.send(()).unwrap();
|
|
||||||
};
|
|
||||||
|
|
||||||
if is_paused {
|
thread::spawn(move || {
|
||||||
thread::sleep(std::time::Duration::from_millis(200));
|
keep_reading(sender, rx_stopper, tx_ack);
|
||||||
} else if event::poll(std::time::Duration::from_millis(150)).unwrap_or_default() {
|
});
|
||||||
// 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() {
|
|
||||||
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();
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Event::Mouse(evt)) => match evt.kind {
|
pub(crate) fn stop(&self) {
|
||||||
MouseEventKind::ScrollUp => {
|
if let Some((stopper, ack)) = &self.stopper {
|
||||||
let msg = MsgIn::External(ExternalMsg::FocusPrevious);
|
stopper.send(true).unwrap();
|
||||||
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
ack.recv().unwrap();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MouseEventKind::ScrollDown => {
|
fn keep_reading(tx_msg_in: Sender<Task>, rx_stopper: Receiver<bool>, tx_ack: Sender<()>) {
|
||||||
let msg = MsgIn::External(ExternalMsg::FocusNext);
|
loop {
|
||||||
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
if rx_stopper.try_recv().unwrap_or(false) {
|
||||||
}
|
tx_ack.send(()).unwrap();
|
||||||
_ => {}
|
break;
|
||||||
},
|
} else if event::poll(std::time::Duration::from_millis(150)).unwrap_or_default() {
|
||||||
|
// 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() {
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
Ok(Event::Resize(_, _)) => {
|
Ok(Event::Mouse(evt)) => match evt.kind {
|
||||||
let msg = MsgIn::External(ExternalMsg::Refresh);
|
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)).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
Err(e) => {
|
MouseEventKind::ScrollDown => {
|
||||||
tx_msg_in
|
let msg = MsgIn::External(ExternalMsg::FocusNext);
|
||||||
.send(Task::new(
|
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
||||||
MsgIn::External(ExternalMsg::LogError(e.to_string())),
|
|
||||||
None,
|
|
||||||
))
|
|
||||||
.unwrap();
|
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
|
},
|
||||||
|
|
||||||
|
Ok(Event::Resize(_, _)) => {
|
||||||
|
let msg = MsgIn::External(ExternalMsg::Refresh);
|
||||||
|
tx_msg_in.send(Task::new(msg, None)).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
Err(e) => {
|
||||||
|
tx_msg_in
|
||||||
|
.send(Task::new(
|
||||||
|
MsgIn::External(ExternalMsg::LogError(e.to_string())),
|
||||||
|
None,
|
||||||
|
))
|
||||||
|
.unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue