| OLD | NEW | 
|---|
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be | 
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. | 
| 4 | 4 | 
| 5 #include "base/message_pump_glib_x.h" | 5 #include "base/message_pump_glib_x.h" | 
| 6 | 6 | 
| 7 #include <gdk/gdkx.h> | 7 #include <gdk/gdkx.h> | 
| 8 #if defined(HAVE_XINPUT2) | 8 #if defined(HAVE_XINPUT2) | 
| 9 #include <X11/extensions/XInput2.h> | 9 #include <X11/extensions/XInput2.h> | 
| 10 #else | 10 #else | 
| (...skipping 61 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 72 | 72 | 
| 73 #endif  // HAVE_XINPUT2 | 73 #endif  // HAVE_XINPUT2 | 
| 74 | 74 | 
| 75 }  // namespace | 75 }  // namespace | 
| 76 | 76 | 
| 77 namespace base { | 77 namespace base { | 
| 78 | 78 | 
| 79 MessagePumpGlibX::MessagePumpGlibX() : base::MessagePumpForUI(), | 79 MessagePumpGlibX::MessagePumpGlibX() : base::MessagePumpForUI(), | 
| 80 #if defined(HAVE_XINPUT2) | 80 #if defined(HAVE_XINPUT2) | 
| 81     xiopcode_(-1), | 81     xiopcode_(-1), | 
| 82     masters_(), | 82     pointer_devices_(), | 
| 83     floats_(), |  | 
| 84 #endif | 83 #endif | 
| 85     gdksource_(NULL), | 84     gdksource_(NULL), | 
| 86     dispatching_event_(false), | 85     dispatching_event_(false), | 
| 87     capture_x_events_(0), | 86     capture_x_events_(0), | 
| 88     capture_gdk_events_(0) { | 87     capture_gdk_events_(0) { | 
| 89   gdk_event_handler_set(&EventDispatcherX, this, NULL); | 88   gdk_event_handler_set(&EventDispatcherX, this, NULL); | 
| 90 | 89 | 
| 91 #if defined(HAVE_XINPUT2) | 90 #if defined(HAVE_XINPUT2) | 
| 92   InitializeXInput2(); | 91   InitializeXInput2(); | 
| 93 #endif | 92 #endif | 
| (...skipping 11 matching lines...) Expand all  Loading... | 
| 105   Display* xdisplay = GDK_DISPLAY_XDISPLAY(gdk_display_get_default()); | 104   Display* xdisplay = GDK_DISPLAY_XDISPLAY(gdk_display_get_default()); | 
| 106 | 105 | 
| 107   // Setup mask for mouse events. | 106   // Setup mask for mouse events. | 
| 108   unsigned char mask[(XI_LASTEVENT + 7)/8]; | 107   unsigned char mask[(XI_LASTEVENT + 7)/8]; | 
| 109   memset(mask, 0, sizeof(mask)); | 108   memset(mask, 0, sizeof(mask)); | 
| 110 | 109 | 
| 111   XISetMask(mask, XI_ButtonPress); | 110   XISetMask(mask, XI_ButtonPress); | 
| 112   XISetMask(mask, XI_ButtonRelease); | 111   XISetMask(mask, XI_ButtonRelease); | 
| 113   XISetMask(mask, XI_Motion); | 112   XISetMask(mask, XI_Motion); | 
| 114 | 113 | 
| 115   // It is not necessary to select for slave devices. XInput2 provides enough | 114   XIEventMask evmasks[pointer_devices_.size()]; | 
| 116   // information to the event callback to decide which slave device triggered |  | 
| 117   // the event, thus decide whether the 'pointer event' is a 'mouse event' or a |  | 
| 118   // 'touch event'. |  | 
| 119   // If the touch device has 'GrabDevice' set and 'SendCoreEvents' unset (which |  | 
| 120   // is possible), then the device is detected as a floating device, and a |  | 
| 121   // floating device is not connected to a master device. So it is necessary to |  | 
| 122   // also select on the floating devices. |  | 
| 123   std::set<int> devices; |  | 
| 124   std::set_union(masters_.begin(), masters_.end(), |  | 
| 125                  floats_.begin(), floats_.end(), |  | 
| 126                  std::inserter(devices, devices.begin())); |  | 
| 127   XIEventMask evmasks[devices.size()]; |  | 
| 128   int count = 0; | 115   int count = 0; | 
| 129   for (std::set<int>::const_iterator iter = devices.begin(); | 116   for (std::set<int>::const_iterator iter = pointer_devices_.begin(); | 
| 130        iter != devices.end(); | 117        iter != pointer_devices_.end(); | 
| 131        ++iter, ++count) { | 118        ++iter, ++count) { | 
| 132     evmasks[count].deviceid = *iter; | 119     evmasks[count].deviceid = *iter; | 
| 133     evmasks[count].mask_len = sizeof(mask); | 120     evmasks[count].mask_len = sizeof(mask); | 
| 134     evmasks[count].mask = mask; | 121     evmasks[count].mask = mask; | 
| 135   } | 122   } | 
| 136 | 123 | 
| 137   XISelectEvents(xdisplay, xwindow, evmasks, devices.size()); | 124   XISelectEvents(xdisplay, xwindow, evmasks, pointer_devices_.size()); | 
| 138 | 125 | 
| 139   // TODO(sad): Setup masks for keyboard events. | 126   // TODO(sad): Setup masks for keyboard events. | 
| 140 | 127 | 
| 141   XFlush(xdisplay); | 128   XFlush(xdisplay); | 
| 142 } | 129 } | 
| 143 #endif  // HAVE_XINPUT2 | 130 #endif  // HAVE_XINPUT2 | 
| 144 | 131 | 
| 145 bool MessagePumpGlibX::RunOnce(GMainContext* context, bool block) { | 132 bool MessagePumpGlibX::RunOnce(GMainContext* context, bool block) { | 
| 146   GdkDisplay* gdisp = gdk_display_get_default(); | 133   GdkDisplay* gdisp = gdk_display_get_default(); | 
| 147   if (!gdisp || !GetDispatcher()) | 134   if (!gdisp || !GetDispatcher()) | 
| (...skipping 142 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 290     xiopcode_ = -1; | 277     xiopcode_ = -1; | 
| 291     return; | 278     return; | 
| 292   } | 279   } | 
| 293 | 280 | 
| 294   // TODO(sad): Here, we only setup so that the X windows created by GTK+ are | 281   // TODO(sad): Here, we only setup so that the X windows created by GTK+ are | 
| 295   // setup for XInput2 events. We need a way to listen for XInput2 events for X | 282   // setup for XInput2 events. We need a way to listen for XInput2 events for X | 
| 296   // windows created by other means (e.g. for context menus). | 283   // windows created by other means (e.g. for context menus). | 
| 297   SetupGtkWidgetRealizeNotifier(this); | 284   SetupGtkWidgetRealizeNotifier(this); | 
| 298 | 285 | 
| 299   // Instead of asking X for the list of devices all the time, let's maintain a | 286   // Instead of asking X for the list of devices all the time, let's maintain a | 
| 300   // list of slave (physical) and master (virtual) pointer devices. | 287   // list of pointer devices we care about. | 
|  | 288   // It is not necessary to select for slave devices. XInput2 provides enough | 
|  | 289   // information to the event callback to decide which slave device triggered | 
|  | 290   // the event, thus decide whether the 'pointer event' is a 'mouse event' or a | 
|  | 291   // 'touch event'. | 
|  | 292   // If the touch device has 'GrabDevice' set and 'SendCoreEvents' unset (which | 
|  | 293   // is possible), then the device is detected as a floating device, and a | 
|  | 294   // floating device is not connected to a master device. So it is necessary to | 
|  | 295   // also select on the floating devices. | 
| 301   int count = 0; | 296   int count = 0; | 
| 302   XIDeviceInfo* devices = XIQueryDevice(xdisplay, XIAllDevices, &count); | 297   XIDeviceInfo* devices = XIQueryDevice(xdisplay, XIAllDevices, &count); | 
| 303   for (int i = 0; i < count; i++) { | 298   for (int i = 0; i < count; i++) { | 
| 304     XIDeviceInfo* devinfo = devices + i; | 299     XIDeviceInfo* devinfo = devices + i; | 
| 305     if (devinfo->use == XIFloatingSlave) { | 300     if (devinfo->use == XIFloatingSlave || devinfo->use == XIMasterPointer) | 
| 306       floats_.insert(devinfo->deviceid); | 301       pointer_devices_.insert(devinfo->deviceid); | 
| 307     } else if (devinfo->use == XIMasterPointer) { |  | 
| 308       masters_.insert(devinfo->deviceid); |  | 
| 309     } |  | 
| 310   } | 302   } | 
| 311   XIFreeDeviceInfo(devices); | 303   XIFreeDeviceInfo(devices); | 
| 312 | 304 | 
| 313   // TODO(sad): Select on root for XI_HierarchyChanged so that floats_ and | 305   // TODO(sad): Select on root for XI_HierarchyChanged so that floats_ and | 
| 314   // masters_ can be kept up-to-date. This is a relatively rare event, so we can | 306   // masters_ can be kept up-to-date. This is a relatively rare event, so we can | 
| 315   // put it off for a later time. | 307   // put it off for a later time. | 
| 316   // Note: It is not necessary to listen for XI_DeviceChanged events. | 308   // Note: It is not necessary to listen for XI_DeviceChanged events. | 
| 317 } | 309 } | 
| 318 #endif  // HAVE_XINPUT2 | 310 #endif  // HAVE_XINPUT2 | 
| 319 | 311 | 
| 320 }  // namespace base | 312 }  // namespace base | 
| OLD | NEW | 
|---|