/* * Clutter. * * An OpenGL based 'interactive canvas' library. * * Copyright © 2011 Intel Corp. * 2011 Giovanni Campagna * * 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, see . * * Author: Emmanuele Bassi */ #include "config.h" #ifdef HAVE_STDINT_H #include #endif /* XXX: We're using the GdkDeviceManager deprecated API because * porting to the new GdkSeat API would imply adding an equivalent * API to Clutter. */ #define GDK_DISABLE_DEPRECATION_WARNINGS #include "clutter-device-manager-gdk.h" #include "clutter-backend-gdk.h" #include "clutter-input-device-gdk.h" #include "clutter-stage-gdk.h" #include "clutter-backend.h" #include "clutter-debug.h" #include "clutter-device-manager-private.h" #include "clutter-event-private.h" #include "clutter-event-translator.h" #include "clutter-stage-private.h" #include "clutter-private.h" #define clutter_device_manager_gdk_get_type _clutter_device_manager_gdk_get_type static void clutter_event_extender_iface_init (ClutterEventExtenderInterface *iface); G_DEFINE_TYPE_WITH_CODE (ClutterDeviceManagerGdk, clutter_device_manager_gdk, CLUTTER_TYPE_DEVICE_MANAGER, G_IMPLEMENT_INTERFACE (CLUTTER_TYPE_EVENT_EXTENDER, clutter_event_extender_iface_init)) enum { PROP_0, PROP_GDK_DISPLAY, PROP_LAST }; static void clutter_device_manager_gdk_copy_event_data (ClutterEventExtender *event_extender, const ClutterEvent *src, ClutterEvent *dest) { GdkEvent *gdk_event; gdk_event = _clutter_event_get_platform_data (src); if (gdk_event != NULL) _clutter_event_set_platform_data (dest, gdk_event_copy (gdk_event)); } static void clutter_device_manager_gdk_free_event_data (ClutterEventExtender *event_extender, ClutterEvent *event) { GdkEvent *gdk_event; gdk_event = _clutter_event_get_platform_data (event); if (gdk_event != NULL) gdk_event_free (gdk_event); } static void clutter_event_extender_iface_init (ClutterEventExtenderInterface *iface) { iface->copy_event_data = clutter_device_manager_gdk_copy_event_data; iface->free_event_data = clutter_device_manager_gdk_free_event_data; } ClutterInputDevice * _clutter_device_manager_gdk_lookup_device (ClutterDeviceManager *manager, GdkDevice *device) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (manager); ClutterInputDevice *clutter_device; clutter_device = g_object_get_data (G_OBJECT (device), "clutter-device"); if (clutter_device != NULL) return clutter_device; clutter_device = _clutter_input_device_gdk_new (manager, device); g_object_set_data_full (G_OBJECT (device), "clutter-device", clutter_device, g_object_unref); manager_gdk->device_cache = g_slist_prepend (manager_gdk->device_cache, g_object_ref (clutter_device)); g_hash_table_replace (manager_gdk->device_by_id, GINT_TO_POINTER (clutter_input_device_get_device_id (clutter_device)), g_object_ref (clutter_device)); return clutter_device; } static void clutter_device_manager_gdk_add_device (ClutterDeviceManager *manager, ClutterInputDevice *device) { /* XXX implement */ } static void clutter_device_manager_gdk_remove_device (ClutterDeviceManager *manager, ClutterInputDevice *device) { /* XXX implement */ } static const GSList * clutter_device_manager_gdk_get_devices (ClutterDeviceManager *manager) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (manager); return manager_gdk->device_cache; } static ClutterInputDevice * clutter_device_manager_gdk_get_device (ClutterDeviceManager *manager, gint id) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (manager); return g_hash_table_lookup (manager_gdk->device_by_id, GINT_TO_POINTER (id)); } static ClutterInputDevice * clutter_device_manager_gdk_get_core_device (ClutterDeviceManager *manager, ClutterInputDeviceType device_type) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (manager); GdkDevice *gdk_device; gdk_device = gdk_device_manager_get_client_pointer (manager_gdk->device_manager); g_assert (gdk_device != NULL); if (device_type == CLUTTER_KEYBOARD_DEVICE) gdk_device = gdk_device_get_associated_device (gdk_device); else if (device_type != CLUTTER_POINTER_DEVICE) return NULL; return _clutter_device_manager_gdk_lookup_device (manager, gdk_device); } static void gdk_device_added (GdkDeviceManager *gdk_manager, GdkDevice *device, ClutterDeviceManager *self) { /* this will do the right thing if the device is not there */ ClutterInputDevice *clutter_device = _clutter_device_manager_gdk_lookup_device (self, device); _clutter_device_manager_add_device (self, clutter_device); } static void gdk_device_removed (GdkDeviceManager *gdk_manager, GdkDevice *device, ClutterDeviceManagerGdk *self) { ClutterInputDevice *clutter_device = g_object_get_data (G_OBJECT (device), "clutter-device"); if (clutter_device == NULL) return; self->device_cache = g_slist_remove (self->device_cache, clutter_device); g_object_unref (clutter_device); g_hash_table_remove (self->device_by_id, GINT_TO_POINTER (clutter_input_device_get_device_id (clutter_device))); _clutter_device_manager_remove_device (CLUTTER_DEVICE_MANAGER (self), clutter_device); } static void gdk_device_foreach_cb (gpointer data, gpointer user_data) { _clutter_device_manager_gdk_lookup_device (user_data, data); } static void clutter_device_manager_gdk_constructed (GObject *gobject) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (gobject); GList *all_devices; g_assert (manager_gdk->device_manager != NULL); all_devices = gdk_device_manager_list_devices (manager_gdk->device_manager, GDK_DEVICE_TYPE_MASTER); g_list_foreach (all_devices, gdk_device_foreach_cb, manager_gdk); g_list_free (all_devices); all_devices = gdk_device_manager_list_devices (manager_gdk->device_manager, GDK_DEVICE_TYPE_SLAVE); g_list_foreach (all_devices, gdk_device_foreach_cb, manager_gdk); g_list_free (all_devices); all_devices = gdk_device_manager_list_devices (manager_gdk->device_manager, GDK_DEVICE_TYPE_FLOATING); g_list_foreach (all_devices, gdk_device_foreach_cb, manager_gdk); g_list_free (all_devices); g_object_connect (manager_gdk->device_manager, "object-signal::device-added", gdk_device_added, gobject, "object-signal::device-removed", gdk_device_removed, gobject, NULL); if (G_OBJECT_CLASS (clutter_device_manager_gdk_parent_class)->constructed) G_OBJECT_CLASS (clutter_device_manager_gdk_parent_class)->constructed (gobject); } static void clutter_device_manager_gdk_set_property (GObject *gobject, guint prop_id, const GValue *value, GParamSpec *pspec) { ClutterDeviceManagerGdk *manager_gdk = CLUTTER_DEVICE_MANAGER_GDK (gobject); GdkDisplay *gdk_display; switch (prop_id) { case PROP_GDK_DISPLAY: gdk_display = GDK_DISPLAY (g_value_get_object (value)); manager_gdk->device_manager = gdk_display_get_device_manager (gdk_display); g_object_ref (manager_gdk->device_manager); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); break; } } static void clutter_device_manager_gdk_class_init (ClutterDeviceManagerGdkClass *klass) { ClutterDeviceManagerClass *manager_class; GObjectClass *gobject_class; GParamSpec *pspec; gobject_class = G_OBJECT_CLASS (klass); gobject_class->constructed = clutter_device_manager_gdk_constructed; gobject_class->set_property = clutter_device_manager_gdk_set_property; manager_class = CLUTTER_DEVICE_MANAGER_CLASS (klass); manager_class->add_device = clutter_device_manager_gdk_add_device; manager_class->remove_device = clutter_device_manager_gdk_remove_device; manager_class->get_devices = clutter_device_manager_gdk_get_devices; manager_class->get_core_device = clutter_device_manager_gdk_get_core_device; manager_class->get_device = clutter_device_manager_gdk_get_device; pspec = g_param_spec_object ("gdk-display", "GdkDisplay", "The GDK display", GDK_TYPE_DISPLAY, CLUTTER_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY); g_object_class_install_property (gobject_class, PROP_GDK_DISPLAY, pspec); } static void clutter_device_manager_gdk_init (ClutterDeviceManagerGdk *self) { self->device_by_id = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify) g_object_unref); }