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, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
22 #include "gdkdevice-xi.h"
24 #include "gdkwindow.h"
25 #include "gdkdeviceprivate.h"
26 #include "gdkprivate-x11.h"
30 #define MAX_DEVICE_CLASSES 13
32 static GQuark quark_window_input_info = 0;
40 static void gdk_device_xi_constructed (GObject *object);
41 static void gdk_device_xi_dispose (GObject *object);
43 static void gdk_device_xi_set_property (GObject *object,
47 static void gdk_device_xi_get_property (GObject *object,
52 static gboolean gdk_device_xi_get_history (GdkDevice *device,
56 GdkTimeCoord ***events,
59 static void gdk_device_xi_get_state (GdkDevice *device,
62 GdkModifierType *mask);
63 static void gdk_device_xi_set_window_cursor (GdkDevice *device,
66 static void gdk_device_xi_warp (GdkDevice *device,
70 static gboolean gdk_device_xi_query_state (GdkDevice *device,
72 GdkWindow **root_window,
73 GdkWindow **child_window,
78 GdkModifierType *mask);
79 static GdkGrabStatus gdk_device_xi_grab (GdkDevice *device,
81 gboolean owner_events,
82 GdkEventMask event_mask,
83 GdkWindow *confine_to,
86 static void gdk_device_xi_ungrab (GdkDevice *device,
89 static GdkWindow* gdk_device_xi_window_at_position (GdkDevice *device,
92 GdkModifierType *mask,
93 gboolean get_toplevel);
95 static void gdk_device_xi_select_window_events (GdkDevice *device,
100 G_DEFINE_TYPE (GdkDeviceXI, gdk_device_xi, GDK_TYPE_DEVICE)
108 gdk_device_xi_class_init (GdkDeviceXIClass *klass)
110 GObjectClass *object_class = G_OBJECT_CLASS (klass);
111 GdkDeviceClass *device_class = GDK_DEVICE_CLASS (klass);
113 quark_window_input_info = g_quark_from_static_string ("gdk-window-input-info");
115 object_class->constructed = gdk_device_xi_constructed;
116 object_class->set_property = gdk_device_xi_set_property;
117 object_class->get_property = gdk_device_xi_get_property;
118 object_class->dispose = gdk_device_xi_dispose;
120 device_class->get_history = gdk_device_xi_get_history;
121 device_class->get_state = gdk_device_xi_get_state;
122 device_class->set_window_cursor = gdk_device_xi_set_window_cursor;
123 device_class->warp = gdk_device_xi_warp;
124 device_class->query_state = gdk_device_xi_query_state;
125 device_class->grab = gdk_device_xi_grab;
126 device_class->ungrab = gdk_device_xi_ungrab;
127 device_class->window_at_position = gdk_device_xi_window_at_position;
128 device_class->select_window_events = gdk_device_xi_select_window_events;
130 g_object_class_install_property (object_class,
132 g_param_spec_int ("device-id",
136 G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
140 gdk_device_xi_init (GdkDeviceXI *device)
145 gdk_device_xi_constructed (GObject *object)
150 device = GDK_DEVICE_XI (object);
151 display = gdk_device_get_display (GDK_DEVICE (object));
153 gdk_error_trap_push ();
154 device->xdevice = XOpenDevice (GDK_DISPLAY_XDISPLAY (display),
157 if (gdk_error_trap_pop ())
158 g_warning ("Device %s can't be opened",
159 gdk_device_get_name (GDK_DEVICE (device)));
161 if (G_OBJECT_CLASS (gdk_device_xi_parent_class)->constructed)
162 G_OBJECT_CLASS (gdk_device_xi_parent_class)->constructed (object);
166 gdk_device_xi_set_property (GObject *object,
171 GdkDeviceXI *device = GDK_DEVICE_XI (object);
176 device->device_id = g_value_get_int (value);
179 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
185 gdk_device_xi_get_property (GObject *object,
190 GdkDeviceXI *device = GDK_DEVICE_XI (object);
195 g_value_set_int (value, device->device_id);
198 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
204 gdk_device_xi_dispose (GObject *object)
206 GdkDeviceXI *device_xi;
209 device_xi = GDK_DEVICE_XI (object);
210 display = gdk_device_get_display (GDK_DEVICE (device_xi));
212 if (device_xi->xdevice)
214 XCloseDevice (GDK_DISPLAY_XDISPLAY (display), device_xi->xdevice);
215 device_xi->xdevice = NULL;
218 if (device_xi->axis_data)
220 g_free (device_xi->axis_data);
221 device_xi->axis_data = NULL;
224 G_OBJECT_CLASS (gdk_device_xi_parent_class)->dispose (object);
228 gdk_device_xi_get_history (GdkDevice *device,
232 GdkTimeCoord ***events,
235 GdkTimeCoord **coords;
236 XDeviceTimeCoord *device_coords;
237 GdkWindow *impl_window;
238 GdkDeviceXI *device_xi;
239 gint n_events_return;
241 gint axis_count_return;
244 device_xi = GDK_DEVICE_XI (device);
245 impl_window = _gdk_window_get_impl_window (window);
247 device_coords = XGetDeviceMotionEvents (GDK_WINDOW_XDISPLAY (impl_window),
257 *n_events = n_events_return;
258 coords = _gdk_device_allocate_history (device, *n_events);
260 for (i = 0; i < *n_events; i++)
262 coords[i]->time = device_coords[i].time;
263 gdk_device_xi_translate_axes (device, window,
264 device_coords[i].data,
269 XFreeDeviceMotionEvents (device_coords);
277 gdk_device_xi_get_state (GdkDevice *device,
280 GdkModifierType *mask)
282 GdkDeviceXI *device_xi;
284 XInputClass *input_class;
288 gdk_window_get_pointer (window, NULL, NULL, mask);
290 device_xi = GDK_DEVICE_XI (device);
291 state = XQueryDeviceState (GDK_WINDOW_XDISPLAY (window),
293 input_class = state->data;
295 for (i = 0; i < state->num_classes; i++)
297 switch (input_class->class)
301 gdk_device_xi_translate_axes (device, window,
302 ((XValuatorState *) input_class)->valuators,
310 if (((XButtonState *)input_class)->num_buttons > 0)
311 *mask |= ((XButtonState *)input_class)->buttons[0] << 7;
312 /* GDK_BUTTON1_MASK = 1 << 8, and button n is stored
313 * in bit 1<<(n%8) in byte n/8. n = 1,2,... */
318 input_class = (XInputClass *)(((char *)input_class)+input_class->length);
321 XFreeDeviceState (state);
325 gdk_device_xi_set_window_cursor (GdkDevice *device,
332 gdk_device_xi_warp (GdkDevice *device,
340 find_events (GdkDevice *device,
342 XEventClass *classes,
345 GdkDeviceXI *device_xi;
349 device_xi = GDK_DEVICE_XI (device);
352 if (mask & GDK_BUTTON_PRESS_MASK)
354 DeviceButtonPress (device_xi->xdevice, device_xi->button_press_type, class);
356 classes[i++] = class;
358 DeviceButtonPressGrab (device_xi->xdevice, 0, class);
360 classes[i++] = class;
363 if (mask & GDK_BUTTON_RELEASE_MASK)
365 DeviceButtonRelease (device_xi->xdevice, device_xi->button_release_type, class);
367 classes[i++] = class;
370 if (mask & (GDK_POINTER_MOTION_MASK |
371 GDK_BUTTON1_MOTION_MASK | GDK_BUTTON2_MOTION_MASK |
372 GDK_BUTTON3_MOTION_MASK | GDK_BUTTON_MOTION_MASK))
374 /* Make sure device->motionnotify_type is set */
375 DeviceMotionNotify (device_xi->xdevice, device_xi->motion_notify_type, class);
377 classes[i++] = class;
378 DeviceStateNotify (device_xi->xdevice, device_xi->state_notify_type, class);
380 classes[i++] = class;
383 if (mask & GDK_KEY_PRESS_MASK)
385 DeviceKeyPress (device_xi->xdevice, device_xi->key_press_type, class);
387 classes[i++] = class;
390 if (mask & GDK_KEY_RELEASE_MASK)
392 DeviceKeyRelease (device_xi->xdevice, device_xi->key_release_type, class);
394 classes[i++] = class;
397 if (mask & GDK_PROXIMITY_IN_MASK)
399 ProximityIn (device_xi->xdevice, device_xi->proximity_in_type, class);
401 classes[i++] = class;
404 if (mask & GDK_PROXIMITY_OUT_MASK)
406 ProximityOut (device_xi->xdevice, device_xi->proximity_out_type, class);
408 classes[i++] = class;
415 gdk_device_xi_query_state (GdkDevice *device,
417 GdkWindow **root_window,
418 GdkWindow **child_window,
423 GdkModifierType *mask)
429 gdk_device_xi_grab (GdkDevice *device,
431 gboolean owner_events,
432 GdkEventMask event_mask,
433 GdkWindow *confine_to,
437 XEventClass event_classes[MAX_DEVICE_CLASSES];
438 gint status, num_classes;
439 GdkDeviceXI *device_xi;
441 device_xi = GDK_DEVICE_XI (device);
442 find_events (device, event_mask, event_classes, &num_classes);
444 status = XGrabDevice (GDK_WINDOW_XDISPLAY (window),
446 GDK_WINDOW_XWINDOW (window),
448 num_classes, event_classes,
449 GrabModeAsync, GrabModeAsync,
452 return _gdk_x11_convert_grab_status (status);
456 gdk_device_xi_ungrab (GdkDevice *device,
460 GdkDeviceXI *device_xi;
462 device_xi = GDK_DEVICE_XI (device);
463 display = gdk_device_get_display (device);
465 XUngrabDevice (GDK_DISPLAY_XDISPLAY (device),
471 gdk_device_xi_window_at_position (GdkDevice *device,
474 GdkModifierType *mask,
475 gboolean get_toplevel)
480 gdk_device_xi_select_window_events (GdkDevice *device,
482 GdkEventMask event_mask)
484 XEventClass event_classes[MAX_DEVICE_CLASSES];
485 GdkDeviceXI *device_xi;
488 event_mask |= (GDK_PROXIMITY_IN_MASK |
489 GDK_PROXIMITY_OUT_MASK);
491 device_xi = GDK_DEVICE_XI (device);
492 find_events (device, event_mask, event_classes, &num_classes);
494 XSelectExtensionEvent (GDK_WINDOW_XDISPLAY (window),
495 GDK_WINDOW_XWINDOW (window),
496 event_classes, num_classes);
500 GdkWindowInputInfo *info;
502 info = g_new0 (GdkWindowInputInfo, 1);
503 g_object_set_qdata_full (G_OBJECT (window),
504 quark_window_input_info,
506 (GDestroyNotify) g_free);
509 g_object_set_qdata (G_OBJECT (window),
510 quark_window_input_info,
515 gdk_device_xi_update_window_info (GdkWindow *window)
517 GdkWindowInputInfo *info;
520 info = g_object_get_qdata (G_OBJECT (window),
521 quark_window_input_info);
526 gdk_window_get_origin (window, &root_x, &root_y);
527 info->root_x = (gdouble) root_x;
528 info->root_y = (gdouble) root_y;
532 gdk_device_xi_get_window_info (GdkWindow *window,
536 GdkWindowInputInfo *info;
538 info = g_object_get_qdata (G_OBJECT (window),
539 quark_window_input_info);
544 *root_x = info->root_x;
545 *root_y = info->root_y;
551 gdk_device_xi_update_axes (GdkDevice *device,
556 GdkDeviceXI *device_xi;
559 device_xi = GDK_DEVICE_XI (device);
560 g_return_if_fail (first_axis >= 0 &&
561 first_axis + axes_count <= gdk_device_get_n_axes (device));
563 if (!device_xi->axis_data)
564 device_xi->axis_data = g_new0 (gint, gdk_device_get_n_axes (device));
566 for (i = 0; i < axes_count; i++)
567 device_xi->axis_data[first_axis + i] = axis_data[i];
571 gdk_device_xi_translate_axes (GdkDevice *device,
578 GdkDeviceXI *device_xi;
579 GdkWindow *impl_window;
580 gdouble root_x, root_y;
581 gdouble temp_x, temp_y;
585 device_xi = GDK_DEVICE_XI (device);
586 impl_window = _gdk_window_get_impl_window (window);
589 if (!gdk_device_xi_get_window_info (impl_window, &root_x, &root_y))
592 n_axes = gdk_device_get_n_axes (device);
594 for (i = 0; i < n_axes; i++)
598 use = _gdk_device_get_axis_use (device, i);
604 if (gdk_device_get_mode (device) == GDK_MODE_WINDOW)
605 _gdk_device_translate_window_coord (device, window,
609 _gdk_device_translate_screen_coord (device, window,
613 if (use == GDK_AXIS_X)
615 else if (use == GDK_AXIS_Y)
620 _gdk_device_translate_axis (device, i, axis_data[i], &axes[i]);