Rename GdkDevice[Manager]Core to GdkX11Device[Manager]Core
This is mainly to avoid clash with the classes of the same name in the quartz backend.
This commit is contained in:
600
gdk/x11/gdkdevice-core-x11.c
Normal file
600
gdk/x11/gdkdevice-core-x11.c
Normal file
@ -0,0 +1,600 @@
|
||||
/* GDK - The GIMP Drawing Kit
|
||||
* Copyright (C) 2009 Carlos Garnacho <carlosg@gnome.org>
|
||||
*
|
||||
* This library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2 of the License, or (at your option) any later version.
|
||||
*
|
||||
* This library is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the
|
||||
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
* Boston, MA 02111-1307, USA.
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
|
||||
#include "gdkx11device-core.h"
|
||||
|
||||
#include "gdkinternals.h"
|
||||
#include "gdkwindow.h"
|
||||
#include "gdkprivate-x11.h"
|
||||
#include "gdkasync.h"
|
||||
|
||||
static gboolean gdk_x11_device_core_get_history (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
guint32 start,
|
||||
guint32 stop,
|
||||
GdkTimeCoord ***events,
|
||||
gint *n_events);
|
||||
static void gdk_x11_device_core_get_state (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
gdouble *axes,
|
||||
GdkModifierType *mask);
|
||||
static void gdk_x11_device_core_set_window_cursor (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkCursor *cursor);
|
||||
static void gdk_x11_device_core_warp (GdkDevice *device,
|
||||
GdkScreen *screen,
|
||||
gint x,
|
||||
gint y);
|
||||
static gboolean gdk_x11_device_core_query_state (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkWindow **root_window,
|
||||
GdkWindow **child_window,
|
||||
gint *root_x,
|
||||
gint *root_y,
|
||||
gint *win_x,
|
||||
gint *win_y,
|
||||
GdkModifierType *mask);
|
||||
static GdkGrabStatus gdk_x11_device_core_grab (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
gboolean owner_events,
|
||||
GdkEventMask event_mask,
|
||||
GdkWindow *confine_to,
|
||||
GdkCursor *cursor,
|
||||
guint32 time_);
|
||||
static void gdk_x11_device_core_ungrab (GdkDevice *device,
|
||||
guint32 time_);
|
||||
static GdkWindow * gdk_x11_device_core_window_at_position (GdkDevice *device,
|
||||
gint *win_x,
|
||||
gint *win_y,
|
||||
GdkModifierType *mask,
|
||||
gboolean get_toplevel);
|
||||
static void gdk_x11_device_core_select_window_events (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkEventMask event_mask);
|
||||
|
||||
G_DEFINE_TYPE (GdkX11DeviceCore, gdk_x11_device_core, GDK_TYPE_DEVICE)
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_class_init (GdkX11DeviceCoreClass *klass)
|
||||
{
|
||||
GdkDeviceClass *device_class = GDK_DEVICE_CLASS (klass);
|
||||
|
||||
device_class->get_history = gdk_x11_device_core_get_history;
|
||||
device_class->get_state = gdk_x11_device_core_get_state;
|
||||
device_class->set_window_cursor = gdk_x11_device_core_set_window_cursor;
|
||||
device_class->warp = gdk_x11_device_core_warp;
|
||||
device_class->query_state = gdk_x11_device_core_query_state;
|
||||
device_class->grab = gdk_x11_device_core_grab;
|
||||
device_class->ungrab = gdk_x11_device_core_ungrab;
|
||||
device_class->window_at_position = gdk_x11_device_core_window_at_position;
|
||||
device_class->select_window_events = gdk_x11_device_core_select_window_events;
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_init (GdkX11DeviceCore *device_core)
|
||||
{
|
||||
GdkDevice *device;
|
||||
|
||||
device = GDK_DEVICE (device_core);
|
||||
|
||||
_gdk_device_add_axis (device, GDK_NONE, GDK_AXIS_X, 0, 0, 1);
|
||||
_gdk_device_add_axis (device, GDK_NONE, GDK_AXIS_Y, 0, 0, 1);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
impl_coord_in_window (GdkWindow *window,
|
||||
int impl_x,
|
||||
int impl_y)
|
||||
{
|
||||
if (impl_x < window->abs_x ||
|
||||
impl_x >= window->abs_x + window->width)
|
||||
return FALSE;
|
||||
|
||||
if (impl_y < window->abs_y ||
|
||||
impl_y >= window->abs_y + window->height)
|
||||
return FALSE;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gdk_x11_device_core_get_history (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
guint32 start,
|
||||
guint32 stop,
|
||||
GdkTimeCoord ***events,
|
||||
gint *n_events)
|
||||
{
|
||||
XTimeCoord *xcoords;
|
||||
GdkTimeCoord **coords;
|
||||
GdkWindow *impl_window;
|
||||
int tmp_n_events;
|
||||
int i, j;
|
||||
|
||||
impl_window = _gdk_window_get_impl_window (window);
|
||||
xcoords = XGetMotionEvents (GDK_WINDOW_XDISPLAY (window),
|
||||
GDK_WINDOW_XID (impl_window),
|
||||
start, stop, &tmp_n_events);
|
||||
if (!xcoords)
|
||||
return FALSE;
|
||||
|
||||
coords = _gdk_device_allocate_history (device, tmp_n_events);
|
||||
|
||||
for (i = 0, j = 0; i < tmp_n_events; i++)
|
||||
{
|
||||
if (impl_coord_in_window (window, xcoords[i].x, xcoords[i].y))
|
||||
{
|
||||
coords[j]->time = xcoords[i].time;
|
||||
coords[j]->axes[0] = xcoords[i].x - window->abs_x;
|
||||
coords[j]->axes[1] = xcoords[i].y - window->abs_y;
|
||||
j++;
|
||||
}
|
||||
}
|
||||
|
||||
XFree (xcoords);
|
||||
|
||||
/* free the events we allocated too much */
|
||||
for (i = j; i < tmp_n_events; i++)
|
||||
{
|
||||
g_free (coords[i]);
|
||||
coords[i] = NULL;
|
||||
}
|
||||
|
||||
tmp_n_events = j;
|
||||
|
||||
if (tmp_n_events == 0)
|
||||
{
|
||||
gdk_device_free_history (coords, tmp_n_events);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (n_events)
|
||||
*n_events = tmp_n_events;
|
||||
|
||||
if (events)
|
||||
*events = coords;
|
||||
else if (coords)
|
||||
gdk_device_free_history (coords, tmp_n_events);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_get_state (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
gdouble *axes,
|
||||
GdkModifierType *mask)
|
||||
{
|
||||
gint x_int, y_int;
|
||||
|
||||
gdk_window_get_pointer (window, &x_int, &y_int, mask);
|
||||
|
||||
if (axes)
|
||||
{
|
||||
axes[0] = x_int;
|
||||
axes[1] = y_int;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_set_window_cursor (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkCursor *cursor)
|
||||
{
|
||||
Cursor xcursor;
|
||||
|
||||
if (!cursor)
|
||||
xcursor = None;
|
||||
else
|
||||
xcursor = gdk_x11_cursor_get_xcursor (cursor);
|
||||
|
||||
XDefineCursor (GDK_WINDOW_XDISPLAY (window),
|
||||
GDK_WINDOW_XID (window),
|
||||
xcursor);
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_warp (GdkDevice *device,
|
||||
GdkScreen *screen,
|
||||
gint x,
|
||||
gint y)
|
||||
{
|
||||
Display *xdisplay;
|
||||
Window dest;
|
||||
|
||||
xdisplay = GDK_DISPLAY_XDISPLAY (gdk_device_get_display (device));
|
||||
dest = GDK_WINDOW_XID (gdk_screen_get_root_window (screen));
|
||||
|
||||
XWarpPointer (xdisplay, None, dest, 0, 0, 0, 0, x, y);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gdk_x11_device_core_query_state (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkWindow **root_window,
|
||||
GdkWindow **child_window,
|
||||
gint *root_x,
|
||||
gint *root_y,
|
||||
gint *win_x,
|
||||
gint *win_y,
|
||||
GdkModifierType *mask)
|
||||
{
|
||||
GdkDisplay *display;
|
||||
GdkScreen *default_screen;
|
||||
Window xroot_window, xchild_window;
|
||||
int xroot_x, xroot_y, xwin_x, xwin_y;
|
||||
unsigned int xmask;
|
||||
|
||||
display = gdk_window_get_display (window);
|
||||
default_screen = gdk_display_get_default_screen (display);
|
||||
|
||||
if (G_LIKELY (GDK_X11_DISPLAY (display)->trusted_client))
|
||||
{
|
||||
if (!XQueryPointer (GDK_WINDOW_XDISPLAY (window),
|
||||
GDK_WINDOW_XID (window),
|
||||
&xroot_window,
|
||||
&xchild_window,
|
||||
&xroot_x, &xroot_y,
|
||||
&xwin_x, &xwin_y,
|
||||
&xmask))
|
||||
return FALSE;
|
||||
}
|
||||
else
|
||||
{
|
||||
XSetWindowAttributes attributes;
|
||||
Display *xdisplay;
|
||||
Window xwindow, w;
|
||||
|
||||
/* FIXME: untrusted clients not multidevice-safe */
|
||||
xdisplay = GDK_SCREEN_XDISPLAY (default_screen);
|
||||
xwindow = GDK_SCREEN_XROOTWIN (default_screen);
|
||||
|
||||
w = XCreateWindow (xdisplay, xwindow, 0, 0, 1, 1, 0,
|
||||
CopyFromParent, InputOnly, CopyFromParent,
|
||||
0, &attributes);
|
||||
XQueryPointer (xdisplay, w,
|
||||
&xroot_window,
|
||||
&xchild_window,
|
||||
&xroot_x, &xroot_y,
|
||||
&xwin_x, &xwin_y,
|
||||
&xmask);
|
||||
XDestroyWindow (xdisplay, w);
|
||||
}
|
||||
|
||||
if (root_window)
|
||||
*root_window = gdk_x11_window_lookup_for_display (display, xroot_window);
|
||||
|
||||
if (child_window)
|
||||
*child_window = gdk_x11_window_lookup_for_display (display, xchild_window);
|
||||
|
||||
if (root_x)
|
||||
*root_x = xroot_x;
|
||||
|
||||
if (root_y)
|
||||
*root_y = xroot_y;
|
||||
|
||||
if (win_x)
|
||||
*win_x = xwin_x;
|
||||
|
||||
if (win_y)
|
||||
*win_y = xwin_y;
|
||||
|
||||
if (mask)
|
||||
*mask = xmask;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static GdkGrabStatus
|
||||
gdk_x11_device_core_grab (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
gboolean owner_events,
|
||||
GdkEventMask event_mask,
|
||||
GdkWindow *confine_to,
|
||||
GdkCursor *cursor,
|
||||
guint32 time_)
|
||||
{
|
||||
GdkDisplay *display;
|
||||
Window xwindow, xconfine_to;
|
||||
gint status;
|
||||
|
||||
display = gdk_device_get_display (device);
|
||||
|
||||
xwindow = GDK_WINDOW_XID (window);
|
||||
|
||||
if (confine_to)
|
||||
confine_to = _gdk_window_get_impl_window (confine_to);
|
||||
|
||||
if (!confine_to || GDK_WINDOW_DESTROYED (confine_to))
|
||||
xconfine_to = None;
|
||||
else
|
||||
xconfine_to = GDK_WINDOW_XID (confine_to);
|
||||
|
||||
#ifdef G_ENABLE_DEBUG
|
||||
if (_gdk_debug_flags & GDK_DEBUG_NOGRABS)
|
||||
status = GrabSuccess;
|
||||
else
|
||||
#endif
|
||||
if (gdk_device_get_source (device) == GDK_SOURCE_KEYBOARD)
|
||||
{
|
||||
/* Device is a keyboard */
|
||||
status = XGrabKeyboard (GDK_DISPLAY_XDISPLAY (display),
|
||||
xwindow,
|
||||
owner_events,
|
||||
GrabModeAsync, GrabModeAsync,
|
||||
time_);
|
||||
}
|
||||
else
|
||||
{
|
||||
Cursor xcursor;
|
||||
guint xevent_mask;
|
||||
gint i;
|
||||
|
||||
/* Device is a pointer */
|
||||
if (!cursor)
|
||||
xcursor = None;
|
||||
else
|
||||
{
|
||||
_gdk_x11_cursor_update_theme (cursor);
|
||||
xcursor = gdk_x11_cursor_get_xcursor (cursor);
|
||||
}
|
||||
|
||||
xevent_mask = 0;
|
||||
|
||||
for (i = 0; i < _gdk_x11_event_mask_table_size; i++)
|
||||
{
|
||||
if (event_mask & (1 << (i + 1)))
|
||||
xevent_mask |= _gdk_x11_event_mask_table[i];
|
||||
}
|
||||
|
||||
/* We don't want to set a native motion hint mask, as we're emulating motion
|
||||
* hints. If we set a native one we just wouldn't get any events.
|
||||
*/
|
||||
xevent_mask &= ~PointerMotionHintMask;
|
||||
|
||||
status = XGrabPointer (GDK_DISPLAY_XDISPLAY (display),
|
||||
xwindow,
|
||||
owner_events,
|
||||
xevent_mask,
|
||||
GrabModeAsync, GrabModeAsync,
|
||||
xconfine_to,
|
||||
xcursor,
|
||||
time_);
|
||||
}
|
||||
|
||||
_gdk_x11_display_update_grab_info (display, device, status);
|
||||
|
||||
return _gdk_x11_convert_grab_status (status);
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_ungrab (GdkDevice *device,
|
||||
guint32 time_)
|
||||
{
|
||||
GdkDisplay *display;
|
||||
gulong serial;
|
||||
|
||||
display = gdk_device_get_display (device);
|
||||
serial = NextRequest (GDK_DISPLAY_XDISPLAY (display));
|
||||
|
||||
if (gdk_device_get_source (device) == GDK_SOURCE_KEYBOARD)
|
||||
XUngrabKeyboard (GDK_DISPLAY_XDISPLAY (display), time_);
|
||||
else
|
||||
XUngrabPointer (GDK_DISPLAY_XDISPLAY (display), time_);
|
||||
|
||||
_gdk_x11_display_update_grab_info_ungrab (display, device, time_, serial);
|
||||
}
|
||||
|
||||
static GdkWindow *
|
||||
gdk_x11_device_core_window_at_position (GdkDevice *device,
|
||||
gint *win_x,
|
||||
gint *win_y,
|
||||
GdkModifierType *mask,
|
||||
gboolean get_toplevel)
|
||||
{
|
||||
GdkDisplay *display;
|
||||
GdkScreen *screen;
|
||||
Display *xdisplay;
|
||||
GdkWindow *window;
|
||||
Window xwindow, root, child, last;
|
||||
int xroot_x, xroot_y, xwin_x, xwin_y;
|
||||
unsigned int xmask;
|
||||
|
||||
last = None;
|
||||
display = gdk_device_get_display (device);
|
||||
screen = gdk_display_get_default_screen (display);
|
||||
|
||||
/* This function really only works if the mouse pointer is held still
|
||||
* during its operation. If it moves from one leaf window to another
|
||||
* than we'll end up with inaccurate values for win_x, win_y
|
||||
* and the result.
|
||||
*/
|
||||
gdk_x11_display_grab (display);
|
||||
|
||||
xdisplay = GDK_SCREEN_XDISPLAY (screen);
|
||||
xwindow = GDK_SCREEN_XROOTWIN (screen);
|
||||
|
||||
if (G_LIKELY (GDK_X11_DISPLAY (display)->trusted_client))
|
||||
{
|
||||
XQueryPointer (xdisplay, xwindow,
|
||||
&root, &child,
|
||||
&xroot_x, &xroot_y,
|
||||
&xwin_x, &xwin_y,
|
||||
&xmask);
|
||||
|
||||
if (root == xwindow)
|
||||
xwindow = child;
|
||||
else
|
||||
xwindow = root;
|
||||
}
|
||||
else
|
||||
{
|
||||
gint i, screens, width, height;
|
||||
GList *toplevels, *list;
|
||||
Window pointer_window, root, child;
|
||||
int rootx = -1, rooty = -1;
|
||||
int winx, winy;
|
||||
unsigned int xmask;
|
||||
|
||||
/* FIXME: untrusted clients case not multidevice-safe */
|
||||
pointer_window = None;
|
||||
screens = gdk_display_get_n_screens (display);
|
||||
|
||||
for (i = 0; i < screens; ++i)
|
||||
{
|
||||
screen = gdk_display_get_screen (display, i);
|
||||
toplevels = gdk_screen_get_toplevel_windows (screen);
|
||||
for (list = toplevels; list != NULL; list = g_list_next (list))
|
||||
{
|
||||
window = GDK_WINDOW (list->data);
|
||||
xwindow = GDK_WINDOW_XID (window);
|
||||
gdk_x11_display_error_trap_push (display);
|
||||
XQueryPointer (xdisplay, xwindow,
|
||||
&root, &child,
|
||||
&rootx, &rooty,
|
||||
&winx, &winy,
|
||||
&xmask);
|
||||
if (gdk_x11_display_error_trap_pop (display))
|
||||
continue;
|
||||
if (child != None)
|
||||
{
|
||||
pointer_window = child;
|
||||
break;
|
||||
}
|
||||
gdk_window_get_geometry (window, NULL, NULL, &width, &height);
|
||||
if (winx >= 0 && winy >= 0 && winx < width && winy < height)
|
||||
{
|
||||
/* A childless toplevel, or below another window? */
|
||||
XSetWindowAttributes attributes;
|
||||
Window w;
|
||||
|
||||
w = XCreateWindow (xdisplay, xwindow, winx, winy, 1, 1, 0,
|
||||
CopyFromParent, InputOnly, CopyFromParent,
|
||||
0, &attributes);
|
||||
XMapWindow (xdisplay, w);
|
||||
XQueryPointer (xdisplay, xwindow,
|
||||
&root, &child,
|
||||
&rootx, &rooty,
|
||||
&winx, &winy,
|
||||
&xmask);
|
||||
XDestroyWindow (xdisplay, w);
|
||||
if (child == w)
|
||||
{
|
||||
pointer_window = xwindow;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
g_list_free (toplevels);
|
||||
if (pointer_window != None)
|
||||
break;
|
||||
}
|
||||
|
||||
xwindow = pointer_window;
|
||||
}
|
||||
|
||||
while (xwindow)
|
||||
{
|
||||
last = xwindow;
|
||||
gdk_x11_display_error_trap_push (display);
|
||||
XQueryPointer (xdisplay, xwindow,
|
||||
&root, &xwindow,
|
||||
&xroot_x, &xroot_y,
|
||||
&xwin_x, &xwin_y,
|
||||
&xmask);
|
||||
if (gdk_x11_display_error_trap_pop (display))
|
||||
break;
|
||||
|
||||
if (get_toplevel && last != root &&
|
||||
(window = gdk_x11_window_lookup_for_display (display, last)) != NULL &&
|
||||
window->window_type != GDK_WINDOW_FOREIGN)
|
||||
{
|
||||
xwindow = last;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
gdk_x11_display_ungrab (display);
|
||||
|
||||
window = gdk_x11_window_lookup_for_display (display, last);
|
||||
|
||||
if (win_x)
|
||||
*win_x = (window) ? xwin_x : -1;
|
||||
|
||||
if (win_y)
|
||||
*win_y = (window) ? xwin_y : -1;
|
||||
|
||||
if (mask)
|
||||
*mask = xmask;
|
||||
|
||||
return window;
|
||||
}
|
||||
|
||||
static void
|
||||
gdk_x11_device_core_select_window_events (GdkDevice *device,
|
||||
GdkWindow *window,
|
||||
GdkEventMask event_mask)
|
||||
{
|
||||
GdkEventMask filter_mask, window_mask;
|
||||
guint xmask = 0;
|
||||
gint i;
|
||||
|
||||
window_mask = gdk_window_get_events (window);
|
||||
filter_mask = (GDK_POINTER_MOTION_MASK &
|
||||
GDK_POINTER_MOTION_HINT_MASK &
|
||||
GDK_BUTTON_MOTION_MASK &
|
||||
GDK_BUTTON1_MOTION_MASK &
|
||||
GDK_BUTTON2_MOTION_MASK &
|
||||
GDK_BUTTON3_MOTION_MASK &
|
||||
GDK_BUTTON_PRESS_MASK &
|
||||
GDK_BUTTON_RELEASE_MASK &
|
||||
GDK_KEY_PRESS_MASK &
|
||||
GDK_KEY_RELEASE_MASK &
|
||||
GDK_ENTER_NOTIFY_MASK &
|
||||
GDK_LEAVE_NOTIFY_MASK &
|
||||
GDK_FOCUS_CHANGE_MASK &
|
||||
GDK_PROXIMITY_IN_MASK &
|
||||
GDK_PROXIMITY_OUT_MASK &
|
||||
GDK_SCROLL_MASK);
|
||||
|
||||
/* Filter out non-device events */
|
||||
event_mask &= filter_mask;
|
||||
|
||||
/* Unset device events on window mask */
|
||||
window_mask &= ~(filter_mask);
|
||||
|
||||
/* Combine masks */
|
||||
event_mask |= window_mask;
|
||||
|
||||
for (i = 0; i < _gdk_x11_event_mask_table_size; i++)
|
||||
{
|
||||
if (event_mask & (1 << (i + 1)))
|
||||
xmask |= _gdk_x11_event_mask_table[i];
|
||||
}
|
||||
|
||||
if (GDK_WINDOW_XID (window) != GDK_WINDOW_XROOTWIN (window))
|
||||
xmask |= StructureNotifyMask | PropertyChangeMask;
|
||||
|
||||
XSelectInput (GDK_WINDOW_XDISPLAY (window),
|
||||
GDK_WINDOW_XID (window),
|
||||
xmask);
|
||||
}
|
||||
Reference in New Issue
Block a user