xcb_grab_key没有错误,但事件循环没有捕获(全局热键)

时间:2016-03-10 02:00:36

标签: linux x11 xcb

我正在尝试在Linux上设置全局热键。

我最初使用的是x11(libX11.so)但是我必须从一个线程中执行此操作。我试过了,但XPendingEventXNextEvent最终会导致应用崩溃。

所以我切换到了xcb(libxcb.so.1)。没有错误,我甚至用xcb_request_check检查但事件循环没有提取任何东西。一旦我开始循环,我只得到一个看起来像这样的事件:

{
    response_type: 0,
    pad0: 10,
    sequence: 2,
    pad: [620, 2162688, 0, 0, 0, 0, 0],
    full_sequence: 2
}

这是我的代码,我实际上是在js-ctypes中执行此操作,但是我将所有内容都删除,只显示简单的不可知代码:

conn = xcb_connect(null, null);

keysyms = xcb_key_symbols_alloc(conn);

keycodesPtr = xcb_key_symbols_get_keycode(keysyms, XK_Space);

setup = xcb_get_setup(conn);

screens = xcb_setup_roots_iterator(setup);
screensCnt = screens.rem;

for (var i=0; i<screensCnt; i++) {
    rez_grab = xcb_grab_key(conn, 1, screens.data.root, XCB_MOD_MASK_ANY, keycodesPtr[0], XCB_GRAB_MODE_ASYNC, XCB_GRAB_MODE_ASYNC);

    rez_err = xcb_request_check(conn, rez_grab);
    // rez_err is null

    xcb_screen_next(&screens);
}

xcb_flush(conn);

// start event loop
while (true) {
    ev = xcb_poll_for_event(conn);

    console.log(ev);

    if (ev != null) {
        free(ev);
    }

    Sleep(50);
}

console.log(ev)向我提供了我之前发布的response_type 0之后的ev,然后null只是xcb_void_cookie_t(2)

有谁知道怎么了? rez_grab作为原始字符串是$('.burger').on('click', function(e) { $('.overlay').toggleClass("hidden"); $("body").toggleClass("no-scroll"); e.preventDefault(); });

非常感谢

1 个答案:

答案 0 :(得分:3)

终于搞清楚了!!啊想出来了!我正在使用XCB_MOD_MASK_ANY,这个常量适用于Debian但不适用于Ubuntu,这是我用来测试的。我把它切换到使用num lock,caps lock等,现在它可以工作了! :)

ostypes.API('xcb_grab_key')(conn, 1, screens.data.contents.root, ostypes.CONST.XCB_MOD_MASK_LOCK, keycodesArr[j], ostypes.CONST.XCB_GRAB_MODE_ASYNC, ostypes.CONST.XCB_GRAB_MODE_ASYNC); // caps lock
ostypes.API('xcb_grab_key')(conn, 1, screens.data.contents.root, ostypes.CONST.XCB_MOD_MASK_2, keycodesArr[j], ostypes.CONST.XCB_GRAB_MODE_ASYNC, ostypes.CONST.XCB_GRAB_MODE_ASYNC); // num lock
ostypes.API('xcb_grab_key')(conn, 1, screens.data.contents.root, ostypes.CONST.XCB_MOD_MASK_LOCK | ostypes.CONST.XCB_MOD_MASK_2, keycodesArr[j], ostypes.CONST.XCB_GRAB_MODE_ASYNC, ostypes.CONST.XCB_GRAB_MODE_ASYNC); // caps lock AND num lock

非常非常疯狂我不知道XCB_MOD_MASK_ANY常量在Ubuntu上没有用 -

var rez_grab = ostypes.API('xcb_grab_key')(conn, 1, screens.data.contents.root, ostypes.CONST.XCB_MOD_MASK_ANY, keycodesArr[j], ostypes.CONST.XCB_GRAB_MODE_ASYNC, ostypes.CONST.XCB_GRAB_MODE_ASYNC);

我也试过@n.m的代码。在ubuntu上它没有用,但是在debian上工作 -

#include <xcb/xcb.h>
#define XK_LATIN1
#include <xcb/xcb_keysyms.h>
#include <X11/keysymdef.h>
#include <stdio.h>
#include <stdlib.h>

int
main ()
{
  xcb_connection_t *conn;

  conn = xcb_connect (NULL, NULL);

  xcb_key_symbols_t * keysyms = xcb_key_symbols_alloc(conn);

  xcb_keycode_t * keycodesPtr = xcb_key_symbols_get_keycode(keysyms, XK_space);

  const xcb_setup_t* setup = xcb_get_setup(conn);

  xcb_screen_iterator_t screens = xcb_setup_roots_iterator(setup);
  int screensCnt = screens.rem;

  for (int i=0; i<screensCnt; i++) {

      xcb_void_cookie_t rez_grab = xcb_grab_key(conn, 1, screens.data->root, XCB_MOD_MASK_ANY, keycodesPtr[0], XCB_GRAB_MODE_ASYNC, XCB_GRAB_MODE_ASYNC);

      const static uint32_t values[] = { XCB_EVENT_MASK_EXPOSURE | XCB_EVENT_MASK_BUTTON_PRESS };

      xcb_change_window_attributes (conn, screens.data->root, XCB_CW_EVENT_MASK, values);

      xcb_screen_next(&screens);
  }

  xcb_flush(conn);

  // start event loop
  while (1) {
      xcb_generic_event_t *ev = xcb_wait_for_event(conn);

      if (ev && ((ev->response_type & ~0x80) == XCB_KEY_PRESS))
      {
          xcb_key_press_event_t *kp = (xcb_key_press_event_t *)ev;
          printf ("Got key press %d\n", (int)kp->event);
      }

      if (ev != NULL) {
          free(ev);
      }
  }

  return 0;
}