Add a new GDK backend
This commit introduces a new flavour for Clutter, that uses GDK
for handling all window system specific interactions (except for
creating the cogl context, as cogl does not know about GDK), including
in particular events. This is not compatible with the X11 (glx)
flavour, and this is reflected by the different soname (libclutter-gdk-1.0.so),
as all X11 specific functions and classes are not available. If you
wish to be compatible, you should check for CLUTTER_WINDOWING_X11.
Other than that, this backend should be on feature parity with X11,
including XInput 2, XSettings and EMWH (with much, much less code)
https://bugzilla.gnome.org/show_bug.cgi?id=657434
2011-08-26 03:09:18 +02:00
|
|
|
/*
|
|
|
|
* Clutter.
|
|
|
|
*
|
|
|
|
* An OpenGL based 'interactive canvas' library.
|
|
|
|
*
|
|
|
|
* Copyright © 2011 Intel Corp.
|
|
|
|
* 2011 Giovanni Campagna <scampa.giovanni@gmail.com>
|
|
|
|
*
|
|
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* Author: Emmanuele Bassi <ebassi@linux.intel.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2012-05-16 17:36:38 +08:00
|
|
|
#ifdef HAVE_STDINT_H
|
Add a new GDK backend
This commit introduces a new flavour for Clutter, that uses GDK
for handling all window system specific interactions (except for
creating the cogl context, as cogl does not know about GDK), including
in particular events. This is not compatible with the X11 (glx)
flavour, and this is reflected by the different soname (libclutter-gdk-1.0.so),
as all X11 specific functions and classes are not available. If you
wish to be compatible, you should check for CLUTTER_WINDOWING_X11.
Other than that, this backend should be on feature parity with X11,
including XInput 2, XSettings and EMWH (with much, much less code)
https://bugzilla.gnome.org/show_bug.cgi?id=657434
2011-08-26 03:09:18 +02:00
|
|
|
#include <stdint.h>
|
2012-05-16 17:36:38 +08:00
|
|
|
#endif
|
Add a new GDK backend
This commit introduces a new flavour for Clutter, that uses GDK
for handling all window system specific interactions (except for
creating the cogl context, as cogl does not know about GDK), including
in particular events. This is not compatible with the X11 (glx)
flavour, and this is reflected by the different soname (libclutter-gdk-1.0.so),
as all X11 specific functions and classes are not available. If you
wish to be compatible, you should check for CLUTTER_WINDOWING_X11.
Other than that, this backend should be on feature parity with X11,
including XInput 2, XSettings and EMWH (with much, much less code)
https://bugzilla.gnome.org/show_bug.cgi?id=657434
2011-08-26 03:09:18 +02:00
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (ClutterDeviceManagerGdk, clutter_device_manager_gdk, CLUTTER_TYPE_DEVICE_MANAGER)
|
|
|
|
|
|
|
|
enum {
|
|
|
|
PROP_0,
|
|
|
|
PROP_GDK_DISPLAY,
|
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|