|
|
@ -20,6 +20,7 @@ |
|
|
|
#include <assert.h> |
|
|
|
#include <err.h> |
|
|
|
#include <time.h> |
|
|
|
#include <sys/time.h> |
|
|
|
|
|
|
|
#include "cursors.h" |
|
|
|
#include "unlock_indicator.h" |
|
|
@ -189,21 +190,27 @@ xcb_window_t open_fullscreen_window(xcb_connection_t *conn, xcb_screen_t *scr, c |
|
|
|
} |
|
|
|
|
|
|
|
/*
|
|
|
|
* Repeatedly tries to grab pointer and keyboard (up to 10000 times). |
|
|
|
* Repeatedly tries to grab pointer and keyboard (up to the specified number of |
|
|
|
* tries). |
|
|
|
* |
|
|
|
* Returns true if the grab succeeded, false if not. |
|
|
|
* |
|
|
|
*/ |
|
|
|
void grab_pointer_and_keyboard(xcb_connection_t *conn, xcb_screen_t *screen, xcb_cursor_t cursor) { |
|
|
|
bool grab_pointer_and_keyboard(xcb_connection_t *conn, xcb_screen_t *screen, xcb_cursor_t cursor, int tries) { |
|
|
|
xcb_grab_pointer_cookie_t pcookie; |
|
|
|
xcb_grab_pointer_reply_t *preply; |
|
|
|
|
|
|
|
xcb_grab_keyboard_cookie_t kcookie; |
|
|
|
xcb_grab_keyboard_reply_t *kreply; |
|
|
|
|
|
|
|
int tries = 10000; |
|
|
|
const suseconds_t screen_redraw_timeout = 100000; /* 100ms */ |
|
|
|
|
|
|
|
/* Using few variables to trigger a redraw_screen() if too many tries */ |
|
|
|
bool redrawn = false; |
|
|
|
time_t start = clock(); |
|
|
|
struct timeval start; |
|
|
|
if (gettimeofday(&start, NULL) == -1) { |
|
|
|
err(EXIT_FAILURE, "gettimeofday"); |
|
|
|
} |
|
|
|
|
|
|
|
while (tries-- > 0) { |
|
|
|
pcookie = xcb_grab_pointer( |
|
|
@ -226,10 +233,17 @@ void grab_pointer_and_keyboard(xcb_connection_t *conn, xcb_screen_t *screen, xcb |
|
|
|
/* Make this quite a bit slower */ |
|
|
|
usleep(50); |
|
|
|
|
|
|
|
/* Measure elapsed time and trigger a screen redraw if elapsed > 250000 */ |
|
|
|
struct timeval now; |
|
|
|
if (gettimeofday(&now, NULL) == -1) { |
|
|
|
err(EXIT_FAILURE, "gettimeofday"); |
|
|
|
} |
|
|
|
|
|
|
|
struct timeval elapsed; |
|
|
|
timersub(&now, &start, &elapsed); |
|
|
|
|
|
|
|
if (!redrawn && |
|
|
|
(tries % 100) == 0 && |
|
|
|
(clock() - start) > 250000) { |
|
|
|
elapsed.tv_usec >= screen_redraw_timeout) { |
|
|
|
redraw_screen(); |
|
|
|
redrawn = true; |
|
|
|
} |
|
|
@ -253,23 +267,24 @@ void grab_pointer_and_keyboard(xcb_connection_t *conn, xcb_screen_t *screen, xcb |
|
|
|
/* Make this quite a bit slower */ |
|
|
|
usleep(50); |
|
|
|
|
|
|
|
/* Measure elapsed time and trigger a screen redraw if elapsed > 250000 */ |
|
|
|
struct timeval now; |
|
|
|
if (gettimeofday(&now, NULL) == -1) { |
|
|
|
err(EXIT_FAILURE, "gettimeofday"); |
|
|
|
} |
|
|
|
|
|
|
|
struct timeval elapsed; |
|
|
|
timersub(&now, &start, &elapsed); |
|
|
|
|
|
|
|
/* Trigger a screen redraw if 100ms elapsed */ |
|
|
|
if (!redrawn && |
|
|
|
(tries % 100) == 0 && |
|
|
|
(clock() - start) > 250000) { |
|
|
|
elapsed.tv_usec >= screen_redraw_timeout) { |
|
|
|
redraw_screen(); |
|
|
|
redrawn = true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
/* After trying for 10000 times, i3lock will display an error message
|
|
|
|
* for 2 sec prior to terminate. */ |
|
|
|
if (tries <= 0) { |
|
|
|
auth_state = STATE_I3LOCK_LOCK_FAILED; |
|
|
|
redraw_screen(); |
|
|
|
sleep(1); |
|
|
|
errx(EXIT_FAILURE, "Cannot grab pointer/keyboard"); |
|
|
|
} |
|
|
|
return (tries > 0); |
|
|
|
} |
|
|
|
|
|
|
|
xcb_cursor_t create_cursor(xcb_connection_t *conn, xcb_screen_t *screen, xcb_window_t win, int choice) { |
|
|
@ -335,6 +350,70 @@ xcb_cursor_t create_cursor(xcb_connection_t *conn, xcb_screen_t *screen, xcb_win |
|
|
|
return cursor; |
|
|
|
} |
|
|
|
|
|
|
|
static xcb_atom_t _NET_ACTIVE_WINDOW = XCB_NONE; |
|
|
|
void _init_net_active_window(xcb_connection_t *conn) { |
|
|
|
if (_NET_ACTIVE_WINDOW != XCB_NONE) { |
|
|
|
/* already initialized */ |
|
|
|
return; |
|
|
|
} |
|
|
|
xcb_generic_error_t *err; |
|
|
|
xcb_intern_atom_reply_t *atom_reply = xcb_intern_atom_reply( |
|
|
|
conn, |
|
|
|
xcb_intern_atom(conn, 0, strlen("_NET_ACTIVE_WINDOW"), "_NET_ACTIVE_WINDOW"), |
|
|
|
&err); |
|
|
|
if (atom_reply == NULL) { |
|
|
|
fprintf(stderr, "X11 Error %d\n", err->error_code); |
|
|
|
free(err); |
|
|
|
return; |
|
|
|
} |
|
|
|
_NET_ACTIVE_WINDOW = atom_reply->atom; |
|
|
|
free(atom_reply); |
|
|
|
} |
|
|
|
|
|
|
|
xcb_window_t find_focused_window(xcb_connection_t *conn, const xcb_window_t root) { |
|
|
|
xcb_window_t result = XCB_NONE; |
|
|
|
|
|
|
|
_init_net_active_window(conn); |
|
|
|
|
|
|
|
xcb_get_property_reply_t *prop_reply = xcb_get_property_reply( |
|
|
|
conn, |
|
|
|
xcb_get_property_unchecked( |
|
|
|
conn, false, root, _NET_ACTIVE_WINDOW, XCB_GET_PROPERTY_TYPE_ANY, 0, 1 /* word */), |
|
|
|
NULL); |
|
|
|
if (prop_reply == NULL) { |
|
|
|
goto out; |
|
|
|
} |
|
|
|
if (xcb_get_property_value_length(prop_reply) == 0) { |
|
|
|
goto out_prop; |
|
|
|
} |
|
|
|
if (prop_reply->type != XCB_ATOM_WINDOW) { |
|
|
|
goto out_prop; |
|
|
|
} |
|
|
|
|
|
|
|
result = *((xcb_window_t *)xcb_get_property_value(prop_reply)); |
|
|
|
|
|
|
|
out_prop: |
|
|
|
free(prop_reply); |
|
|
|
out: |
|
|
|
return result; |
|
|
|
} |
|
|
|
|
|
|
|
void set_focused_window(xcb_connection_t *conn, const xcb_window_t root, const xcb_window_t window) { |
|
|
|
xcb_client_message_event_t ev; |
|
|
|
memset(&ev, '\0', sizeof(xcb_client_message_event_t)); |
|
|
|
|
|
|
|
_init_net_active_window(conn); |
|
|
|
|
|
|
|
ev.response_type = XCB_CLIENT_MESSAGE; |
|
|
|
ev.window = window; |
|
|
|
ev.type = _NET_ACTIVE_WINDOW; |
|
|
|
ev.format = 32; |
|
|
|
ev.data.data32[0] = 2; /* 2 = pager */ |
|
|
|
|
|
|
|
xcb_send_event(conn, false, root, XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT, (char *)&ev); |
|
|
|
xcb_flush(conn); |
|
|
|
} |
|
|
|
|
|
|
|
xcb_pixmap_t capture_bg_pixmap(xcb_connection_t *conn, xcb_screen_t *scr, u_int32_t * resolution) { |
|
|
|
xcb_pixmap_t bg_pixmap = xcb_generate_id(conn); |
|
|
|
xcb_create_pixmap(conn, scr->root_depth, bg_pixmap, scr->root, resolution[0], resolution[1]); |
|
|
|