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