2013-05-03 18:51:22 +01:00
|
|
|
/*
|
|
|
|
* Wayland Support
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013 Intel Corporation
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "wayland/meta-wayland-seat.h"
|
2014-04-22 17:58:01 -04:00
|
|
|
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "wayland/meta-wayland-data-device.h"
|
|
|
|
#include "wayland/meta-wayland-private.h"
|
|
|
|
#include "wayland/meta-wayland-tablet-seat.h"
|
|
|
|
#include "wayland/meta-wayland-versions.h"
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2014-07-23 15:33:03 +02:00
|
|
|
#define CAPABILITY_ENABLED(prev, cur, capability) ((cur & (capability)) && !(prev & (capability)))
|
|
|
|
#define CAPABILITY_DISABLED(prev, cur, capability) ((prev & (capability)) && !(cur & (capability)))
|
|
|
|
|
2013-05-03 18:51:22 +01:00
|
|
|
static void
|
|
|
|
unbind_resource (struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
wl_list_remove (wl_resource_get_link (resource));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
seat_get_pointer (struct wl_client *client,
|
|
|
|
struct wl_resource *resource,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
MetaWaylandSeat *seat = wl_resource_get_user_data (resource);
|
2016-04-01 16:39:30 +08:00
|
|
|
MetaWaylandPointer *pointer = seat->pointer;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
wayland: Avoid a race in wl_seat capabilities
The way wl_seat capabilities work, by notifying clients of capabilities
changes, and clients consequently requesting the relevant interface
objects (pointer, keyboard, touch) is inherently racy.
On quick VT changes for example, capabilities on the seat will be added
and removed, and by the time the client receives the capability change
notification and requests the relevant keyboard, pointer or touch,
another VT switch might have occurred and the wl_pointer, wl_keyboard or
wl_touch already destroyed, leading to a protocol error which kills the
client.
To avoid this, create the objects when requested regardless of the
capabilities.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/1797
Related: https://bugzilla.gnome.org/show_bug.cgi?id=790932
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/77>
2017-11-28 10:54:08 +01:00
|
|
|
meta_wayland_pointer_create_new_resource (pointer, client, resource, id);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
seat_get_keyboard (struct wl_client *client,
|
|
|
|
struct wl_resource *resource,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
MetaWaylandSeat *seat = wl_resource_get_user_data (resource);
|
2016-04-01 16:39:30 +08:00
|
|
|
MetaWaylandKeyboard *keyboard = seat->keyboard;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
wayland: Avoid a race in wl_seat capabilities
The way wl_seat capabilities work, by notifying clients of capabilities
changes, and clients consequently requesting the relevant interface
objects (pointer, keyboard, touch) is inherently racy.
On quick VT changes for example, capabilities on the seat will be added
and removed, and by the time the client receives the capability change
notification and requests the relevant keyboard, pointer or touch,
another VT switch might have occurred and the wl_pointer, wl_keyboard or
wl_touch already destroyed, leading to a protocol error which kills the
client.
To avoid this, create the objects when requested regardless of the
capabilities.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/1797
Related: https://bugzilla.gnome.org/show_bug.cgi?id=790932
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/77>
2017-11-28 10:54:08 +01:00
|
|
|
meta_wayland_keyboard_create_new_resource (keyboard, client, resource, id);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
seat_get_touch (struct wl_client *client,
|
|
|
|
struct wl_resource *resource,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
2014-04-24 14:50:47 +02:00
|
|
|
MetaWaylandSeat *seat = wl_resource_get_user_data (resource);
|
2016-04-01 16:39:30 +08:00
|
|
|
MetaWaylandTouch *touch = seat->touch;
|
2014-04-24 14:50:47 +02:00
|
|
|
|
wayland: Avoid a race in wl_seat capabilities
The way wl_seat capabilities work, by notifying clients of capabilities
changes, and clients consequently requesting the relevant interface
objects (pointer, keyboard, touch) is inherently racy.
On quick VT changes for example, capabilities on the seat will be added
and removed, and by the time the client receives the capability change
notification and requests the relevant keyboard, pointer or touch,
another VT switch might have occurred and the wl_pointer, wl_keyboard or
wl_touch already destroyed, leading to a protocol error which kills the
client.
To avoid this, create the objects when requested regardless of the
capabilities.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/1797
Related: https://bugzilla.gnome.org/show_bug.cgi?id=790932
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/77>
2017-11-28 10:54:08 +01:00
|
|
|
meta_wayland_touch_create_new_resource (touch, client, resource, id);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2020-08-17 01:51:20 +02:00
|
|
|
static void
|
|
|
|
seat_release (struct wl_client *client,
|
|
|
|
struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
wl_resource_destroy (resource);
|
|
|
|
}
|
|
|
|
|
2014-04-17 16:36:45 -04:00
|
|
|
static const struct wl_seat_interface seat_interface = {
|
|
|
|
seat_get_pointer,
|
|
|
|
seat_get_keyboard,
|
2020-08-17 01:51:20 +02:00
|
|
|
seat_get_touch,
|
|
|
|
seat_release
|
2014-04-17 16:36:45 -04:00
|
|
|
};
|
2013-05-03 18:51:22 +01:00
|
|
|
|
|
|
|
static void
|
|
|
|
bind_seat (struct wl_client *client,
|
|
|
|
void *data,
|
|
|
|
guint32 version,
|
|
|
|
guint32 id)
|
|
|
|
{
|
|
|
|
MetaWaylandSeat *seat = data;
|
|
|
|
struct wl_resource *resource;
|
|
|
|
|
2014-08-04 10:24:59 -04:00
|
|
|
resource = wl_resource_create (client, &wl_seat_interface, version, id);
|
2013-09-10 13:03:37 +02:00
|
|
|
wl_resource_set_implementation (resource, &seat_interface, seat, unbind_resource);
|
2013-05-03 18:51:22 +01:00
|
|
|
wl_list_insert (&seat->base_resource_list, wl_resource_get_link (resource));
|
|
|
|
|
2014-07-23 15:33:03 +02:00
|
|
|
wl_seat_send_capabilities (resource, seat->capabilities);
|
2013-09-10 13:03:37 +02:00
|
|
|
|
2014-05-12 17:00:49 -04:00
|
|
|
if (version >= WL_SEAT_NAME_SINCE_VERSION)
|
2013-09-10 13:03:37 +02:00
|
|
|
wl_seat_send_name (resource, "seat0");
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2014-07-23 15:33:03 +02:00
|
|
|
static uint32_t
|
2019-10-04 14:01:19 +02:00
|
|
|
lookup_device_capabilities (ClutterSeat *seat)
|
2014-07-23 15:33:03 +02:00
|
|
|
{
|
2019-10-04 14:01:19 +02:00
|
|
|
GList *devices, *l;
|
2014-07-23 15:33:03 +02:00
|
|
|
uint32_t capabilities = 0;
|
|
|
|
|
2019-10-04 14:01:19 +02:00
|
|
|
devices = clutter_seat_list_devices (seat);
|
2014-07-23 15:33:03 +02:00
|
|
|
|
|
|
|
for (l = devices; l; l = l->next)
|
|
|
|
{
|
2022-03-08 17:26:43 +01:00
|
|
|
ClutterInputCapabilities device_capabilities;
|
2014-07-23 15:33:03 +02:00
|
|
|
|
2020-08-31 18:47:23 +02:00
|
|
|
/* Only look for physical devices, logical devices have rather generic
|
2014-07-23 15:33:03 +02:00
|
|
|
* keyboard/pointer device types, which is not truly representative of
|
2020-08-31 18:47:23 +02:00
|
|
|
* the physical devices connected to them.
|
2014-07-23 15:33:03 +02:00
|
|
|
*/
|
2020-08-31 18:47:23 +02:00
|
|
|
if (clutter_input_device_get_device_mode (l->data) == CLUTTER_INPUT_MODE_LOGICAL)
|
2014-07-23 15:33:03 +02:00
|
|
|
continue;
|
|
|
|
|
2022-03-08 17:26:43 +01:00
|
|
|
device_capabilities = clutter_input_device_get_capabilities (l->data);
|
|
|
|
|
|
|
|
if (device_capabilities & CLUTTER_INPUT_CAPABILITY_POINTER)
|
|
|
|
capabilities |= WL_SEAT_CAPABILITY_POINTER;
|
|
|
|
if (device_capabilities & CLUTTER_INPUT_CAPABILITY_KEYBOARD)
|
|
|
|
capabilities |= WL_SEAT_CAPABILITY_KEYBOARD;
|
|
|
|
if (device_capabilities & CLUTTER_INPUT_CAPABILITY_TOUCH)
|
|
|
|
capabilities |= WL_SEAT_CAPABILITY_TOUCH;
|
2014-07-23 15:33:03 +02:00
|
|
|
}
|
|
|
|
|
2019-10-04 14:01:19 +02:00
|
|
|
g_list_free (devices);
|
|
|
|
|
2014-07-23 15:33:03 +02:00
|
|
|
return capabilities;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_wayland_seat_set_capabilities (MetaWaylandSeat *seat,
|
|
|
|
uint32_t flags)
|
|
|
|
{
|
|
|
|
struct wl_resource *resource;
|
|
|
|
uint32_t prev_flags;
|
|
|
|
|
|
|
|
prev_flags = seat->capabilities;
|
|
|
|
|
|
|
|
if (prev_flags == flags)
|
|
|
|
return;
|
|
|
|
|
|
|
|
seat->capabilities = flags;
|
|
|
|
|
|
|
|
if (CAPABILITY_ENABLED (prev_flags, flags, WL_SEAT_CAPABILITY_POINTER))
|
2016-09-12 23:20:36 +08:00
|
|
|
meta_wayland_pointer_enable (seat->pointer);
|
2014-07-23 15:33:03 +02:00
|
|
|
else if (CAPABILITY_DISABLED (prev_flags, flags, WL_SEAT_CAPABILITY_POINTER))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_pointer_disable (seat->pointer);
|
2014-07-23 15:33:03 +02:00
|
|
|
|
|
|
|
if (CAPABILITY_ENABLED (prev_flags, flags, WL_SEAT_CAPABILITY_KEYBOARD))
|
|
|
|
{
|
|
|
|
MetaDisplay *display;
|
|
|
|
|
2016-09-12 23:20:36 +08:00
|
|
|
meta_wayland_keyboard_enable (seat->keyboard);
|
2014-07-23 15:33:03 +02:00
|
|
|
display = meta_get_display ();
|
|
|
|
|
|
|
|
/* Post-initialization, ensure the input focus is in sync */
|
|
|
|
if (display)
|
|
|
|
meta_display_sync_wayland_input_focus (display);
|
|
|
|
}
|
|
|
|
else if (CAPABILITY_DISABLED (prev_flags, flags, WL_SEAT_CAPABILITY_KEYBOARD))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_keyboard_disable (seat->keyboard);
|
2014-07-23 15:33:03 +02:00
|
|
|
|
|
|
|
if (CAPABILITY_ENABLED (prev_flags, flags, WL_SEAT_CAPABILITY_TOUCH))
|
2016-09-12 23:20:36 +08:00
|
|
|
meta_wayland_touch_enable (seat->touch);
|
2014-07-23 15:33:03 +02:00
|
|
|
else if (CAPABILITY_DISABLED (prev_flags, flags, WL_SEAT_CAPABILITY_TOUCH))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_touch_disable (seat->touch);
|
2014-07-23 15:33:03 +02:00
|
|
|
|
|
|
|
/* Broadcast capability changes */
|
|
|
|
wl_resource_for_each (resource, &seat->base_resource_list)
|
|
|
|
{
|
|
|
|
wl_seat_send_capabilities (resource, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2019-10-04 14:01:19 +02:00
|
|
|
meta_wayland_seat_update_capabilities (MetaWaylandSeat *seat,
|
|
|
|
ClutterSeat *clutter_seat)
|
2014-07-23 15:33:03 +02:00
|
|
|
{
|
|
|
|
uint32_t capabilities;
|
|
|
|
|
2019-10-04 14:01:19 +02:00
|
|
|
capabilities = lookup_device_capabilities (clutter_seat);
|
2014-07-23 15:33:03 +02:00
|
|
|
meta_wayland_seat_set_capabilities (seat, capabilities);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2019-10-04 14:01:19 +02:00
|
|
|
meta_wayland_seat_devices_updated (ClutterSeat *clutter_seat,
|
|
|
|
ClutterInputDevice *input_device,
|
|
|
|
MetaWaylandSeat *seat)
|
2014-07-23 15:33:03 +02:00
|
|
|
{
|
2019-10-04 14:01:19 +02:00
|
|
|
meta_wayland_seat_update_capabilities (seat, clutter_seat);
|
2014-07-23 15:33:03 +02:00
|
|
|
}
|
|
|
|
|
2014-04-22 18:05:44 -04:00
|
|
|
static MetaWaylandSeat *
|
2016-01-26 13:31:48 +01:00
|
|
|
meta_wayland_seat_new (MetaWaylandCompositor *compositor,
|
|
|
|
struct wl_display *display)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
|
|
|
MetaWaylandSeat *seat = g_new0 (MetaWaylandSeat, 1);
|
2019-10-04 14:01:19 +02:00
|
|
|
ClutterSeat *clutter_seat;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
|
|
|
wl_list_init (&seat->base_resource_list);
|
2014-07-23 15:33:03 +02:00
|
|
|
seat->wl_display = display;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2016-09-12 23:20:36 +08:00
|
|
|
seat->pointer = g_object_new (META_TYPE_WAYLAND_POINTER,
|
|
|
|
"seat", seat,
|
|
|
|
NULL);
|
|
|
|
seat->keyboard = g_object_new (META_TYPE_WAYLAND_KEYBOARD,
|
|
|
|
"seat", seat,
|
|
|
|
NULL);
|
|
|
|
seat->touch = g_object_new (META_TYPE_WAYLAND_TOUCH,
|
|
|
|
"seat", seat,
|
|
|
|
NULL);
|
2016-04-01 16:39:30 +08:00
|
|
|
|
2018-08-13 18:59:02 +02:00
|
|
|
seat->text_input = meta_wayland_text_input_new (seat);
|
2017-12-06 12:55:25 +01:00
|
|
|
|
2014-07-10 10:31:08 -04:00
|
|
|
meta_wayland_data_device_init (&seat->data_device);
|
2020-05-13 18:07:27 +02:00
|
|
|
meta_wayland_data_device_primary_init (&seat->primary_data_device);
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2019-10-04 14:01:19 +02:00
|
|
|
clutter_seat = clutter_backend_get_default_seat (clutter_get_default_backend ());
|
|
|
|
meta_wayland_seat_update_capabilities (seat, clutter_seat);
|
|
|
|
g_signal_connect (clutter_seat, "device-added",
|
2014-07-23 15:33:03 +02:00
|
|
|
G_CALLBACK (meta_wayland_seat_devices_updated), seat);
|
2019-10-04 14:01:19 +02:00
|
|
|
g_signal_connect (clutter_seat, "device-removed",
|
2014-07-23 15:33:03 +02:00
|
|
|
G_CALLBACK (meta_wayland_seat_devices_updated), seat);
|
|
|
|
|
2013-09-10 13:45:27 +02:00
|
|
|
wl_global_create (display, &wl_seat_interface, META_WL_SEAT_VERSION, seat, bind_seat);
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2016-01-26 13:31:48 +01:00
|
|
|
meta_wayland_tablet_manager_ensure_seat (compositor->tablet_manager, seat);
|
|
|
|
|
2013-05-03 18:51:22 +01:00
|
|
|
return seat;
|
|
|
|
}
|
|
|
|
|
2014-04-22 18:05:44 -04:00
|
|
|
void
|
|
|
|
meta_wayland_seat_init (MetaWaylandCompositor *compositor)
|
|
|
|
{
|
2016-01-26 13:31:48 +01:00
|
|
|
compositor->seat = meta_wayland_seat_new (compositor,
|
|
|
|
compositor->wayland_display);
|
2014-04-22 18:05:44 -04:00
|
|
|
}
|
|
|
|
|
2014-04-17 18:57:25 -04:00
|
|
|
void
|
|
|
|
meta_wayland_seat_free (MetaWaylandSeat *seat)
|
|
|
|
{
|
2019-10-04 14:01:19 +02:00
|
|
|
ClutterSeat *clutter_seat;
|
2014-07-23 15:33:03 +02:00
|
|
|
|
2019-10-04 14:01:19 +02:00
|
|
|
clutter_seat = clutter_backend_get_default_seat (clutter_get_default_backend ());
|
|
|
|
g_signal_handlers_disconnect_by_data (clutter_seat, seat);
|
2014-07-23 15:33:03 +02:00
|
|
|
meta_wayland_seat_set_capabilities (seat, 0);
|
2014-04-17 18:57:25 -04:00
|
|
|
|
2016-04-01 16:39:30 +08:00
|
|
|
g_object_unref (seat->pointer);
|
|
|
|
g_object_unref (seat->keyboard);
|
|
|
|
g_object_unref (seat->touch);
|
2021-04-12 14:57:27 +02:00
|
|
|
|
2018-08-13 18:59:02 +02:00
|
|
|
meta_wayland_text_input_destroy (seat->text_input);
|
2016-04-01 16:39:30 +08:00
|
|
|
|
2019-05-17 14:56:31 -05:00
|
|
|
g_free (seat);
|
2014-04-17 18:57:25 -04:00
|
|
|
}
|
|
|
|
|
2015-09-17 16:13:31 +02:00
|
|
|
static gboolean
|
|
|
|
event_is_synthesized_crossing (const ClutterEvent *event)
|
|
|
|
{
|
|
|
|
ClutterInputDevice *device;
|
|
|
|
|
|
|
|
if (event->type != CLUTTER_ENTER && event->type != CLUTTER_LEAVE)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
device = clutter_event_get_source_device (event);
|
2020-08-31 18:47:23 +02:00
|
|
|
return clutter_input_device_get_device_mode (device) == CLUTTER_INPUT_MODE_LOGICAL;
|
2015-09-17 16:13:31 +02:00
|
|
|
}
|
|
|
|
|
2015-02-18 10:43:18 -05:00
|
|
|
static gboolean
|
|
|
|
event_from_supported_hardware_device (MetaWaylandSeat *seat,
|
|
|
|
const ClutterEvent *event)
|
|
|
|
{
|
2022-03-08 17:26:43 +01:00
|
|
|
ClutterInputDevice *input_device;
|
|
|
|
ClutterInputMode input_mode;
|
|
|
|
ClutterInputCapabilities capabilities;
|
|
|
|
gboolean hardware_device = FALSE;
|
|
|
|
gboolean supported_device = FALSE;
|
2015-02-18 10:43:18 -05:00
|
|
|
|
|
|
|
input_device = clutter_event_get_source_device (event);
|
|
|
|
|
|
|
|
if (input_device == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
input_mode = clutter_input_device_get_device_mode (input_device);
|
|
|
|
|
2020-08-31 18:47:23 +02:00
|
|
|
if (input_mode != CLUTTER_INPUT_MODE_PHYSICAL)
|
2015-02-18 10:43:18 -05:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
hardware_device = TRUE;
|
|
|
|
|
2022-03-08 17:26:43 +01:00
|
|
|
capabilities = clutter_input_device_get_capabilities (input_device);
|
2015-02-18 10:43:18 -05:00
|
|
|
|
2022-03-08 17:26:43 +01:00
|
|
|
if ((capabilities &
|
|
|
|
(CLUTTER_INPUT_CAPABILITY_POINTER |
|
|
|
|
CLUTTER_INPUT_CAPABILITY_KEYBOARD |
|
|
|
|
CLUTTER_INPUT_CAPABILITY_TOUCH)) != 0)
|
|
|
|
supported_device = TRUE;
|
2015-02-18 10:43:18 -05:00
|
|
|
|
|
|
|
out:
|
|
|
|
return hardware_device && supported_device;
|
|
|
|
}
|
|
|
|
|
2014-04-17 19:07:43 -04:00
|
|
|
void
|
|
|
|
meta_wayland_seat_update (MetaWaylandSeat *seat,
|
|
|
|
const ClutterEvent *event)
|
|
|
|
{
|
2018-04-24 14:22:03 +02:00
|
|
|
if (!(clutter_event_get_flags (event) & CLUTTER_EVENT_FLAG_INPUT_METHOD) &&
|
|
|
|
!event_from_supported_hardware_device (seat, event) &&
|
2015-09-17 16:13:31 +02:00
|
|
|
!event_is_synthesized_crossing (event))
|
2015-02-18 10:43:18 -05:00
|
|
|
return;
|
|
|
|
|
2014-04-17 19:07:43 -04:00
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case CLUTTER_MOTION:
|
|
|
|
case CLUTTER_BUTTON_PRESS:
|
|
|
|
case CLUTTER_BUTTON_RELEASE:
|
|
|
|
case CLUTTER_SCROLL:
|
2015-09-17 16:13:31 +02:00
|
|
|
case CLUTTER_ENTER:
|
|
|
|
case CLUTTER_LEAVE:
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_pointer (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_pointer_update (seat->pointer, event);
|
2014-04-17 19:07:43 -04:00
|
|
|
break;
|
2014-04-17 17:52:11 -04:00
|
|
|
|
2014-04-17 19:07:43 -04:00
|
|
|
case CLUTTER_KEY_PRESS:
|
|
|
|
case CLUTTER_KEY_RELEASE:
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_keyboard (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_keyboard_update (seat->keyboard, (const ClutterKeyEvent *) event);
|
2014-04-17 19:07:43 -04:00
|
|
|
break;
|
2014-04-17 17:52:11 -04:00
|
|
|
|
2014-04-24 14:50:47 +02:00
|
|
|
case CLUTTER_TOUCH_BEGIN:
|
|
|
|
case CLUTTER_TOUCH_UPDATE:
|
|
|
|
case CLUTTER_TOUCH_END:
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_touch (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_touch_update (seat->touch, event);
|
2014-04-24 14:50:47 +02:00
|
|
|
break;
|
|
|
|
|
2014-04-17 19:07:43 -04:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-13 21:41:29 +01:00
|
|
|
gboolean
|
|
|
|
meta_wayland_seat_handle_event (MetaWaylandSeat *seat,
|
|
|
|
const ClutterEvent *event)
|
|
|
|
{
|
2018-04-24 14:22:03 +02:00
|
|
|
if (!(clutter_event_get_flags (event) & CLUTTER_EVENT_FLAG_INPUT_METHOD) &&
|
|
|
|
!event_from_supported_hardware_device (seat, event))
|
2015-02-18 10:43:18 -05:00
|
|
|
return FALSE;
|
|
|
|
|
2021-07-27 16:50:50 +02:00
|
|
|
if (event->type == CLUTTER_BUTTON_PRESS ||
|
|
|
|
event->type == CLUTTER_TOUCH_BEGIN)
|
|
|
|
{
|
|
|
|
meta_wayland_text_input_handle_event (seat->text_input, event);
|
|
|
|
}
|
|
|
|
|
2013-05-03 18:51:22 +01:00
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case CLUTTER_MOTION:
|
|
|
|
case CLUTTER_BUTTON_PRESS:
|
|
|
|
case CLUTTER_BUTTON_RELEASE:
|
2014-04-17 19:10:39 -04:00
|
|
|
case CLUTTER_SCROLL:
|
2015-07-22 16:43:25 +02:00
|
|
|
case CLUTTER_TOUCHPAD_SWIPE:
|
2015-07-22 16:46:55 +02:00
|
|
|
case CLUTTER_TOUCHPAD_PINCH:
|
2021-04-18 20:24:52 +02:00
|
|
|
case CLUTTER_TOUCHPAD_HOLD:
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_pointer (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
return meta_wayland_pointer_handle_event (seat->pointer, event);
|
2014-04-17 19:10:39 -04:00
|
|
|
|
2018-01-17 14:00:44 +01:00
|
|
|
break;
|
2013-05-03 18:51:22 +01:00
|
|
|
case CLUTTER_KEY_PRESS:
|
|
|
|
case CLUTTER_KEY_RELEASE:
|
2018-08-13 18:59:02 +02:00
|
|
|
if (meta_wayland_text_input_handle_event (seat->text_input, event))
|
|
|
|
return TRUE;
|
|
|
|
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_keyboard (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
return meta_wayland_keyboard_handle_event (seat->keyboard,
|
wayland: Don't handle input events after capability was removed
The seat capability updating is synchronous, but input events are
asynchronous (first queued then emitted). This means we may end up in a
situation where we from libinput first may receive a key event,
immediately followed by a device-removed event. Clutter will first
queue the key event, then remove the device, immediately triggering the
seat capability removal.
Later, when the clutter stage processes the queued events, the
previously queued key event will be processed, eventually making it
into MetaWaylandSeat. Before this patch, MetaWaylandSeat would still
forward the key event to MetaWaylandKeyboard, even though it had
'released' it. Doing this would cause referencing potentially freed
memory, such as the xkb state that was unreferenced when the seat
removed the capability.
In order to avoid processing these lingering events, for now, just drop
them on the floor if the capability has been removed.
Eventually, the event queuing etc needs to be redesigned to work better
when used in a Wayland compositor, but for now at least don't access
freed memory.
https://bugzilla.gnome.org/show_bug.cgi?id=770727
2016-09-02 11:50:29 +08:00
|
|
|
(const ClutterKeyEvent *) event);
|
2018-01-17 14:00:44 +01:00
|
|
|
break;
|
2014-04-24 14:50:47 +02:00
|
|
|
case CLUTTER_TOUCH_BEGIN:
|
|
|
|
case CLUTTER_TOUCH_UPDATE:
|
|
|
|
case CLUTTER_TOUCH_END:
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_touch (seat))
|
2016-04-01 16:39:30 +08:00
|
|
|
return meta_wayland_touch_handle_event (seat->touch, event);
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2020-02-17 11:25:14 +01:00
|
|
|
break;
|
|
|
|
case CLUTTER_IM_COMMIT:
|
|
|
|
case CLUTTER_IM_DELETE:
|
|
|
|
case CLUTTER_IM_PREEDIT:
|
|
|
|
if (meta_wayland_text_input_handle_event (seat->text_input, event))
|
|
|
|
return TRUE;
|
|
|
|
|
2018-01-17 14:00:44 +01:00
|
|
|
break;
|
2013-05-03 18:51:22 +01:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2013-09-04 15:01:11 +02:00
|
|
|
|
|
|
|
return FALSE;
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2014-07-10 10:13:54 -04:00
|
|
|
void
|
|
|
|
meta_wayland_seat_set_input_focus (MetaWaylandSeat *seat,
|
|
|
|
MetaWaylandSurface *surface)
|
|
|
|
{
|
2016-05-10 17:29:57 +02:00
|
|
|
MetaWaylandTabletSeat *tablet_seat;
|
|
|
|
MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default ();
|
|
|
|
|
2016-09-21 11:52:41 +08:00
|
|
|
if (meta_wayland_seat_has_keyboard (seat))
|
2016-05-10 17:29:57 +02:00
|
|
|
{
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_keyboard_set_focus (seat->keyboard, surface);
|
2016-05-10 17:29:57 +02:00
|
|
|
meta_wayland_data_device_set_keyboard_focus (&seat->data_device);
|
2020-05-13 18:07:27 +02:00
|
|
|
meta_wayland_data_device_primary_set_keyboard_focus (&seat->primary_data_device);
|
2016-05-10 17:29:57 +02:00
|
|
|
}
|
2015-02-18 10:43:42 -05:00
|
|
|
|
2016-05-10 17:29:57 +02:00
|
|
|
tablet_seat = meta_wayland_tablet_manager_ensure_seat (compositor->tablet_manager, seat);
|
|
|
|
meta_wayland_tablet_seat_set_pad_focus (tablet_seat, surface);
|
2017-12-06 12:55:25 +01:00
|
|
|
|
2018-08-13 18:59:02 +02:00
|
|
|
meta_wayland_text_input_set_focus (seat->text_input, surface);
|
2014-07-10 10:13:54 -04:00
|
|
|
}
|
|
|
|
|
2014-05-22 10:57:02 -04:00
|
|
|
gboolean
|
2014-07-22 01:28:39 +02:00
|
|
|
meta_wayland_seat_get_grab_info (MetaWaylandSeat *seat,
|
2016-04-04 18:07:42 +02:00
|
|
|
MetaWaylandSurface *surface,
|
|
|
|
uint32_t serial,
|
|
|
|
gboolean require_pressed,
|
|
|
|
gfloat *x,
|
|
|
|
gfloat *y)
|
2014-05-22 10:57:02 -04:00
|
|
|
{
|
2017-06-23 08:02:21 -07:00
|
|
|
MetaWaylandCompositor *compositor;
|
|
|
|
MetaWaylandTabletSeat *tablet_seat;
|
|
|
|
GList *tools, *l;
|
2014-07-22 01:28:39 +02:00
|
|
|
|
2017-06-23 08:02:21 -07:00
|
|
|
compositor = meta_wayland_compositor_get_default ();
|
|
|
|
tablet_seat = meta_wayland_tablet_manager_ensure_seat (compositor->tablet_manager, seat);
|
|
|
|
tools = g_hash_table_get_values (tablet_seat->tools);
|
2014-07-22 01:28:39 +02:00
|
|
|
|
2017-06-23 08:02:21 -07:00
|
|
|
if (meta_wayland_seat_has_touch (seat))
|
2014-07-22 01:28:39 +02:00
|
|
|
{
|
2017-06-23 08:02:21 -07:00
|
|
|
ClutterEventSequence *sequence;
|
|
|
|
sequence = meta_wayland_touch_find_grab_sequence (seat->touch,
|
|
|
|
surface,
|
|
|
|
serial);
|
|
|
|
if (sequence)
|
|
|
|
{
|
|
|
|
meta_wayland_touch_get_press_coords (seat->touch, sequence, x, y);
|
|
|
|
return TRUE;
|
|
|
|
}
|
2014-07-22 01:28:39 +02:00
|
|
|
}
|
2015-02-18 10:43:42 -05:00
|
|
|
|
2017-06-23 08:02:21 -07:00
|
|
|
if (meta_wayland_seat_has_pointer (seat))
|
|
|
|
{
|
|
|
|
if ((!require_pressed || seat->pointer->button_count > 0) &&
|
|
|
|
meta_wayland_pointer_can_grab_surface (seat->pointer, surface, serial))
|
2015-02-18 10:43:42 -05:00
|
|
|
{
|
|
|
|
if (x)
|
2016-04-01 16:39:30 +08:00
|
|
|
*x = seat->pointer->grab_x;
|
2015-02-18 10:43:42 -05:00
|
|
|
if (y)
|
2016-04-01 16:39:30 +08:00
|
|
|
*y = seat->pointer->grab_y;
|
2017-06-23 08:02:21 -07:00
|
|
|
|
|
|
|
return TRUE;
|
2015-02-18 10:43:42 -05:00
|
|
|
}
|
|
|
|
}
|
2014-07-22 01:28:39 +02:00
|
|
|
|
2017-06-23 08:02:21 -07:00
|
|
|
for (l = tools; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaWaylandTabletTool *tool = l->data;
|
|
|
|
|
|
|
|
if ((!require_pressed || tool->button_count > 0) &&
|
|
|
|
meta_wayland_tablet_tool_can_grab_surface (tool, surface, serial))
|
|
|
|
{
|
|
|
|
if (x)
|
|
|
|
*x = tool->grab_x;
|
|
|
|
if (y)
|
|
|
|
*y = tool->grab_y;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
2014-05-22 10:57:02 -04:00
|
|
|
}
|
2015-10-09 16:42:06 +02:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_seat_can_popup (MetaWaylandSeat *seat,
|
|
|
|
uint32_t serial)
|
|
|
|
{
|
2019-10-28 18:07:31 +01:00
|
|
|
MetaWaylandCompositor *compositor;
|
|
|
|
MetaWaylandTabletSeat *tablet_seat;
|
|
|
|
|
|
|
|
compositor = meta_wayland_compositor_get_default ();
|
|
|
|
tablet_seat =
|
|
|
|
meta_wayland_tablet_manager_ensure_seat (compositor->tablet_manager, seat);
|
|
|
|
|
2016-04-01 16:39:30 +08:00
|
|
|
return (meta_wayland_pointer_can_popup (seat->pointer, serial) ||
|
|
|
|
meta_wayland_keyboard_can_popup (seat->keyboard, serial) ||
|
2019-10-28 18:07:31 +01:00
|
|
|
meta_wayland_touch_can_popup (seat->touch, serial) ||
|
|
|
|
meta_wayland_tablet_seat_can_popup (tablet_seat, serial));
|
2015-10-09 16:42:06 +02:00
|
|
|
}
|
2016-09-12 23:12:40 +08:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_seat_has_keyboard (MetaWaylandSeat *seat)
|
|
|
|
{
|
|
|
|
return (seat->capabilities & WL_SEAT_CAPABILITY_KEYBOARD) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_seat_has_pointer (MetaWaylandSeat *seat)
|
|
|
|
{
|
|
|
|
return (seat->capabilities & WL_SEAT_CAPABILITY_POINTER) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_seat_has_touch (MetaWaylandSeat *seat)
|
|
|
|
{
|
|
|
|
return (seat->capabilities & WL_SEAT_CAPABILITY_TOUCH) != 0;
|
|
|
|
}
|