2019-09-15 16:29:11 +00:00
|
|
|
/*
|
|
|
|
* This file is part of espanso.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 Federico Terzi
|
|
|
|
*
|
|
|
|
* espanso 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.
|
|
|
|
*
|
|
|
|
* espanso 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 espanso. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
use crate::matcher::{Match, MatchReceiver, MatchContentType};
|
2019-09-13 13:03:03 +00:00
|
|
|
use crate::keyboard::KeyboardManager;
|
2019-09-07 14:13:13 +00:00
|
|
|
use crate::config::ConfigManager;
|
2019-09-07 08:43:23 +00:00
|
|
|
use crate::config::BackendType;
|
2019-09-06 22:38:13 +00:00
|
|
|
use crate::clipboard::ClipboardManager;
|
2019-09-15 11:03:21 +00:00
|
|
|
use log::{info, warn, error};
|
2019-09-12 21:24:55 +00:00
|
|
|
use crate::ui::{UIManager, MenuItem, MenuItemType};
|
2019-09-14 10:19:11 +00:00
|
|
|
use crate::event::{ActionEventReceiver, ActionType};
|
2019-09-15 11:03:21 +00:00
|
|
|
use crate::extension::Extension;
|
2019-09-12 21:24:55 +00:00
|
|
|
use std::cell::RefCell;
|
2019-09-13 13:03:03 +00:00
|
|
|
use std::process::exit;
|
2019-09-15 11:03:21 +00:00
|
|
|
use std::collections::HashMap;
|
2019-11-28 21:56:00 +00:00
|
|
|
use std::path::PathBuf;
|
2019-09-15 11:03:21 +00:00
|
|
|
use regex::{Regex, Captures};
|
2019-08-31 14:07:45 +00:00
|
|
|
|
2019-09-13 13:03:03 +00:00
|
|
|
pub struct Engine<'a, S: KeyboardManager, C: ClipboardManager, M: ConfigManager<'a>,
|
2019-09-08 11:37:58 +00:00
|
|
|
U: UIManager> {
|
2019-09-13 13:03:03 +00:00
|
|
|
keyboard_manager: &'a S,
|
2019-09-07 14:13:13 +00:00
|
|
|
clipboard_manager: &'a C,
|
|
|
|
config_manager: &'a M,
|
2019-09-08 11:37:58 +00:00
|
|
|
ui_manager: &'a U,
|
2019-09-15 11:03:21 +00:00
|
|
|
|
|
|
|
extension_map: HashMap<String, Box<dyn Extension>>,
|
|
|
|
|
2019-09-12 21:24:55 +00:00
|
|
|
enabled: RefCell<bool>,
|
2019-08-31 14:07:45 +00:00
|
|
|
}
|
|
|
|
|
2019-09-13 13:03:03 +00:00
|
|
|
impl <'a, S: KeyboardManager, C: ClipboardManager, M: ConfigManager<'a>, U: UIManager>
|
2019-09-08 11:37:58 +00:00
|
|
|
Engine<'a, S, C, M, U> {
|
2019-09-15 11:03:21 +00:00
|
|
|
pub fn new(keyboard_manager: &'a S, clipboard_manager: &'a C,
|
|
|
|
config_manager: &'a M, ui_manager: &'a U,
|
|
|
|
extensions: Vec<Box<dyn Extension>>) -> Engine<'a, S, C, M, U> {
|
|
|
|
// Register all the extensions
|
|
|
|
let mut extension_map = HashMap::new();
|
|
|
|
for extension in extensions.into_iter() {
|
|
|
|
extension_map.insert(extension.name(), extension);
|
|
|
|
}
|
|
|
|
|
2019-09-12 21:24:55 +00:00
|
|
|
let enabled = RefCell::new(true);
|
2019-09-15 11:03:21 +00:00
|
|
|
|
|
|
|
Engine{keyboard_manager,
|
|
|
|
clipboard_manager,
|
|
|
|
config_manager,
|
|
|
|
ui_manager,
|
|
|
|
extension_map,
|
|
|
|
enabled
|
|
|
|
}
|
2019-09-12 21:24:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn build_menu(&self) -> Vec<MenuItem> {
|
|
|
|
let mut menu = Vec::new();
|
|
|
|
|
|
|
|
let enabled = self.enabled.borrow();
|
|
|
|
let toggle_text = if *enabled {
|
|
|
|
"Disable"
|
|
|
|
}else{
|
|
|
|
"Enable"
|
|
|
|
}.to_owned();
|
|
|
|
menu.push(MenuItem{
|
|
|
|
item_type: MenuItemType::Button,
|
|
|
|
item_name: toggle_text,
|
2019-09-12 21:53:17 +00:00
|
|
|
item_id: ActionType::Toggle as i32,
|
2019-09-12 21:24:55 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
menu.push(MenuItem{
|
|
|
|
item_type: MenuItemType::Separator,
|
|
|
|
item_name: "".to_owned(),
|
|
|
|
item_id: 999,
|
|
|
|
});
|
|
|
|
|
|
|
|
menu.push(MenuItem{
|
|
|
|
item_type: MenuItemType::Button,
|
|
|
|
item_name: "Exit".to_owned(),
|
2019-09-12 21:53:17 +00:00
|
|
|
item_id: ActionType::Exit as i32,
|
2019-09-12 21:24:55 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
menu
|
2019-08-31 14:07:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-15 11:03:21 +00:00
|
|
|
lazy_static! {
|
2019-09-15 13:46:24 +00:00
|
|
|
static ref VAR_REGEX: Regex = Regex::new("\\{\\{\\s*(?P<name>\\w+)\\s*\\}\\}").unwrap();
|
2019-09-15 11:03:21 +00:00
|
|
|
}
|
|
|
|
|
2019-09-13 13:03:03 +00:00
|
|
|
impl <'a, S: KeyboardManager, C: ClipboardManager, M: ConfigManager<'a>, U: UIManager>
|
2019-09-08 11:37:58 +00:00
|
|
|
MatchReceiver for Engine<'a, S, C, M, U>{
|
|
|
|
|
2019-10-19 21:31:05 +00:00
|
|
|
fn on_match(&self, m: &Match, trailing_separator: Option<char>) {
|
2019-09-09 15:13:58 +00:00
|
|
|
let config = self.config_manager.active_config();
|
|
|
|
|
|
|
|
if config.disabled {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-07 15:59:34 +00:00
|
|
|
|
2019-10-19 21:31:05 +00:00
|
|
|
let char_count = if trailing_separator.is_none() {
|
|
|
|
m.trigger.chars().count() as i32
|
|
|
|
}else{
|
|
|
|
m.trigger.chars().count() as i32 + 1 // Count also the separator
|
|
|
|
};
|
|
|
|
|
|
|
|
self.keyboard_manager.delete_string(char_count);
|
2019-09-06 08:21:33 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Manage the different types of matches
|
|
|
|
match &m.content {
|
|
|
|
// Text Match
|
|
|
|
MatchContentType::Text(content) => {
|
|
|
|
let mut target_string = if content._has_vars {
|
|
|
|
let mut output_map = HashMap::new();
|
|
|
|
|
|
|
|
for variable in content.vars.iter() {
|
|
|
|
let extension = self.extension_map.get(&variable.var_type);
|
|
|
|
if let Some(extension) = extension {
|
|
|
|
let ext_out = extension.calculate(&variable.params);
|
|
|
|
if let Some(output) = ext_out {
|
|
|
|
output_map.insert(variable.name.clone(), output);
|
|
|
|
}else{
|
|
|
|
output_map.insert(variable.name.clone(), "".to_owned());
|
|
|
|
warn!("Could not generate output for variable: {}", variable.name);
|
|
|
|
}
|
|
|
|
}else{
|
|
|
|
error!("No extension found for variable type: {}", variable.var_type);
|
|
|
|
}
|
|
|
|
}
|
2019-09-15 11:03:21 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Replace the variables
|
|
|
|
let result = VAR_REGEX.replace_all(&content.replace, |caps: &Captures| {
|
|
|
|
let var_name = caps.name("name").unwrap().as_str();
|
|
|
|
let output = output_map.get(var_name);
|
|
|
|
output.unwrap()
|
|
|
|
});
|
|
|
|
|
|
|
|
result.to_string()
|
|
|
|
}else{ // No variables, simple text substitution
|
|
|
|
content.replace.clone()
|
|
|
|
};
|
|
|
|
|
|
|
|
// If a trailing separator was counted in the match, add it back to the target string
|
|
|
|
if let Some(trailing_separator) = trailing_separator {
|
|
|
|
if trailing_separator == '\r' { // If the trailing separator is a carriage return,
|
|
|
|
target_string.push('\n'); // convert it to new line
|
2019-09-15 11:03:21 +00:00
|
|
|
}else{
|
2019-11-28 21:56:00 +00:00
|
|
|
target_string.push(trailing_separator);
|
2019-09-15 11:03:21 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-24 16:48:55 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Convert Windows style newlines into unix styles
|
|
|
|
target_string = target_string.replace("\r\n", "\n");
|
2019-10-24 16:48:55 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Calculate cursor rewind moves if a Cursor Hint is present
|
|
|
|
let index = target_string.find("$|$");
|
|
|
|
let cursor_rewind = if let Some(index) = index {
|
|
|
|
// Convert the byte index to a char index
|
|
|
|
let char_str = &target_string[0..index];
|
|
|
|
let char_index = char_str.chars().count();
|
|
|
|
let total_size = target_string.chars().count();
|
2019-10-24 16:48:55 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Remove the $|$ placeholder
|
|
|
|
target_string = target_string.replace("$|$", "");
|
2019-09-06 08:21:33 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
// Calculate the amount of rewind moves needed (LEFT ARROW).
|
|
|
|
// Subtract also 3, equal to the number of chars of the placeholder "$|$"
|
|
|
|
let moves = (total_size - char_index - 3) as i32;
|
|
|
|
Some(moves)
|
2019-09-07 08:43:23 +00:00
|
|
|
}else{
|
2019-11-28 21:56:00 +00:00
|
|
|
None
|
|
|
|
};
|
|
|
|
|
|
|
|
match config.backend {
|
|
|
|
BackendType::Inject => {
|
|
|
|
// Send the expected string. On linux, newlines are managed automatically
|
|
|
|
// while on windows and macos, we need to emulate a Enter key press.
|
|
|
|
|
|
|
|
if cfg!(target_os = "linux") {
|
|
|
|
self.keyboard_manager.send_string(&target_string);
|
|
|
|
}else{
|
|
|
|
// To handle newlines, substitute each "\n" char with an Enter key press.
|
|
|
|
let splits = target_string.split('\n');
|
|
|
|
|
|
|
|
for (i, split) in splits.enumerate() {
|
|
|
|
if i > 0 {
|
|
|
|
self.keyboard_manager.send_enter();
|
|
|
|
}
|
|
|
|
|
|
|
|
self.keyboard_manager.send_string(split);
|
|
|
|
}
|
2019-09-07 08:43:23 +00:00
|
|
|
}
|
2019-11-28 21:56:00 +00:00
|
|
|
},
|
|
|
|
BackendType::Clipboard => {
|
2019-12-11 17:12:40 +00:00
|
|
|
let previous_clipboard_content = self.clipboard_manager.get_clipboard().unwrap_or(String::from(""));
|
2019-11-28 21:56:00 +00:00
|
|
|
self.clipboard_manager.set_clipboard(&target_string);
|
2019-11-29 21:09:02 +00:00
|
|
|
self.keyboard_manager.trigger_paste(&config.paste_shortcut);
|
2019-12-11 17:12:40 +00:00
|
|
|
self.clipboard_manager.set_clipboard(&previous_clipboard_content);
|
2019-11-28 21:56:00 +00:00
|
|
|
},
|
|
|
|
}
|
2019-09-06 08:21:33 +00:00
|
|
|
|
2019-11-28 21:56:00 +00:00
|
|
|
if let Some(moves) = cursor_rewind {
|
|
|
|
// Simulate left arrow key presses to bring the cursor into the desired position
|
|
|
|
self.keyboard_manager.move_cursor_left(moves);
|
2019-09-07 08:43:23 +00:00
|
|
|
}
|
|
|
|
},
|
2019-11-28 21:56:00 +00:00
|
|
|
|
|
|
|
// Image Match
|
|
|
|
MatchContentType::Image(content) => {
|
2019-11-28 22:06:12 +00:00
|
|
|
// Make sure the image exist beforehand
|
2019-11-28 23:01:26 +00:00
|
|
|
if content.path.exists() {
|
|
|
|
self.clipboard_manager.set_clipboard_image(&content.path);
|
2019-11-29 21:09:02 +00:00
|
|
|
self.keyboard_manager.trigger_paste(&config.paste_shortcut);
|
2019-11-28 22:06:12 +00:00
|
|
|
}else{
|
2019-11-28 23:01:26 +00:00
|
|
|
error!("Image not found in path: {:?}", content.path);
|
2019-11-28 22:06:12 +00:00
|
|
|
}
|
2019-09-07 08:43:23 +00:00
|
|
|
},
|
2019-09-06 08:21:33 +00:00
|
|
|
}
|
2019-08-31 14:07:45 +00:00
|
|
|
}
|
2019-09-08 11:37:58 +00:00
|
|
|
|
2019-09-14 18:13:09 +00:00
|
|
|
fn on_enable_update(&self, status: bool) {
|
2019-09-08 11:37:58 +00:00
|
|
|
let message = if status {
|
|
|
|
"espanso enabled"
|
|
|
|
}else{
|
|
|
|
"espanso disabled"
|
|
|
|
};
|
|
|
|
|
|
|
|
info!("Toggled: {}", message);
|
|
|
|
|
2019-09-12 21:24:55 +00:00
|
|
|
let mut enabled_ref = self.enabled.borrow_mut();
|
|
|
|
*enabled_ref = status;
|
|
|
|
|
2019-09-08 11:37:58 +00:00
|
|
|
self.ui_manager.notify(message);
|
|
|
|
}
|
2019-09-12 20:14:41 +00:00
|
|
|
}
|
|
|
|
|
2019-09-13 13:03:03 +00:00
|
|
|
impl <'a, S: KeyboardManager, C: ClipboardManager,
|
2019-09-12 20:14:41 +00:00
|
|
|
M: ConfigManager<'a>, U: UIManager> ActionEventReceiver for Engine<'a, S, C, M, U>{
|
|
|
|
|
2019-09-14 10:19:11 +00:00
|
|
|
fn on_action_event(&self, e: ActionType) {
|
2019-09-12 21:24:55 +00:00
|
|
|
match e {
|
2019-09-14 10:19:11 +00:00
|
|
|
ActionType::IconClick => {
|
2019-09-12 21:24:55 +00:00
|
|
|
self.ui_manager.show_menu(self.build_menu());
|
|
|
|
},
|
2019-09-14 10:19:11 +00:00
|
|
|
ActionType::Exit => {
|
|
|
|
info!("Terminating espanso.");
|
2019-09-16 09:59:23 +00:00
|
|
|
self.ui_manager.cleanup();
|
2019-09-14 10:19:11 +00:00
|
|
|
exit(0);
|
|
|
|
},
|
|
|
|
_ => {}
|
2019-09-12 21:24:55 +00:00
|
|
|
}
|
2019-09-12 20:14:41 +00:00
|
|
|
}
|
2019-08-31 14:07:45 +00:00
|
|
|
}
|