From 23b074481a24405d785af863d2a7771eb84be017 Mon Sep 17 00:00:00 2001 From: Carlos Garnacho Date: Thu, 19 Jun 2014 23:13:38 +0200 Subject: [PATCH] display: Set an X11 passive touch grab on the root window Touch events will be caught first by the compositor this way, whenever the MetaGestureTracker notifies of the accepted/rejected state of a sequence, XIAllowTouchEvents() will be called on it accordingly, so it is handled exclusively by the compositor or punted to clients. --- src/core/display.c | 58 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 58 insertions(+) diff --git a/src/core/display.c b/src/core/display.c index 07d05b446..3d9d8ee24 100644 --- a/src/core/display.c +++ b/src/core/display.c @@ -51,6 +51,7 @@ #include "meta-cursor-tracker-private.h" #include "meta-backend.h" #include "backends/x11/meta-backend-x11.h" +#include #ifdef HAVE_RANDR #include @@ -413,6 +414,57 @@ meta_set_gnome_wm_keybindings (const char *wm_keybindings) gnome_wm_keybindings = wm_keybindings; } +static void +gesture_tracker_state_changed (MetaGestureTracker *tracker, + ClutterEventSequence *sequence, + MetaSequenceState state, + MetaDisplay *display) +{ + MetaBackendX11 *backend = META_BACKEND_X11 (meta_get_backend ()); + int event_mode; + + if (state == META_SEQUENCE_ACCEPTED) + event_mode = XIAcceptTouch; + else if (state == META_SEQUENCE_REJECTED) + event_mode = XIRejectTouch; + else + return; + + XIAllowTouchEvents (meta_backend_x11_get_xdisplay (backend), + META_VIRTUAL_CORE_POINTER_ID, + clutter_x11_event_sequence_get_touch_detail (sequence), + DefaultRootWindow (display->xdisplay), event_mode); +} + +static void +meta_display_grab_window_touch (MetaDisplay *display, + Window xwindow) +{ + MetaBackendX11 *backend = META_BACKEND_X11 (meta_get_backend ()); + unsigned char mask_bits[XIMaskLen (XI_LASTEVENT)] = { 0 }; + XIEventMask mask = { META_VIRTUAL_CORE_POINTER_ID, sizeof (mask_bits), mask_bits }; + XIGrabModifiers mods = { XIAnyModifier, 0 }; + + XISetMask (mask.mask, XI_TouchBegin); + XISetMask (mask.mask, XI_TouchUpdate); + XISetMask (mask.mask, XI_TouchEnd); + + XIGrabTouchBegin (meta_backend_x11_get_xdisplay (backend), + META_VIRTUAL_CORE_POINTER_ID, + xwindow, False, &mask, 1, &mods); +} + +static void +meta_display_ungrab_window_touch (MetaDisplay *display, + Window xwindow) +{ + MetaBackendX11 *backend = META_BACKEND_X11 (meta_get_backend ()); + XIGrabModifiers mods = { XIAnyModifier, 0 }; + + XIUngrabTouchBegin (meta_backend_x11_get_xdisplay (backend), + META_VIRTUAL_CORE_POINTER_ID, xwindow, 1, &mods); +} + /** * meta_display_open: * @@ -810,6 +862,9 @@ meta_display_open (void) /* Set up touch support */ the_display->gesture_tracker = meta_gesture_tracker_new (0); + g_signal_connect (the_display->gesture_tracker, "state-changed", + G_CALLBACK (gesture_tracker_state_changed), the_display); + meta_display_grab_window_touch (the_display, DefaultRootWindow (the_display->xdisplay)); /* We know that if mutter is running as a Wayland compositor, * we start out with no windows. @@ -980,6 +1035,9 @@ meta_display_close (MetaDisplay *display, meta_display_remove_autoraise_callback (display); + meta_display_ungrab_window_touch (display, DefaultRootWindow (display->xdisplay)); + g_clear_object (&display->gesture_tracker); + if (display->focus_timeout_id) g_source_remove (display->focus_timeout_id); display->focus_timeout_id = 0;