From 7e5556e0f247076e8f547fca313d957eeca46366 Mon Sep 17 00:00:00 2001 From: Amr Bashir Date: Sat, 20 Aug 2022 18:58:29 +0200 Subject: [PATCH] fix(linux/globalShortcut): grab the shortcut with extra mods, closes #307 (#540) --- .changes/linux-global-shortcut.md | 5 +++ src/platform_impl/linux/global_shortcut.rs | 36 ++++++++++++++++------ 2 files changed, 31 insertions(+), 10 deletions(-) create mode 100644 .changes/linux-global-shortcut.md diff --git a/.changes/linux-global-shortcut.md b/.changes/linux-global-shortcut.md new file mode 100644 index 000000000..1a50ef057 --- /dev/null +++ b/.changes/linux-global-shortcut.md @@ -0,0 +1,5 @@ +--- +"tao": "patch" +--- + +On Linux, fix global shortcut are never triggered when a Lock key is ON, eg. NumLock, CapsLock. diff --git a/src/platform_impl/linux/global_shortcut.rs b/src/platform_impl/linux/global_shortcut.rs index 52fa3f1a8..f06403321 100644 --- a/src/platform_impl/linux/global_shortcut.rs +++ b/src/platform_impl/linux/global_shortcut.rs @@ -74,19 +74,32 @@ impl ShortcutManager { } } + // XGrabKey works only with the exact state (modifiers) + // and since X11 considers NumLock, ScrollLock and CapsLock a modifier when it is ON, + // we also need to register our shortcut combined with these extra modifiers as well + const IGNORED_MODS: [u32; 4] = [ + 0, // modifier only + xlib::Mod2Mask, // NumLock + xlib::LockMask, // CapsLock + xlib::Mod2Mask | xlib::LockMask, + ]; + match thread_receiver.try_recv() { Ok(HotkeyMessage::RegisterHotkey(_, modifiers, key)) => { let keycode = (xlib.XKeysymToKeycode)(display, key.into()) as i32; - let result = (xlib.XGrabKey)( - display, - keycode, - modifiers, - root, - 0, - xlib::GrabModeAsync, - xlib::GrabModeAsync, - ); + let mut result = 0; + for m in IGNORED_MODS { + result = (xlib.XGrabKey)( + display, + keycode, + modifiers | m, + root, + 0, + xlib::GrabModeAsync, + xlib::GrabModeAsync, + ); + } if result == 0 { if let Err(err) = thread_sender .clone() @@ -107,7 +120,10 @@ impl ShortcutManager { } } Ok(HotkeyMessage::UnregisterHotkey(id)) => { - let result = (xlib.XUngrabKey)(display, id.0, id.1, root); + let mut result = 0; + for m in IGNORED_MODS { + result = (xlib.XUngrabKey)(display, id.0, id.1 | m, root); + } if result == 0 { if let Err(err) = thread_sender .clone()