/* * Wayland Support * * Copyright (C) 2015 Red Hat * * 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, see . * * Author: Carlos Garnacho */ #include "config.h" #include #include #include "wayland/meta-wayland-private.h" #include "wayland/meta-wayland-tablet-manager.h" #include "wayland/meta-wayland-tablet-seat.h" #include "wayland/meta-wayland-tablet-tool.h" #include "tablet-unstable-v2-server-protocol.h" static void unbind_resource (struct wl_resource *resource) { wl_list_remove (wl_resource_get_link (resource)); } static gboolean is_tablet_device (ClutterInputDevice *device) { ClutterInputCapabilities capabilities; if (clutter_input_device_get_device_mode (device) == CLUTTER_INPUT_MODE_LOGICAL) return FALSE; capabilities = clutter_input_device_get_capabilities (device); return (capabilities & (CLUTTER_INPUT_CAPABILITY_TABLET_TOOL | CLUTTER_INPUT_CAPABILITY_TABLET_PAD)) != 0; } static void tablet_manager_get_tablet_seat (struct wl_client *client, struct wl_resource *resource, guint32 id, struct wl_resource *seat_resource) { MetaWaylandTabletManager *tablet_manager = wl_resource_get_user_data (resource); MetaWaylandSeat *seat = wl_resource_get_user_data (seat_resource); MetaWaylandTabletSeat *tablet_seat; tablet_seat = meta_wayland_tablet_manager_ensure_seat (tablet_manager, seat); meta_wayland_tablet_seat_create_new_resource (tablet_seat, client, resource, id); } static void tablet_manager_destroy (struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy (resource); } static const struct zwp_tablet_manager_v2_interface tablet_manager_interface = { tablet_manager_get_tablet_seat, tablet_manager_destroy }; static void bind_tablet_manager (struct wl_client *client, void *data, uint32_t version, uint32_t id) { MetaWaylandCompositor *compositor = data; MetaWaylandTabletManager *tablet_manager = compositor->tablet_manager; struct wl_resource *resource; resource = wl_resource_create (client, &zwp_tablet_manager_v2_interface, MIN (version, 1), id); wl_resource_set_implementation (resource, &tablet_manager_interface, tablet_manager, unbind_resource); wl_resource_set_user_data (resource, tablet_manager); wl_list_insert (&tablet_manager->resource_list, wl_resource_get_link (resource)); } static MetaWaylandTabletManager * meta_wayland_tablet_manager_new (MetaWaylandCompositor *compositor) { MetaWaylandTabletManager *tablet_manager; tablet_manager = g_new0 (MetaWaylandTabletManager, 1); tablet_manager->compositor = compositor; tablet_manager->wl_display = compositor->wayland_display; tablet_manager->seats = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify) meta_wayland_tablet_seat_free); wl_list_init (&tablet_manager->resource_list); wl_global_create (tablet_manager->wl_display, &zwp_tablet_manager_v2_interface, 1, compositor, bind_tablet_manager); return tablet_manager; } void meta_wayland_tablet_manager_init (MetaWaylandCompositor *compositor) { compositor->tablet_manager = meta_wayland_tablet_manager_new (compositor); } void meta_wayland_tablet_manager_finalize (MetaWaylandCompositor *compositor) { g_hash_table_destroy (compositor->tablet_manager->seats); g_clear_pointer (&compositor->tablet_manager, g_free); } static MetaWaylandTabletSeat * meta_wayland_tablet_manager_lookup_seat (MetaWaylandTabletManager *manager, ClutterInputDevice *device) { MetaWaylandTabletSeat *tablet_seat; MetaWaylandSeat *seat; GHashTableIter iter; if (!device || !is_tablet_device (device)) return NULL; g_hash_table_iter_init (&iter, manager->seats); while (g_hash_table_iter_next (&iter, (gpointer*) &seat, (gpointer*) &tablet_seat)) { if (meta_wayland_tablet_seat_lookup_tablet (tablet_seat, device) || meta_wayland_tablet_seat_lookup_pad (tablet_seat, device)) return tablet_seat; } return NULL; } gboolean meta_wayland_tablet_manager_consumes_event (MetaWaylandTabletManager *manager, const ClutterEvent *event) { ClutterInputDevice *device = clutter_event_get_source_device (event); return meta_wayland_tablet_manager_lookup_seat (manager, device) != NULL; } void meta_wayland_tablet_manager_update (MetaWaylandTabletManager *manager, const ClutterEvent *event) { ClutterInputDevice *device = clutter_event_get_source_device (event); MetaWaylandTabletSeat *tablet_seat; tablet_seat = meta_wayland_tablet_manager_lookup_seat (manager, device); if (!tablet_seat) return; switch (clutter_event_type (event)) { case CLUTTER_PROXIMITY_IN: case CLUTTER_PROXIMITY_OUT: case CLUTTER_BUTTON_PRESS: case CLUTTER_BUTTON_RELEASE: case CLUTTER_MOTION: case CLUTTER_PAD_BUTTON_PRESS: case CLUTTER_PAD_BUTTON_RELEASE: case CLUTTER_PAD_RING: case CLUTTER_PAD_STRIP: meta_wayland_tablet_seat_update (tablet_seat, event); break; default: break; } } gboolean meta_wayland_tablet_manager_handle_event (MetaWaylandTabletManager *manager, const ClutterEvent *event) { ClutterInputDevice *device = clutter_event_get_source_device (event); MetaWaylandTabletSeat *tablet_seat; tablet_seat = meta_wayland_tablet_manager_lookup_seat (manager, device); if (!tablet_seat) return CLUTTER_EVENT_PROPAGATE; switch (clutter_event_type (event)) { case CLUTTER_PROXIMITY_IN: case CLUTTER_PROXIMITY_OUT: case CLUTTER_BUTTON_PRESS: case CLUTTER_BUTTON_RELEASE: case CLUTTER_MOTION: case CLUTTER_PAD_BUTTON_PRESS: case CLUTTER_PAD_BUTTON_RELEASE: case CLUTTER_PAD_RING: case CLUTTER_PAD_STRIP: return meta_wayland_tablet_seat_handle_event (tablet_seat, event); default: return CLUTTER_EVENT_PROPAGATE; } } MetaWaylandTabletSeat * meta_wayland_tablet_manager_ensure_seat (MetaWaylandTabletManager *manager, MetaWaylandSeat *seat) { MetaWaylandTabletSeat *tablet_seat; tablet_seat = g_hash_table_lookup (manager->seats, seat); if (!tablet_seat) { tablet_seat = meta_wayland_tablet_seat_new (manager, seat); g_hash_table_insert (manager->seats, seat, tablet_seat); } return tablet_seat; }