Моя цель - иметь программу, которая спит в фоновом режиме, но может быть активирована пользователем через некоторую "горячую клавишу". Из копания вокруг руководства 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;
}