Use a nested event loop which polls and saves motion notify events for later

This should speed up resizing/dragging quite a bit, thus fixing ticket #51
This commit is contained in:
Michael Stapelberg 2009-06-21 13:44:44 +02:00
parent 626e6b2b6f
commit a5489d6546

View File

@ -254,9 +254,11 @@ static void drag_pointer(xcb_connection_t *conn, Client *client, xcb_button_pres
/* Go into our own event loop */ /* Go into our own event loop */
xcb_flush(conn); xcb_flush(conn);
xcb_generic_event_t *inside_event; xcb_generic_event_t *inside_event, *last_motion_notify = NULL;
/* Ive always wanted to have my own eventhandler… */ /* Ive always wanted to have my own eventhandler… */
while ((inside_event = xcb_wait_for_event(conn))) { while ((inside_event = xcb_wait_for_event(conn))) {
/* We now handle all events we can get using xcb_poll_for_event */
do {
/* Same as get_event_handler in xcb */ /* Same as get_event_handler in xcb */
int nr = inside_event->response_type; int nr = inside_event->response_type;
if (nr == 0) { if (nr == 0) {
@ -269,16 +271,14 @@ static void drag_pointer(xcb_connection_t *conn, Client *client, xcb_button_pres
nr &= XCB_EVENT_RESPONSE_TYPE_MASK; nr &= XCB_EVENT_RESPONSE_TYPE_MASK;
assert(nr >= 2); assert(nr >= 2);
/* Check if we need to escape this loop */
if (nr == XCB_BUTTON_RELEASE)
break;
switch (nr) { switch (nr) {
case XCB_MOTION_NOTIFY: case XCB_BUTTON_RELEASE:
new_x = ((xcb_motion_notify_event_t*)inside_event)->root_x; goto done;
new_y = ((xcb_motion_notify_event_t*)inside_event)->root_y;
callback(&old_rect, new_x, new_y); case XCB_MOTION_NOTIFY:
/* motion_notify events are saved for later */
FREE(last_motion_notify);
last_motion_notify = inside_event;
break; break;
default: default:
@ -287,9 +287,20 @@ static void drag_pointer(xcb_connection_t *conn, Client *client, xcb_button_pres
xcb_event_handle(&evenths, inside_event); xcb_event_handle(&evenths, inside_event);
break; break;
} }
if (last_motion_notify != inside_event)
free(inside_event); free(inside_event);
} } while ((inside_event = xcb_poll_for_event(conn)) != NULL);
if (last_motion_notify == NULL)
continue;
new_x = ((xcb_motion_notify_event_t*)last_motion_notify)->root_x;
new_y = ((xcb_motion_notify_event_t*)last_motion_notify)->root_y;
callback(&old_rect, new_x, new_y);
FREE(last_motion_notify);
}
done:
xcb_ungrab_pointer(conn, XCB_CURRENT_TIME); xcb_ungrab_pointer(conn, XCB_CURRENT_TIME);
xcb_flush(conn); xcb_flush(conn);
} }