我的目标是让一个程序在后台睡觉,但可以由用户通过一些“热键”激活。从挖掘Xlib手册和Xlib O'reilly手册开始,我认为正确的方法是使用XGrabKey。但是我对这个过程的理解是不正确的,因为简单的概念证明不起作用。
我的理解是,如果我使用根窗口调用XGrabKey作为grab_window,并且owner_events为false,那么只要按下热键,事件就会仅发送到根窗口。如果我然后从根窗口选择KeyPress事件,然后侦听X事件,我应该在按下热键时获得按键事件。我在下面粘贴了一个最小的例子。
我期望的是,当程序运行时,无论哪个窗口有焦点,如果按下Ctrl + Shift + K,我的程序应输出“热键被按下!”在控制台中,然后终止。
此外,我的理解是,如果XGrabKey失败,默认错误处理程序将显示一条消息,因为它不是我假设调用成功。
显然,我的理解在某种程度上是有缺陷的。谁能指出我正确的方向?
#include <iostream>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
using namespace std;
int main()
{
Display* dpy = XOpenDisplay(0);
Window root = DefaultRootWindow(dpy);
XEvent ev;
unsigned int modifiers = ControlMask | ShiftMask;
int keycode = XKeysymToKeycode(dpy,XK_Y);
Window grab_window = root;
Bool owner_events = False;
int pointer_mode = GrabModeAsync;
int keyboard_mode = GrabModeAsync;
XGrabKey(dpy, keycode, modifiers, grab_window, owner_events, pointer_mode,
keyboard_mode);
XSelectInput(dpy, root, KeyPressMask );
while(true)
{
bool shouldQuit = false;
XNextEvent(dpy, &ev);
switch(ev.type)
{
case KeyPress:
cout << "Hot key pressed!" << endl;
XUngrabKey(dpy,keycode,modifiers,grab_window);
shouldQuit = true;
default:
break;
}
if(shouldQuit)
break;
}
XCloseDisplay(dpy);
return 0;
}
答案 0 :(得分:21)
你的程序在这里工作。我的猜测是你有另一个修饰符活跃,比如NumLock。 GrabKey仅适用于精确修饰符掩码。
例如,这是来自metacity窗口管理器的一些(GPL)代码
/* Grab/ungrab, ignoring all annoying modifiers like NumLock etc. */
static void
meta_change_keygrab (MetaDisplay *display,
Window xwindow,
gboolean grab,
int keysym,
unsigned int keycode,
int modmask)
{
unsigned int ignored_mask;
/* Grab keycode/modmask, together with
* all combinations of ignored modifiers.
* X provides no better way to do this.
*/
meta_topic (META_DEBUG_KEYBINDINGS,
"%s keybinding %s keycode %d mask 0x%x on 0x%lx\n",
grab ? "Grabbing" : "Ungrabbing",
keysym_name (keysym), keycode,
modmask, xwindow);
/* efficiency, avoid so many XSync() */
meta_error_trap_push (display);
ignored_mask = 0;
while (ignored_mask <= display->ignored_modifier_mask)
{
if (ignored_mask & ~(display->ignored_modifier_mask))
{
/* Not a combination of ignored modifiers
* (it contains some non-ignored modifiers)
*/
++ignored_mask;
continue;
}
if (meta_is_debugging ())
meta_error_trap_push_with_return (display);
if (grab)
XGrabKey (display->xdisplay, keycode,
modmask | ignored_mask,
xwindow,
True,
GrabModeAsync, GrabModeSync);
else
XUngrabKey (display->xdisplay, keycode,
modmask | ignored_mask,
xwindow);
if (meta_is_debugging ())
{
int result;
result = meta_error_trap_pop_with_return (display, FALSE);
if (grab && result != Success)
{
if (result == BadAccess)
meta_warning (_("Some other program is already using the key %s with modifiers %x as a binding\n"), keysym_name (keysym), modmask | ignored_mask);
else
meta_topic (META_DEBUG_KEYBINDINGS,
"Failed to grab key %s with modifiers %x\n",
keysym_name (keysym), modmask | ignored_mask);
}
}
++ignored_mask;
}
meta_error_trap_pop (display, FALSE);
}
答案 1 :(得分:9)
使用面具ControlMask | ShiftMask
,如果持有另一个修饰键,则无法获得该键。这首先听起来不错,但是存在一个陷阱: NumLock , CapsLock 等等都被视为修饰符。
您有两种选择:
XGrabKey()
,每次为您感兴趣的每个明确组合致电一次。XGrabKey()
致电AnyModifier
并使用event.xkey.state
检查修改符是否符合预期。标题文件<X.h>
定义ShiftMask
,LockMask
,ControlMask
,Mod1Mask
,Mod2Mask
,Mod3Mask
,{{ 1}},Mod4Mask
和Mod5Mask
。
关键是:
AnyModifier
警告我通过尝试找到了Mask | Value | Key
------------+-------+------------
ShiftMask | 1 | Shift
LockMask | 2 | Caps Lock
ControlMask | 4 | Ctrl
Mod1Mask | 8 | Alt
Mod2Mask | 16 | Num Lock
Mod3Mask | 32 | Scroll Lock
Mod4Mask | 64 | Windows
Mod5Mask | 128 | ???
密钥,但我不知道这是否适用于所有计算机/配置/版本/操作系统。
在您的情况下,您可能希望确保ModNMask
已设置,ShiftMask | CtrlMask
已清除,其他则被忽略。
我这样做是为了设置按键抓取:
Mod1Mask | Mod4Mask
这是为了检查是否设置了正确的修饰符:
XGrabKey(dpy, keycode, AnyModifier, grab_window, owner_events, pointer_mode, keyboard_mode);
答案 2 :(得分:8)
如果你在X11上使用/定位gtk,那么有一个C库,它有一个更简单的接口:
https://github.com/engla/keybinder
包括Python,Lua和Vala绑定。 (还提到here。)