Add textarea and run commands from it

This commit is contained in:
2023-11-01 21:53:07 +01:00
parent 1c13d2c784
commit 72018ff53c
5 changed files with 69 additions and 8 deletions

View File

@ -7,7 +7,6 @@
], ],
"keybindings": { "keybindings": {
"Home": { "Home": {
"<q>": "/quit",
"<Ctrl-d>": "/quit", "<Ctrl-d>": "/quit",
"<Ctrl-c>": "/quit", "<Ctrl-c>": "/quit",
"<Ctrl-z>": "/suspend", "<Ctrl-z>": "/suspend",

View File

@ -53,6 +53,7 @@ ansi-to-tui = "3.1.0"
crossterm = { version = "0.27.0", features = ["serde", "event-stream"] } crossterm = { version = "0.27.0", features = ["serde", "event-stream"] }
ratatui = { version = "0.24.0", features = ["serde", "macros"] } ratatui = { version = "0.24.0", features = ["serde", "macros"] }
strip-ansi-escapes = "0.2.0" strip-ansi-escapes = "0.2.0"
tui-textarea = "0.3.0"
[dev-dependencies] [dev-dependencies]
pretty_assertions = "1.4.0" pretty_assertions = "1.4.0"

View File

@ -17,6 +17,7 @@ pub enum Action {
Error(String), Error(String),
PreviousBuffer, PreviousBuffer,
NextBuffer, NextBuffer,
RunCommand(String),
Help, Help,
} }

View File

@ -266,6 +266,10 @@ impl App {
} }
})?; })?;
}, },
Action::RunCommand(ref command_line) => {
log::info!("Got command: {command_line}");
crate::commands::run_command(&command_line, &self, &action_tx)?;
},
_ => {}, _ => {},
} }
for component in self.components.iter_mut() { for component in self.components.iter_mut() {
@ -304,8 +308,7 @@ impl App {
tui::Event::Key(key) => { tui::Event::Key(key) => {
if let Some(keymap) = self.config.keybindings.get(&self.mode) { if let Some(keymap) = self.config.keybindings.get(&self.mode) {
if let Some(command_line) = keymap.get(&vec![key]) { if let Some(command_line) = keymap.get(&vec![key]) {
log::info!("Got command: {command_line}"); action_tx.send(Action::RunCommand(command_line.clone()))?;
crate::commands::run_command(command_line, &self, &action_tx)?;
} else { } else {
let mut last_tick_key_events = self.last_tick_key_events.borrow_mut(); let mut last_tick_key_events = self.last_tick_key_events.borrow_mut();
@ -315,8 +318,7 @@ impl App {
// Check for multi-key combinations // Check for multi-key combinations
if let Some(command_line) = keymap.get(&*last_tick_key_events) { if let Some(command_line) = keymap.get(&*last_tick_key_events) {
log::info!("Got command: {command_line}"); action_tx.send(Action::RunCommand(command_line.clone()))?;
crate::commands::run_command(command_line, &self, &action_tx)?;
} }
} }
}; };

View File

@ -1,11 +1,12 @@
use std::{collections::HashMap, time::Duration}; use std::{collections::HashMap, time::Duration};
use color_eyre::eyre::{Result, WrapErr}; use color_eyre::eyre::{Result, WrapErr};
use crossterm::event::{KeyCode, KeyEvent}; use crossterm::event::{KeyCode, KeyEvent, KeyEventKind, KeyModifiers};
use ratatui::{prelude::*, widgets::*}; use ratatui::{prelude::*, widgets::*};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use tokio::sync::mpsc::UnboundedSender; use tokio::sync::mpsc::UnboundedSender;
use tokio::sync::OnceCell; use tokio::sync::OnceCell;
use tui_textarea::{Input, Key, TextArea};
use super::{Buflist, Component}; use super::{Buflist, Component};
use crate::{ use crate::{
@ -18,11 +19,21 @@ pub struct Home {
command_tx: Option<UnboundedSender<Action>>, command_tx: Option<UnboundedSender<Action>>,
config: OnceCell<Config>, config: OnceCell<Config>,
buflist: Buflist, buflist: Buflist,
textarea: TextArea<'static>,
} }
impl Home { impl Home {
pub fn new() -> Self { pub fn new() -> Self {
Self::default() let mut self_ = Self::default();
self_.configure_textarea();
self_
}
fn configure_textarea(&mut self) {
self
.textarea
.set_block(Block::default().borders(Borders::ALL));
self.textarea.set_cursor_line_style(Style::default());
} }
} }
@ -42,6 +53,41 @@ impl Component for Home {
Ok(()) Ok(())
} }
fn handle_key_events(&mut self, key: KeyEvent) -> Result<Option<Action>> {
match key {
KeyEvent {
code: KeyCode::Enter,
modifiers: KeyModifiers::NONE,
kind: KeyEventKind::Press,
state: _,
} => {
// User pressed enter; clear the textarea by replacing it with a new one,
// and get the content of the old one.
let mut textarea = TextArea::default();
std::mem::swap(&mut textarea, &mut self.textarea);
let command = textarea.into_lines().join("\n");
self.configure_textarea();
Ok(Some(Action::RunCommand(command)))
},
KeyEvent {
code: KeyCode::Enter,
modifiers: KeyModifiers::SHIFT,
kind: KeyEventKind::Press,
state: _,
} => {
// FIXME: For some reason this does nothing because Crossterm does not emit
// Shift-Enter events (at least on Foot via SSH).
// However, tui-textarea implements Alt-Tab to insert a newline, so it's okay.
self.textarea.insert_newline();
Ok(Some(Action::Render))
},
_ => {
self.textarea.input(key);
Ok(Some(Action::Render))
},
}
}
fn update(&mut self, action: &Action) -> Result<Option<Action>> { fn update(&mut self, action: &Action) -> Result<Option<Action>> {
self.buflist.update(action)?; self.buflist.update(action)?;
match action { match action {
@ -66,10 +112,22 @@ impl Component for Home {
.buflist .buflist
.draw(frame, layout[0], buffers) .draw(frame, layout[0], buffers)
.context("Error drawing buflist")?; .context("Error drawing buflist")?;
let layout = Layout::default()
.direction(Direction::Vertical)
.constraints(vec![
Constraint::Min(5),
Constraint::Length(self.textarea.lines().len() as u16 + 2), // +2 for borders
])
.split(layout[1]);
frame.render_widget( frame.render_widget(
Paragraph::new(buffers.active_buffer().content()).block(Block::new().borders(Borders::ALL)), Paragraph::new(buffers.active_buffer().content()).block(Block::new().borders(Borders::ALL)),
layout[1], layout[0],
); );
frame.render_widget(self.textarea.widget(), layout[1]);
Ok(()) Ok(())
} }
} }