Recount-TUI/src/app.rs

142 lines
4 KiB
Rust
Raw Normal View History

2023-08-07 22:27:27 -07:00
use std::sync::Arc;
2023-08-12 11:54:09 -07:00
use crossterm::event::{self, Event, KeyCode};
2023-08-07 22:27:27 -07:00
use tokio;
2023-08-09 22:39:00 -07:00
2023-08-07 22:27:27 -07:00
use log::warn;
2023-08-11 22:46:43 -07:00
use crate::db::data_cache::DataCache;
2023-08-12 11:54:09 -07:00
use crate::db::DB;
use crate::uis::history::HistoryState;
use crate::uis::navigation_frame::NavigationState;
2023-08-12 11:54:09 -07:00
use crate::uis::new_transaction::NewTransactionTabState;
2023-05-31 22:27:26 -07:00
2023-05-31 18:00:17 -07:00
pub type AppResult<T> = std::result::Result<T, Box<dyn std::error::Error>>;
2023-05-28 22:18:30 -07:00
pub enum InputMode {
Insert,
Normal,
2023-05-31 22:27:26 -07:00
}
2023-05-28 22:18:30 -07:00
pub struct App<'a> {
pub running: bool,
2023-05-31 22:27:26 -07:00
pub states: States<'a>,
2023-05-31 22:27:26 -07:00
pub input_mode: InputMode,
2023-08-07 22:27:27 -07:00
pub db: Arc<tokio::sync::Mutex<DB>>,
2023-08-11 22:46:43 -07:00
pub data_cache: DataCache,
2023-05-28 22:18:30 -07:00
}
impl<'a> App<'a> {
2023-08-11 22:46:43 -07:00
pub fn new(db: DB, data_cache: DataCache) -> App<'a> {
2023-05-31 18:00:17 -07:00
App {
running: true,
2023-08-11 22:46:43 -07:00
states: States::new(),
input_mode: InputMode::Normal,
2023-08-07 22:27:27 -07:00
db: Arc::new(tokio::sync::Mutex::new(db)),
2023-08-11 22:46:43 -07:00
data_cache,
2023-05-31 18:00:17 -07:00
}
2023-05-28 22:18:30 -07:00
}
pub fn poll_events(&mut self) -> AppResult<()> {
if let Event::Key(key) = event::read()? {
// 'q' needs to be handled at the top level so it can't be
// accidentally handed to a dead end
if let KeyCode::Char('q') = key.code {
if let Some(_) = self.states.nav_state.message.clone() {
self.states.nav_state.message = None;
} else {
if let ActiveFrame::Navigation = self.states.active_frame {
self.running = false;
} else {
self.states.active_frame = ActiveFrame::Navigation;
}
}
2023-08-07 21:01:13 -07:00
} else if let KeyCode::Esc = key.code {
if let Some(_) = self.states.nav_state.message.clone() {
self.states.nav_state.message = None;
} else {
if let ActiveFrame::Navigation = self.states.active_frame {
self.running = false;
} else {
self.states.active_frame = ActiveFrame::Navigation;
}
}
}
2023-08-07 22:27:27 -07:00
if let KeyCode::Char('r') = key.code {
self.refresh();
}
match self.states.active_frame {
ActiveFrame::Navigation => {
NavigationState::handle_event(key, self);
}
ActiveFrame::History => {
HistoryState::handle_event(key, self);
}
ActiveFrame::NewTransaction => {
NewTransactionTabState::handle_event(key, self);
}
}
2023-05-28 22:18:30 -07:00
}
2023-08-12 11:54:09 -07:00
return Ok(());
2023-05-31 22:27:26 -07:00
}
2023-08-07 22:27:27 -07:00
pub fn refresh(&mut self) {
2023-08-17 22:24:12 -07:00
let db_clone1 = Arc::clone(&self.db);
let db_clone2 = Arc::clone(&self.db);
let db_clone3 = Arc::clone(&self.db);
2023-08-12 11:54:09 -07:00
tokio::spawn(async move {
2023-08-17 22:24:12 -07:00
let res = db_clone1.lock().await.get_all_transactions().await;
match res {
Ok(_) => {}
Err(e) => warn!("{}", e),
}
});
tokio::spawn(async move {
let res = db_clone2.lock().await.get_all_accounts().await;
match res {
Ok(_) => {}
Err(e) => warn!("{}", e),
}
});
tokio::spawn(async move {
let res = db_clone3.lock().await.get_all_buckets().await;
2023-08-07 22:27:27 -07:00
match res {
2023-08-12 11:54:09 -07:00
Ok(_) => {}
Err(e) => warn!("{}", e),
2023-08-07 22:27:27 -07:00
}
});
}
}
2023-05-31 22:27:26 -07:00
pub enum ActiveFrame {
Navigation,
NewTransaction,
History,
}
pub struct States<'a> {
pub nav_state: NavigationState<'a>,
pub transactions: NewTransactionTabState<'a>,
pub history: HistoryState,
pub active_frame: ActiveFrame,
}
impl<'a> States<'a> {
2023-08-11 22:46:43 -07:00
pub fn new() -> States<'a> {
States {
nav_state: NavigationState::new(),
transactions: NewTransactionTabState::new(),
history: HistoryState::new(),
2023-05-31 22:27:26 -07:00
active_frame: ActiveFrame::Navigation,
}
2023-05-31 22:27:26 -07:00
}
}