Add cursor to textarea
This commit is contained in:
7
Cargo.lock
generated
7
Cargo.lock
generated
@@ -1727,6 +1727,12 @@ dependencies = [
|
|||||||
"getrandom 0.2.16",
|
"getrandom 0.2.16",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rat-cursor"
|
||||||
|
version = "1.2.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7d7c7dfdb4219fa10891a06232b071460a66ccd15ce627e3764d63c0371eb9f5"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ratatui"
|
name = "ratatui"
|
||||||
version = "0.29.0"
|
version = "0.29.0"
|
||||||
@@ -2234,6 +2240,7 @@ dependencies = [
|
|||||||
"futures",
|
"futures",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"libsqlite3-sys",
|
"libsqlite3-sys",
|
||||||
|
"rat-cursor",
|
||||||
"ratatui",
|
"ratatui",
|
||||||
"reqwest",
|
"reqwest",
|
||||||
"robotstxt",
|
"robotstxt",
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ lazy_static = "1.5.0"
|
|||||||
rust-ini = "0.21.3"
|
rust-ini = "0.21.3"
|
||||||
robotstxt = "0.3.0"
|
robotstxt = "0.3.0"
|
||||||
scraper = "0.24.0"
|
scraper = "0.24.0"
|
||||||
|
rat-cursor = "1.2.1"
|
||||||
|
|
||||||
[dependencies.tui-input]
|
[dependencies.tui-input]
|
||||||
version = "0.14.0"
|
version = "0.14.0"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::any::Any;
|
use std::any::{Any};
|
||||||
use crate::event::{AppEvent, EventHandler};
|
use crate::event::{AppEvent, EventHandler};
|
||||||
use ratatui::{DefaultTerminal, Frame};
|
use ratatui::{DefaultTerminal, Frame};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
@@ -6,10 +6,11 @@ use color_eyre::Result;
|
|||||||
use crossterm::event::{Event, KeyEvent};
|
use crossterm::event::{Event, KeyEvent};
|
||||||
use crossterm::event::Event as CrosstermEvent;
|
use crossterm::event::Event as CrosstermEvent;
|
||||||
use diesel::{Connection, SqliteConnection};
|
use diesel::{Connection, SqliteConnection};
|
||||||
|
use rat_cursor::HasScreenCursor;
|
||||||
use crate::config::types::ApplicationConfig;
|
use crate::config::types::ApplicationConfig;
|
||||||
use crate::constants::{APP_CONFIG_DIR, APP_CONIFG_FILE_PATH, APP_DATA_DIR};
|
use crate::constants::{APP_CONFIG_DIR, APP_CONIFG_FILE_PATH, APP_DATA_DIR};
|
||||||
use crate::widgets::views::{View};
|
use crate::widgets::views::{View};
|
||||||
use crate::widgets::views::main_view::MainView;
|
use crate::widgets::views::MainView;
|
||||||
|
|
||||||
pub(crate) struct App {
|
pub(crate) struct App {
|
||||||
events: EventHandler,
|
events: EventHandler,
|
||||||
@@ -99,6 +100,9 @@ impl App {
|
|||||||
if let Some(view) = self.state.view.as_mut() {
|
if let Some(view) = self.state.view.as_mut() {
|
||||||
if let Some(main_view) = view.downcast_mut::<MainView>() {
|
if let Some(main_view) = view.downcast_mut::<MainView>() {
|
||||||
frame.render_stateful_widget(MainView::new(&self.app_config), frame.area(), &mut main_view.state);
|
frame.render_stateful_widget(MainView::new(&self.app_config), frame.area(), &mut main_view.state);
|
||||||
|
if let Some(pos) = main_view.screen_cursor() {
|
||||||
|
frame.set_cursor_position(pos);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1 +1,2 @@
|
|||||||
pub mod textarea;
|
mod textarea;
|
||||||
|
pub use textarea::*;
|
||||||
@@ -1,21 +1,29 @@
|
|||||||
use crossterm::event::{Event};
|
use crossterm::event::{Event};
|
||||||
use ratatui::buffer::Buffer;
|
use ratatui::buffer::Buffer;
|
||||||
use ratatui::layout::Rect;
|
use ratatui::layout::{Constraint, Direction, Layout, Rect};
|
||||||
use ratatui::prelude::StatefulWidget;
|
use ratatui::prelude::StatefulWidget;
|
||||||
use ratatui::text::Text;
|
use ratatui::text::Text;
|
||||||
use ratatui::widgets::{Block, Borders, Paragraph, Widget};
|
use ratatui::widgets::{Block, Borders, Paragraph, Widget};
|
||||||
use tui_input::backend::crossterm::EventHandler;
|
use tui_input::backend::crossterm::{EventHandler};
|
||||||
use tui_input::Input;
|
use tui_input::{Input};
|
||||||
use color_eyre::Result;
|
use color_eyre::Result;
|
||||||
use ratatui::crossterm::cursor::SetCursorStyle;
|
use rat_cursor::HasScreenCursor;
|
||||||
|
use ratatui::style::{Color, Stylize};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct TextArea {
|
pub struct TextArea {
|
||||||
input: Input,
|
input: Input,
|
||||||
title: String,
|
title: String,
|
||||||
|
style: TextAreaStyle,
|
||||||
|
area: Option<Rect>,
|
||||||
pub active: bool,
|
pub active: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub enum TextAreaStyle {
|
||||||
|
Block, SingleLine
|
||||||
|
}
|
||||||
|
|
||||||
impl StatefulWidget for TextArea {
|
impl StatefulWidget for TextArea {
|
||||||
type State = TextArea;
|
type State = TextArea;
|
||||||
fn render(self, area: Rect, buf: &mut Buffer, state: &mut Self::State)
|
fn render(self, area: Rect, buf: &mut Buffer, state: &mut Self::State)
|
||||||
@@ -23,21 +31,57 @@ impl StatefulWidget for TextArea {
|
|||||||
Self: Sized
|
Self: Sized
|
||||||
{
|
{
|
||||||
let input_value = state.input.value().to_string();
|
let input_value = state.input.value().to_string();
|
||||||
|
let title = self.title.clone();
|
||||||
|
if matches!(self.style, TextAreaStyle::Block) {
|
||||||
let block = Block::default()
|
let block = Block::default()
|
||||||
.borders(Borders::ALL)
|
.borders(Borders::ALL)
|
||||||
.title(state.title.clone());
|
.title(title);
|
||||||
let paragraph = Paragraph::new(Text::from(input_value))
|
let paragraph = Paragraph::new(Text::from(input_value))
|
||||||
.block(block);
|
.block(block);
|
||||||
paragraph.render(area, buf);
|
paragraph.render(area, buf);
|
||||||
}
|
}
|
||||||
|
else if matches!(self.style, TextAreaStyle::SingleLine) {
|
||||||
|
let chunks = Layout::default()
|
||||||
|
.direction(Direction::Horizontal)
|
||||||
|
.constraints([
|
||||||
|
Constraint::Percentage(10),
|
||||||
|
Constraint::Fill(0),
|
||||||
|
])
|
||||||
|
.split(area);
|
||||||
|
let text = Text::from(self.title);
|
||||||
|
let line = Paragraph::new(text);
|
||||||
|
let input_text = Text::from(input_value)
|
||||||
|
.fg(Color::White);
|
||||||
|
let paragraph = Paragraph::new(input_text)
|
||||||
|
.bg(Color::Green);
|
||||||
|
state.area = Some(chunks[1]);
|
||||||
|
line.render(chunks[0], buf);
|
||||||
|
paragraph.render(chunks[1], buf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HasScreenCursor for TextArea {
|
||||||
|
fn screen_cursor(&self) -> Option<(u16, u16)> {
|
||||||
|
if self.area.is_none() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
let area = self.area.unwrap();
|
||||||
|
let width = area.width.max(3) - 3;
|
||||||
|
let scroll = self.input.visual_scroll(width as usize);
|
||||||
|
let x = (self.input.visual_cursor().max(scroll) - scroll) as u16;
|
||||||
|
Some((area.x + x, area.y))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TextArea {
|
impl TextArea {
|
||||||
pub fn new(title: &str, placeholder_text: &str) -> Self {
|
pub fn new(title: &str, placeholder_text: &str, style: Option<TextAreaStyle>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
input: Input::new(placeholder_text.to_string()),
|
input: Input::new(placeholder_text.to_string()),
|
||||||
title: title.to_string(),
|
title: title.to_string(),
|
||||||
active: false,
|
active: false,
|
||||||
|
style: style.unwrap_or(TextAreaStyle::SingleLine),
|
||||||
|
area: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ use ratatui::buffer::Buffer;
|
|||||||
use ratatui::layout::{Constraint, Direction, Layout, Margin, Rect};
|
use ratatui::layout::{Constraint, Direction, Layout, Margin, Rect};
|
||||||
use ratatui::prelude::{StatefulWidget, Widget};
|
use ratatui::prelude::{StatefulWidget, Widget};
|
||||||
use ratatui::widgets::{Block, Borders};
|
use ratatui::widgets::{Block, Borders};
|
||||||
use crate::widgets::components::textarea::TextArea;
|
use crate::widgets::components::TextArea;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct AddFolderPopup {
|
pub struct AddFolderPopup {
|
||||||
@@ -11,7 +11,7 @@ pub struct AddFolderPopup {
|
|||||||
|
|
||||||
impl AddFolderPopup {
|
impl AddFolderPopup {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
let mut textarea = TextArea::new("Folder Path", "");
|
let mut textarea = TextArea::new("Folder Path", "", None);
|
||||||
textarea.active = true;
|
textarea.active = true;
|
||||||
Self {
|
Self {
|
||||||
textarea
|
textarea
|
||||||
@@ -38,7 +38,7 @@ impl StatefulWidget for AddFolderPopup {
|
|||||||
let chunks = Layout::default()
|
let chunks = Layout::default()
|
||||||
.direction(Direction::Vertical)
|
.direction(Direction::Vertical)
|
||||||
.constraints(vec![
|
.constraints(vec![
|
||||||
Constraint::Min(2)
|
Constraint::Length(1)
|
||||||
])
|
])
|
||||||
.split(popup_area.inner(Margin::new(1, 1)));
|
.split(popup_area.inner(Margin::new(1, 1)));
|
||||||
self.textarea.render(chunks[0], buf, &mut state.textarea);
|
self.textarea.render(chunks[0], buf, &mut state.textarea);
|
||||||
|
|||||||
@@ -1,14 +1,16 @@
|
|||||||
use std::any::Any;
|
use std::any::Any;
|
||||||
use crossterm::event::{Event, KeyCode, KeyEvent, KeyEventKind};
|
use crossterm::event::{Event, KeyCode, KeyEvent, KeyEventKind};
|
||||||
use crossterm::event::KeyCode::Char;
|
use crossterm::event::KeyCode::Char;
|
||||||
|
use rat_cursor::HasScreenCursor;
|
||||||
use ratatui::buffer::Buffer;
|
use ratatui::buffer::Buffer;
|
||||||
|
use ratatui::Frame;
|
||||||
use ratatui::layout::{Constraint, Direction, Layout, Rect};
|
use ratatui::layout::{Constraint, Direction, Layout, Rect};
|
||||||
use ratatui::prelude::{Color, Line, Span, Style, Text, Widget};
|
use ratatui::prelude::{Color, Line, Span, Style, Text, Widget};
|
||||||
use ratatui::widgets::{Block, Borders, Paragraph, StatefulWidget};
|
use ratatui::widgets::{Block, Borders, Paragraph, StatefulWidget};
|
||||||
use crate::config::types::ApplicationConfig;
|
use crate::config::types::ApplicationConfig;
|
||||||
use crate::constants::APP_CONIFG_FILE_PATH;
|
use crate::constants::APP_CONIFG_FILE_PATH;
|
||||||
use crate::widgets::popups::folder::AddFolderPopup;
|
use crate::widgets::popups::folder::AddFolderPopup;
|
||||||
use crate::widgets::views::{AppStatus, View};
|
use crate::widgets::views::{View};
|
||||||
|
|
||||||
pub struct MainView {
|
pub struct MainView {
|
||||||
app_config: ApplicationConfig,
|
app_config: ApplicationConfig,
|
||||||
@@ -18,7 +20,14 @@ pub struct MainView {
|
|||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct MainViewState {
|
pub struct MainViewState {
|
||||||
popup: Option<Box<dyn Any>>,
|
popup: Option<Box<dyn Any>>,
|
||||||
status: AppStatus,
|
status: Status,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
enum Status {
|
||||||
|
Running,
|
||||||
|
Exiting,
|
||||||
|
Popup
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MainView {
|
impl MainView {
|
||||||
@@ -26,28 +35,25 @@ impl MainView {
|
|||||||
Self {
|
Self {
|
||||||
state: MainViewState {
|
state: MainViewState {
|
||||||
popup: None,
|
popup: None,
|
||||||
status: AppStatus::Running
|
status: Status::Running
|
||||||
},
|
},
|
||||||
app_config: app_conf.clone(),
|
app_config: app_conf.clone()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn quit(&mut self) -> color_eyre::Result<()> {
|
fn quit(&mut self) -> color_eyre::Result<()> {
|
||||||
if self.state.popup.is_none() {
|
if self.state.popup.is_none() {
|
||||||
self.state.status = AppStatus::Exiting;
|
self.state.status = Status::Exiting;
|
||||||
self.app_config
|
self.app_config
|
||||||
.clone()
|
.clone()
|
||||||
.write_to_file(&APP_CONIFG_FILE_PATH.to_path_buf())?;
|
.write_to_file(&APP_CONIFG_FILE_PATH.to_path_buf())?;
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
self.state.popup = None;
|
|
||||||
}
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn folder_popup(&mut self) {
|
fn folder_popup(&mut self) {
|
||||||
self.state.popup = Some(Box::new(AddFolderPopup::new()));
|
self.state.popup = Some(Box::new(AddFolderPopup::new()));
|
||||||
self.state.status = AppStatus::Input;
|
self.state.status = Status::Popup;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -61,10 +67,11 @@ impl View for MainView {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn handle_key_input(&mut self, key: &KeyEvent) -> color_eyre::Result<()> {
|
fn handle_key_input(&mut self, key: &KeyEvent) -> color_eyre::Result<()> {
|
||||||
if matches!(self.state.status, AppStatus::Input) && matches!(key.code, KeyCode::Esc) {
|
if matches!(self.state.status, Status::Popup) && matches!(key.code, KeyCode::Esc) {
|
||||||
self.state.status = AppStatus::Running;
|
self.state.status = Status::Running;
|
||||||
|
self.state.popup = None;
|
||||||
}
|
}
|
||||||
if matches!(key.kind, KeyEventKind::Press) && !matches!(self.state.status, AppStatus::Input) {
|
if !matches!(self.state.status, Status::Popup) && matches!(key.kind, KeyEventKind::Press) {
|
||||||
match key.code {
|
match key.code {
|
||||||
Char('q') => self.quit()?,
|
Char('q') => self.quit()?,
|
||||||
Char('a') => self.folder_popup(),
|
Char('a') => self.folder_popup(),
|
||||||
@@ -75,7 +82,7 @@ impl View for MainView {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn is_running(&self) -> bool {
|
fn is_running(&self) -> bool {
|
||||||
!matches!(self.state.status, AppStatus::Exiting)
|
!matches!(self.state.status, Status::Exiting)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -105,6 +112,16 @@ impl StatefulWidget for MainView {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl HasScreenCursor for MainView {
|
||||||
|
fn screen_cursor(&self) -> Option<(u16, u16)> {
|
||||||
|
if let Some(popup) = &self.state.popup &&
|
||||||
|
let Some(add_folder) = popup.downcast_ref::<AddFolderPopup>() {
|
||||||
|
return add_folder.textarea.screen_cursor()
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl MainView {
|
impl MainView {
|
||||||
fn render_game_list(state: &mut MainViewState, area: Rect, buf: &mut Buffer) {
|
fn render_game_list(state: &mut MainViewState, area: Rect, buf: &mut Buffer) {
|
||||||
let game_list = Block::new()
|
let game_list = Block::new()
|
||||||
@@ -128,8 +145,8 @@ impl MainView {
|
|||||||
let mut navigation_text = vec![
|
let mut navigation_text = vec![
|
||||||
Span::styled("(q) quit / (a) add folders", Style::default().fg(Color::Green)),
|
Span::styled("(q) quit / (a) add folders", Style::default().fg(Color::Green)),
|
||||||
];
|
];
|
||||||
if matches!(state.status, AppStatus::Input) {
|
if matches!(state.status, Status::Popup) {
|
||||||
navigation_text[0] = Span::styled("Input Mode", Style::default().fg(Color::Green));
|
navigation_text[0] = Span::styled("(Esc) close", Style::default().fg(Color::Green));
|
||||||
}
|
}
|
||||||
let line = Line::from(navigation_text);
|
let line = Line::from(navigation_text);
|
||||||
let footer = Paragraph::new(line);
|
let footer = Paragraph::new(line);
|
||||||
|
|||||||
@@ -1,14 +1,7 @@
|
|||||||
use std::any::Any;
|
mod main_view;
|
||||||
|
|
||||||
use crossterm::event::{Event, KeyEvent};
|
use crossterm::event::{Event, KeyEvent};
|
||||||
|
pub use main_view::MainView;
|
||||||
pub mod main_view;
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy)]
|
|
||||||
pub enum AppStatus {
|
|
||||||
Running,
|
|
||||||
Exiting,
|
|
||||||
Input
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait View {
|
pub trait View {
|
||||||
fn handle_input(&mut self, event: &Event) -> color_eyre::Result<()>;
|
fn handle_input(&mut self, event: &Event) -> color_eyre::Result<()>;
|
||||||
|
|||||||
Reference in New Issue
Block a user