我正在用XCB编写一个程序,该程序需要检测窗口何时获得焦点或失去焦点。到目前为止,我已经有了它,但是它只是挂在xcb_wait_for_event
调用上,从未进入循环。我在这里缺少举办根事件的想法吗?还是我只是要解决这个完全错误的问题,还有比听根源更好的方法?
#include <stdio.h>
#include <stdlib.h>
#include <xcb/xcb.h>
int main (int argc, char **argv)
{
xcb_connection_t* conn = xcb_connect(NULL, NULL);
if (xcb_connection_has_error(conn)) {
printf("Cannot open daemon connection.");
return 0;
}
xcb_screen_t* screen = xcb_setup_roots_iterator(xcb_get_setup(conn)).data;
uint32_t values[] = { XCB_EVENT_MASK_FOCUS_CHANGE };
xcb_change_window_attributes(
conn,
screen->root,
XCB_CW_EVENT_MASK,
values);
xcb_generic_event_t *ev;
while ((ev = xcb_wait_for_event(conn))) {
printf("IN LOOP\n");
switch (ev->response_type & 0x7F) {
case XCB_FOCUS_IN:
case XCB_FOCUS_OUT:
printf("IN CASE\n");
break;
default:
printf("IN DEFAULT\n");
break;
}
free(ev);
}
return 0;
}
答案 0 :(得分:1)
仅当您选择了这些事件的窗口接收焦点或失去焦点时,才会发送焦点事件,请参阅https://www.x.org/releases/X11R7.5/doc/x11proto/proto.html:
重点关注 FocusOut
[...]
这些事件在输入焦点更改时生成,并报告给选择该窗口上的FocusChange的客户端。
要使用此功能,您必须在所有窗口上选择此事件掩码,还要注意是否有新窗口的创建。
我建议采用另一种方法:在根窗口上监视PropertyChangeNotify事件,以查看_NET_ACTIVE_WINDOW
属性何时更改。根据EWMH,此属性应由WM保持最新状态。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <xcb/xcb.h>
static xcb_atom_t intern_atom(xcb_connection_t *conn, const char *atom)
{
xcb_atom_t result = XCB_NONE;
xcb_intern_atom_reply_t *r = xcb_intern_atom_reply(conn,
xcb_intern_atom(conn, 0, strlen(atom), atom), NULL);
if (r)
result = r->atom;
free(r);
return result;
}
int main (int argc, char **argv)
{
xcb_connection_t* conn = xcb_connect(NULL, NULL);
if (xcb_connection_has_error(conn)) {
printf("Cannot open daemon connection.");
return 0;
}
xcb_atom_t active_window = intern_atom(conn, "_NET_ACTIVE_WINDOW");
xcb_screen_t* screen = xcb_setup_roots_iterator(xcb_get_setup(conn)).data;
uint32_t values[] = { XCB_EVENT_MASK_PROPERTY_CHANGE };
xcb_change_window_attributes(
conn,
screen->root,
XCB_CW_EVENT_MASK,
values);
xcb_flush(conn);
xcb_generic_event_t *ev;
while ((ev = xcb_wait_for_event(conn))) {
printf("IN LOOP\n");
switch (ev->response_type & 0x7F) {
case XCB_PROPERTY_NOTIFY: {
xcb_property_notify_event_t *e = (void *) ev;
if (e->atom == active_window)
puts("active window changed");
break;
}
default:
printf("IN DEFAULT\n");
break;
}
free(ev);
}
return 0;
}