feat(core): wire up modulo forms
This commit is contained in:
		
							parent
							
								
									de236a89d2
								
							
						
					
					
						commit
						d7ebd2a4dd
					
				
							
								
								
									
										2
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| 
						 | 
				
			
			@ -302,6 +302,8 @@ dependencies = [
 | 
			
		|||
 "lazy_static",
 | 
			
		||||
 "log",
 | 
			
		||||
 "maplit",
 | 
			
		||||
 "serde",
 | 
			
		||||
 "serde_json",
 | 
			
		||||
 "simplelog",
 | 
			
		||||
 "thiserror",
 | 
			
		||||
]
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -32,4 +32,6 @@ clap = "2.33.3"
 | 
			
		|||
lazy_static = "1.4.0"
 | 
			
		||||
crossbeam = "0.8.0"
 | 
			
		||||
enum-as-inner = "0.3.3"
 | 
			
		||||
dirs = "3.0.1"
 | 
			
		||||
dirs = "3.0.1"
 | 
			
		||||
serde = { version = "1.0.123", features = ["derive"] }
 | 
			
		||||
serde_json = "1.0.62"
 | 
			
		||||
| 
						 | 
				
			
			@ -39,6 +39,8 @@ impl <'a> TextInjector for EventInjectorAdapter<'a> {
 | 
			
		|||
  }
 | 
			
		||||
  
 | 
			
		||||
  fn inject_text(&self, text: &str) -> anyhow::Result<()> {
 | 
			
		||||
    // TODO: wait for modifiers release
 | 
			
		||||
 | 
			
		||||
    // Handle CRLF or LF line endings correctly
 | 
			
		||||
    let split_sequence = if text.contains("\r\n") {
 | 
			
		||||
      "\r\n"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,6 +33,8 @@ impl<'a> KeyInjectorAdapter<'a> {
 | 
			
		|||
 | 
			
		||||
impl<'a> KeyInjector for KeyInjectorAdapter<'a> {
 | 
			
		||||
  fn inject_sequence(&self, keys: &[crate::engine::event::input::Key]) -> anyhow::Result<()> {
 | 
			
		||||
    // TODO: wait for modifiers release
 | 
			
		||||
    
 | 
			
		||||
    let converted_keys: Vec<_> = keys.iter().map(convert_to_inject_key).collect();
 | 
			
		||||
    self.injector.send_keys(&converted_keys, Default::default()) // TODO: handle options
 | 
			
		||||
  }
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -22,6 +22,8 @@ use espanso_config::{config::ConfigStore, matches::store::MatchStore};
 | 
			
		|||
use espanso_path::Paths;
 | 
			
		||||
use ui::selector::MatchSelectorAdapter;
 | 
			
		||||
 | 
			
		||||
use super::ui::icon::IconPaths;
 | 
			
		||||
 | 
			
		||||
pub mod executor;
 | 
			
		||||
pub mod match_cache;
 | 
			
		||||
pub mod matcher;
 | 
			
		||||
| 
						 | 
				
			
			@ -30,10 +32,17 @@ pub mod render;
 | 
			
		|||
pub mod source;
 | 
			
		||||
pub mod ui;
 | 
			
		||||
 | 
			
		||||
pub fn initialize_and_spawn(paths: Paths, config_store: Box<dyn ConfigStore>, match_store: Box<dyn MatchStore>) -> Result<()> {
 | 
			
		||||
pub fn initialize_and_spawn(
 | 
			
		||||
  paths: Paths,
 | 
			
		||||
  config_store: Box<dyn ConfigStore>,
 | 
			
		||||
  match_store: Box<dyn MatchStore>,
 | 
			
		||||
  icon_paths: IconPaths,
 | 
			
		||||
) -> Result<()> {
 | 
			
		||||
  std::thread::Builder::new()
 | 
			
		||||
    .name("engine thread".to_string())
 | 
			
		||||
    .spawn(move || {
 | 
			
		||||
      // TODO: properly order the initializations if necessary
 | 
			
		||||
 | 
			
		||||
      let app_info_provider =
 | 
			
		||||
        espanso_info::get_provider().expect("unable to initialize app info provider");
 | 
			
		||||
      let config_manager =
 | 
			
		||||
| 
						 | 
				
			
			@ -42,15 +51,19 @@ pub fn initialize_and_spawn(paths: Paths, config_store: Box<dyn ConfigStore>, ma
 | 
			
		|||
        super::engine::matcher::convert::MatchConverter::new(&*config_store, &*match_store);
 | 
			
		||||
      let match_cache = super::engine::match_cache::MatchCache::load(&*config_store, &*match_store);
 | 
			
		||||
 | 
			
		||||
      let detect_source =
 | 
			
		||||
        super::engine::source::detect::init_and_spawn().expect("failed to initialize detector module");
 | 
			
		||||
      let modulo_manager = ui::modulo::ModuloManager::new();
 | 
			
		||||
 | 
			
		||||
      let detect_source = super::engine::source::detect::init_and_spawn()
 | 
			
		||||
        .expect("failed to initialize detector module");
 | 
			
		||||
      let sources: Vec<&dyn crate::engine::funnel::Source> = vec![&detect_source];
 | 
			
		||||
      let funnel = crate::engine::funnel::default(&sources);
 | 
			
		||||
 | 
			
		||||
      let matcher = super::engine::matcher::rolling::RollingMatcherAdapter::new(
 | 
			
		||||
        &match_converter.get_rolling_matches(),
 | 
			
		||||
      );
 | 
			
		||||
      let matchers: Vec<&dyn crate::engine::process::Matcher<super::engine::matcher::MatcherState>> = vec![&matcher];
 | 
			
		||||
      let matchers: Vec<
 | 
			
		||||
        &dyn crate::engine::process::Matcher<super::engine::matcher::MatcherState>,
 | 
			
		||||
      > = vec![&matcher];
 | 
			
		||||
      let selector = MatchSelectorAdapter::new();
 | 
			
		||||
      let multiplexer = super::engine::multiplex::MultiplexAdapter::new(&match_cache);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -72,6 +85,8 @@ pub fn initialize_and_spawn(paths: Paths, config_store: Box<dyn ConfigStore>, ma
 | 
			
		|||
        &paths.packages,
 | 
			
		||||
      );
 | 
			
		||||
      let shell_extension = espanso_render::extension::shell::ShellExtension::new(&paths.config);
 | 
			
		||||
      let form_adapter = ui::modulo::form::ModuloFormProviderAdapter::new(&modulo_manager, icon_paths.form_icon);
 | 
			
		||||
      let form_extension = espanso_render::extension::form::FormExtension::new(&form_adapter);
 | 
			
		||||
      let renderer = espanso_render::create(vec![
 | 
			
		||||
        &clipboard_extension,
 | 
			
		||||
        &date_extension,
 | 
			
		||||
| 
						 | 
				
			
			@ -79,6 +94,7 @@ pub fn initialize_and_spawn(paths: Paths, config_store: Box<dyn ConfigStore>, ma
 | 
			
		|||
        &random_extension,
 | 
			
		||||
        &script_extension,
 | 
			
		||||
        &shell_extension,
 | 
			
		||||
        &form_extension,
 | 
			
		||||
      ]);
 | 
			
		||||
      let renderer_adapter =
 | 
			
		||||
        super::engine::render::RendererAdapter::new(&match_cache, &config_manager, &renderer);
 | 
			
		||||
| 
						 | 
				
			
			@ -92,11 +108,13 @@ pub fn initialize_and_spawn(paths: Paths, config_store: Box<dyn ConfigStore>, ma
 | 
			
		|||
        &match_cache,
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      let event_injector = super::engine::executor::event_injector::EventInjectorAdapter::new(&*injector);
 | 
			
		||||
      let clipboard_injector = super::engine::executor::clipboard_injector::ClipboardInjectorAdapter::new(
 | 
			
		||||
        &*injector,
 | 
			
		||||
        &*clipboard,
 | 
			
		||||
      );
 | 
			
		||||
      let event_injector =
 | 
			
		||||
        super::engine::executor::event_injector::EventInjectorAdapter::new(&*injector);
 | 
			
		||||
      let clipboard_injector =
 | 
			
		||||
        super::engine::executor::clipboard_injector::ClipboardInjectorAdapter::new(
 | 
			
		||||
          &*injector,
 | 
			
		||||
          &*clipboard,
 | 
			
		||||
        );
 | 
			
		||||
      let key_injector = super::engine::executor::key_injector::KeyInjectorAdapter::new(&*injector);
 | 
			
		||||
      let dispatcher = crate::engine::dispatch::default(
 | 
			
		||||
        &event_injector,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -17,4 +17,5 @@
 | 
			
		|||
 * along with espanso.  If not, see <https://www.gnu.org/licenses/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
pub mod modulo;
 | 
			
		||||
pub mod selector;
 | 
			
		||||
							
								
								
									
										125
									
								
								espanso/src/cli/worker/engine/ui/modulo/form.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								espanso/src/cli/worker/engine/ui/modulo/form.rs
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,125 @@
 | 
			
		|||
/*
 | 
			
		||||
 * This file is part of espanso.
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright (C) 2019-2021 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/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
use std::{collections::HashMap, path::PathBuf};
 | 
			
		||||
 | 
			
		||||
use super::ModuloManager;
 | 
			
		||||
use anyhow::Result;
 | 
			
		||||
use espanso_render::extension::form::{FormProvider, FormProviderResult};
 | 
			
		||||
use log::{error};
 | 
			
		||||
use serde::Serialize;
 | 
			
		||||
use serde_json::{Map, Value};
 | 
			
		||||
 | 
			
		||||
pub struct ModuloFormProviderAdapter<'a> {
 | 
			
		||||
  manager: &'a ModuloManager,
 | 
			
		||||
  icon_path: Option<String>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl<'a> ModuloFormProviderAdapter<'a> {
 | 
			
		||||
  pub fn new(manager: &'a ModuloManager, icon_path: Option<PathBuf>) -> Self {
 | 
			
		||||
    Self {
 | 
			
		||||
      manager,
 | 
			
		||||
      icon_path: icon_path.map(|path| path.to_string_lossy().to_string()),
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl<'a> FormProvider for ModuloFormProviderAdapter<'a> {
 | 
			
		||||
  fn show(
 | 
			
		||||
    &self,
 | 
			
		||||
    layout: &str,
 | 
			
		||||
    fields: &espanso_render::Params,
 | 
			
		||||
    _: &espanso_render::Params,
 | 
			
		||||
  ) -> FormProviderResult {
 | 
			
		||||
    let modulo_form_config = ModuloFormConfig {
 | 
			
		||||
      icon: self.icon_path.as_deref(),
 | 
			
		||||
      title: "espanso",
 | 
			
		||||
      layout,
 | 
			
		||||
      fields: convert_params_into_object(fields),
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    match serde_json::to_string(&modulo_form_config) {
 | 
			
		||||
      Ok(json_config) => {
 | 
			
		||||
        match self
 | 
			
		||||
          .manager
 | 
			
		||||
          .invoke(&["form", "-j", "-i", "-"], &json_config)
 | 
			
		||||
        {
 | 
			
		||||
          Ok(output) => {
 | 
			
		||||
            let json: Result<HashMap<String, String>, _> = serde_json::from_str(&output);
 | 
			
		||||
            match json {
 | 
			
		||||
              Ok(json) => {
 | 
			
		||||
                if json.is_empty() {
 | 
			
		||||
                  return FormProviderResult::Aborted;
 | 
			
		||||
                } else {
 | 
			
		||||
                  return FormProviderResult::Success(json);
 | 
			
		||||
                }
 | 
			
		||||
              }
 | 
			
		||||
              Err(error) => {
 | 
			
		||||
                return FormProviderResult::Error(error.into());
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
          Err(err) => {
 | 
			
		||||
            return FormProviderResult::Error(err.into());
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      Err(err) => {
 | 
			
		||||
        return FormProviderResult::Error(err.into());
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Debug, Serialize)]
 | 
			
		||||
struct ModuloFormConfig<'a> {
 | 
			
		||||
  icon: Option<&'a str>,
 | 
			
		||||
  title: &'a str,
 | 
			
		||||
  layout: &'a str,
 | 
			
		||||
  fields: Map<String, Value>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// TODO: test
 | 
			
		||||
fn convert_params_into_object(params: &espanso_render::Params) -> Map<String, Value> {
 | 
			
		||||
  let mut obj = Map::new();
 | 
			
		||||
  for (field, value) in params {
 | 
			
		||||
    obj.insert(field.clone(), convert_value(value));
 | 
			
		||||
  }
 | 
			
		||||
  obj
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// TODO: test
 | 
			
		||||
fn convert_value(value: &espanso_render::Value) -> Value {
 | 
			
		||||
  match value {
 | 
			
		||||
    espanso_render::Value::Null => Value::Null,
 | 
			
		||||
    espanso_render::Value::Bool(value) => Value::Bool(*value),
 | 
			
		||||
    espanso_render::Value::Number(num) => match num {
 | 
			
		||||
      espanso_render::Number::Integer(val) => Value::Number((*val).into()),
 | 
			
		||||
      espanso_render::Number::Float(val) => {
 | 
			
		||||
        Value::Number(serde_json::Number::from_f64(*val).unwrap_or_else(|| {
 | 
			
		||||
          error!("unable to convert float value to json");
 | 
			
		||||
          0.into()
 | 
			
		||||
        }))
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    espanso_render::Value::String(value) => Value::String(value.clone()),
 | 
			
		||||
    espanso_render::Value::Array(arr) => Value::Array(arr.into_iter().map(convert_value).collect()),
 | 
			
		||||
    espanso_render::Value::Object(obj) => Value::Object(convert_params_into_object(obj)),
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										155
									
								
								espanso/src/cli/worker/engine/ui/modulo/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								espanso/src/cli/worker/engine/ui/modulo/mod.rs
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,155 @@
 | 
			
		|||
/*
 | 
			
		||||
 * This file is part of espanso.
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright (C) 2019-2021 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/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
use anyhow::Result;
 | 
			
		||||
use log::{error, info};
 | 
			
		||||
use std::io::Write;
 | 
			
		||||
use std::process::Command;
 | 
			
		||||
use thiserror::Error;
 | 
			
		||||
 | 
			
		||||
pub mod form;
 | 
			
		||||
 | 
			
		||||
pub struct ModuloManager {
 | 
			
		||||
  modulo_path: Option<String>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl ModuloManager {
 | 
			
		||||
  pub fn new() -> Self {
 | 
			
		||||
    let mut modulo_path: Option<String> = None;
 | 
			
		||||
    // Check if the `MODULO_PATH` env variable is configured
 | 
			
		||||
    if let Some(_modulo_path) = std::env::var_os("MODULO_PATH") {
 | 
			
		||||
      info!("using modulo from env variable at {:?}", _modulo_path);
 | 
			
		||||
      modulo_path = Some(_modulo_path.to_string_lossy().to_string())
 | 
			
		||||
    } else {
 | 
			
		||||
      // Check in the same directory of espanso
 | 
			
		||||
      if let Ok(exe_path) = std::env::current_exe() {
 | 
			
		||||
        if let Some(parent) = exe_path.parent() {
 | 
			
		||||
          let possible_path = parent.join("modulo");
 | 
			
		||||
          let possible_path = possible_path.to_string_lossy().to_string();
 | 
			
		||||
 | 
			
		||||
          if let Ok(output) = Command::new(&possible_path).arg("--version").output() {
 | 
			
		||||
            if output.status.success() {
 | 
			
		||||
              info!("using modulo from exe directory at {:?}", possible_path);
 | 
			
		||||
              modulo_path = Some(possible_path);
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Otherwise check if present in the PATH
 | 
			
		||||
      if modulo_path.is_none() {
 | 
			
		||||
        if let Ok(output) = Command::new("modulo").arg("--version").output() {
 | 
			
		||||
          if output.status.success() {
 | 
			
		||||
            info!("using modulo executable found in PATH");
 | 
			
		||||
            modulo_path = Some("modulo".to_owned());
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Self { modulo_path }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // pub fn is_valid(&self) -> bool {
 | 
			
		||||
  //   self.modulo_path.is_some()
 | 
			
		||||
  // }
 | 
			
		||||
 | 
			
		||||
  // pub fn get_version(&self) -> Option<String> {
 | 
			
		||||
  //   if let Some(ref modulo_path) = self.modulo_path {
 | 
			
		||||
  //     if let Ok(output) = Command::new(modulo_path).arg("--version").output() {
 | 
			
		||||
  //       let version = String::from_utf8_lossy(&output.stdout);
 | 
			
		||||
  //       return Some(version.to_string());
 | 
			
		||||
  //     }
 | 
			
		||||
  //   }
 | 
			
		||||
 | 
			
		||||
  //   None
 | 
			
		||||
  // }
 | 
			
		||||
 | 
			
		||||
  pub fn invoke(&self, args: &[&str], body: &str) -> Result<String> {
 | 
			
		||||
    if let Some(modulo_path) = &self.modulo_path {
 | 
			
		||||
      let mut command = Command::new(modulo_path);
 | 
			
		||||
      command
 | 
			
		||||
        .args(args)
 | 
			
		||||
        .stdin(std::process::Stdio::piped())
 | 
			
		||||
        .stdout(std::process::Stdio::piped())
 | 
			
		||||
        .stderr(std::process::Stdio::piped());
 | 
			
		||||
 | 
			
		||||
      crate::util::set_command_flags(&mut command);
 | 
			
		||||
 | 
			
		||||
      let child = command.spawn();
 | 
			
		||||
 | 
			
		||||
      match child {
 | 
			
		||||
        Ok(mut child) => {
 | 
			
		||||
          if let Some(stdin) = child.stdin.as_mut() {
 | 
			
		||||
            match stdin.write_all(body.as_bytes()) {
 | 
			
		||||
              Ok(_) => {
 | 
			
		||||
                // Get the output
 | 
			
		||||
                match child.wait_with_output() {
 | 
			
		||||
                  Ok(child_output) => {
 | 
			
		||||
                    let output = String::from_utf8_lossy(&child_output.stdout);
 | 
			
		||||
 | 
			
		||||
                    // Check also if the program reports an error
 | 
			
		||||
                    let error = String::from_utf8_lossy(&child_output.stderr);
 | 
			
		||||
                    if !error.is_empty() {
 | 
			
		||||
                      error!("modulo reported an error: {}", error);
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    if !output.trim().is_empty() {
 | 
			
		||||
                      return Ok(output.to_string());
 | 
			
		||||
                    } else {
 | 
			
		||||
                      return Err(ModuloError::EmptyOutput.into());
 | 
			
		||||
                    }
 | 
			
		||||
                  }
 | 
			
		||||
                  Err(error) => {
 | 
			
		||||
                    return Err(ModuloError::Error(error).into());
 | 
			
		||||
                  }
 | 
			
		||||
                }
 | 
			
		||||
              }
 | 
			
		||||
              Err(error) => {
 | 
			
		||||
                return Err(ModuloError::Error(error).into());
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
          } else {
 | 
			
		||||
            return Err(ModuloError::StdinError.into());
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        Err(error) => {
 | 
			
		||||
          return Err(ModuloError::Error(error).into());
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      return Err(ModuloError::MissingModulo.into());
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Error, Debug)]
 | 
			
		||||
pub enum ModuloError {
 | 
			
		||||
  #[error("attempt to invoke modulo even though it's not configured")]
 | 
			
		||||
  MissingModulo,
 | 
			
		||||
 | 
			
		||||
  #[error("modulo returned an empty output")]
 | 
			
		||||
  EmptyOutput,
 | 
			
		||||
 | 
			
		||||
  #[error("could not connect to modulo stdin")]
 | 
			
		||||
  StdinError,
 | 
			
		||||
 | 
			
		||||
  #[error("error occurred during modulo invocation")]
 | 
			
		||||
  Error(#[from] std::io::Error),
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -55,6 +55,7 @@ fn worker_main(args: CliModuleArgs) {
 | 
			
		|||
    icon_paths: convert_icon_paths_to_tray_vec(&icon_paths),
 | 
			
		||||
    notification_icon_path: icon_paths
 | 
			
		||||
      .logo
 | 
			
		||||
      .as_ref()
 | 
			
		||||
      .map(|path| path.to_string_lossy().to_string()),
 | 
			
		||||
    ..Default::default()
 | 
			
		||||
  })
 | 
			
		||||
| 
						 | 
				
			
			@ -66,7 +67,7 @@ fn worker_main(args: CliModuleArgs) {
 | 
			
		|||
 | 
			
		||||
  // TODO: pass the remote
 | 
			
		||||
  // Initialize the engine on another thread and start it
 | 
			
		||||
  engine::initialize_and_spawn(paths.clone(), config_store, match_store)
 | 
			
		||||
  engine::initialize_and_spawn(paths.clone(), config_store, match_store, icon_paths)
 | 
			
		||||
    .expect("unable to initialize engine");
 | 
			
		||||
 | 
			
		||||
  eventloop.run(Box::new(move |event| {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,6 +33,8 @@ const WINDOWS_RED_ICO_BINARY: &[u8] = include_bytes!("../../../res/windows/espan
 | 
			
		|||
 | 
			
		||||
#[derive(Debug, Default)]
 | 
			
		||||
pub struct IconPaths {
 | 
			
		||||
  pub form_icon: Option<PathBuf>,
 | 
			
		||||
 | 
			
		||||
  pub tray_icon_normal: Option<PathBuf>,
 | 
			
		||||
  pub tray_icon_disabled: Option<PathBuf>,
 | 
			
		||||
  pub tray_icon_system_disabled: Option<PathBuf>, // TODO: secure input
 | 
			
		||||
| 
						 | 
				
			
			@ -43,6 +45,7 @@ pub struct IconPaths {
 | 
			
		|||
#[cfg(target_os = "windows")]
 | 
			
		||||
pub fn load_icon_paths(runtime_dir: &Path) -> Result<IconPaths> {
 | 
			
		||||
  Ok(IconPaths {
 | 
			
		||||
    form_icon: Some(extract_icon(WINDOWS_ICO_BINARY, &runtime_dir.join("form.ico"))?),
 | 
			
		||||
    tray_icon_normal: Some(extract_icon(WINDOWS_ICO_BINARY, &runtime_dir.join("normal.ico"))?),
 | 
			
		||||
    tray_icon_disabled: Some(extract_icon(WINDOWS_RED_ICO_BINARY, &runtime_dir.join("disabled.ico"))?),
 | 
			
		||||
    logo: Some(extract_icon(ICON_BINARY, &runtime_dir.join("icon.png"))?),
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -28,6 +28,7 @@ use simplelog::{
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
mod cli;
 | 
			
		||||
mod util;
 | 
			
		||||
mod engine;
 | 
			
		||||
mod logging;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										33
									
								
								espanso/src/util.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								espanso/src/util.rs
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,33 @@
 | 
			
		|||
/*
 | 
			
		||||
 * This file is part of espanso.
 | 
			
		||||
 *
 | 
			
		||||
 * Copyright (C) 2019-2021 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/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
use std::process::Command;
 | 
			
		||||
 | 
			
		||||
#[cfg(target_os = "windows")]
 | 
			
		||||
pub fn set_command_flags(command: &mut Command) {
 | 
			
		||||
  use std::os::windows::process::CommandExt;
 | 
			
		||||
  // Avoid showing the shell window
 | 
			
		||||
  // See: https://github.com/federico-terzi/espanso/issues/249
 | 
			
		||||
  command.creation_flags(0x08000000);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[cfg(not(target_os = "windows"))]
 | 
			
		||||
pub fn set_command_flags(_: &mut Command) {
 | 
			
		||||
  // NOOP on Linux and macOS
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user