refactor(hotkey): use tauri global shortcut

This commit is contained in:
GyDi
2022-09-23 15:31:01 +08:00
committed by GitHub
parent be1f9e66e3
commit 186a922d06
6 changed files with 58 additions and 102 deletions

View File

@@ -1,22 +1,23 @@
use crate::{data::*, feat, log_if_err};
use anyhow::{bail, Result};
use std::collections::HashMap;
use tauri_hotkey::{parse_hotkey, HotkeyManager};
use tauri::{AppHandle, GlobalShortcutManager};
pub struct Hotkey {
current: Vec<String>, // 保存当前的热键设置
manager: HotkeyManager,
manager: Option<AppHandle>,
}
impl Hotkey {
pub fn new() -> Hotkey {
Hotkey {
current: Vec::new(),
manager: HotkeyManager::new(),
manager: None,
}
}
pub fn init(&mut self) -> Result<()> {
pub fn init(&mut self, app_handle: AppHandle) -> Result<()> {
self.manager = Some(app_handle);
let data = Data::global();
let verge = data.verge.lock();
@@ -27,9 +28,9 @@ impl Hotkey {
let key = iter.next();
if func.is_some() && key.is_some() {
log_if_err!(self.register(func.unwrap(), key.unwrap()));
log_if_err!(self.register(key.unwrap(), func.unwrap()));
} else {
log::error!(target: "app", "invalid hotkey \"{}\":\"{}\"", func.unwrap_or("None"), key.unwrap_or("None"));
log::error!(target: "app", "invalid hotkey \"{}\":\"{}\"", key.unwrap_or("None"), func.unwrap_or("None"));
}
}
self.current = hotkeys.clone();
@@ -38,18 +39,25 @@ impl Hotkey {
Ok(())
}
fn register(&mut self, func: &str, key: &str) -> Result<()> {
let hotkey = parse_hotkey(key.trim())?;
fn get_manager(&self) -> Result<impl GlobalShortcutManager> {
if self.manager.is_none() {
bail!("failed to get hotkey manager");
}
Ok(self.manager.as_ref().unwrap().global_shortcut_manager())
}
if self.manager.is_registered(&hotkey) {
self.manager.unregister(&hotkey)?;
fn register(&mut self, hotkey: &str, func: &str) -> Result<()> {
let mut manager = self.get_manager()?;
if manager.is_registered(hotkey)? {
manager.unregister(hotkey)?;
}
let f = match func.trim() {
"clash_mode_rule" => || feat::change_clash_mode("rule"),
"clash_mode_direct" => || feat::change_clash_mode("direct"),
"clash_mode_global" => || feat::change_clash_mode("global"),
"clash_moda_script" => || feat::change_clash_mode("script"),
"clash_mode_direct" => || feat::change_clash_mode("direct"),
"clash_mode_script" => || feat::change_clash_mode("script"),
"toggle_system_proxy" => || feat::toggle_system_proxy(),
"enable_system_proxy" => || feat::enable_system_proxy(),
"disable_system_proxy" => || feat::disable_system_proxy(),
@@ -60,15 +68,14 @@ impl Hotkey {
_ => bail!("invalid function \"{func}\""),
};
self.manager.register(hotkey, f)?;
log::info!(target: "app", "register hotkey {func} {key}");
manager.register(hotkey, f)?;
log::info!(target: "app", "register hotkey {hotkey} {func}");
Ok(())
}
fn unregister(&mut self, key: &str) -> Result<()> {
let hotkey = parse_hotkey(key.trim())?;
self.manager.unregister(&hotkey)?;
log::info!(target: "app", "unregister hotkey {key}");
fn unregister(&mut self, hotkey: &str) -> Result<()> {
self.get_manager()?.unregister(&hotkey)?;
log::info!(target: "app", "unregister hotkey {hotkey}");
Ok(())
}
@@ -77,20 +84,15 @@ impl Hotkey {
let old_map = Self::get_map_from_vec(&current);
let new_map = Self::get_map_from_vec(&new_hotkeys);
for diff in Self::get_diff(old_map, new_map).iter() {
match diff {
Diff::Del(key) => {
let _ = self.unregister(key);
}
Diff::Mod(key, func) => {
let _ = self.unregister(key);
log_if_err!(self.register(func, key));
}
Diff::Add(key, func) => {
log_if_err!(self.register(func, key));
}
}
}
let (del, add) = Self::get_diff(old_map, new_map);
del.iter().for_each(|key| {
let _ = self.unregister(key);
});
add.iter().for_each(|(key, func)| {
log_if_err!(self.register(key, func));
});
self.current = new_hotkeys;
Ok(())
@@ -116,38 +118,36 @@ impl Hotkey {
fn get_diff<'a>(
old_map: HashMap<&'a str, &'a str>,
new_map: HashMap<&'a str, &'a str>,
) -> Vec<Diff<'a>> {
let mut list = vec![];
) -> (Vec<&'a str>, Vec<(&'a str, &'a str)>) {
let mut del_list = vec![];
let mut add_list = vec![];
old_map
.iter()
.for_each(|(key, func)| match new_map.get(key) {
old_map.iter().for_each(|(&key, func)| {
match new_map.get(key) {
Some(new_func) => {
if new_func != func {
list.push(Diff::Mod(key, new_func));
del_list.push(key);
add_list.push((key, *new_func));
}
}
None => list.push(Diff::Del(key)),
});
None => del_list.push(key),
};
});
new_map.iter().for_each(|(key, func)| {
new_map.iter().for_each(|(&key, &func)| {
if old_map.get(key).is_none() {
list.push(Diff::Add(key, func));
add_list.push((key, func));
}
});
list
(del_list, add_list)
}
}
impl Drop for Hotkey {
fn drop(&mut self) {
let _ = self.manager.unregister_all();
if let Ok(mut manager) = self.get_manager() {
let _ = manager.unregister_all();
}
}
}
enum Diff<'a> {
Del(&'a str), // key
Add(&'a str, &'a str), // key, func
Mod(&'a str, &'a str), // key, func
}

View File

@@ -49,7 +49,7 @@ impl Core {
Service::kill_old_clash();
let mut handle = self.handle.lock();
handle.set_inner(app_handle);
handle.set_inner(app_handle.clone());
drop(handle);
let mut service = self.service.lock();
@@ -69,7 +69,7 @@ impl Core {
drop(handle);
let mut hotkey = self.hotkey.lock();
log_if_err!(hotkey.init());
log_if_err!(hotkey.init(app_handle));
drop(hotkey);
// timer initialize