From cb4f5d8e1c488d8e3e0771ed578e9c3201d07195 Mon Sep 17 00:00:00 2001 From: Lily Hopkins Date: Mon, 6 May 2024 12:37:34 +0100 Subject: [PATCH 1/2] chore: bumped rc version --- Cargo.lock | 16 ++++++++-------- Cargo.toml | 2 +- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index b7845d9..ed46867 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3672,7 +3672,7 @@ dependencies = [ [[package]] name = "testangel" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "arboard", "base64 0.22.1", @@ -3709,7 +3709,7 @@ dependencies = [ [[package]] name = "testangel-engine" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "testangel-engine-macros", "testangel-ipc", @@ -3717,11 +3717,11 @@ dependencies = [ [[package]] name = "testangel-engine-macros" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" [[package]] name = "testangel-evidence" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "lazy_static", "testangel-engine", @@ -3729,7 +3729,7 @@ dependencies = [ [[package]] name = "testangel-ipc" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "schemars", "serde", @@ -3738,7 +3738,7 @@ dependencies = [ [[package]] name = "testangel-rand" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "lazy_static", "rand", @@ -3749,7 +3749,7 @@ dependencies = [ [[package]] name = "testangel-time" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "lazy_static", "testangel-engine", @@ -3758,7 +3758,7 @@ dependencies = [ [[package]] name = "testangel-user-interaction" -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" dependencies = [ "lazy_static", "rfd", diff --git a/Cargo.toml b/Cargo.toml index f20712a..b165313 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,5 +1,5 @@ [workspace.package] -version = "0.21.0-rc.10" +version = "0.21.0-rc.11" edition = "2021" [workspace] From 38cb435453121711e7b2e62bffd8eedc0b5fad15 Mon Sep 17 00:00:00 2001 From: Lily Hopkins Date: Mon, 6 May 2024 12:44:04 +0100 Subject: [PATCH 2/2] fix: set needs saving on text change fixes #202 --- testangel/src/ui/actions/mod.rs | 1413 ++++++++++++++++--------------- 1 file changed, 711 insertions(+), 702 deletions(-) diff --git a/testangel/src/ui/actions/mod.rs b/testangel/src/ui/actions/mod.rs index df24329..bc74be8 100644 --- a/testangel/src/ui/actions/mod.rs +++ b/testangel/src/ui/actions/mod.rs @@ -1,702 +1,711 @@ -use std::{collections::HashMap, fs, path::PathBuf, rc::Rc, sync::Arc}; - -use adw::prelude::*; -use relm4::{ - adw, gtk, Component, ComponentController, ComponentParts, ComponentSender, Controller, - RelmWidgetExt, -}; -use sourceview::StyleSchemeManager; -use testangel::{ - action_loader::ActionMap, - ipc::EngineList, - types::action_v1::ActionV1, - types::{Action, VersionedFile}, -}; - -use super::{file_filters, lang}; -use sourceview5 as sourceview; - -pub mod header; -mod metadata_component; - -pub enum SaveOrOpenActionError { - IoError(std::io::Error), - ParsingError(ron::error::SpannedError), - SerializingError(ron::Error), - ActionNotVersionCompatible, - MissingInstruction(String), -} - -impl ToString for SaveOrOpenActionError { - fn to_string(&self) -> String { - match self { - Self::IoError(e) => lang::lookup_with_args("action-save-open-error-io-error", { - let mut map = HashMap::new(); - map.insert("error", e.to_string().into()); - map - }), - Self::ParsingError(e) => { - lang::lookup_with_args("action-save-open-error-parsing-error", { - let mut map = HashMap::new(); - map.insert("error", e.to_string().into()); - map - }) - } - Self::SerializingError(e) => { - lang::lookup_with_args("action-save-open-error-serializing-error", { - let mut map = HashMap::new(); - map.insert("error", e.to_string().into()); - map - }) - } - Self::ActionNotVersionCompatible => { - lang::lookup("action-save-open-error-action-not-version-compatible") - } - Self::MissingInstruction(e) => { - lang::lookup_with_args("action-save-open-error-missing-instruction", { - let mut map = HashMap::new(); - map.insert("error", e.to_string().into()); - map - }) - } - } - } -} - -#[derive(Clone, Debug)] -pub enum ActionInputs { - /// Do nothing - NoOp, - /// The map of actions has changed and should be updated - ActionsMapChanged(Arc), - /// Create a new action - NewAction, - /// Actually create the new action - _NewAction, - /// Prompt the user to open an action. This will ask to save first if needed. - OpenAction, - /// Actually show the user the open file dialog - _OpenAction, - /// Actually open an action after the user has finished selecting - __OpenAction(PathBuf), - /// Save the action, prompting if needed to set file path - SaveAction, - /// Save the action as a new file, always prompting for a file path - SaveAsAction, - /// Ask where to save if needed, then save - _SaveActionThen(Box), - /// Actually write the action to disk, then emit then input. First bool is whether a new ID should be used. - __SaveActionThen(bool, PathBuf, Box), - /// Close the action, prompting if needing to save first - CloseAction, - /// Actually close the action - _CloseAction, - /// Add the step with the ID provided - AddStep(String), - /// The metadata has been updated and the action should be updated to reflect that - MetadataUpdated(metadata_component::MetadataOutput), -} -#[derive(Clone, Debug)] -pub enum ActionOutputs { - /// Inform other parts that actions may have changed, reload them! - ReloadActions, -} - -#[derive(Debug)] -pub struct ActionsModel { - action_map: Arc, - engine_list: Arc, - - open_action: Option, - open_path: Option, - needs_saving: bool, - header: Rc>, - metadata: Controller, - source_view: sourceview::View, -} - -impl ActionsModel { - /// Get an [`Rc`] clone of the header controller - pub fn header_controller_rc(&self) -> Rc> { - self.header.clone() - } - - /// Create the absolute barebones of a message dialog, allowing for custom button and response mapping. - fn create_message_dialog_skeleton(&self, title: S, message: S) -> adw::MessageDialog - where - S: AsRef, - { - adw::MessageDialog::builder() - .transient_for(&self.header.widget().toplevel_window().expect( - "ActionsModel::create_message_dialog cannot be called until the header is attached", - )) - .title(title.as_ref()) - .heading(title.as_ref()) - .body(message.as_ref()) - .modal(true) - .build() - } - - /// Create a message dialog attached to the toplevel window. This includes default implementations of an 'OK' button. - fn create_message_dialog(&self, title: S, message: S) -> adw::MessageDialog - where - S: AsRef, - { - let dialog = self.create_message_dialog_skeleton(title, message); - dialog.add_response("ok", &lang::lookup("ok")); - dialog.set_default_response(Some("ok")); - dialog.set_close_response("ok"); - dialog - } - - /// Just open a brand new action - fn new_action(&mut self) { - self.open_path = None; - self.needs_saving = true; - let action = Action::default(); - self.source_view.buffer().set_text(&action.script); - self.open_action = Some(action); - self.header - .emit(header::ActionsHeaderInput::ChangeActionOpen( - self.open_action.is_some(), - )); - self.metadata - .emit(metadata_component::MetadataInput::ChangeAction( - Action::default(), - )); - } - - /// Open an action. This does not ask to save first. - fn open_action(&mut self, file: PathBuf) -> Result<(), SaveOrOpenActionError> { - let mut data = fs::read_to_string(&file).map_err(SaveOrOpenActionError::IoError)?; - - let versioned_file: VersionedFile = - ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; - if versioned_file.version() == 1 { - // Upgrade from instruction list to lua script - // This doesn't save anything, just changes what loads to something compatible - let action_v1: ActionV1 = - ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; - let action_upgraded = action_v1.upgrade_action(self.engine_list.clone()); - data = ron::to_string(&action_upgraded) - .map_err(SaveOrOpenActionError::SerializingError)?; - } else if versioned_file.version() != 2 { - return Err(SaveOrOpenActionError::ActionNotVersionCompatible); - } - - let action: Action = ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; - // Validate that all instructions used in the script are available, or return a MissingInstruction err - action - .check_instructions_available(self.engine_list.clone()) - .map_err(|missing| SaveOrOpenActionError::MissingInstruction(missing[0].clone()))?; - self.source_view.buffer().set_text(&action.script); - - self.open_action = Some(action.clone()); - self.header - .emit(header::ActionsHeaderInput::ChangeActionOpen( - self.open_action.is_some(), - )); - self.metadata - .emit(metadata_component::MetadataInput::ChangeAction( - action.clone(), - )); - self.open_path = Some(file); - self.needs_saving = false; - log::debug!("New action open."); - log::debug!("Action: {:?}", self.open_action); - Ok(()) - } - - /// Ask the user if they want to save this file. If they response yes, this will also trigger the save function. - /// This function will only ask the user if needed, otherwise it will emit immediately. - fn prompt_to_save(&self, sender: &relm4::Sender, then: ActionInputs) { - if self.needs_saving { - let question = self.create_message_dialog_skeleton( - lang::lookup("action-save-before"), - lang::lookup("action-save-before-message"), - ); - question.add_response("discard", &lang::lookup("discard")); - question.add_response("save", &lang::lookup("save")); - question.set_response_appearance("discard", adw::ResponseAppearance::Destructive); - question.set_default_response(Some("save")); - question.set_close_response("discard"); - let sender_c = sender.clone(); - let then_c = then.clone(); - question.connect_response(Some("save"), move |_, _| { - sender_c.emit(ActionInputs::_SaveActionThen(Box::new(then_c.clone()))); - }); - let sender_c = sender.clone(); - question.connect_response(Some("discard"), move |_, _| { - sender_c.emit(then.clone()); - }); - question.set_visible(true); - } else { - sender.emit(then); - } - } - - /// Ask the user where to save the flow, or just save if that's good enough - fn ask_where_to_save( - &mut self, - sender: &relm4::Sender, - transient_for: &impl IsA, - always_ask_where: bool, - then: ActionInputs, - ) { - if always_ask_where || self.open_path.is_none() { - // Ask where - let dialog = gtk::FileDialog::builder() - .modal(true) - .title(lang::lookup("header-save")) - .initial_folder(>k::gio::File::for_path( - std::env::var("TA_ACTION_DIR").unwrap_or("./actions".to_string()), - )) - .filters(&file_filters::filter_list(vec![ - file_filters::actions(), - file_filters::all(), - ])) - .build(); - - let sender_c = sender.clone(); - dialog.save( - Some(transient_for), - Some(&relm4::gtk::gio::Cancellable::new()), - move |res| { - if let Ok(file) = res { - let path = file.path().unwrap(); - sender_c.emit(ActionInputs::__SaveActionThen( - true, - path, - Box::new(then.clone()), - )); - } - }, - ); - } else { - sender.emit(ActionInputs::__SaveActionThen( - false, - self.open_path.clone().unwrap(), - Box::new(then), - )); - } - } - - /// Just save the action to disk with the current `open_path` as the destination - fn save_action(&mut self) -> Result<(), SaveOrOpenActionError> { - // Get content - let buffer = self.source_view.buffer(); - let script = buffer.text(&buffer.start_iter(), &buffer.end_iter(), false); - - // Update script - let action = self.open_action.as_mut().unwrap(); - action.script = script.to_string(); - - // Loop through all possible instruction luanames in the environment, then save a vector of which are used by this action - action.required_instructions.clear(); - for engine in self.engine_list.inner().clone() { - let engine_lua_name = engine.lua_name.clone(); - for instruction in engine.instructions.clone() { - let instruction_lua_name = instruction.lua_name().clone(); - let built_call = format!("{}.{}", engine_lua_name, instruction_lua_name); - if script.contains(&built_call) { - action.required_instructions.push(instruction.id().clone()); - } - } - } - - let save_path = self.open_path.as_ref().unwrap(); - let data = ron::to_string(self.open_action.as_ref().unwrap()) - .map_err(SaveOrOpenActionError::SerializingError)?; - fs::write(save_path, data).map_err(SaveOrOpenActionError::IoError)?; - self.needs_saving = false; - Ok(()) - } - - /// Close this action without checking first - fn close_action(&mut self) { - self.open_action = None; - self.open_path = None; - self.needs_saving = false; - self.header - .emit(header::ActionsHeaderInput::ChangeActionOpen( - self.open_action.is_some(), - )); - } -} - -#[relm4::component(pub)] -impl Component for ActionsModel { - type Init = (Arc, Arc); - type Input = ActionInputs; - type Output = ActionOutputs; - type CommandOutput = (); - - view! { - #[root] - toast_target = adw::ToastOverlay { - gtk::ScrolledWindow { - set_vexpand: true, - set_hscrollbar_policy: gtk::PolicyType::Never, - - if model.open_action.is_none() { - adw::StatusPage { - set_title: &lang::lookup("nothing-open"), - set_description: Some(&lang::lookup("action-nothing-open-description")), - set_icon_name: Some(relm4_icons::icon_names::LIGHTBULB), - set_vexpand: true, - } - } else { - gtk::Box { - set_orientation: gtk::Orientation::Vertical, - set_margin_all: 10, - set_spacing: 10, - - model.metadata.widget(), - - gtk::Separator { - set_orientation: gtk::Orientation::Horizontal, - }, - - #[local_ref] - source_view -> sourceview::View, - } - } - }, - }, - } - - fn init( - init: Self::Init, - root: Self::Root, - sender: ComponentSender, - ) -> ComponentParts { - let header = Rc::new( - header::ActionsHeader::builder() - .launch((init.1.clone(), init.0.clone())) - .forward(sender.input_sender(), |msg| match msg { - header::ActionsHeaderOutput::NewAction => ActionInputs::NewAction, - header::ActionsHeaderOutput::OpenAction => ActionInputs::OpenAction, - header::ActionsHeaderOutput::SaveAction => ActionInputs::SaveAction, - header::ActionsHeaderOutput::SaveAsAction => ActionInputs::SaveAsAction, - header::ActionsHeaderOutput::CloseAction => ActionInputs::CloseAction, - header::ActionsHeaderOutput::AddStep(step) => ActionInputs::AddStep(step), - }), - ); - - // Setup source view style manager - StyleSchemeManager::default().append_search_path("styles"); - - let model = ActionsModel { - action_map: init.0, - engine_list: init.1, - open_action: None, - open_path: None, - needs_saving: false, - header, - metadata: metadata_component::Metadata::builder() - .launch(()) - .forward(sender.input_sender(), |msg| { - ActionInputs::MetadataUpdated(msg) - }), - source_view: sourceview::View::builder() - .show_line_numbers(true) - .monospace(true) - .indent_on_tab(true) - .indent_width(2) - .insert_spaces_instead_of_tabs(true) - .show_right_margin(true) - .auto_indent(true) - .vexpand(true) - .buffer( - &sourceview::Buffer::builder() - .highlight_syntax(true) - .language( - &sourceview::LanguageManager::builder() - .search_path(vec![ - "language-specs", // Windows and Local - "/usr/share/gtksourceview-5/language-specs/", // Linux - &std::env::var("GTKSV_LANGSPEC").unwrap_or_default() // Other environments - ]) - .build() - .language("lua") - .expect("lua syntax highlighting not found - maybe use GTKSV_LANGSPEC to specify another search path?"), - ) - .build(), - ) - .build(), - }; - - let source_view = &model.source_view; - let widgets = view_output!(); - - ComponentParts { model, widgets } - } - - fn update_with_view( - &mut self, - widgets: &mut Self::Widgets, - message: Self::Input, - sender: ComponentSender, - root: &Self::Root, - ) { - match message { - ActionInputs::NoOp => (), - - ActionInputs::MetadataUpdated(meta) => { - if let Some(action) = self.open_action.as_mut() { - if let Some(new_name) = meta.new_name { - action.friendly_name = new_name; - } - if let Some(new_group) = meta.new_group { - action.group = new_group; - } - if let Some(new_author) = meta.new_author { - action.author = new_author; - } - if let Some(new_description) = meta.new_description { - action.description = new_description; - } - if let Some(new_visible) = meta.new_visible { - action.visible = new_visible; - } - self.needs_saving = true; - } - } - - ActionInputs::ActionsMapChanged(new_map) => { - self.action_map = new_map.clone(); - self.header - .emit(header::ActionsHeaderInput::ActionsMapChanged(new_map)); - } - - ActionInputs::NewAction => { - self.prompt_to_save(sender.input_sender(), ActionInputs::_NewAction); - } - ActionInputs::_NewAction => { - self.new_action(); - } - ActionInputs::OpenAction => { - self.prompt_to_save(sender.input_sender(), ActionInputs::_OpenAction); - } - ActionInputs::_OpenAction => { - let dialog = gtk::FileDialog::builder() - .modal(true) - .title(lang::lookup("header-open")) - .filters(&file_filters::filter_list(vec![ - file_filters::actions(), - file_filters::all(), - ])) - .initial_folder(>k::gio::File::for_path( - std::env::var("TA_ACTION_DIR").unwrap_or("./actions".to_string()), - )) - .build(); - - let sender_c = sender.clone(); - dialog.open( - Some(&root.toplevel_window().unwrap()), - Some(&relm4::gtk::gio::Cancellable::new()), - move |res| { - if let Ok(file) = res { - let path = file.path().unwrap(); - sender_c.input(ActionInputs::__OpenAction(path)); - } - }, - ); - } - ActionInputs::__OpenAction(path) => { - match self.open_action(path) { - Ok(_) => { - // Nothing more to do... - } - Err(e) => { - // Show error dialog - self.create_message_dialog( - lang::lookup("action-error-opening"), - e.to_string(), - ) - .set_visible(true); - } - } - } - ActionInputs::SaveAction => { - if self.open_action.is_some() { - // unwrap rationale: this cannot be triggered if not attached to a window - self.ask_where_to_save( - sender.input_sender(), - &root.toplevel_window().unwrap(), - false, - ActionInputs::NoOp, - ); - } - } - ActionInputs::SaveAsAction => { - if self.open_action.is_some() { - // unwrap rationale: this cannot be triggered if not attached to a window - self.ask_where_to_save( - sender.input_sender(), - &root.toplevel_window().unwrap(), - true, - ActionInputs::NoOp, - ); - } - } - ActionInputs::_SaveActionThen(then) => { - // unwrap rationale: this cannot be triggered if not attached to a window - self.ask_where_to_save( - sender.input_sender(), - &root.toplevel_window().unwrap(), - false, - *then, - ); - } - ActionInputs::__SaveActionThen(new_id, path, then) => { - self.open_path = Some(path.with_extension("taaction")); - if new_id { - if let Some(action) = &mut self.open_action { - action.id = uuid::Uuid::new_v4().to_string(); - } - } - if let Err(e) = self.save_action() { - self.create_message_dialog(lang::lookup("action-error-saving"), e.to_string()) - .set_visible(true); - } else { - widgets - .toast_target - .add_toast(adw::Toast::new(&lang::lookup("action-saved"))); - sender.input_sender().emit(*then); - } - let _ = sender.output(ActionOutputs::ReloadActions); - } - ActionInputs::CloseAction => { - self.prompt_to_save(sender.input_sender(), ActionInputs::_CloseAction); - } - ActionInputs::_CloseAction => { - self.close_action(); - } - - ActionInputs::AddStep(step_id) => { - if self.open_action.is_none() { - self.new_action(); - } - - // unwrap rationale: the header can't ask to add an action that doesn't exist - let engine = self - .engine_list - .get_engine_by_instruction_id(&step_id) - .unwrap(); - let instruction = self.engine_list.get_instruction_by_id(&step_id).unwrap(); - // Build LoC - let mut params = String::new(); - for param_id in instruction.parameter_order() { - use convert_case::{Case, Casing}; - - let (param_name, _param_kind) = instruction.parameters().get(param_id).unwrap(); - // remove invalid chars - let mut sanitised_name = String::new(); - for c in param_name.chars() { - if c.is_ascii_alphanumeric() || c.is_ascii_whitespace() { - sanitised_name.push(c); - } - } - params.push_str(&format!("{}, ", sanitised_name.to_case(Case::Snake))); - } - // remove last ", " - let _ = params.pop(); - let _ = params.pop(); - - let loc = if instruction.outputs().is_empty() { - format!("{}.{}({})", engine.lua_name, instruction.lua_name(), params) - } else { - let mut returns = String::new(); - for return_id in instruction.output_order() { - use convert_case::{Case, Casing}; - - let (name, _kind) = instruction.outputs()[return_id].clone(); - returns.push_str(&format!("{}, ", name.to_case(Case::Snake))); - } - - // Remove last ", " - let _ = returns.pop(); - let _ = returns.pop(); - - format!( - "local {} = {}.{}({})", - returns, - engine.lua_name, - instruction.lua_name(), - params - ) - }; - // Append LoC - let buffer = self.source_view.buffer(); - let text = buffer - .text(&buffer.start_iter(), &buffer.end_iter(), false) - .to_string(); - let mut newline_after = true; - - // Decide if cursor needs moving down a line (or into function) - let cursor_pos = buffer.cursor_position(); - log::debug!( - "Inserting step, cursor pos: {} (text len: {})", - cursor_pos, - text.len() - ); - if cursor_pos == 0 || cursor_pos == text.len() as i32 { - // Move cursor into function - log::debug!("Offsetting cursor into function"); - for (i, l) in text.lines().enumerate() { - if l.contains("function run_action") { - log::debug!("Function on line {}", i); - if let Some(text_iter) = buffer.iter_at_line_offset(i as i32 + 1, 2) { - buffer.place_cursor(&text_iter); - } - } - } - } else { - // If line is not empty, add new line - let mut line_starts_at = 0; - let mut line_ends_at = text.len(); - let mut line_num = 0; - - for (idx, c) in text.char_indices() { - if c == '\n' { - if idx < cursor_pos as usize { - line_starts_at = idx + 1; - line_num += 1; - } else { - line_ends_at = idx; - break; - } - } - } - - // Move cursor to end and insert newline if needed - let line = &text[line_starts_at..line_ends_at]; - log::debug!("cursor on line: {:?}", line); - if !line.trim().is_empty() { - // Offset cursor to end of line - log::debug!( - "Moving cursor to end of line {} (pos {})", - line_num, - line.len() - ); - if let Some(iter) = buffer.iter_at_line_offset(line_num, line.len() as i32) - { - buffer.place_cursor(&iter); - buffer.insert_at_cursor("\n "); - newline_after = false; - } - } - } - - buffer.insert_interactive_at_cursor( - &format!("{}{}", loc, if newline_after { "\n " } else { "" }), - true, - ); - - self.needs_saving = true; - } - } - self.update_view(widgets, sender); - } -} +use std::{collections::HashMap, fs, path::PathBuf, rc::Rc, sync::Arc}; + +use adw::prelude::*; +use relm4::{ + adw, gtk, Component, ComponentController, ComponentParts, ComponentSender, Controller, + RelmWidgetExt, +}; +use sourceview::StyleSchemeManager; +use testangel::{ + action_loader::ActionMap, + ipc::EngineList, + types::action_v1::ActionV1, + types::{Action, VersionedFile}, +}; + +use super::{file_filters, lang}; +use sourceview5 as sourceview; + +pub mod header; +mod metadata_component; + +pub enum SaveOrOpenActionError { + IoError(std::io::Error), + ParsingError(ron::error::SpannedError), + SerializingError(ron::Error), + ActionNotVersionCompatible, + MissingInstruction(String), +} + +impl ToString for SaveOrOpenActionError { + fn to_string(&self) -> String { + match self { + Self::IoError(e) => lang::lookup_with_args("action-save-open-error-io-error", { + let mut map = HashMap::new(); + map.insert("error", e.to_string().into()); + map + }), + Self::ParsingError(e) => { + lang::lookup_with_args("action-save-open-error-parsing-error", { + let mut map = HashMap::new(); + map.insert("error", e.to_string().into()); + map + }) + } + Self::SerializingError(e) => { + lang::lookup_with_args("action-save-open-error-serializing-error", { + let mut map = HashMap::new(); + map.insert("error", e.to_string().into()); + map + }) + } + Self::ActionNotVersionCompatible => { + lang::lookup("action-save-open-error-action-not-version-compatible") + } + Self::MissingInstruction(e) => { + lang::lookup_with_args("action-save-open-error-missing-instruction", { + let mut map = HashMap::new(); + map.insert("error", e.to_string().into()); + map + }) + } + } + } +} + +#[derive(Clone, Debug)] +pub enum ActionInputs { + /// Do nothing + NoOp, + /// The map of actions has changed and should be updated + ActionsMapChanged(Arc), + /// Create a new action + NewAction, + /// Actually create the new action + _NewAction, + /// Prompt the user to open an action. This will ask to save first if needed. + OpenAction, + /// Actually show the user the open file dialog + _OpenAction, + /// Actually open an action after the user has finished selecting + __OpenAction(PathBuf), + /// Save the action, prompting if needed to set file path + SaveAction, + /// Save the action as a new file, always prompting for a file path + SaveAsAction, + /// Ask where to save if needed, then save + _SaveActionThen(Box), + /// Actually write the action to disk, then emit then input. First bool is whether a new ID should be used. + __SaveActionThen(bool, PathBuf, Box), + /// Close the action, prompting if needing to save first + CloseAction, + /// Actually close the action + _CloseAction, + /// Add the step with the ID provided + AddStep(String), + /// The metadata has been updated and the action should be updated to reflect that + MetadataUpdated(metadata_component::MetadataOutput), +} +#[derive(Clone, Debug)] +pub enum ActionOutputs { + /// Inform other parts that actions may have changed, reload them! + ReloadActions, +} + +#[derive(Debug)] +pub struct ActionsModel { + action_map: Arc, + engine_list: Arc, + + open_action: Option, + open_path: Option, + needs_saving: bool, + header: Rc>, + metadata: Controller, + source_view: sourceview::View, +} + +impl ActionsModel { + /// Get an [`Rc`] clone of the header controller + pub fn header_controller_rc(&self) -> Rc> { + self.header.clone() + } + + /// Create the absolute barebones of a message dialog, allowing for custom button and response mapping. + fn create_message_dialog_skeleton(&self, title: S, message: S) -> adw::MessageDialog + where + S: AsRef, + { + adw::MessageDialog::builder() + .transient_for(&self.header.widget().toplevel_window().expect( + "ActionsModel::create_message_dialog cannot be called until the header is attached", + )) + .title(title.as_ref()) + .heading(title.as_ref()) + .body(message.as_ref()) + .modal(true) + .build() + } + + /// Create a message dialog attached to the toplevel window. This includes default implementations of an 'OK' button. + fn create_message_dialog(&self, title: S, message: S) -> adw::MessageDialog + where + S: AsRef, + { + let dialog = self.create_message_dialog_skeleton(title, message); + dialog.add_response("ok", &lang::lookup("ok")); + dialog.set_default_response(Some("ok")); + dialog.set_close_response("ok"); + dialog + } + + /// Just open a brand new action + fn new_action(&mut self) { + self.open_path = None; + self.needs_saving = true; + let action = Action::default(); + self.source_view.buffer().set_text(&action.script); + self.open_action = Some(action); + self.header + .emit(header::ActionsHeaderInput::ChangeActionOpen( + self.open_action.is_some(), + )); + self.metadata + .emit(metadata_component::MetadataInput::ChangeAction( + Action::default(), + )); + } + + /// Open an action. This does not ask to save first. + fn open_action(&mut self, file: PathBuf) -> Result<(), SaveOrOpenActionError> { + let mut data = fs::read_to_string(&file).map_err(SaveOrOpenActionError::IoError)?; + + let versioned_file: VersionedFile = + ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; + if versioned_file.version() == 1 { + // Upgrade from instruction list to lua script + // This doesn't save anything, just changes what loads to something compatible + let action_v1: ActionV1 = + ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; + let action_upgraded = action_v1.upgrade_action(self.engine_list.clone()); + data = ron::to_string(&action_upgraded) + .map_err(SaveOrOpenActionError::SerializingError)?; + } else if versioned_file.version() != 2 { + return Err(SaveOrOpenActionError::ActionNotVersionCompatible); + } + + let action: Action = ron::from_str(&data).map_err(SaveOrOpenActionError::ParsingError)?; + // Validate that all instructions used in the script are available, or return a MissingInstruction err + action + .check_instructions_available(self.engine_list.clone()) + .map_err(|missing| SaveOrOpenActionError::MissingInstruction(missing[0].clone()))?; + self.source_view.buffer().set_text(&action.script); + + self.open_action = Some(action.clone()); + self.header + .emit(header::ActionsHeaderInput::ChangeActionOpen( + self.open_action.is_some(), + )); + self.metadata + .emit(metadata_component::MetadataInput::ChangeAction( + action.clone(), + )); + self.open_path = Some(file); + self.needs_saving = false; + log::debug!("New action open."); + log::debug!("Action: {:?}", self.open_action); + Ok(()) + } + + /// Ask the user if they want to save this file. If they response yes, this will also trigger the save function. + /// This function will only ask the user if needed, otherwise it will emit immediately. + fn prompt_to_save(&self, sender: &relm4::Sender, then: ActionInputs) { + if self.needs_saving { + let question = self.create_message_dialog_skeleton( + lang::lookup("action-save-before"), + lang::lookup("action-save-before-message"), + ); + question.add_response("discard", &lang::lookup("discard")); + question.add_response("save", &lang::lookup("save")); + question.set_response_appearance("discard", adw::ResponseAppearance::Destructive); + question.set_default_response(Some("save")); + question.set_close_response("discard"); + let sender_c = sender.clone(); + let then_c = then.clone(); + question.connect_response(Some("save"), move |_, _| { + sender_c.emit(ActionInputs::_SaveActionThen(Box::new(then_c.clone()))); + }); + let sender_c = sender.clone(); + question.connect_response(Some("discard"), move |_, _| { + sender_c.emit(then.clone()); + }); + question.set_visible(true); + } else { + sender.emit(then); + } + } + + /// Ask the user where to save the flow, or just save if that's good enough + fn ask_where_to_save( + &mut self, + sender: &relm4::Sender, + transient_for: &impl IsA, + always_ask_where: bool, + then: ActionInputs, + ) { + if always_ask_where || self.open_path.is_none() { + // Ask where + let dialog = gtk::FileDialog::builder() + .modal(true) + .title(lang::lookup("header-save")) + .initial_folder(>k::gio::File::for_path( + std::env::var("TA_ACTION_DIR").unwrap_or("./actions".to_string()), + )) + .filters(&file_filters::filter_list(vec![ + file_filters::actions(), + file_filters::all(), + ])) + .build(); + + let sender_c = sender.clone(); + dialog.save( + Some(transient_for), + Some(&relm4::gtk::gio::Cancellable::new()), + move |res| { + if let Ok(file) = res { + let path = file.path().unwrap(); + sender_c.emit(ActionInputs::__SaveActionThen( + true, + path, + Box::new(then.clone()), + )); + } + }, + ); + } else { + sender.emit(ActionInputs::__SaveActionThen( + false, + self.open_path.clone().unwrap(), + Box::new(then), + )); + } + } + + /// Just save the action to disk with the current `open_path` as the destination + fn save_action(&mut self) -> Result<(), SaveOrOpenActionError> { + // Get content + let buffer = self.source_view.buffer(); + let script = buffer.text(&buffer.start_iter(), &buffer.end_iter(), false); + + // Update script + let action = self.open_action.as_mut().unwrap(); + action.script = script.to_string(); + + // Loop through all possible instruction luanames in the environment, then save a vector of which are used by this action + action.required_instructions.clear(); + for engine in self.engine_list.inner().clone() { + let engine_lua_name = engine.lua_name.clone(); + for instruction in engine.instructions.clone() { + let instruction_lua_name = instruction.lua_name().clone(); + let built_call = format!("{}.{}", engine_lua_name, instruction_lua_name); + if script.contains(&built_call) { + action.required_instructions.push(instruction.id().clone()); + } + } + } + + let save_path = self.open_path.as_ref().unwrap(); + let data = ron::to_string(self.open_action.as_ref().unwrap()) + .map_err(SaveOrOpenActionError::SerializingError)?; + fs::write(save_path, data).map_err(SaveOrOpenActionError::IoError)?; + self.needs_saving = false; + Ok(()) + } + + /// Close this action without checking first + fn close_action(&mut self) { + self.open_action = None; + self.open_path = None; + self.needs_saving = false; + self.header + .emit(header::ActionsHeaderInput::ChangeActionOpen( + self.open_action.is_some(), + )); + } +} + +#[relm4::component(pub)] +impl Component for ActionsModel { + type Init = (Arc, Arc); + type Input = ActionInputs; + type Output = ActionOutputs; + type CommandOutput = (); + + view! { + #[root] + toast_target = adw::ToastOverlay { + gtk::ScrolledWindow { + set_vexpand: true, + set_hscrollbar_policy: gtk::PolicyType::Never, + + if model.open_action.is_none() { + adw::StatusPage { + set_title: &lang::lookup("nothing-open"), + set_description: Some(&lang::lookup("action-nothing-open-description")), + set_icon_name: Some(relm4_icons::icon_names::LIGHTBULB), + set_vexpand: true, + } + } else { + gtk::Box { + set_orientation: gtk::Orientation::Vertical, + set_margin_all: 10, + set_spacing: 10, + + model.metadata.widget(), + + gtk::Separator { + set_orientation: gtk::Orientation::Horizontal, + }, + + #[local_ref] + source_view -> sourceview::View, + } + } + }, + }, + } + + fn init( + init: Self::Init, + root: Self::Root, + sender: ComponentSender, + ) -> ComponentParts { + let header = Rc::new( + header::ActionsHeader::builder() + .launch((init.1.clone(), init.0.clone())) + .forward(sender.input_sender(), |msg| match msg { + header::ActionsHeaderOutput::NewAction => ActionInputs::NewAction, + header::ActionsHeaderOutput::OpenAction => ActionInputs::OpenAction, + header::ActionsHeaderOutput::SaveAction => ActionInputs::SaveAction, + header::ActionsHeaderOutput::SaveAsAction => ActionInputs::SaveAsAction, + header::ActionsHeaderOutput::CloseAction => ActionInputs::CloseAction, + header::ActionsHeaderOutput::AddStep(step) => ActionInputs::AddStep(step), + }), + ); + + // Setup source view style manager + StyleSchemeManager::default().append_search_path("styles"); + + let model = ActionsModel { + action_map: init.0, + engine_list: init.1, + open_action: None, + open_path: None, + needs_saving: false, + header, + metadata: metadata_component::Metadata::builder() + .launch(()) + .forward(sender.input_sender(), |msg| { + ActionInputs::MetadataUpdated(msg) + }), + source_view: sourceview::View::builder() + .show_line_numbers(true) + .monospace(true) + .indent_on_tab(true) + .indent_width(2) + .insert_spaces_instead_of_tabs(true) + .show_right_margin(true) + .auto_indent(true) + .vexpand(true) + .buffer( + &sourceview::Buffer::builder() + .highlight_syntax(true) + .language( + &sourceview::LanguageManager::builder() + .search_path(vec![ + "language-specs", // Windows and Local + "/usr/share/gtksourceview-5/language-specs/", // Linux + &std::env::var("GTKSV_LANGSPEC").unwrap_or_default() // Other environments + ]) + .build() + .language("lua") + .expect("lua syntax highlighting not found - maybe use GTKSV_LANGSPEC to specify another search path?"), + ) + .build(), + ) + .build(), + }; + + let source_view = &model.source_view; + let widgets = view_output!(); + + ComponentParts { model, widgets } + } + + fn update_with_view( + &mut self, + widgets: &mut Self::Widgets, + message: Self::Input, + sender: ComponentSender, + root: &Self::Root, + ) { + match message { + ActionInputs::NoOp => (), + + ActionInputs::MetadataUpdated(meta) => { + if let Some(action) = self.open_action.as_mut() { + if let Some(new_name) = meta.new_name { + action.friendly_name = new_name; + } + if let Some(new_group) = meta.new_group { + action.group = new_group; + } + if let Some(new_author) = meta.new_author { + action.author = new_author; + } + if let Some(new_description) = meta.new_description { + action.description = new_description; + } + if let Some(new_visible) = meta.new_visible { + action.visible = new_visible; + } + self.needs_saving = true; + } + } + + ActionInputs::ActionsMapChanged(new_map) => { + self.action_map = new_map.clone(); + self.header + .emit(header::ActionsHeaderInput::ActionsMapChanged(new_map)); + } + + ActionInputs::NewAction => { + self.prompt_to_save(sender.input_sender(), ActionInputs::_NewAction); + } + ActionInputs::_NewAction => { + self.new_action(); + } + ActionInputs::OpenAction => { + self.prompt_to_save(sender.input_sender(), ActionInputs::_OpenAction); + } + ActionInputs::_OpenAction => { + let dialog = gtk::FileDialog::builder() + .modal(true) + .title(lang::lookup("header-open")) + .filters(&file_filters::filter_list(vec![ + file_filters::actions(), + file_filters::all(), + ])) + .initial_folder(>k::gio::File::for_path( + std::env::var("TA_ACTION_DIR").unwrap_or("./actions".to_string()), + )) + .build(); + + let sender_c = sender.clone(); + dialog.open( + Some(&root.toplevel_window().unwrap()), + Some(&relm4::gtk::gio::Cancellable::new()), + move |res| { + if let Ok(file) = res { + let path = file.path().unwrap(); + sender_c.input(ActionInputs::__OpenAction(path)); + } + }, + ); + } + ActionInputs::__OpenAction(path) => { + match self.open_action(path) { + Ok(_) => { + // Nothing more to do... + } + Err(e) => { + // Show error dialog + self.create_message_dialog( + lang::lookup("action-error-opening"), + e.to_string(), + ) + .set_visible(true); + } + } + } + ActionInputs::SaveAction => { + if self.open_action.is_some() { + // unwrap rationale: this cannot be triggered if not attached to a window + self.ask_where_to_save( + sender.input_sender(), + &root.toplevel_window().unwrap(), + false, + ActionInputs::NoOp, + ); + } + } + ActionInputs::SaveAsAction => { + if self.open_action.is_some() { + // unwrap rationale: this cannot be triggered if not attached to a window + self.ask_where_to_save( + sender.input_sender(), + &root.toplevel_window().unwrap(), + true, + ActionInputs::NoOp, + ); + } + } + ActionInputs::_SaveActionThen(then) => { + // unwrap rationale: this cannot be triggered if not attached to a window + self.ask_where_to_save( + sender.input_sender(), + &root.toplevel_window().unwrap(), + false, + *then, + ); + } + ActionInputs::__SaveActionThen(new_id, path, then) => { + self.open_path = Some(path.with_extension("taaction")); + if new_id { + if let Some(action) = &mut self.open_action { + action.id = uuid::Uuid::new_v4().to_string(); + } + } + if let Err(e) = self.save_action() { + self.create_message_dialog(lang::lookup("action-error-saving"), e.to_string()) + .set_visible(true); + } else { + widgets + .toast_target + .add_toast(adw::Toast::new(&lang::lookup("action-saved"))); + sender.input_sender().emit(*then); + } + let _ = sender.output(ActionOutputs::ReloadActions); + } + ActionInputs::CloseAction => { + // Establish if needs_saving needs updating from text change + if let Some(action) = &self.open_action { + let buf = self.source_view.buffer(); + if action.script != buf.text(&buf.start_iter(), &buf.end_iter(), false) { + log::debug!("Needs saving due to text change."); + self.needs_saving = true; + } + } + + self.prompt_to_save(sender.input_sender(), ActionInputs::_CloseAction); + } + ActionInputs::_CloseAction => { + self.close_action(); + } + + ActionInputs::AddStep(step_id) => { + if self.open_action.is_none() { + self.new_action(); + } + + // unwrap rationale: the header can't ask to add an action that doesn't exist + let engine = self + .engine_list + .get_engine_by_instruction_id(&step_id) + .unwrap(); + let instruction = self.engine_list.get_instruction_by_id(&step_id).unwrap(); + // Build LoC + let mut params = String::new(); + for param_id in instruction.parameter_order() { + use convert_case::{Case, Casing}; + + let (param_name, _param_kind) = instruction.parameters().get(param_id).unwrap(); + // remove invalid chars + let mut sanitised_name = String::new(); + for c in param_name.chars() { + if c.is_ascii_alphanumeric() || c.is_ascii_whitespace() { + sanitised_name.push(c); + } + } + params.push_str(&format!("{}, ", sanitised_name.to_case(Case::Snake))); + } + // remove last ", " + let _ = params.pop(); + let _ = params.pop(); + + let loc = if instruction.outputs().is_empty() { + format!("{}.{}({})", engine.lua_name, instruction.lua_name(), params) + } else { + let mut returns = String::new(); + for return_id in instruction.output_order() { + use convert_case::{Case, Casing}; + + let (name, _kind) = instruction.outputs()[return_id].clone(); + returns.push_str(&format!("{}, ", name.to_case(Case::Snake))); + } + + // Remove last ", " + let _ = returns.pop(); + let _ = returns.pop(); + + format!( + "local {} = {}.{}({})", + returns, + engine.lua_name, + instruction.lua_name(), + params + ) + }; + // Append LoC + let buffer = self.source_view.buffer(); + let text = buffer + .text(&buffer.start_iter(), &buffer.end_iter(), false) + .to_string(); + let mut newline_after = true; + + // Decide if cursor needs moving down a line (or into function) + let cursor_pos = buffer.cursor_position(); + log::debug!( + "Inserting step, cursor pos: {} (text len: {})", + cursor_pos, + text.len() + ); + if cursor_pos == 0 || cursor_pos == text.len() as i32 { + // Move cursor into function + log::debug!("Offsetting cursor into function"); + for (i, l) in text.lines().enumerate() { + if l.contains("function run_action") { + log::debug!("Function on line {}", i); + if let Some(text_iter) = buffer.iter_at_line_offset(i as i32 + 1, 2) { + buffer.place_cursor(&text_iter); + } + } + } + } else { + // If line is not empty, add new line + let mut line_starts_at = 0; + let mut line_ends_at = text.len(); + let mut line_num = 0; + + for (idx, c) in text.char_indices() { + if c == '\n' { + if idx < cursor_pos as usize { + line_starts_at = idx + 1; + line_num += 1; + } else { + line_ends_at = idx; + break; + } + } + } + + // Move cursor to end and insert newline if needed + let line = &text[line_starts_at..line_ends_at]; + log::debug!("cursor on line: {:?}", line); + if !line.trim().is_empty() { + // Offset cursor to end of line + log::debug!( + "Moving cursor to end of line {} (pos {})", + line_num, + line.len() + ); + if let Some(iter) = buffer.iter_at_line_offset(line_num, line.len() as i32) + { + buffer.place_cursor(&iter); + buffer.insert_at_cursor("\n "); + newline_after = false; + } + } + } + + buffer.insert_interactive_at_cursor( + &format!("{}{}", loc, if newline_after { "\n " } else { "" }), + true, + ); + + self.needs_saving = true; + } + } + self.update_view(widgets, sender); + } +}