2005-11-22 10:03:32 +00:00
|
|
|
/* GDK - The GIMP Drawing Kit
|
|
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
|
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
|
|
* files for a list of changes. These files are distributed with
|
|
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
|
|
*/
|
|
|
|
|
2008-06-22 14:28:52 +00:00
|
|
|
#include "config.h"
|
2005-11-22 10:03:32 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "gdkprivate-quartz.h"
|
|
|
|
#include "gdkinput.h"
|
|
|
|
#include "gdkprivate.h"
|
|
|
|
#include "gdkinputprivate.h"
|
2010-05-25 22:38:44 +00:00
|
|
|
#include <gdkdevice.h>
|
|
|
|
#include <gdkdeviceprivate.h>
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
/* Addition used for extension_events mask */
|
|
|
|
#define GDK_ALL_DEVICES_MASK (1<<30)
|
2005-11-22 10:03:32 +00:00
|
|
|
|
|
|
|
GdkDevice *_gdk_core_pointer = NULL;
|
|
|
|
|
|
|
|
/* Global variables */
|
|
|
|
|
|
|
|
gchar *_gdk_input_gxid_host;
|
|
|
|
gint _gdk_input_gxid_port;
|
|
|
|
gint _gdk_input_ignore_core;
|
|
|
|
GList *_gdk_input_windows;
|
|
|
|
GList *_gdk_input_devices;
|
|
|
|
|
|
|
|
|
|
|
|
GList *
|
|
|
|
gdk_devices_list (void)
|
|
|
|
{
|
|
|
|
return _gdk_input_devices;
|
|
|
|
}
|
|
|
|
|
|
|
|
GList *
|
|
|
|
gdk_display_list_devices (GdkDisplay *dpy)
|
|
|
|
{
|
|
|
|
return _gdk_input_devices;
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
static void
|
|
|
|
_gdk_input_select_device_events (GdkWindow *impl_window,
|
|
|
|
GdkDevice *device)
|
2010-05-25 15:54:16 +00:00
|
|
|
{
|
2010-05-25 22:38:44 +00:00
|
|
|
guint event_mask;
|
|
|
|
GdkWindowObject *w;
|
|
|
|
GdkInputWindow *iw;
|
|
|
|
GdkInputMode mode;
|
|
|
|
gboolean has_cursor;
|
|
|
|
GdkDeviceType type;
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
event_mask = 0;
|
|
|
|
iw = ((GdkWindowObject *)impl_window)->input_window;
|
|
|
|
|
|
|
|
g_object_get (device,
|
|
|
|
"type", &type,
|
|
|
|
"input-mode", &mode,
|
|
|
|
"has-cursor", &has_cursor,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (iw == NULL ||
|
|
|
|
mode == GDK_MODE_DISABLED ||
|
|
|
|
type == GDK_DEVICE_TYPE_MASTER)
|
2010-05-25 15:54:16 +00:00
|
|
|
return;
|
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
for (l = _gdk_input_windows; l != NULL; l = l->next)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
2010-05-25 22:38:44 +00:00
|
|
|
w = l->data;
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
if (has_cursor || (w->extension_events & GDK_ALL_DEVICES_MASK))
|
|
|
|
{
|
|
|
|
event_mask = w->extension_events;
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
if (event_mask)
|
|
|
|
event_mask |= GDK_PROXIMITY_OUT_MASK
|
|
|
|
| GDK_BUTTON_PRESS_MASK
|
|
|
|
| GDK_BUTTON_RELEASE_MASK;
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
gdk_window_set_device_events ((GdkWindow *) w, device, event_mask);
|
|
|
|
}
|
2005-11-22 10:03:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gint
|
|
|
|
_gdk_input_enable_window (GdkWindow *window, GdkDevicePrivate *gdkdev)
|
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gint
|
|
|
|
_gdk_input_disable_window (GdkWindow *window, GdkDevicePrivate *gdkdev)
|
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GdkInputWindow *
|
|
|
|
_gdk_input_window_find(GdkWindow *window)
|
|
|
|
{
|
|
|
|
GList *tmp_list;
|
|
|
|
|
|
|
|
for (tmp_list=_gdk_input_windows; tmp_list; tmp_list=tmp_list->next)
|
|
|
|
if (((GdkInputWindow *)(tmp_list->data))->window == window)
|
|
|
|
return (GdkInputWindow *)(tmp_list->data);
|
|
|
|
|
|
|
|
return NULL; /* Not found */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: this routine currently needs to be called between creation
|
|
|
|
and the corresponding configure event (because it doesn't get the
|
|
|
|
root_relative_geometry). This should work with
|
|
|
|
gtk_window_set_extension_events, but will likely fail in other
|
|
|
|
cases */
|
|
|
|
|
|
|
|
void
|
2010-05-25 22:38:44 +00:00
|
|
|
gdk_input_set_extension_events (GdkWindow *window,
|
|
|
|
gint mask,
|
|
|
|
GdkExtensionMode mode)
|
2005-11-22 10:03:32 +00:00
|
|
|
{
|
|
|
|
GdkWindowObject *window_private;
|
2010-05-25 22:38:44 +00:00
|
|
|
GdkWindowObject *impl_window;
|
2005-11-22 10:03:32 +00:00
|
|
|
GList *tmp_list;
|
|
|
|
GdkInputWindow *iw;
|
|
|
|
|
|
|
|
g_return_if_fail (window != NULL);
|
2009-01-21 17:35:37 +00:00
|
|
|
g_return_if_fail (GDK_WINDOW_IS_QUARTZ (window));
|
2005-11-22 10:03:32 +00:00
|
|
|
|
|
|
|
window_private = (GdkWindowObject*) window;
|
2010-05-25 22:38:44 +00:00
|
|
|
impl_window = (GdkWindowObject *)_gdk_window_get_impl_window (window);
|
2005-11-22 10:03:32 +00:00
|
|
|
|
|
|
|
if (mode == GDK_EXTENSION_EVENTS_NONE)
|
|
|
|
mask = 0;
|
|
|
|
|
|
|
|
if (mask != 0)
|
|
|
|
{
|
2010-05-25 22:38:44 +00:00
|
|
|
iw = g_new (GdkInputWindow, 1);
|
2005-11-22 10:03:32 +00:00
|
|
|
|
|
|
|
iw->window = window;
|
|
|
|
iw->mode = mode;
|
|
|
|
|
|
|
|
iw->obscuring = NULL;
|
|
|
|
iw->num_obscuring = 0;
|
|
|
|
iw->grabbed = FALSE;
|
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
_gdk_input_windows = g_list_append (_gdk_input_windows, iw);
|
2005-11-22 10:03:32 +00:00
|
|
|
window_private->extension_events = mask;
|
|
|
|
|
|
|
|
/* Add enter window events to the event mask */
|
|
|
|
/* FIXME, this is not needed for XINPUT_NONE */
|
|
|
|
gdk_window_set_events (window,
|
2010-05-25 22:38:44 +00:00
|
|
|
gdk_window_get_events (window) |
|
|
|
|
GDK_ENTER_NOTIFY_MASK);
|
2005-11-22 10:03:32 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
iw = _gdk_input_window_find (window);
|
|
|
|
if (iw)
|
2010-05-25 22:38:44 +00:00
|
|
|
{
|
|
|
|
_gdk_input_windows = g_list_remove (_gdk_input_windows,iw);
|
|
|
|
g_free (iw);
|
|
|
|
}
|
2005-11-22 10:03:32 +00:00
|
|
|
|
|
|
|
window_private->extension_events = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (tmp_list = _gdk_input_devices; tmp_list; tmp_list = tmp_list->next)
|
|
|
|
{
|
2010-05-25 22:38:44 +00:00
|
|
|
GdkDevice *dev = tmp_list->data;
|
2005-11-22 10:03:32 +00:00
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
_gdk_input_select_device_events (GDK_WINDOW (impl_window), dev);
|
2005-11-22 10:03:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_gdk_input_window_destroy (GdkWindow *window)
|
|
|
|
{
|
|
|
|
GdkInputWindow *input_window;
|
|
|
|
|
|
|
|
input_window = _gdk_input_window_find (window);
|
|
|
|
g_return_if_fail (input_window != NULL);
|
|
|
|
|
|
|
|
_gdk_input_windows = g_list_remove (_gdk_input_windows,input_window);
|
|
|
|
g_free (input_window);
|
|
|
|
}
|
|
|
|
|
2010-05-25 22:38:44 +00:00
|
|
|
void
|
|
|
|
_gdk_input_check_extension_events (GdkDevice *device)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
void
|
|
|
|
_gdk_input_init (void)
|
|
|
|
{
|
2010-05-25 22:38:44 +00:00
|
|
|
GdkDeviceManager *device_manager;
|
|
|
|
GList *list, *l;
|
|
|
|
|
|
|
|
device_manager = gdk_display_get_device_manager (_gdk_display);
|
|
|
|
|
|
|
|
/* For backward compatibility, just add floating devices that are
|
|
|
|
* not keyboards.
|
|
|
|
*/
|
|
|
|
list = gdk_device_manager_list_devices (device_manager,
|
|
|
|
GDK_DEVICE_TYPE_FLOATING);
|
|
|
|
for (l = list; l; l = l->next)
|
|
|
|
{
|
|
|
|
GdkDevice *device = l->data;
|
|
|
|
|
|
|
|
if (device->source == GDK_SOURCE_KEYBOARD)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
_gdk_input_devices = g_list_prepend (_gdk_input_devices, l->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free (list);
|
|
|
|
|
|
|
|
/* Now set "core" pointer to the first master device that is a pointer.
|
|
|
|
*/
|
|
|
|
list = gdk_device_manager_list_devices (device_manager,
|
|
|
|
GDK_DEVICE_TYPE_MASTER);
|
|
|
|
|
|
|
|
for (l = list; l; l = l->next)
|
|
|
|
{
|
|
|
|
GdkDevice *device = list->data;
|
|
|
|
|
|
|
|
if (device->source != GDK_SOURCE_MOUSE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
_gdk_display->core_pointer = device;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free (list);
|
|
|
|
|
|
|
|
/* Add the core pointer to the devices list */
|
|
|
|
_gdk_input_devices = g_list_prepend (_gdk_input_devices,
|
|
|
|
_gdk_display->core_pointer);
|
|
|
|
|
2005-11-22 10:03:32 +00:00
|
|
|
_gdk_input_ignore_core = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_gdk_input_exit (void)
|
|
|
|
{
|
|
|
|
GList *tmp_list;
|
|
|
|
GdkDevicePrivate *gdkdev;
|
|
|
|
|
|
|
|
for (tmp_list = _gdk_input_devices; tmp_list; tmp_list = tmp_list->next)
|
|
|
|
{
|
|
|
|
gdkdev = (GdkDevicePrivate *)(tmp_list->data);
|
|
|
|
if (gdkdev != (GdkDevicePrivate *)_gdk_core_pointer)
|
|
|
|
{
|
|
|
|
gdk_device_set_mode ((GdkDevice *)gdkdev, GDK_MODE_DISABLED);
|
|
|
|
|
|
|
|
g_free (gdkdev->info.name);
|
|
|
|
g_free (gdkdev->info.axes);
|
|
|
|
g_free (gdkdev->info.keys);
|
|
|
|
g_free (gdkdev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free (_gdk_input_devices);
|
|
|
|
|
|
|
|
for (tmp_list = _gdk_input_windows; tmp_list; tmp_list = tmp_list->next)
|
|
|
|
{
|
|
|
|
g_free (tmp_list->data);
|
|
|
|
}
|
|
|
|
g_list_free (_gdk_input_windows);
|
|
|
|
}
|