feat(core): implement search bar

This commit is contained in:
Federico Terzi 2021-08-13 20:35:22 +02:00
parent d569d96dc6
commit 80a197376c
10 changed files with 147 additions and 6 deletions

View File

@ -19,11 +19,14 @@
use std::cell::Cell; use std::cell::Cell;
use espanso_config::config::Config;
use crate::engine::event::EventType; use crate::engine::event::EventType;
use super::context::Context; use super::context::Context;
mod debug; mod debug;
mod search;
const MIN_BUILTIN_MATCH_ID: i32 = 1_000_000_000; const MIN_BUILTIN_MATCH_ID: i32 = 1_000_000_000;
@ -34,11 +37,17 @@ pub struct BuiltInMatch {
pub action: fn(context: &dyn Context) -> EventType, pub action: fn(context: &dyn Context) -> EventType,
} }
pub fn get_builtin_matches() -> Vec<BuiltInMatch> { pub fn get_builtin_matches(config: &dyn Config) -> Vec<BuiltInMatch> {
vec![ let mut matches = vec![
debug::create_match_paste_active_config_info(), debug::create_match_paste_active_config_info(),
debug::create_match_paste_active_app_info(), debug::create_match_paste_active_app_info(),
] ];
if let Some(search_trigger) = config.search_trigger() {
matches.push(search::create_match_trigger_search_bar(&search_trigger));
}
matches
} }
pub fn is_builtin_match(id: i32) -> bool { pub fn is_builtin_match(id: i32) -> bool {

View 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 crate::{cli::worker::builtin::generate_next_builtin_id, engine::event::{EventType}};
use super::BuiltInMatch;
pub fn create_match_trigger_search_bar(shortcut: &str) -> BuiltInMatch {
BuiltInMatch {
id: generate_next_builtin_id(),
label: "Open search bar",
triggers: vec![shortcut.to_string()],
action: |_| {
EventType::ShowSearchBar
},
}
}

View File

@ -79,7 +79,7 @@ pub fn initialize_and_spawn(
let modulo_search_ui = crate::gui::modulo::search::ModuloSearchUI::new(&modulo_manager); let modulo_search_ui = crate::gui::modulo::search::ModuloSearchUI::new(&modulo_manager);
let context: Box<dyn Context> = Box::new(super::context::DefaultContext::new(&config_manager, &*app_info_provider)); let context: Box<dyn Context> = Box::new(super::context::DefaultContext::new(&config_manager, &*app_info_provider));
let builtin_matches = super::builtin::get_builtin_matches(); let builtin_matches = super::builtin::get_builtin_matches(&*config_manager.default());
let combined_match_cache = CombinedMatchCache::load(&match_cache, &builtin_matches); let combined_match_cache = CombinedMatchCache::load(&match_cache, &builtin_matches);
let match_converter = MatchConverter::new(&*config_store, &*match_store, &builtin_matches); let match_converter = MatchConverter::new(&*config_store, &*match_store, &builtin_matches);
@ -183,6 +183,7 @@ pub fn initialize_and_spawn(
&path_provider, &path_provider,
disable_options, disable_options,
&config_manager, &config_manager,
&combined_match_cache,
); );
let event_injector = EventInjectorAdapter::new(&*injector, &config_manager); let event_injector = EventInjectorAdapter::new(&*injector, &config_manager);

View File

@ -43,6 +43,10 @@ impl<'a> MatchCache<'a> {
Self { cache } Self { cache }
} }
fn ids(&self) -> Vec<i32> {
self.cache.keys().copied().collect()
}
} }
impl<'a> super::engine::process::middleware::render::MatchProvider<'a> for MatchCache<'a> { impl<'a> super::engine::process::middleware::render::MatchProvider<'a> for MatchCache<'a> {
@ -150,3 +154,11 @@ impl<'a> super::engine::process::middleware::multiplex::MatchProvider<'a>
}) })
} }
} }
impl<'a> crate::engine::process::MatchProvider for CombinedMatchCache<'a> {
fn get_all_matches_ids(&self) -> Vec<i32> {
let mut ids: Vec<i32> = self.builtin_match_cache.keys().copied().collect();
ids.extend(self.user_match_cache.ids());
ids
}
}

View File

@ -92,6 +92,7 @@ pub enum EventType {
ShowContextMenu(ui::ShowContextMenuEvent), ShowContextMenu(ui::ShowContextMenuEvent),
IconStatusChange(ui::IconStatusChangeEvent), IconStatusChange(ui::IconStatusChangeEvent),
DisplaySecureInputTroubleshoot, DisplaySecureInputTroubleshoot,
ShowSearchBar,
// Other // Other
LaunchSecureInputAutoFix, LaunchSecureInputAutoFix,

View File

@ -19,13 +19,13 @@
use log::trace; use log::trace;
use super::{DisableOptions, MatchFilter, MatchInfoProvider, MatchSelector, Matcher, MatcherMiddlewareConfigProvider, Middleware, Multiplexer, PathProvider, Processor, Renderer, middleware::{ use super::{DisableOptions, MatchFilter, MatchInfoProvider, MatchProvider, MatchSelector, Matcher, MatcherMiddlewareConfigProvider, Middleware, Multiplexer, PathProvider, Processor, Renderer, middleware::{
match_select::MatchSelectMiddleware, matcher::MatcherMiddleware, multiplex::MultiplexMiddleware, match_select::MatchSelectMiddleware, matcher::MatcherMiddleware, multiplex::MultiplexMiddleware,
render::RenderMiddleware, action::{ActionMiddleware, EventSequenceProvider}, cursor_hint::CursorHintMiddleware, cause::CauseCompensateMiddleware, render::RenderMiddleware, action::{ActionMiddleware, EventSequenceProvider}, cursor_hint::CursorHintMiddleware, cause::CauseCompensateMiddleware,
delay_modifiers::{DelayForModifierReleaseMiddleware, ModifierStatusProvider}, markdown::MarkdownMiddleware, delay_modifiers::{DelayForModifierReleaseMiddleware, ModifierStatusProvider}, markdown::MarkdownMiddleware,
past_discard::PastEventsDiscardMiddleware, past_discard::PastEventsDiscardMiddleware,
}}; }};
use crate::engine::{event::{Event, EventType}, process::middleware::{context_menu::ContextMenuMiddleware, disable::DisableMiddleware, exit::ExitMiddleware, icon_status::IconStatusMiddleware, image_resolve::ImageResolverMiddleware}}; use crate::engine::{event::{Event, EventType}, process::middleware::{context_menu::ContextMenuMiddleware, disable::DisableMiddleware, exit::ExitMiddleware, icon_status::IconStatusMiddleware, image_resolve::ImageResolverMiddleware, search::SearchMiddleware}};
use std::collections::VecDeque; use std::collections::VecDeque;
pub struct DefaultProcessor<'a> { pub struct DefaultProcessor<'a> {
@ -46,6 +46,7 @@ impl<'a> DefaultProcessor<'a> {
path_provider: &'a dyn PathProvider, path_provider: &'a dyn PathProvider,
disable_options: DisableOptions, disable_options: DisableOptions,
matcher_options_provider: &'a dyn MatcherMiddlewareConfigProvider, matcher_options_provider: &'a dyn MatcherMiddlewareConfigProvider,
match_provider: &'a dyn MatchProvider,
) -> DefaultProcessor<'a> { ) -> DefaultProcessor<'a> {
Self { Self {
event_queue: VecDeque::new(), event_queue: VecDeque::new(),
@ -63,6 +64,7 @@ impl<'a> DefaultProcessor<'a> {
Box::new(CursorHintMiddleware::new()), Box::new(CursorHintMiddleware::new()),
Box::new(ExitMiddleware::new()), Box::new(ExitMiddleware::new()),
Box::new(ActionMiddleware::new(match_info_provider, event_sequence_provider)), Box::new(ActionMiddleware::new(match_info_provider, event_sequence_provider)),
Box::new(SearchMiddleware::new(match_provider)),
Box::new(MarkdownMiddleware::new()), Box::new(MarkdownMiddleware::new()),
Box::new(DelayForModifierReleaseMiddleware::new(modifier_status_provider)), Box::new(DelayForModifierReleaseMiddleware::new(modifier_status_provider)),
], ],

View File

@ -32,3 +32,4 @@ pub mod markdown;
pub mod multiplex; pub mod multiplex;
pub mod past_discard; pub mod past_discard;
pub mod render; pub mod render;
pub mod search;

View File

@ -0,0 +1,75 @@
/*
* 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;
use super::super::Middleware;
use crate::engine::{event::{
internal::{DetectedMatch, MatchesDetectedEvent},
Event, EventType,
}};
pub trait MatchProvider {
fn get_all_matches_ids(&self) -> Vec<i32>;
}
pub struct SearchMiddleware<'a> {
match_provider: &'a dyn MatchProvider,
}
impl<'a> SearchMiddleware<'a> {
pub fn new(match_provider: &'a dyn MatchProvider) -> Self {
Self { match_provider }
}
}
impl<'a> Middleware for SearchMiddleware<'a> {
fn name(&self) -> &'static str {
"search"
}
fn next(&self, event: Event, dispatch: &mut dyn FnMut(Event)) -> Event {
if let EventType::ShowSearchBar = event.etype {
let detected_matches = Event::caused_by(
event.source_id,
EventType::MatchesDetected(MatchesDetectedEvent {
matches: self
.match_provider
.get_all_matches_ids()
.into_iter()
.map(|id| DetectedMatch {
id,
trigger: None,
left_separator: None,
right_separator: None,
args: HashMap::new(),
})
.collect(),
}),
);
dispatch(detected_matches);
return Event::caused_by(event.source_id, EventType::NOOP);
}
event
}
}
// TODO: test

View File

@ -97,6 +97,7 @@ pub use middleware::delay_modifiers::ModifierStatusProvider;
pub use middleware::image_resolve::PathProvider; pub use middleware::image_resolve::PathProvider;
pub use middleware::disable::DisableOptions; pub use middleware::disable::DisableOptions;
pub use middleware::matcher::MatcherMiddlewareConfigProvider; pub use middleware::matcher::MatcherMiddlewareConfigProvider;
pub use middleware::search::MatchProvider;
pub fn default<'a, MatcherState>( pub fn default<'a, MatcherState>(
matchers: &'a [&'a dyn Matcher<'a, MatcherState>], matchers: &'a [&'a dyn Matcher<'a, MatcherState>],
@ -110,6 +111,7 @@ pub fn default<'a, MatcherState>(
path_provider: &'a dyn PathProvider, path_provider: &'a dyn PathProvider,
disable_options: DisableOptions, disable_options: DisableOptions,
matcher_options_provider: &'a dyn MatcherMiddlewareConfigProvider, matcher_options_provider: &'a dyn MatcherMiddlewareConfigProvider,
match_provider: &'a dyn MatchProvider,
) -> impl Processor + 'a { ) -> impl Processor + 'a {
default::DefaultProcessor::new( default::DefaultProcessor::new(
matchers, matchers,
@ -123,5 +125,6 @@ pub fn default<'a, MatcherState>(
path_provider, path_provider,
disable_options, disable_options,
matcher_options_provider, matcher_options_provider,
match_provider,
) )
} }

View File

@ -74,6 +74,10 @@ macro_rules! generate_patchable_config {
fn is_match<'b>(&self, app: &AppProperties<'b>) -> bool { fn is_match<'b>(&self, app: &AppProperties<'b>) -> bool {
self.base.is_match(app) self.base.is_match(app)
} }
fn search_trigger(&self) -> Option<String> {
self.base.search_trigger()
}
} }
}; };
} }