Only re-render when an event happened
For now we don't even filter out invisible events, but it already cuts down considerably on CPU usage.
This commit is contained in:
@ -8,7 +8,10 @@ use serde::{
|
|||||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
pub enum Action {
|
pub enum Action {
|
||||||
Tick,
|
Tick,
|
||||||
|
/// Apply any pending update to the screen
|
||||||
Render,
|
Render,
|
||||||
|
/// Notify there is a pending update
|
||||||
|
ShouldRender,
|
||||||
Resize(u16, u16),
|
Resize(u16, u16),
|
||||||
Suspend,
|
Suspend,
|
||||||
Resume,
|
Resume,
|
||||||
|
40
src/app.rs
40
src/app.rs
@ -205,7 +205,22 @@ impl App {
|
|||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut changes_since_last_render = true;
|
||||||
loop {
|
loop {
|
||||||
|
if changes_since_last_render {
|
||||||
|
tui.draw(|f| {
|
||||||
|
for component in self.components.iter_mut() {
|
||||||
|
let r = component.draw(f, f.size(), &self.buffers);
|
||||||
|
if let Err(e) = r {
|
||||||
|
action_tx
|
||||||
|
.send(Action::Error(format!("Failed to draw: {:?}", e)))
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})?;
|
||||||
|
changes_since_last_render = false;
|
||||||
|
}
|
||||||
|
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
e = tui.next() => {
|
e = tui.next() => {
|
||||||
if let Some(e) = e {
|
if let Some(e) = e {
|
||||||
@ -221,9 +236,9 @@ impl App {
|
|||||||
let (client, sync_response) = sync_response.expect("sync_responses_rx unexpectedly closed");
|
let (client, sync_response) = sync_response.expect("sync_responses_rx unexpectedly closed");
|
||||||
self.handle_sync_response(&action_tx, client, sync_response).await.context("Error while handling sync response")?;
|
self.handle_sync_response(&action_tx, client, sync_response).await.context("Error while handling sync response")?;
|
||||||
}
|
}
|
||||||
poll_updates = futures::future::join_all(
|
poll_updates = self.buffers.poll_updates() => {
|
||||||
self.buffers.iter_mut().map(|buf| buf.poll_updates())
|
changes_since_last_render = true;
|
||||||
) => {}
|
}
|
||||||
sync_result = sync_results.next() => {
|
sync_result = sync_results.next() => {
|
||||||
if !self.should_quit.load(Ordering::Acquire) {
|
if !self.should_quit.load(Ordering::Acquire) {
|
||||||
panic!("Sync ended unexpected: {:?}", sync_result);
|
panic!("Sync ended unexpected: {:?}", sync_result);
|
||||||
@ -243,18 +258,21 @@ impl App {
|
|||||||
Action::Suspend => self.should_suspend = true,
|
Action::Suspend => self.should_suspend = true,
|
||||||
Action::Resume => self.should_suspend = false,
|
Action::Resume => self.should_suspend = false,
|
||||||
Action::NextBuffer => {
|
Action::NextBuffer => {
|
||||||
|
changes_since_last_render = true;
|
||||||
// self.buffer implements wrap-around itself
|
// self.buffer implements wrap-around itself
|
||||||
self
|
self
|
||||||
.buffers
|
.buffers
|
||||||
.set_active_index(self.buffers.active_index() as isize + 1)
|
.set_active_index(self.buffers.active_index() as isize + 1)
|
||||||
},
|
},
|
||||||
Action::PreviousBuffer => {
|
Action::PreviousBuffer => {
|
||||||
|
changes_since_last_render = true;
|
||||||
// self.buffer implements wrap-around itself
|
// self.buffer implements wrap-around itself
|
||||||
self
|
self
|
||||||
.buffers
|
.buffers
|
||||||
.set_active_index(self.buffers.active_index() as isize - 1)
|
.set_active_index(self.buffers.active_index() as isize - 1)
|
||||||
},
|
},
|
||||||
Action::Resize(w, h) => {
|
Action::Resize(w, h) => {
|
||||||
|
changes_since_last_render = true;
|
||||||
tui.resize(Rect::new(0, 0, w, h))?;
|
tui.resize(Rect::new(0, 0, w, h))?;
|
||||||
tui.draw(|f| {
|
tui.draw(|f| {
|
||||||
for component in self.components.iter_mut() {
|
for component in self.components.iter_mut() {
|
||||||
@ -267,19 +285,14 @@ impl App {
|
|||||||
}
|
}
|
||||||
})?;
|
})?;
|
||||||
},
|
},
|
||||||
|
Action::ShouldRender => {
|
||||||
|
changes_since_last_render = true;
|
||||||
|
},
|
||||||
Action::Render => {
|
Action::Render => {
|
||||||
tui.draw(|f| {
|
// Merely used as a way to signify it's time to render any pending update
|
||||||
for component in self.components.iter_mut() {
|
|
||||||
let r = component.draw(f, f.size(), &self.buffers);
|
|
||||||
if let Err(e) = r {
|
|
||||||
action_tx
|
|
||||||
.send(Action::Error(format!("Failed to draw: {:?}", e)))
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})?;
|
|
||||||
},
|
},
|
||||||
Action::RunCommand(ref command_line) => {
|
Action::RunCommand(ref command_line) => {
|
||||||
|
changes_since_last_render = true;
|
||||||
log::info!("Got command: {command_line}");
|
log::info!("Got command: {command_line}");
|
||||||
crate::commands::run_command(&command_line, &self, &action_tx)?;
|
crate::commands::run_command(&command_line, &self, &action_tx)?;
|
||||||
},
|
},
|
||||||
@ -291,6 +304,7 @@ impl App {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.should_suspend {
|
if self.should_suspend {
|
||||||
tui.suspend()?;
|
tui.suspend()?;
|
||||||
action_tx.send(Action::Resume)?;
|
action_tx.send(Action::Resume)?;
|
||||||
|
@ -14,6 +14,8 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
use futures::stream::FuturesUnordered;
|
||||||
|
use futures::StreamExt;
|
||||||
use matrix_sdk::async_trait;
|
use matrix_sdk::async_trait;
|
||||||
use nonempty::NonEmpty;
|
use nonempty::NonEmpty;
|
||||||
|
|
||||||
@ -23,10 +25,13 @@ mod room;
|
|||||||
pub use room::RoomBuffer;
|
pub use room::RoomBuffer;
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait Buffer: Send {
|
pub trait Buffer: Send + Sync {
|
||||||
/// A short human-readable name for the room, eg. to show in compact buflist
|
/// A short human-readable name for the room, eg. to show in compact buflist
|
||||||
fn short_name(&self) -> String;
|
fn short_name(&self) -> String;
|
||||||
async fn poll_updates(&mut self) {}
|
/// Returns if there are any updates to apply.
|
||||||
|
async fn poll_updates(&mut self) {
|
||||||
|
std::future::pending().await
|
||||||
|
}
|
||||||
fn content(&self) -> Vec<ratatui::text::Text>; // TODO: make this lazy, only the last few are used
|
fn content(&self) -> Vec<ratatui::text::Text>; // TODO: make this lazy, only the last few are used
|
||||||
/// Called when the user is being showned the oldest items this buffer returned.
|
/// Called when the user is being showned the oldest items this buffer returned.
|
||||||
///
|
///
|
||||||
@ -47,6 +52,16 @@ impl Buffers {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn poll_updates(&mut self) {
|
||||||
|
self
|
||||||
|
.iter_mut()
|
||||||
|
.map(|buf| buf.poll_updates())
|
||||||
|
.collect::<FuturesUnordered<_>>()
|
||||||
|
.next()
|
||||||
|
.await
|
||||||
|
.expect("poll_updates reached the end of the never-ending stream");
|
||||||
|
}
|
||||||
|
|
||||||
pub fn iter(&self) -> impl Iterator<Item = &dyn Buffer> {
|
pub fn iter(&self) -> impl Iterator<Item = &dyn Buffer> {
|
||||||
self.buffers.iter().map(|buffer_box| &**buffer_box)
|
self.buffers.iter().map(|buffer_box| &**buffer_box)
|
||||||
}
|
}
|
||||||
|
@ -96,7 +96,7 @@ impl Component for Home {
|
|||||||
// Shift-Enter events (at least on Foot via SSH).
|
// Shift-Enter events (at least on Foot via SSH).
|
||||||
// However, tui-textarea implements Alt-Tab to insert a newline, so it's okay.
|
// However, tui-textarea implements Alt-Tab to insert a newline, so it's okay.
|
||||||
self.textarea.insert_newline();
|
self.textarea.insert_newline();
|
||||||
Ok(Some(Action::Render))
|
Ok(Some(Action::ShouldRender))
|
||||||
},
|
},
|
||||||
_ => {
|
_ => {
|
||||||
assert!(
|
assert!(
|
||||||
@ -104,7 +104,7 @@ impl Component for Home {
|
|||||||
"backlog.handle_key_events returned Some"
|
"backlog.handle_key_events returned Some"
|
||||||
);
|
);
|
||||||
self.textarea.input(key);
|
self.textarea.input(key);
|
||||||
Ok(Some(Action::Render))
|
Ok(Some(Action::ShouldRender))
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user