Change drag_pointer return type to int
Patch status: needinfo
Patch by Tony Crisci
To apply this patch, use:
curl http://cr.i3wm.org/patch/233/raw.patch | git am
b/include/floating.h
| 14 |
@@ -138,10 +138,11 @@ void floating_toggle_hide(xcb_connection_t *conn, Workspace *workspace); |
| 15 |
* Every time you move your mouse, an XCB_MOTION_NOTIFY event will be received |
| 16 |
* and the given callback will be called with the parameters specified (client, |
| 17 |
* border on which the click originally was), the original rect of the client, |
| 18 |
- * the event and the new coordinates (x, y). |
| 19 |
+ * the event and the new coordinates (x, y). A return value of 1 indicates the |
| 20 |
+ * user has cancelled the drag and the action should be undone. |
| 21 |
* |
| 22 |
*/ |
| 23 |
-void drag_pointer(Con *con, const xcb_button_press_event_t *event, |
| 24 |
+int drag_pointer(Con *con, const xcb_button_press_event_t *event, |
| 25 |
xcb_window_t confine_to, border_t border, int cursor, |
| 26 |
callback_t callback, const void *extra); |
| 27 |
|
b/src/floating.c
| 32 |
@@ -561,7 +561,7 @@ void floating_resize_window(Con *con, const bool proportional, |
| 33 |
* the event and the new coordinates (x, y). |
| 34 |
* |
| 35 |
*/ |
| 36 |
-void drag_pointer(Con *con, const xcb_button_press_event_t *event, xcb_window_t |
| 37 |
+int drag_pointer(Con *con, const xcb_button_press_event_t *event, xcb_window_t |
| 38 |
confine_to, border_t border, int cursor, callback_t callback, const void *extra) |
| 39 |
{
|
| 40 |
uint32_t new_x, new_y; |
| 41 |
@@ -587,16 +587,38 @@ void drag_pointer(Con *con, const xcb_button_press_event_t *event, xcb_window_t |
| 42 |
|
| 43 |
if ((reply = xcb_grab_pointer_reply(conn, cookie, NULL)) == NULL) {
|
| 44 |
ELOG("Could not grab pointer\n");
|
| 45 |
- return; |
| 46 |
+ return 0; |
| 47 |
} |
| 48 |
|
| 49 |
free(reply); |
| 50 |
|
| 51 |
+ /* Grab the keyboard */ |
| 52 |
+ xcb_grab_keyboard_cookie_t keyb_cookie; |
| 53 |
+ xcb_grab_keyboard_reply_t *keyb_reply; |
| 54 |
+ |
| 55 |
+ keyb_cookie = xcb_grab_keyboard(conn, |
| 56 |
+ false, /* get all keyboard events */ |
| 57 |
+ root, /* grab the root window */ |
| 58 |
+ XCB_CURRENT_TIME, |
| 59 |
+ XCB_GRAB_MODE_ASYNC, /* continue processing pointer events as normal */ |
| 60 |
+ XCB_GRAB_MODE_ASYNC /* keyboard mode */ |
| 61 |
+ ); |
| 62 |
+ |
| 63 |
+ if ((keyb_reply = xcb_grab_keyboard_reply(conn, keyb_cookie, NULL)) == NULL) {
|
| 64 |
+ ELOG("Could not grab keyboard\n");
|
| 65 |
+ return 0; |
| 66 |
+ } |
| 67 |
+ |
| 68 |
+ free(keyb_reply); |
| 69 |
+ |
| 70 |
/* Go into our own event loop */ |
| 71 |
xcb_flush(conn); |
| 72 |
|
| 73 |
xcb_generic_event_t *inside_event, *last_motion_notify = NULL; |
| 74 |
+ xcb_key_press_event_t *inside_keyboard_event = NULL; |
| 75 |
bool loop_done = false; |
| 76 |
+ /* The return value, set to 1 on user cancel */ |
| 77 |
+ int drag_result = 0; |
| 78 |
/* I’ve always wanted to have my own eventhandler… */ |
| 79 |
while (!loop_done && (inside_event = xcb_wait_for_event(conn))) {
|
| 80 |
/* We now handle all events we can get using xcb_poll_for_event */ |
| 81 |
@@ -621,13 +643,25 @@ void drag_pointer(Con *con, const xcb_button_press_event_t *event, xcb_window_t |
| 82 |
break; |
| 83 |
|
| 84 |
case XCB_UNMAP_NOTIFY: |
| 85 |
- case XCB_KEY_PRESS: |
| 86 |
- case XCB_KEY_RELEASE: |
| 87 |
DLOG("Unmap-notify, aborting\n");
|
| 88 |
handle_event(type, inside_event); |
| 89 |
loop_done = true; |
| 90 |
break; |
| 91 |
|
| 92 |
+ case XCB_KEY_PRESS: |
| 93 |
+ case XCB_KEY_RELEASE: |
| 94 |
+ /* Cancel the drag if the ESC key was pressed */ |
| 95 |
+ inside_keyboard_event = (xcb_key_press_event_t *)inside_event; |
| 96 |
+ |
| 97 |
+ if ((int)inside_keyboard_event->detail == 9) {
|
| 98 |
+ DLOG("ESC key was pressed, drag cancelled.");
|
| 99 |
+ loop_done = true; |
| 100 |
+ drag_result = 1; |
| 101 |
+ } |
| 102 |
+ |
| 103 |
+ handle_event(type, inside_event); |
| 104 |
+ break; |
| 105 |
+ |
| 106 |
default: |
| 107 |
DLOG("Passing to original handler\n");
|
| 108 |
/* Use original handler */ |
| 109 |
@@ -648,8 +682,12 @@ void drag_pointer(Con *con, const xcb_button_press_event_t *event, xcb_window_t |
| 110 |
FREE(last_motion_notify); |
| 111 |
} |
| 112 |
|
| 113 |
- xcb_ungrab_pointer(conn, XCB_CURRENT_TIME); |
| 114 |
xcb_flush(conn); |
| 115 |
+ |
| 116 |
+ xcb_ungrab_keyboard(conn, XCB_CURRENT_TIME); |
| 117 |
+ xcb_ungrab_pointer(conn, XCB_CURRENT_TIME); |
| 118 |
+ |
| 119 |
+ return drag_result; |
| 120 |
} |
| 121 |
|
| 122 |
/* |