custom history list (#524)

* use custom list impl

* fmt

* segment

* clean up

* fix offsets

* fix scroll back space

* small touch ups
This commit is contained in:
Conrad Ludgate 2022-09-12 20:39:41 +01:00 committed by GitHub
parent e8c8415278
commit db2a00f456
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 232 additions and 140 deletions

View file

@ -11,6 +11,7 @@ use super::history::ListMode;
mod cursor; mod cursor;
mod duration; mod duration;
mod event; mod event;
mod history_list;
mod interactive; mod interactive;
pub use duration::format_duration; pub use duration::format_duration;

View file

@ -14,6 +14,10 @@ impl Cursor {
self.source.as_str() self.source.as_str()
} }
pub fn into_inner(self) -> String {
self.source
}
/// Returns the string before the cursor /// Returns the string before the cursor
pub fn substring(&self) -> &str { pub fn substring(&self) -> &str {
&self.source[..self.index] &self.source[..self.index]

View file

@ -0,0 +1,175 @@
use std::time::Duration;
use atuin_client::history::History;
use tui::{
buffer::Buffer,
layout::Rect,
style::{Color, Modifier, Style},
widgets::{Block, StatefulWidget, Widget},
};
use super::format_duration;
pub struct HistoryList<'a> {
history: &'a [History],
block: Option<Block<'a>>,
}
#[derive(Default)]
pub struct ListState {
offset: usize,
selected: usize,
}
impl ListState {
pub fn selected(&self) -> usize {
self.selected
}
pub fn select(&mut self, index: usize) {
self.selected = index;
}
}
impl<'a> StatefulWidget for HistoryList<'a> {
type State = ListState;
fn render(mut self, area: Rect, buf: &mut Buffer, state: &mut Self::State) {
let list_area = match self.block.take() {
Some(b) => {
let inner_area = b.inner(area);
b.render(area, buf);
inner_area
}
None => area,
};
if list_area.width < 1 || list_area.height < 1 || self.history.is_empty() {
return;
}
let list_height = list_area.height as usize;
let (start, end) = self.get_items_bounds(state.selected, state.offset, list_height);
state.offset = start;
let mut s = DrawState {
buf,
list_area,
x: 0,
y: 0,
state,
};
for item in self.history.iter().skip(state.offset).take(end - start) {
s.index();
s.duration(item);
s.time(item);
s.command(item);
// reset line
s.y += 1;
s.x = 0;
}
}
}
impl<'a> HistoryList<'a> {
pub fn new(history: &'a [History]) -> Self {
Self {
history,
block: None,
}
}
pub fn block(mut self, block: Block<'a>) -> Self {
self.block = Some(block);
self
}
fn get_items_bounds(&self, selected: usize, offset: usize, height: usize) -> (usize, usize) {
let offset = offset.min(self.history.len().saturating_sub(1));
let max_scroll_space = height.min(10);
if offset + height < selected + max_scroll_space {
let end = selected + max_scroll_space;
(end - height, end)
} else if selected < offset {
(selected, selected + height)
} else {
(offset, offset + height)
}
}
}
struct DrawState<'a> {
buf: &'a mut Buffer,
list_area: Rect,
x: u16,
y: u16,
state: &'a ListState,
}
// longest line prefix I could come up with
#[allow(clippy::cast_possible_truncation)] // we know that this is <65536 length
pub const PREFIX_LENGTH: u16 = " > 123ms 59s ago".len() as u16;
impl DrawState<'_> {
fn index(&mut self) {
// these encode the slices of `" > "`, `" {n} "`, or `" "` in a compact form.
// Yes, this is a hack, but it makes me feel happy
static SLICES: &str = " > 1 2 3 4 5 6 7 8 9 ";
let i = self.y as usize + self.state.offset;
let i = i.checked_sub(self.state.selected);
let i = i.unwrap_or(10).min(10) * 2;
self.draw(&SLICES[i..i + 3], Style::default());
}
fn duration(&mut self, h: &History) {
let status = Style::default().fg(if h.success() {
Color::Green
} else {
Color::Red
});
let duration = Duration::from_nanos(u64::try_from(h.duration).unwrap_or(0));
self.draw(&format_duration(duration), status);
}
#[allow(clippy::cast_possible_truncation)] // we know that time.len() will be <6
fn time(&mut self, h: &History) {
let style = Style::default().fg(Color::Blue);
// Account for the chance that h.timestamp is "in the future"
// This would mean that "since" is negative, and the unwrap here
// would fail.
// If the timestamp would otherwise be in the future, display
// the time since as 0.
let since = chrono::Utc::now() - h.timestamp;
let time = format_duration(since.to_std().unwrap_or_default());
// pad the time a little bit before we write. this aligns things nicely
self.x = PREFIX_LENGTH - 4 - time.len() as u16;
self.draw(&time, style);
self.draw(" ago", style);
}
fn command(&mut self, h: &History) {
let mut style = Style::default();
if self.y as usize + self.state.offset == self.state.selected {
style = style.fg(Color::Red).add_modifier(Modifier::BOLD);
}
for section in h.command.split_ascii_whitespace() {
self.x += 1;
self.draw(section, style);
}
}
fn draw(&mut self, s: &str, style: Style) {
let cx = self.list_area.left() + self.x;
let cy = self.list_area.bottom() - self.y - 1;
let w = (self.list_area.width - self.x) as usize;
self.x += self.buf.set_stringn(cx, cy, s, w, style).0 - cx;
}
}

View file

@ -1,4 +1,4 @@
use std::{io::stdout, ops::Sub, time::Duration}; use std::io::stdout;
use eyre::Result; use eyre::Result;
use termion::{ use termion::{
@ -7,10 +7,10 @@ use termion::{
}; };
use tui::{ use tui::{
backend::{Backend, TermionBackend}, backend::{Backend, TermionBackend},
layout::{Alignment, Constraint, Corner, Direction, Layout}, layout::{Alignment, Constraint, Direction, Layout},
style::{Color, Modifier, Style}, style::{Color, Modifier, Style},
text::{Span, Spans, Text}, text::{Span, Spans, Text},
widgets::{Block, BorderType, Borders, List, ListItem, ListState, Paragraph}, widgets::{Block, BorderType, Borders, Paragraph},
Frame, Terminal, Frame, Terminal,
}; };
use unicode_width::UnicodeWidthStr; use unicode_width::UnicodeWidthStr;
@ -26,98 +26,24 @@ use atuin_client::{
use super::{ use super::{
cursor::Cursor, cursor::Cursor,
event::{Event, Events}, event::{Event, Events},
format_duration, history_list::{HistoryList, ListState, PREFIX_LENGTH},
}; };
use crate::VERSION; use crate::VERSION;
struct State { struct State {
history_count: i64,
input: Cursor, input: Cursor,
filter_mode: FilterMode, filter_mode: FilterMode,
results: Vec<History>,
results_state: ListState, results_state: ListState,
context: Context, context: Context,
} }
fn duration(h: &History) -> String {
let duration = Duration::from_nanos(u64::try_from(h.duration).unwrap_or(0));
format_duration(duration)
}
fn ago(h: &History) -> String {
let ago = chrono::Utc::now().sub(h.timestamp);
// Account for the chance that h.timestamp is "in the future"
// This would mean that "ago" is negative, and the unwrap here
// would fail.
// If the timestamp would otherwise be in the future, display
// the time ago as 0.
let ago = ago.to_std().unwrap_or_default();
format_duration(ago) + " ago"
}
impl State {
fn render_results<T: tui::backend::Backend>(
&mut self,
f: &mut tui::Frame<T>,
r: tui::layout::Rect,
b: tui::widgets::Block,
) {
let max_length = 12; // '123ms' + '59s ago'
let results: Vec<ListItem> = self
.results
.iter()
.enumerate()
.map(|(i, m)| {
// these encode the slices of `" > "`, `" {n} "`, or `" "` in a compact form.
// Yes, this is a hack, but it makes me feel happy
let slices = " > 1 2 3 4 5 6 7 8 9 ";
let index = self.results_state.selected().and_then(|s| i.checked_sub(s));
let slice_index = index.unwrap_or(10).min(10) * 2;
let status_colour = if m.success() {
Color::Green
} else {
Color::Red
};
let ago = ago(m);
let duration = format!("{:width$}", duration(m), width = max_length - ago.len());
let command = m.command.replace(['\n', '\t'], " ");
let mut command = Span::raw(command);
if slice_index == 0 {
command.style = Style::default().fg(Color::Red).add_modifier(Modifier::BOLD);
}
let spans = Spans::from(vec![
Span::raw(&slices[slice_index..slice_index + 3]),
Span::styled(duration, Style::default().fg(status_colour)),
Span::raw(" "),
Span::styled(ago, Style::default().fg(Color::Blue)),
Span::raw(" "),
command,
]);
ListItem::new(spans)
})
.collect();
let results = List::new(results).block(b).start_corner(Corner::BottomLeft);
f.render_stateful_widget(results, r, &mut self.results_state);
}
}
impl State { impl State {
async fn query_results( async fn query_results(
&mut self, &mut self,
search_mode: SearchMode, search_mode: SearchMode,
db: &mut impl Database, db: &mut impl Database,
) -> Result<()> { ) -> Result<Vec<History>> {
let i = self.input.as_str(); let i = self.input.as_str();
let results = if i.is_empty() { let results = if i.is_empty() {
db.list(self.filter_mode, &self.context, Some(200), true) db.list(self.filter_mode, &self.context, Some(200), true)
@ -127,38 +53,19 @@ impl State {
.await? .await?
}; };
self.results = results; self.results_state.select(0);
Ok(results)
if self.results.is_empty() {
self.results_state.select(None);
} else {
self.results_state.select(Some(0));
} }
Ok(()) fn handle_input(&mut self, input: &TermEvent, len: usize) -> Option<usize> {
}
fn handle_input(&mut self, input: &TermEvent) -> Option<&str> {
match input { match input {
TermEvent::Key(Key::Esc | Key::Ctrl('c' | 'd' | 'g')) => return Some(""), TermEvent::Key(Key::Esc | Key::Ctrl('c' | 'd' | 'g')) => return Some(usize::MAX),
TermEvent::Key(Key::Char('\n')) => { TermEvent::Key(Key::Char('\n')) => {
let i = self.results_state.selected().unwrap_or(0); return Some(self.results_state.selected());
return Some(
self.results
.get(i)
.map_or(self.input.as_str(), |h| h.command.as_str()),
);
} }
TermEvent::Key(Key::Alt(c @ '1'..='9')) => { TermEvent::Key(Key::Alt(c @ '1'..='9')) => {
let c = c.to_digit(10)? as usize; let c = c.to_digit(10)? as usize;
let i = self.results_state.selected()? + c; return Some(self.results_state.selected() + c);
return Some(
self.results
.get(i)
.map_or(self.input.as_str(), |h| h.command.as_str()),
);
} }
TermEvent::Key(Key::Left | Key::Ctrl('h')) => { TermEvent::Key(Key::Left | Key::Ctrl('h')) => {
self.input.left(); self.input.left();
@ -195,20 +102,13 @@ impl State {
} }
TermEvent::Key(Key::Down | Key::Ctrl('n' | 'j')) TermEvent::Key(Key::Down | Key::Ctrl('n' | 'j'))
| TermEvent::Mouse(MouseEvent::Press(MouseButton::WheelDown, _, _)) => { | TermEvent::Mouse(MouseEvent::Press(MouseButton::WheelDown, _, _)) => {
let i = self let i = self.results_state.selected().saturating_sub(1);
.results_state self.results_state.select(i);
.selected() // try get current selection
.map_or(0, |i| i.saturating_sub(1)); // subtract 1 if possible
self.results_state.select(Some(i));
} }
TermEvent::Key(Key::Up | Key::Ctrl('p' | 'k')) TermEvent::Key(Key::Up | Key::Ctrl('p' | 'k'))
| TermEvent::Mouse(MouseEvent::Press(MouseButton::WheelUp, _, _)) => { | TermEvent::Mouse(MouseEvent::Press(MouseButton::WheelUp, _, _)) => {
let i = self let i = self.results_state.selected() + 1;
.results_state self.results_state.select(i.min(len - 1));
.selected()
.map_or(0, |i| i + 1) // increment the selected index
.min(self.results.len() - 1); // clamp it to the last entry
self.results_state.select(Some(i));
} }
_ => {} _ => {}
}; };
@ -217,7 +117,7 @@ impl State {
} }
#[allow(clippy::cast_possible_truncation)] #[allow(clippy::cast_possible_truncation)]
fn draw<T: Backend>(&mut self, f: &mut Frame<'_, T>, history_count: i64) { fn draw<T: Backend>(&mut self, f: &mut Frame<'_, T>, results: &[History]) {
let chunks = Layout::default() let chunks = Layout::default()
.direction(Direction::Vertical) .direction(Direction::Vertical)
.margin(0) .margin(0)
@ -256,21 +156,22 @@ impl State {
let help = Paragraph::new(Text::from(Spans::from(help))); let help = Paragraph::new(Text::from(Spans::from(help)));
let stats = Paragraph::new(Text::from(Span::raw(format!( let stats = Paragraph::new(Text::from(Span::raw(format!(
"history count: {history_count} ", "history count: {} ",
self.history_count
)))); ))));
f.render_widget(title, top_left_chunks[1]); f.render_widget(title, top_left_chunks[1]);
f.render_widget(help, top_left_chunks[2]); f.render_widget(help, top_left_chunks[2]);
f.render_widget(stats.alignment(Alignment::Right), top_right_chunks[1]); f.render_widget(stats.alignment(Alignment::Right), top_right_chunks[1]);
self.render_results( let results = HistoryList::new(results).block(
f,
chunks[1],
Block::default() Block::default()
.borders(Borders::TOP | Borders::LEFT | Borders::RIGHT) .borders(Borders::TOP | Borders::LEFT | Borders::RIGHT)
.border_type(BorderType::Rounded), // .title("History"), .border_type(BorderType::Rounded),
); );
f.render_stateful_widget(results, chunks[1], &mut self.results_state);
let input = format!( let input = format!(
"[{:^14}] {}", "[{:^14}] {}",
self.filter_mode.as_str(), self.filter_mode.as_str(),
@ -291,14 +192,14 @@ impl State {
let width = UnicodeWidthStr::width(self.input.substring()); let width = UnicodeWidthStr::width(self.input.substring());
f.set_cursor( f.set_cursor(
// Put cursor past the end of the input text // Put cursor past the end of the input text
chunks[2].x + width as u16 + 18, chunks[2].x + width as u16 + PREFIX_LENGTH + 2,
// Move one line down, from the border to the input line // Move one line down, from the border to the input line
chunks[2].y + 1, chunks[2].y + 1,
); );
} }
#[allow(clippy::cast_possible_truncation)] #[allow(clippy::cast_possible_truncation)]
fn draw_compact<T: Backend>(&mut self, f: &mut Frame<'_, T>, history_count: i64) { fn draw_compact<T: Backend>(&mut self, f: &mut Frame<'_, T>, results: &[History]) {
let chunks = Layout::default() let chunks = Layout::default()
.direction(Direction::Vertical) .direction(Direction::Vertical)
.margin(0) .margin(0)
@ -339,7 +240,7 @@ impl State {
let stats = Paragraph::new(Text::from(Span::raw(format!( let stats = Paragraph::new(Text::from(Span::raw(format!(
"history count: {}", "history count: {}",
history_count, self.history_count,
)))) ))))
.style(Style::default().fg(Color::DarkGray)) .style(Style::default().fg(Color::DarkGray))
.alignment(Alignment::Right); .alignment(Alignment::Right);
@ -348,21 +249,22 @@ impl State {
f.render_widget(help, header_chunks[1]); f.render_widget(help, header_chunks[1]);
f.render_widget(stats, header_chunks[2]); f.render_widget(stats, header_chunks[2]);
self.render_results(f, chunks[1], Block::default()); let results = HistoryList::new(results);
f.render_stateful_widget(results, chunks[1], &mut self.results_state);
let input = format!( let input = format!(
"[{:^14}] {}", "[{:^14}] {}",
self.filter_mode.as_str(), self.filter_mode.as_str(),
self.input.as_str(), self.input.as_str(),
); );
let input = Paragraph::new(input).block(Block::default()); let input = Paragraph::new(input);
f.render_widget(input, chunks[2]); f.render_widget(input, chunks[2]);
let extra_width = UnicodeWidthStr::width(self.input.substring()); let extra_width = UnicodeWidthStr::width(self.input.substring());
f.set_cursor( f.set_cursor(
// Put cursor past the end of the input text // Put cursor past the end of the input text
chunks[2].x + extra_width as u16 + 17, chunks[2].x + extra_width as u16 + PREFIX_LENGTH + 1,
// Move one line down, from the border to the input line // Move one line down, from the border to the input line
chunks[2].y + 1, chunks[2].y + 1,
); );
@ -393,36 +295,35 @@ pub async fn history(
// Put the cursor at the end of the query by default // Put the cursor at the end of the query by default
input.end(); input.end();
let mut app = State { let mut app = State {
history_count: db.history_count().await?,
input, input,
results: Vec::new(),
results_state: ListState::default(), results_state: ListState::default(),
context: current_context(), context: current_context(),
filter_mode, filter_mode,
}; };
app.query_results(search_mode, db).await?; let mut results = app.query_results(search_mode, db).await?;
loop { let index = 'render: loop {
let history_count = db.history_count().await?;
let initial_input = app.input.as_str().to_owned(); let initial_input = app.input.as_str().to_owned();
let initial_filter_mode = app.filter_mode; let initial_filter_mode = app.filter_mode;
// Handle input // Handle input
if let Event::Input(input) = events.next()? { if let Event::Input(input) = events.next()? {
if let Some(output) = app.handle_input(&input) { if let Some(i) = app.handle_input(&input, results.len()) {
return Ok(output.to_owned()); break 'render i;
} }
} }
// After we receive input process the whole event channel before query/render. // After we receive input process the whole event channel before query/render.
while let Ok(Event::Input(input)) = events.try_next() { while let Ok(Event::Input(input)) = events.try_next() {
if let Some(output) = app.handle_input(&input) { if let Some(i) = app.handle_input(&input, results.len()) {
return Ok(output.to_owned()); break 'render i;
} }
} }
if initial_input != app.input.as_str() || initial_filter_mode != app.filter_mode { if initial_input != app.input.as_str() || initial_filter_mode != app.filter_mode {
app.query_results(search_mode, db).await?; results = app.query_results(search_mode, db).await?;
} }
let compact = match style { let compact = match style {
@ -433,9 +334,20 @@ pub async fn history(
atuin_client::settings::Style::Full => false, atuin_client::settings::Style::Full => false,
}; };
if compact { if compact {
terminal.draw(|f| app.draw_compact(f, history_count))?; terminal.draw(|f| app.draw_compact(f, &results))?;
} else { } else {
terminal.draw(|f| app.draw(f, history_count))?; terminal.draw(|f| app.draw(f, &results))?;
} }
};
if index < results.len() {
// index is in bounds so we return that entry
Ok(results.swap_remove(index).command)
} else if index == usize::MAX {
// index is max which implies an early exit
Ok(String::new())
} else {
// out of bounds usually implies no selected entry so we return the input
Ok(app.input.into_inner())
} }
} }

View file

@ -1,5 +1,5 @@
#![warn(clippy::pedantic, clippy::nursery)] #![warn(clippy::pedantic, clippy::nursery)]
#![allow(clippy::use_self)] // not 100% reliable #![allow(clippy::use_self, clippy::missing_const_for_fn)] // not 100% reliable
use clap::{AppSettings, Parser}; use clap::{AppSettings, Parser};
use eyre::Result; use eyre::Result;