Skip to content

Commit

Permalink
refactor FloatContent interface
Browse files Browse the repository at this point in the history
  • Loading branch information
cartercanedy committed Oct 2, 2024
1 parent e3460af commit 62427a5
Show file tree
Hide file tree
Showing 4 changed files with 96 additions and 109 deletions.
27 changes: 9 additions & 18 deletions tui/src/confirmation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,7 @@ use std::borrow::Cow;
use crate::{float::FloatContent, hint::Shortcut};

use crossterm::event::{KeyCode, KeyEvent};
use ratatui::{
layout::Alignment,
prelude::*,
widgets::{Block, Borders, Clear, List},
};
use ratatui::{prelude::*, widgets::List};

pub enum ConfirmStatus {
Confirm,
Expand Down Expand Up @@ -57,19 +53,15 @@ impl ConfirmPrompt {
}

impl FloatContent for ConfirmPrompt {
fn draw(&mut self, frame: &mut Frame, area: Rect) {
let block = Block::default()
.borders(Borders::ALL)
.title(" Confirm selections ")
.title_bottom(" [y] to continue, [n] to abort ")
.title_alignment(Alignment::Center)
.title_style(Style::default().bold())
.style(Style::default());

frame.render_widget(block.clone(), area);
fn top_title(&self) -> Option<Line<'_>> {
Some(Line::from(" Confirm selections ").style(Style::default().bold()))
}

let inner_area = block.inner(area);
fn bottom_title(&self) -> Option<Line<'_>> {
Some(Line::from(" [y] to continue, [n] to abort ").italic())
}

fn draw(&mut self, frame: &mut Frame, area: Rect) {
let paths_text = self
.names
.iter()
Expand All @@ -80,8 +72,7 @@ impl FloatContent for ConfirmPrompt {
})
.collect::<Text>();

frame.render_widget(Clear, inner_area);
frame.render_widget(List::new(paths_text), inner_area);
frame.render_widget(List::new(paths_text), area);
}

fn handle_key_event(&mut self, key: &KeyEvent) -> bool {
Expand Down
24 changes: 22 additions & 2 deletions tui/src/float.rs
Original file line number Diff line number Diff line change
@@ -1,13 +1,18 @@
use crossterm::event::{KeyCode, KeyEvent};
use ratatui::{
layout::{Constraint, Direction, Layout, Rect},
layout::{Alignment, Constraint, Direction, Layout, Rect},
style::{Style, Stylize},
text::Line,
widgets::{Block, Borders, Clear},
Frame,
};

use crate::hint::Shortcut;

pub trait FloatContent {
fn draw(&mut self, frame: &mut Frame, area: Rect);
fn top_title(&self) -> Option<Line<'_>>;
fn bottom_title(&self) -> Option<Line<'_>>;
fn handle_key_event(&mut self, key: &KeyEvent) -> bool;
fn is_finished(&self) -> bool;
fn get_shortcut_list(&self) -> (&str, Box<[Shortcut]>);
Expand Down Expand Up @@ -50,7 +55,22 @@ impl<Content: FloatContent + ?Sized> Float<Content> {

pub fn draw(&mut self, frame: &mut Frame, parent_area: Rect) {
let popup_area = self.floating_window(parent_area);
self.content.draw(frame, popup_area);
let mut block = Block::new()
.borders(Borders::ALL)
.title_alignment(Alignment::Center)
.style(Style::new().reset());

if let Some(top_title) = self.content.top_title() {
block = block.title_top(top_title);
}

if let Some(bottom_title) = self.content.bottom_title() {
block = block.title_bottom(bottom_title);
}

frame.render_widget(Clear, popup_area);
frame.render_widget(&block, popup_area);
self.content.draw(frame, block.inner(popup_area));
}

// Returns true if the floating window is finished.
Expand Down
34 changes: 15 additions & 19 deletions tui/src/floating_text.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ use ratatui::{
layout::Rect,
style::{Style, Stylize},
text::Line,
widgets::{Block, Borders, Clear, List},
widgets::{Block, List},
Frame,
};

Expand Down Expand Up @@ -210,21 +210,20 @@ impl FloatingText {
}

impl FloatContent for FloatingText {
fn top_title(&self) -> Option<Line<'_>> {
let title_text = format!(" {} ", self.mode_title);
let title_line = Line::from(title_text)
.centered()
.style(Style::default().reversed());
Some(title_line)
}

fn bottom_title(&self) -> Option<Line<'_>> {
None
}

fn draw(&mut self, frame: &mut Frame, area: Rect) {
// Define the Block with a border and background color
let block = Block::default()
.borders(Borders::ALL)
.title(self.mode_title)
.title_alignment(ratatui::layout::Alignment::Center)
.title_style(Style::default().reversed())
.style(Style::default());

// Draw the Block first
frame.render_widget(block.clone(), area);

// Calculate the inner area to ensure text is not drawn over the border
let inner_area = block.inner(area);
let Rect { height, .. } = inner_area;
let Rect { height, .. } = area;
let lines = self
.src
.iter()
Expand Down Expand Up @@ -267,11 +266,8 @@ impl FloatContent for FloatingText {
.block(Block::default())
.highlight_style(Style::default().reversed());

// Clear the text underneath the floats rendered area
frame.render_widget(Clear, inner_area);

// Render the list inside the bordered area
frame.render_widget(list, inner_area);
frame.render_widget(list, area);
}

fn handle_key_event(&mut self, key: &KeyEvent) -> bool {
Expand Down
120 changes: 50 additions & 70 deletions tui/src/running_command.rs
Original file line number Diff line number Diff line change
@@ -1,32 +1,39 @@
use std::{
cell::{Cell, RefCell},
io::Write,
sync::{Arc, Mutex},
thread::JoinHandle,
};

use crate::{float::FloatContent, hint::Shortcut};

use crossterm::event::{KeyCode, KeyEvent, KeyModifiers};
use linutil_core::Command;

use oneshot::{channel, Receiver};

use portable_pty::{
ChildKiller, CommandBuilder, ExitStatus, MasterPty, NativePtySystem, PtySize, PtySystem,
};

use ratatui::{
layout::{Rect, Size},
style::{Color, Style, Stylize},
text::{Line, Span},
widgets::{Block, Borders},
layout::Rect,
style::{Style, Stylize},
text::Line,
Frame,
};
use std::{
io::Write,
sync::{Arc, Mutex},
thread::JoinHandle,
};

use tui_term::{
vt100::{self, Screen},
widget::PseudoTerminal,
};

use linutil_core::Command;

pub struct RunningCommand {
/// A buffer to save all the command output (accumulates, until the command exits)
buffer: Arc<Mutex<Vec<u8>>>,
/// A handle for the thread running the command
command_thread: Option<JoinHandle<ExitStatus>>,
command_thread: Cell<Option<JoinHandle<ExitStatus>>>,
/// A handle to kill the running process; it's an option because it can only be used once
child_killer: Option<Receiver<Box<dyn ChildKiller + Send + Sync>>>,
/// A join handle for the thread that reads command output and sends it to the main thread
Expand All @@ -36,56 +43,31 @@ pub struct RunningCommand {
/// Used for sending keys to the emulated terminal
writer: Box<dyn Write + Send>,
/// Only set after the process has ended
status: Option<ExitStatus>,
status: RefCell<Option<ExitStatus>>,
scroll_offset: usize,
}

impl FloatContent for RunningCommand {
fn draw(&mut self, frame: &mut Frame, area: Rect) {
// Calculate the inner size of the terminal area, considering borders
let inner_size = Size {
width: area.width - 2, // Adjust for border width
height: area.height - 2,
};

// Define the block for the terminal display
let block = if !self.is_finished() {
// Display a block indicating the command is running
Block::default()
.borders(Borders::ALL)
.title_top(Line::from("Running the command....").centered())
.title_style(Style::default().reversed())
.title_bottom(Line::from("Press Ctrl-C to KILL the command"))
fn top_title(&self) -> Option<Line<'_>> {
let (content, content_style) = if !self.is_finished() {
(" Running command... ", Style::default().reversed())
} else if self.wait_command().success() {
(" Success ", Style::default().green().reversed())
} else {
// Display a block with the command's exit status
let mut title_line = if self.get_exit_status().success() {
Line::from(
Span::default()
.content("SUCCESS!")
.style(Style::default().fg(Color::Green).reversed()),
)
} else {
Line::from(
Span::default()
.content("FAILED!")
.style(Style::default().fg(Color::Red).reversed()),
)
};
(" Failed ", Style::default().red().reversed())
};

title_line.push_span(
Span::default()
.content(" press <ENTER> to close this window ")
.style(Style::default()),
);
Some(Line::from(content).style(content_style))
}

Block::default()
.borders(Borders::ALL)
.title_top(title_line.centered())
};
fn bottom_title(&self) -> Option<Line<'_>> {
Some(Line::from("Press Ctrl-C to KILL the command"))
}

fn draw(&mut self, frame: &mut Frame, area: Rect) {
// Process the buffer and create the pseudo-terminal widget
let screen = self.screen(inner_size);
let pseudo_term = PseudoTerminal::new(&screen).block(block);
let screen = self.screen(area);
let pseudo_term = PseudoTerminal::new(&screen);

// Render the widget on the frame
frame.render_widget(pseudo_term, area);
Expand Down Expand Up @@ -116,12 +98,7 @@ impl FloatContent for RunningCommand {
}

fn is_finished(&self) -> bool {
// Check if the command thread has finished
if let Some(command_thread) = &self.command_thread {
command_thread.is_finished()
} else {
true
}
self.status.borrow().is_some()
}

fn get_shortcut_list(&self) -> (&str, Box<[Shortcut]>) {
Expand Down Expand Up @@ -229,17 +206,17 @@ impl RunningCommand {
let writer = pair.master.take_writer().unwrap();
Self {
buffer: command_buffer,
command_thread: Some(command_handle),
child_killer: Some(rx),
command_thread: Some(command_handle).into(),
child_killer: rx.into(),
_reader_thread: reader_handle,
pty_master: pair.master,
writer,
status: None,
status: None.into(),
scroll_offset: 0,
}
}

fn screen(&mut self, size: Size) -> Screen {
fn screen(&mut self, size: Rect) -> Screen {
// Resize the emulated pty
self.pty_master
.resize(PtySize {
Expand All @@ -263,14 +240,16 @@ impl RunningCommand {
}

/// This function will block if the command is not finished
fn get_exit_status(&mut self) -> ExitStatus {
if self.command_thread.is_some() {
let handle = self.command_thread.take().unwrap();
let exit_status = handle.join().unwrap();
self.status = Some(exit_status.clone());
exit_status
} else {
self.status.as_ref().unwrap().clone()
fn wait_command(&self) -> ExitStatus {
let status = { self.status.borrow().clone() };
match status {
Some(status) => status,
None => {
let handle = self.command_thread.take().unwrap();
let exit_status = handle.join().unwrap();
self.status.replace(Some(exit_status.clone()));
exit_status
}
}
}

Expand All @@ -279,6 +258,7 @@ impl RunningCommand {
if !self.is_finished() {
let mut killer = self.child_killer.take().unwrap().recv().unwrap();
killer.kill().unwrap();
self.wait_command();
}
}

Expand Down

0 comments on commit 62427a5

Please sign in to comment.