mod calcifer; mod tools; use calcifer::code_editor::ColorTheme; use eframe::egui; use egui::FontFamily::Proportional; use egui::FontId; use egui::TextStyle::{Body, Button, Heading, Monospace, Small}; use std::{ops::Range, path::Path, sync::Arc, thread, time}; use calcifer::code_editor::themes::DEFAULT_THEMES; #[cfg(debug_assertions)] mod build { pub const SAVE_PATH: &str = "/home/penwing/Documents/.save/debug/calcifer_save.json"; pub const TITLE: &str = " debug"; } #[cfg(not(debug_assertions))] mod build { pub const SAVE_PATH: &str = "/home/penwing/Documents/.save/calcifer_save.json"; pub const TITLE: &str = ""; } use build::SAVE_PATH; use build::TITLE; const TERMINAL_HEIGHT: f32 = 200.0; const TERMINAL_RANGE: Range = 100.0..500.0; const RED: egui::Color32 = egui::Color32::from_rgb(235, 108, 99); const TIME_LABELS: [&str; 7] = [ "input", "settings", "tree", "terminal", "tabs", "content", "windows", ]; const MAX_FPS: f32 = 30.0; const PATH_ROOT: &str = "/home/penwing/Documents/"; const DISPLAY_PATH_DEPTH: usize = 3; const MAX_TABS: usize = 20; fn main() -> Result<(), eframe::Error> { let icon_data = tools::load_icon().unwrap_or_default(); let options = eframe::NativeOptions { viewport: egui::ViewportBuilder::default() .with_inner_size([1200.0, 800.0]) .with_icon(Arc::new(icon_data)), ..Default::default() }; // Attempt to load previous state let app_state: tools::AppState = if Path::new(SAVE_PATH).exists() { match tools::load_state(SAVE_PATH) { Ok(app_state) => app_state, Err(_) => tools::AppState::default(), } } else { tools::AppState::default() }; eframe::run_native( &format!("Calcifer{}", TITLE), options, Box::new(move |_cc| Box::from(Calcifer::from_app_state(app_state))), ) } struct Calcifer { selected_tab: tools::TabNumber, tabs: Vec, command: String, command_history: Vec, theme: ColorTheme, font_size: f32, tree_visible: bool, profiler_visible: bool, terminal_visible: bool, close_tab_confirm: tools::confirm::ConfirmWindow, tab_to_close: usize, refresh_confirm: tools::confirm::ConfirmWindow, exit_confirm: tools::confirm::ConfirmWindow, search_menu: tools::search::SearchWindow, settings_menu: tools::settings::SettingsWindow, shortcuts_menu: tools::shortcuts::ShortcutsWindow, time_watch: Vec, next_frame: time::Instant, } impl Default for Calcifer { fn default() -> Self { Self { selected_tab: tools::TabNumber::from_index(0), tabs: vec![tools::Tab::default()], command: String::new(), command_history: Vec::new(), theme: DEFAULT_THEMES[0], font_size: 14.0, tree_visible: false, profiler_visible: false, terminal_visible: false, close_tab_confirm: tools::confirm::ConfirmWindow::new( "You have some unsaved changes, Do you still want to close this document ?", "Confirm Close", ), tab_to_close: 0, refresh_confirm: tools::confirm::ConfirmWindow::new( "You have some unsaved changes, Do you still want to refresh this document ?", "Confirm Refresh", ), exit_confirm: tools::confirm::ConfirmWindow::new("", "Confirm Exit"), search_menu: tools::search::SearchWindow::default(), settings_menu: tools::settings::SettingsWindow::new(DEFAULT_THEMES[0]), shortcuts_menu: tools::shortcuts::ShortcutsWindow::new(), time_watch: vec![0.0; TIME_LABELS.len()], next_frame: time::Instant::now(), } } } impl eframe::App for Calcifer { fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) { thread::sleep(time::Duration::from_secs_f32( ((1.0 / MAX_FPS) - self.next_frame.elapsed().as_secs_f32()).max(0.0), )); self.next_frame = time::Instant::now(); let mut watch = time::Instant::now(); let mut style = (*ctx.style()).clone(); style.text_styles = [ (Heading, FontId::new(self.font_size * 1.6, Proportional)), (Body, FontId::new(self.font_size, Proportional)), (Monospace, FontId::new(self.font_size, Proportional)), (Button, FontId::new(self.font_size, Proportional)), (Small, FontId::new(self.font_size, Proportional)), ] .into(); ctx.set_style(style); if ctx.input(|i| i.key_pressed(egui::Key::R) && i.modifiers.ctrl) && !self.refresh_confirm.visible { if self.tabs[self.selected_tab.to_index()].saved { self.tabs[self.selected_tab.to_index()].refresh(); } else { self.refresh_confirm.ask(); } } if ctx.input(|i| i.key_pressed(egui::Key::S) && i.modifiers.ctrl) { self.handle_save_file(self.save_tab()); } if ctx.input(|i| i.key_pressed(egui::Key::S) && i.modifiers.ctrl && i.modifiers.shift) { self.handle_save_file(self.save_tab_as()); } if ctx.input(|i| i.key_pressed(egui::Key::ArrowLeft) && i.modifiers.alt) { self.move_through_tabs(false); } if ctx.input(|i| i.key_pressed(egui::Key::ArrowRight) && i.modifiers.alt) { self.move_through_tabs(true); } if ctx.input(|i| i.zoom_delta() > 1.0) { self.font_size = (self.font_size * 1.1).min(30.0); } if ctx.input(|i| i.zoom_delta() < 1.0) { self.font_size = (self.font_size / 1.1).max(10.0); } if ctx.input(|i| i.key_pressed(egui::Key::F) && i.modifiers.ctrl) { self.search_menu.visible = !self.search_menu.visible; self.search_menu.initialized = !self.search_menu.visible; } if ctx.input(|i| i.viewport().close_requested()) { let mut unsaved_tabs: Vec = vec![]; for (index, tab) in self.tabs.iter().enumerate() { if !tab.saved { unsaved_tabs.push(index); } } if !unsaved_tabs.is_empty() { let mut unsaved_tabs_names: String = "".to_string(); for index in unsaved_tabs.iter() { unsaved_tabs_names.push_str(&self.tabs[*index].get_name()); } egui::Context::send_viewport_cmd(ctx, egui::ViewportCommand::CancelClose); self.exit_confirm.prompt = format!( "You have some unsaved changes :\n{}\nDo you still want to exit ?", unsaved_tabs_names ); self.exit_confirm.ask(); } } self.time_watch[0] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_settings(ctx); self.time_watch[1] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_tree_panel(ctx); self.time_watch[2] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_bottom_tray(ctx); self.draw_terminal_panel(ctx); self.time_watch[3] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_tab_panel(ctx); self.time_watch[4] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_content_panel(ctx); self.time_watch[5] = watch.elapsed().as_micros() as f32 / 1000.0; watch = time::Instant::now(); self.draw_windows(ctx); self.time_watch[6] = watch.elapsed().as_micros() as f32 / 1000.0; } fn on_exit(&mut self, _gl: std::option::Option<&eframe::glow::Context>) { self.save_state(); } }