1 /* GDK - The GIMP Drawing Kit
2 * Copyright (C) 2009 Carlos Garnacho <carlosg@gnome.org>
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library. If not, see <http://www.gnu.org/licenses/>.
20 #include "gdkx11device-xi.h"
22 #include "gdkdeviceprivate-xi.h"
26 #include "gdkwindow.h"
28 #include "gdkprivate-x11.h"
33 G_DEFINE_TYPE (GdkX11DeviceXI, gdk_x11_device_xi, GDK_TYPE_DEVICE)
37 #define MAX_DEVICE_CLASSES 13
39 static GQuark quark_window_input_info = 0;
47 static void gdk_x11_device_xi_constructed (GObject *object);
48 static void gdk_x11_device_xi_dispose (GObject *object);
50 static void gdk_x11_device_xi_set_property (GObject *object,
54 static void gdk_x11_device_xi_get_property (GObject *object,
59 static gboolean gdk_x11_device_xi_get_history (GdkDevice *device,
63 GdkTimeCoord ***events,
66 static void gdk_x11_device_xi_get_state (GdkDevice *device,
69 GdkModifierType *mask);
70 static void gdk_x11_device_xi_set_window_cursor (GdkDevice *device,
73 static void gdk_x11_device_xi_warp (GdkDevice *device,
77 static gboolean gdk_x11_device_xi_query_state (GdkDevice *device,
79 GdkWindow **root_window,
80 GdkWindow **child_window,
85 GdkModifierType *mask);
86 static GdkGrabStatus gdk_x11_device_xi_grab (GdkDevice *device,
88 gboolean owner_events,
89 GdkEventMask event_mask,
90 GdkWindow *confine_to,
93 static void gdk_x11_device_xi_ungrab (GdkDevice *device,
96 static GdkWindow* gdk_x11_device_xi_window_at_position (GdkDevice *device,
99 GdkModifierType *mask,
100 gboolean get_toplevel);
102 static void gdk_x11_device_xi_select_window_events (GdkDevice *device,
113 gdk_x11_device_xi_class_init (GdkX11DeviceXIClass *klass)
115 GObjectClass *object_class = G_OBJECT_CLASS (klass);
116 GdkDeviceClass *device_class = GDK_DEVICE_CLASS (klass);
118 quark_window_input_info = g_quark_from_static_string ("gdk-window-input-info");
120 object_class->constructed = gdk_x11_device_xi_constructed;
121 object_class->set_property = gdk_x11_device_xi_set_property;
122 object_class->get_property = gdk_x11_device_xi_get_property;
123 object_class->dispose = gdk_x11_device_xi_dispose;
125 device_class->get_history = gdk_x11_device_xi_get_history;
126 device_class->get_state = gdk_x11_device_xi_get_state;
127 device_class->set_window_cursor = gdk_x11_device_xi_set_window_cursor;
128 device_class->warp = gdk_x11_device_xi_warp;
129 device_class->query_state = gdk_x11_device_xi_query_state;
130 device_class->grab = gdk_x11_device_xi_grab;
131 device_class->ungrab = gdk_x11_device_xi_ungrab;
132 device_class->window_at_position = gdk_x11_device_xi_window_at_position;
133 device_class->select_window_events = gdk_x11_device_xi_select_window_events;
135 g_object_class_install_property (object_class,
137 g_param_spec_int ("device-id",
141 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
145 gdk_x11_device_xi_init (GdkX11DeviceXI *device)
150 gdk_x11_device_xi_constructed (GObject *object)
152 GdkX11DeviceXI *device;
155 device = GDK_X11_DEVICE_XI (object);
156 display = gdk_device_get_display (GDK_DEVICE (object));
158 gdk_x11_display_error_trap_push (display);
159 device->xdevice = XOpenDevice (GDK_DISPLAY_XDISPLAY (display),
162 if (gdk_x11_display_error_trap_pop (display))
163 g_warning ("Device %s can't be opened",
164 gdk_device_get_name (GDK_DEVICE (device)));
166 if (G_OBJECT_CLASS (gdk_x11_device_xi_parent_class)->constructed)
167 G_OBJECT_CLASS (gdk_x11_device_xi_parent_class)->constructed (object);
171 gdk_x11_device_xi_set_property (GObject *object,
176 GdkX11DeviceXI *device = GDK_X11_DEVICE_XI (object);
181 device->device_id = g_value_get_int (value);
184 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
190 gdk_x11_device_xi_get_property (GObject *object,
195 GdkX11DeviceXI *device = GDK_X11_DEVICE_XI (object);
200 g_value_set_int (value, device->device_id);
203 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
209 gdk_x11_device_xi_dispose (GObject *object)
211 GdkX11DeviceXI *device_xi;
214 device_xi = GDK_X11_DEVICE_XI (object);
215 display = gdk_device_get_display (GDK_DEVICE (device_xi));
217 if (device_xi->xdevice)
219 XCloseDevice (GDK_DISPLAY_XDISPLAY (display), device_xi->xdevice);
220 device_xi->xdevice = NULL;
223 if (device_xi->axis_data)
225 g_free (device_xi->axis_data);
226 device_xi->axis_data = NULL;
229 G_OBJECT_CLASS (gdk_x11_device_xi_parent_class)->dispose (object);
233 gdk_x11_device_xi_get_history (GdkDevice *device,
237 GdkTimeCoord ***events,
240 GdkTimeCoord **coords;
241 XDeviceTimeCoord *device_coords;
242 GdkWindow *impl_window;
243 GdkX11DeviceXI *device_xi;
244 gint n_events_return;
246 gint axis_count_return;
249 device_xi = GDK_X11_DEVICE_XI (device);
250 impl_window = _gdk_window_get_impl_window (window);
252 device_coords = XGetDeviceMotionEvents (GDK_WINDOW_XDISPLAY (impl_window),
262 *n_events = n_events_return;
263 coords = _gdk_device_allocate_history (device, *n_events);
265 for (i = 0; i < *n_events; i++)
267 coords[i]->time = device_coords[i].time;
268 _gdk_x11_device_xi_translate_axes (device, window,
269 device_coords[i].data,
274 XFreeDeviceMotionEvents (device_coords);
282 gdk_x11_device_xi_get_state (GdkDevice *device,
285 GdkModifierType *mask)
287 GdkX11DeviceXI *device_xi;
289 XInputClass *input_class;
293 gdk_window_get_device_position (window, device, NULL, NULL, mask);
295 device_xi = GDK_X11_DEVICE_XI (device);
296 state = XQueryDeviceState (GDK_WINDOW_XDISPLAY (window),
298 input_class = state->data;
300 for (i = 0; i < state->num_classes; i++)
302 switch (input_class->class)
306 _gdk_x11_device_xi_translate_axes (device, window,
307 ((XValuatorState *) input_class)->valuators,
315 if (((XButtonState *)input_class)->num_buttons > 0)
316 *mask |= ((XButtonState *)input_class)->buttons[0] << 7;
317 /* GDK_BUTTON1_MASK = 1 << 8, and button n is stored
318 * in bit 1<<(n%8) in byte n/8. n = 1,2,... */
323 input_class = (XInputClass *)(((char *)input_class)+input_class->length);
326 XFreeDeviceState (state);
330 gdk_x11_device_xi_set_window_cursor (GdkDevice *device,
337 gdk_x11_device_xi_warp (GdkDevice *device,
345 find_events (GdkDevice *device,
347 XEventClass *classes,
350 GdkX11DeviceXI *device_xi;
354 device_xi = GDK_X11_DEVICE_XI (device);
357 if (mask & GDK_BUTTON_PRESS_MASK)
359 DeviceButtonPress (device_xi->xdevice, device_xi->button_press_type, class);
361 classes[i++] = class;
363 DeviceButtonPressGrab (device_xi->xdevice, 0, class);
365 classes[i++] = class;
368 if (mask & GDK_BUTTON_RELEASE_MASK)
370 DeviceButtonRelease (device_xi->xdevice, device_xi->button_release_type, class);
372 classes[i++] = class;
375 if (mask & (GDK_POINTER_MOTION_MASK |
376 GDK_BUTTON1_MOTION_MASK | GDK_BUTTON2_MOTION_MASK |
377 GDK_BUTTON3_MOTION_MASK | GDK_BUTTON_MOTION_MASK))
379 /* Make sure device->motionnotify_type is set */
380 DeviceMotionNotify (device_xi->xdevice, device_xi->motion_notify_type, class);
382 classes[i++] = class;
383 DeviceStateNotify (device_xi->xdevice, device_xi->state_notify_type, class);
385 classes[i++] = class;
388 if (mask & GDK_KEY_PRESS_MASK)
390 DeviceKeyPress (device_xi->xdevice, device_xi->key_press_type, class);
392 classes[i++] = class;
395 if (mask & GDK_KEY_RELEASE_MASK)
397 DeviceKeyRelease (device_xi->xdevice, device_xi->key_release_type, class);
399 classes[i++] = class;
402 if (mask & GDK_PROXIMITY_IN_MASK)
404 ProximityIn (device_xi->xdevice, device_xi->proximity_in_type, class);
406 classes[i++] = class;
409 if (mask & GDK_PROXIMITY_OUT_MASK)
411 ProximityOut (device_xi->xdevice, device_xi->proximity_out_type, class);
413 classes[i++] = class;
420 gdk_x11_device_xi_query_state (GdkDevice *device,
422 GdkWindow **root_window,
423 GdkWindow **child_window,
428 GdkModifierType *mask)
434 gdk_x11_device_xi_grab (GdkDevice *device,
436 gboolean owner_events,
437 GdkEventMask event_mask,
438 GdkWindow *confine_to,
442 XEventClass event_classes[MAX_DEVICE_CLASSES];
443 gint status, num_classes;
444 GdkX11DeviceXI *device_xi;
447 device_xi = GDK_X11_DEVICE_XI (device);
448 display = gdk_device_get_display (device);
449 find_events (device, event_mask, event_classes, &num_classes);
451 #ifdef G_ENABLE_DEBUG
452 if (_gdk_debug_flags & GDK_DEBUG_NOGRABS)
453 status = GrabSuccess;
456 status = XGrabDevice (GDK_DISPLAY_XDISPLAY (display),
458 GDK_WINDOW_XID (window),
460 num_classes, event_classes,
461 GrabModeAsync, GrabModeAsync,
464 _gdk_x11_display_update_grab_info (display, device, status);
466 return _gdk_x11_convert_grab_status (status);
470 gdk_x11_device_xi_ungrab (GdkDevice *device,
473 GdkX11DeviceXI *device_xi;
476 unsigned long serial;
478 device_xi = GDK_X11_DEVICE_XI (device);
479 display = gdk_device_get_display (device);
480 xdisplay = GDK_DISPLAY_XDISPLAY (display);
482 serial = NextRequest (xdisplay);
484 XUngrabDevice (xdisplay, device_xi->xdevice, time_);
486 _gdk_x11_display_update_grab_info_ungrab (display, device, time_, serial);
490 gdk_x11_device_xi_window_at_position (GdkDevice *device,
493 GdkModifierType *mask,
494 gboolean get_toplevel)
500 gdk_x11_device_xi_select_window_events (GdkDevice *device,
502 GdkEventMask event_mask)
504 XEventClass event_classes[MAX_DEVICE_CLASSES];
507 event_mask |= (GDK_PROXIMITY_IN_MASK |
508 GDK_PROXIMITY_OUT_MASK);
510 find_events (device, event_mask, event_classes, &num_classes);
512 XSelectExtensionEvent (GDK_WINDOW_XDISPLAY (window),
513 GDK_WINDOW_XID (window),
514 event_classes, num_classes);
518 GdkWindowInputInfo *info;
520 info = g_new0 (GdkWindowInputInfo, 1);
521 g_object_set_qdata_full (G_OBJECT (window),
522 quark_window_input_info,
524 (GDestroyNotify) g_free);
527 g_object_set_qdata (G_OBJECT (window),
528 quark_window_input_info,
533 _gdk_x11_device_xi_update_window_info (GdkWindow *window)
535 GdkWindowInputInfo *info;
538 info = g_object_get_qdata (G_OBJECT (window),
539 quark_window_input_info);
544 gdk_window_get_origin (window, &root_x, &root_y);
545 info->root_x = (gdouble) root_x;
546 info->root_y = (gdouble) root_y;
550 gdk_x11_device_xi_get_window_info (GdkWindow *window,
554 GdkWindowInputInfo *info;
556 info = g_object_get_qdata (G_OBJECT (window),
557 quark_window_input_info);
562 *root_x = info->root_x;
563 *root_y = info->root_y;
569 _gdk_x11_device_xi_update_axes (GdkDevice *device,
574 GdkX11DeviceXI *device_xi;
577 device_xi = GDK_X11_DEVICE_XI (device);
578 g_return_if_fail (first_axis >= 0 &&
579 first_axis + axes_count <= gdk_device_get_n_axes (device));
581 if (!device_xi->axis_data)
582 device_xi->axis_data = g_new0 (gint, gdk_device_get_n_axes (device));
584 for (i = 0; i < axes_count; i++)
585 device_xi->axis_data[first_axis + i] = axis_data[i];
589 _gdk_x11_device_xi_translate_axes (GdkDevice *device,
596 GdkWindow *impl_window;
597 gdouble root_x, root_y;
598 gdouble temp_x, temp_y;
602 impl_window = _gdk_window_get_impl_window (window);
605 if (!gdk_x11_device_xi_get_window_info (impl_window, &root_x, &root_y))
608 n_axes = gdk_device_get_n_axes (device);
610 for (i = 0; i < n_axes; i++)
614 use = gdk_device_get_axis_use (device, i);
620 if (gdk_device_get_mode (device) == GDK_MODE_WINDOW)
621 _gdk_device_translate_window_coord (device, window,
625 _gdk_device_translate_screen_coord (device, window,
629 if (use == GDK_AXIS_X)
631 else if (use == GDK_AXIS_Y)
636 _gdk_device_translate_axis (device, i, axis_data[i], &axes[i]);
648 #else /* XINPUT_XFREE */
651 gdk_x11_device_xi_class_init (GdkX11DeviceXIClass *klass)
656 gdk_x11_device_xi_init (GdkX11DeviceXI *device)
660 #endif /* XINPUT_XFREE */