mirror of
https://github.com/0PandaDEV/Qopy.git
synced 2025-04-21 13:14:04 +02:00
added ability to set custom hotkey
This commit is contained in:
parent
79dd783bf7
commit
5bb6f2249a
8 changed files with 250 additions and 205 deletions
Binary file not shown.
|
@ -121,8 +121,7 @@ pub async fn save_keybind(
|
|||
keybind: Vec<String>,
|
||||
pool: State<'_, SqlitePool>,
|
||||
) -> Result<(), String> {
|
||||
let setting = KeybindSetting { keybind };
|
||||
let json = serde_json::to_string(&setting).map_err(|e| e.to_string())?;
|
||||
let json = serde_json::to_string(&keybind).map_err(|e| e.to_string())?;
|
||||
|
||||
sqlx::query(
|
||||
"INSERT OR REPLACE INTO settings (key, value) VALUES ('keybind', ?)"
|
||||
|
|
|
@ -1,123 +1,47 @@
|
|||
use rdev::{listen, Event, EventType, Key};
|
||||
use tauri::{Manager, Emitter};
|
||||
use std::sync::atomic::{AtomicBool, Ordering};
|
||||
use std::sync::{Arc, Mutex};
|
||||
use std::collections::HashSet;
|
||||
use serde::Serialize;
|
||||
|
||||
use crate::api::database::get_keybind;
|
||||
use crate::utils::commands::center_window_on_current_monitor;
|
||||
use rdev::{listen, EventType, Key};
|
||||
use tauri::Manager;
|
||||
|
||||
static IS_CAPTURING_KEYBIND: AtomicBool = AtomicBool::new(false);
|
||||
|
||||
#[derive(Debug, Clone, Serialize)]
|
||||
struct CapturedKeybind {
|
||||
modifiers: Vec<String>,
|
||||
key: String,
|
||||
}
|
||||
|
||||
struct KeybindState {
|
||||
pressed_keys: HashSet<Key>,
|
||||
}
|
||||
|
||||
impl KeybindState {
|
||||
fn new() -> Self {
|
||||
Self {
|
||||
pressed_keys: HashSet::new(),
|
||||
}
|
||||
}
|
||||
fn key_to_string(key: &Key) -> String {
|
||||
format!("{:?}", key)
|
||||
}
|
||||
|
||||
#[warn(dead_code)]
|
||||
pub fn setup(app_handle: tauri::AppHandle) {
|
||||
let app_handle_clone = app_handle.clone();
|
||||
let keybind_state = Arc::new(Mutex::new(KeybindState::new()));
|
||||
|
||||
std::thread::spawn(move || {
|
||||
if let Err(e) = listen(move |event| {
|
||||
let mut state = keybind_state.lock().unwrap();
|
||||
if IS_CAPTURING_KEYBIND.load(Ordering::SeqCst) {
|
||||
handle_keybind_capture(&app_handle_clone, event, &mut state);
|
||||
} else {
|
||||
handle_normal_hotkey(&app_handle_clone, event, &mut state);
|
||||
}
|
||||
}) {
|
||||
eprintln!("Error setting up event listener: {:?}", e);
|
||||
}
|
||||
});
|
||||
}
|
||||
let pool = app_handle.state::<sqlx::SqlitePool>();
|
||||
let rt = app_handle.state::<tokio::runtime::Runtime>();
|
||||
|
||||
fn handle_normal_hotkey(app_handle: &tauri::AppHandle, event: Event, state: &mut KeybindState) {
|
||||
match event.event_type {
|
||||
EventType::KeyPress(Key::MetaLeft) | EventType::KeyPress(Key::MetaRight) => {
|
||||
state.pressed_keys.insert(Key::MetaLeft);
|
||||
}
|
||||
EventType::KeyRelease(Key::MetaLeft) | EventType::KeyRelease(Key::MetaRight) => {
|
||||
state.pressed_keys.remove(&Key::MetaLeft);
|
||||
}
|
||||
EventType::KeyPress(Key::KeyV) => {
|
||||
if state.pressed_keys.contains(&Key::MetaLeft) {
|
||||
state.pressed_keys.clear();
|
||||
if let Some(window) = app_handle.get_webview_window("main") {
|
||||
let _ = window.show();
|
||||
let _ = window.set_focus();
|
||||
center_window_on_current_monitor(&window);
|
||||
let keybind = rt.block_on(async { get_keybind(pool).await.unwrap_or_default() });
|
||||
|
||||
println!("Listening for keybind: {:?}", keybind);
|
||||
|
||||
let mut pressed_keys = vec![false; keybind.len()];
|
||||
|
||||
listen(move |event| {
|
||||
match event.event_type {
|
||||
EventType::KeyPress(key) => {
|
||||
if let Some(index) = keybind.iter().position(|k| k == &key_to_string(&key)) {
|
||||
pressed_keys[index] = true;
|
||||
}
|
||||
}
|
||||
EventType::KeyRelease(key) => {
|
||||
if let Some(index) = keybind.iter().position(|k| k == &key_to_string(&key)) {
|
||||
pressed_keys[index] = false;
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
fn handle_keybind_capture(app_handle: &tauri::AppHandle, event: Event, state: &mut KeybindState) {
|
||||
match event.event_type {
|
||||
EventType::KeyPress(key) => {
|
||||
state.pressed_keys.insert(key);
|
||||
update_captured_keybind(app_handle, &state.pressed_keys);
|
||||
}
|
||||
EventType::KeyRelease(key) => {
|
||||
state.pressed_keys.remove(&key);
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
fn update_captured_keybind(app_handle: &tauri::AppHandle, pressed_keys: &HashSet<Key>) {
|
||||
let modifiers: Vec<String> = vec![Key::ControlLeft, Key::ShiftLeft, Key::Alt, Key::MetaLeft]
|
||||
.into_iter()
|
||||
.filter(|key| pressed_keys.contains(key))
|
||||
.map(|key| key_to_string(key))
|
||||
.collect();
|
||||
|
||||
let key = pressed_keys.iter()
|
||||
.find(|&&key| !vec![Key::ControlLeft, Key::ShiftLeft, Key::Alt, Key::MetaLeft].contains(&key))
|
||||
.map(|&key| key_to_string(key));
|
||||
|
||||
if let Some(key) = key {
|
||||
let captured_keybind = CapturedKeybind {
|
||||
modifiers,
|
||||
key,
|
||||
};
|
||||
if let Err(e) = app_handle.emit("keybind_captured", captured_keybind) {
|
||||
eprintln!("Error emitting keybind_captured event: {:?}", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn key_to_string(key: Key) -> String {
|
||||
match key {
|
||||
Key::ControlLeft | Key::ControlRight => "Ctrl".to_string(),
|
||||
Key::ShiftLeft | Key::ShiftRight => "Shift".to_string(),
|
||||
Key::Alt => "Alt".to_string(),
|
||||
Key::MetaLeft | Key::MetaRight => "Meta".to_string(),
|
||||
_ => format!("{:?}", key),
|
||||
}
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
pub fn start_keybind_capture() {
|
||||
IS_CAPTURING_KEYBIND.store(true, Ordering::SeqCst);
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
pub fn stop_keybind_capture() {
|
||||
IS_CAPTURING_KEYBIND.store(false, Ordering::SeqCst);
|
||||
if pressed_keys.iter().all(|&k| k) {
|
||||
pressed_keys.iter_mut().for_each(|k| *k = false);
|
||||
let window = app_handle.get_webview_window("main").unwrap();
|
||||
window.show().unwrap();
|
||||
window.set_focus().unwrap();
|
||||
center_window_on_current_monitor(&window);
|
||||
}
|
||||
})
|
||||
.unwrap();
|
||||
});
|
||||
}
|
|
@ -31,7 +31,7 @@ fn main() {
|
|||
.setup(|app| {
|
||||
let app_handle = app.handle().clone();
|
||||
|
||||
#[cfg(not(target_os = "macos"))]
|
||||
// #[cfg(not(target_os = "macos"))]
|
||||
api::hotkeys::setup(app_handle.clone());
|
||||
api::tray::setup(app)?;
|
||||
let _ = api::database::setup(app);
|
||||
|
@ -74,8 +74,6 @@ fn main() {
|
|||
api::clipboard::get_image_path,
|
||||
api::clipboard::write_and_paste,
|
||||
api::clipboard::read_image,
|
||||
api::hotkeys::start_keybind_capture,
|
||||
api::hotkeys::stop_keybind_capture,
|
||||
api::database::save_keybind,
|
||||
api::database::get_keybind
|
||||
])
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue