// bibiman - a TUI for managing BibLaTeX databases // Copyright (C) 2024 lukeflo // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . ///// use crate::bibiman::{CurrentArea, FormerArea}; use color_eyre::eyre::{Context, Ok, Result}; // use super::Event; use crate::cliargs::CLIArgs; use crate::tui::commands::InputCmdAction; use crate::tui::popup::PopupKind; use crate::tui::{self, Tui}; use crate::{bibiman::Bibiman, tui::commands::CmdAction}; use std::process::{Command, Stdio}; use tui::Event; use tui_input::backend::crossterm::EventHandler; use tui_input::Input; // Application. #[derive(Debug)] pub struct App { // Is the application running? pub running: bool, // bibimain pub bibiman: Bibiman, // Input mode pub input: Input, // Input mode bool pub input_mode: bool, } impl App { // Constructs a new instance of [`App`]. pub fn new(args: CLIArgs) -> Result { // Self::default() let running = true; let input = Input::default(); let bibiman = Bibiman::new(args)?; Ok(Self { running, bibiman, input, input_mode: false, }) } pub async fn run(&mut self) -> Result<()> { let mut tui = tui::Tui::new()?; tui.enter()?; // Start the main loop. while self.running { // Render the user interface. tui.draw(self)?; // Handle events. match tui.next().await? { Event::Tick => self.tick(), // Event::Key(key_event) => handle_key_events(key_event, self, &mut tui)?, // Event::Mouse(_) => {} Event::Key(key_event) => { // Automatically close message popups on next keypress if let Some(PopupKind::MessageConfirm) = self.bibiman.popup_area.popup_kind { self.bibiman.close_popup() } else if let Some(PopupKind::MessageError) = self.bibiman.popup_area.popup_kind { self.bibiman.close_popup() } let command = if self.input_mode { CmdAction::Input(InputCmdAction::parse(key_event, &self.input)) } else { CmdAction::from(key_event) }; self.run_command(command, &mut tui)? } Event::Mouse(mouse_event) => { self.run_command(CmdAction::from(mouse_event), &mut tui)? } Event::Resize(_, _) => {} } } // Exit the user interface. tui.exit()?; Ok(()) } // Handles the tick event of the terminal. pub fn tick(&self) {} // General commands // Set running to false to quit the application. pub fn quit(&mut self) { self.running = false; } pub fn run_command(&mut self, cmd: CmdAction, tui: &mut Tui) -> Result<()> { match cmd { CmdAction::Input(cmd) => match cmd { InputCmdAction::Nothing => {} InputCmdAction::Handle(event) => { self.input.handle_event(&event); self.bibiman.search_list_by_pattern(&self.input); } InputCmdAction::Enter => { self.input_mode = true; // Logic for TABS to be added self.bibiman.enter_search_area(); } InputCmdAction::Confirm => { self.input = Input::default(); self.input_mode = false; // Logic for TABS to be added self.bibiman.confirm_search(); } InputCmdAction::Exit => { self.input = Input::default(); self.input_mode = false; self.bibiman.break_search(); } }, CmdAction::SelectNextRow(amount) => match self.bibiman.current_area { // Here add logic to select TAB CurrentArea::EntryArea => { self.bibiman.select_next_entry(amount); } CurrentArea::TagArea => { self.bibiman.select_next_tag(amount); } CurrentArea::PopupArea => { if let Some(PopupKind::Help) = self.bibiman.popup_area.popup_kind { self.bibiman.popup_area.popup_scroll_down(); } else if let Some(PopupKind::Selection) = self.bibiman.popup_area.popup_kind { self.bibiman.popup_area.popup_state.scroll_down_by(1) } } _ => {} }, CmdAction::SelectPrevRow(amount) => match self.bibiman.current_area { // Here add logic to select TAB CurrentArea::EntryArea => { self.bibiman.select_previous_entry(amount); } CurrentArea::TagArea => { self.bibiman.select_previous_tag(amount); } CurrentArea::PopupArea => { if let Some(PopupKind::Help) = self.bibiman.popup_area.popup_kind { self.bibiman.popup_area.popup_scroll_up(); } else if let Some(PopupKind::Selection) = self.bibiman.popup_area.popup_kind { self.bibiman.popup_area.popup_state.scroll_up_by(1) } } _ => {} }, CmdAction::SelectNextCol => { if let CurrentArea::EntryArea = self.bibiman.current_area { self.bibiman.select_next_column(); } } CmdAction::SelectPrevCol => { if let CurrentArea::EntryArea = self.bibiman.current_area { self.bibiman.select_prev_column(); } } CmdAction::ScrollInfoDown => { self.bibiman.scroll_info_down(); } CmdAction::ScrollInfoUp => { self.bibiman.scroll_info_up(); } CmdAction::Bottom => match self.bibiman.current_area { CurrentArea::EntryArea => { self.bibiman.select_last_entry(); } CurrentArea::TagArea => { self.bibiman.select_last_tag(); } _ => {} }, CmdAction::Top => match self.bibiman.current_area { CurrentArea::EntryArea => { self.bibiman.select_first_entry(); } CurrentArea::TagArea => { self.bibiman.select_first_tag(); } _ => {} }, CmdAction::ToggleArea => { self.bibiman.toggle_area(); } CmdAction::SearchList => {} CmdAction::Reset => { if let CurrentArea::PopupArea = self.bibiman.current_area { if let Some(PopupKind::Help) = self.bibiman.popup_area.popup_kind { self.bibiman.popup_area.popup_scroll_pos = 0; self.bibiman.close_popup() } else if let Some(PopupKind::Selection) = self.bibiman.popup_area.popup_kind { self.bibiman.close_popup() } } else { self.bibiman.reset_current_list(); } } CmdAction::Confirm => { if let CurrentArea::TagArea = self.bibiman.current_area { self.bibiman.filter_for_tags(); } else if let CurrentArea::PopupArea = self.bibiman.current_area { if let Some(PopupKind::Help) = self.bibiman.popup_area.popup_kind { self.bibiman.close_popup(); } else if let Some(PopupKind::Selection) = self.bibiman.popup_area.popup_kind { // Index of selected entry let entry_idx = self .bibiman .entry_table .entry_table_state .selected() .unwrap(); // Index of selected popup field let popup_idx = self.bibiman.popup_area.popup_state.selected().unwrap(); // Choose ressource depending an selected popup field if self.bibiman.popup_area.popup_list[popup_idx].contains("Weblink") { let object = prepare_weblink( self.bibiman.entry_table.entry_table_items[entry_idx].doi_url(), ); open_connected_res(&object)?; } else if self.bibiman.popup_area.popup_list[popup_idx].contains("File") { let object = self.bibiman.entry_table.entry_table_items[entry_idx].filepath(); open_connected_res(object)?; } else { eprintln!("Unable to find ressource to open"); }; // run command to open file/Url self.bibiman.close_popup() } } } CmdAction::SortList => { if let CurrentArea::EntryArea = self.bibiman.current_area { self.bibiman.entry_table.sort_entry_table(true); } } CmdAction::YankItem => { if let CurrentArea::EntryArea = self.bibiman.current_area { let citekey: &str = &self.bibiman.entry_table.entry_table_items[self .bibiman .entry_table .entry_table_state .selected() .unwrap()] .citekey; Bibiman::yank_text(citekey); self.bibiman.popup_area.popup_message( "Yanked citekey to clipboard: ", citekey, // self.bibiman.get_selected_citekey(), true, ); } } CmdAction::EditFile => { if let CurrentArea::EntryArea = self.bibiman.current_area { self.bibiman.run_editor(tui)?; } } CmdAction::Open => { if let CurrentArea::EntryArea = self.bibiman.current_area { let idx = self .bibiman .entry_table .entry_table_state .selected() .unwrap(); let entry = self.bibiman.entry_table.entry_table_items[idx].clone(); if entry.filepath.is_some() || entry.doi_url.is_some() { let mut items = vec![]; if entry.doi_url.is_some() { items.push("Weblink (DOI/URL)".to_owned()) } if entry.filepath.is_some() { items.push("File (PDF/EPUB)".to_owned()) } self.bibiman.popup_area.popup_selection(items); self.bibiman.former_area = Some(FormerArea::EntryArea); self.bibiman.current_area = CurrentArea::PopupArea; self.bibiman.popup_area.popup_state.select(Some(0)) } else { self.bibiman.popup_area.popup_message( "Selected entry has no connected ressources: ", &entry.citekey, false, ) } } } CmdAction::ShowHelp => { self.bibiman.show_help(); } CmdAction::Exit => { self.quit(); } CmdAction::Nothing => {} } Ok(()) } } pub fn open_connected_res(object: &str) -> Result<()> { // Build command to execute pdf-reader. 'xdg-open' is Linux standard let cmd = { match std::env::consts::OS { "linux" => String::from("xdg-open"), "macos" => String::from("open"), "windows" => String::from("start"), _ => panic!("Couldn't detect OS for setting correct opener"), } }; // Pass filepath as argument, pipe stdout and stderr to /dev/null // to keep the TUI clean (where is it piped on Windows???) let _ = Command::new(&cmd) .arg(object) .stdout(Stdio::null()) .stderr(Stdio::null()) .spawn() .wrap_err("Opening file not possible"); Ok(()) } pub fn prepare_weblink(url: &str) -> String { if url.starts_with("10.") { let prefix = "https://doi.org/".to_string(); prefix + url } else if url.starts_with("www.") { let prefix = "https://".to_string(); prefix + url } else { url.to_string() } }