753 lines
25 KiB
C
753 lines
25 KiB
C
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
||
|
|
||
|
/*
|
||
|
* Copyright (C) 2021 Red Hat Inc.
|
||
|
*
|
||
|
* 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"
|
||
|
|
||
|
#include "backends/meta-backend-private.h"
|
||
|
#include "backends/meta-eis-client.h"
|
||
|
#include "backends/meta-monitor-manager-private.h"
|
||
|
#include "backends/meta-viewport-info.h"
|
||
|
|
||
|
#include "clutter/clutter-mutter.h"
|
||
|
#include "core/meta-anonymous-file.h"
|
||
|
|
||
|
#define MAX_BUTTON 128
|
||
|
#define MAX_KEY 0x2ff /* KEY_MAX as of 5.13 */
|
||
|
|
||
|
typedef struct _MetaEisDevice MetaEisDevice;
|
||
|
|
||
|
struct _MetaEisDevice
|
||
|
{
|
||
|
struct eis_device *eis_device;
|
||
|
ClutterVirtualInputDevice *device;
|
||
|
|
||
|
guchar button_state[(MAX_BUTTON + 7) / 8];
|
||
|
guchar key_state[(MAX_KEY + 7) / 8];
|
||
|
};
|
||
|
|
||
|
struct _MetaEisClient
|
||
|
{
|
||
|
GObject parent_instance;
|
||
|
MetaEis *meta_eis;
|
||
|
|
||
|
MetaViewportInfo *viewports;
|
||
|
struct eis_client *eis_client;
|
||
|
struct eis_seat *eis_seat;
|
||
|
|
||
|
GHashTable *eis_devices; /* eis_device => MetaEisDevice*/
|
||
|
};
|
||
|
|
||
|
G_DEFINE_TYPE (MetaEisClient, meta_eis_client, G_TYPE_OBJECT)
|
||
|
|
||
|
typedef void (* MetaEisDeviceConfigFunc) (MetaEisClient *meta_eis_client,
|
||
|
struct eis_device *device);
|
||
|
|
||
|
static bool
|
||
|
bit_is_set (const guchar *array,
|
||
|
int bit)
|
||
|
{
|
||
|
return !!(array[bit / 8] & (1 << (bit % 8)));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
bit_set (guchar *array,
|
||
|
int bit)
|
||
|
{
|
||
|
array[bit / 8] |= (1 << (bit % 8));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
bit_clear (guchar *array,
|
||
|
int bit)
|
||
|
{
|
||
|
array[bit / 8] &= ~(1 << (bit % 8));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
notify_key (MetaEisDevice *meta_eis_device,
|
||
|
uint32_t key,
|
||
|
gboolean is_press)
|
||
|
{
|
||
|
ClutterKeyState state;
|
||
|
|
||
|
state = is_press ? CLUTTER_KEY_STATE_PRESSED : CLUTTER_KEY_STATE_RELEASED;
|
||
|
clutter_virtual_input_device_notify_key (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
key,
|
||
|
state);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
notify_button (MetaEisDevice *meta_eis_device,
|
||
|
uint32_t button,
|
||
|
gboolean is_press)
|
||
|
{
|
||
|
ClutterButtonState state;
|
||
|
|
||
|
state = is_press ? CLUTTER_BUTTON_STATE_PRESSED : CLUTTER_BUTTON_STATE_RELEASED;
|
||
|
clutter_virtual_input_device_notify_button (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
button,
|
||
|
state);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
remove_device (MetaEisClient *meta_eis_client,
|
||
|
struct eis_device *eis_device,
|
||
|
gboolean remove_from_hashtable)
|
||
|
{
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
struct eis_keymap *eis_keymap = eis_device_keyboard_get_keymap (eis_device);
|
||
|
|
||
|
if (eis_keymap)
|
||
|
{
|
||
|
MetaAnonymousFile *f = eis_keymap_get_user_data (eis_keymap);
|
||
|
if (f)
|
||
|
meta_anonymous_file_free (f);
|
||
|
}
|
||
|
|
||
|
eis_device_pause (eis_device);
|
||
|
eis_device_remove (eis_device);
|
||
|
g_clear_pointer (&meta_eis_device->eis_device, eis_device_unref);
|
||
|
g_clear_object (&meta_eis_device->device);
|
||
|
|
||
|
if (remove_from_hashtable)
|
||
|
g_hash_table_remove (meta_eis_client->eis_devices, eis_device);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
drop_device (gpointer htkey,
|
||
|
gpointer value,
|
||
|
gpointer data)
|
||
|
{
|
||
|
MetaEisClient *meta_eis_client = data;
|
||
|
struct eis_device *eis_device = htkey;
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
uint32_t key, button;
|
||
|
|
||
|
for (key = 0; key < MAX_KEY; key++)
|
||
|
{
|
||
|
if (bit_is_set (meta_eis_device->key_state, key))
|
||
|
notify_key (meta_eis_device, key, FALSE);
|
||
|
}
|
||
|
|
||
|
for (button = 0; button < MAX_BUTTON; button++)
|
||
|
{
|
||
|
if (bit_is_set (meta_eis_device->button_state, key))
|
||
|
notify_button (meta_eis_device, button, FALSE);
|
||
|
}
|
||
|
|
||
|
remove_device (meta_eis_client, eis_device, FALSE);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_device_free (MetaEisDevice *device)
|
||
|
{
|
||
|
eis_device_unref (device->eis_device);
|
||
|
free (device);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
configure_rel (MetaEisClient *meta_eis_client,
|
||
|
struct eis_device *eis_device)
|
||
|
{
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_POINTER);
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_BUTTON);
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_SCROLL);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
configure_keyboard (MetaEisClient *meta_eis_client,
|
||
|
struct eis_device *eis_device)
|
||
|
{
|
||
|
size_t len;
|
||
|
MetaAnonymousFile *f;
|
||
|
int fd = -1;
|
||
|
char *data;
|
||
|
struct xkb_keymap *xkb_keymap;
|
||
|
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_KEYBOARD);
|
||
|
|
||
|
xkb_keymap =
|
||
|
meta_backend_get_keymap (meta_eis_get_backend (meta_eis_client->meta_eis));
|
||
|
if (!xkb_keymap)
|
||
|
return;
|
||
|
|
||
|
data = xkb_keymap_get_as_string (xkb_keymap, XKB_KEYMAP_FORMAT_TEXT_V1);
|
||
|
if (!data)
|
||
|
return;
|
||
|
|
||
|
len = strlen (data);
|
||
|
f = meta_anonymous_file_new (len, (uint8_t*)data);
|
||
|
if (f)
|
||
|
fd = meta_anonymous_file_open_fd (f, META_ANONYMOUS_FILE_MAPMODE_SHARED);
|
||
|
|
||
|
g_free (data);
|
||
|
if (fd != -1)
|
||
|
{
|
||
|
struct eis_keymap *eis_keymap;
|
||
|
|
||
|
eis_keymap = eis_device_new_keymap (eis_device, EIS_KEYMAP_TYPE_XKB,
|
||
|
fd, len);
|
||
|
/* libeis dup()s the fd */
|
||
|
meta_anonymous_file_close_fd (fd);
|
||
|
/* The memfile must be kept alive while the device is alive */
|
||
|
eis_keymap_set_user_data (eis_keymap, f);
|
||
|
eis_keymap_add (eis_keymap);
|
||
|
eis_keymap_unref (eis_keymap);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
configure_abs (MetaEisClient *meta_eis_client,
|
||
|
struct eis_device *eis_device)
|
||
|
{
|
||
|
int idx = 0;
|
||
|
cairo_rectangle_int_t rect;
|
||
|
float scale;
|
||
|
|
||
|
if (!meta_eis_client->viewports)
|
||
|
return; /* FIXME: should be an error */
|
||
|
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_POINTER_ABSOLUTE);
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_BUTTON);
|
||
|
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_SCROLL);
|
||
|
|
||
|
while (meta_viewport_info_get_view_info (meta_eis_client->viewports, idx++, &rect, &scale))
|
||
|
{
|
||
|
struct eis_region *r = eis_device_new_region (eis_device);
|
||
|
eis_region_set_offset (r, rect.x, rect.y);
|
||
|
eis_region_set_size (r, rect.width, rect.height);
|
||
|
eis_region_set_physical_scale (r, scale);
|
||
|
eis_region_add (r);
|
||
|
eis_region_unref (r);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
add_device (MetaEisClient *meta_eis_client,
|
||
|
struct eis_seat *eis_seat,
|
||
|
ClutterInputDeviceType type,
|
||
|
const char *name_suffix,
|
||
|
MetaEisDeviceConfigFunc extra_config_func)
|
||
|
{
|
||
|
MetaBackend *backend = meta_eis_get_backend (meta_eis_client->meta_eis);
|
||
|
MetaEisDevice *meta_eis_device;
|
||
|
ClutterSeat *seat = meta_backend_get_default_seat (backend);
|
||
|
ClutterVirtualInputDevice *device;
|
||
|
struct eis_device *eis_device;
|
||
|
gchar *name;
|
||
|
|
||
|
device = clutter_seat_create_virtual_device (seat, type);
|
||
|
eis_device = eis_seat_new_device (eis_seat);
|
||
|
name = g_strdup_printf ("%s %s", eis_client_get_name (meta_eis_client->eis_client),
|
||
|
name_suffix);
|
||
|
eis_device_configure_name (eis_device, name);
|
||
|
if (extra_config_func)
|
||
|
extra_config_func (meta_eis_client, eis_device);
|
||
|
|
||
|
meta_eis_device = g_new0 (MetaEisDevice, 1);
|
||
|
meta_eis_device->eis_device = eis_device_ref (eis_device);
|
||
|
meta_eis_device->device = device;
|
||
|
eis_device_set_user_data (eis_device, meta_eis_device);
|
||
|
|
||
|
g_hash_table_insert (meta_eis_client->eis_devices,
|
||
|
eis_device, /* owns the initial ref now */
|
||
|
meta_eis_device);
|
||
|
|
||
|
eis_device_add (eis_device);
|
||
|
eis_device_resume (eis_device);
|
||
|
g_free (name);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_motion_relative (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
double dx, dy;
|
||
|
|
||
|
dx = eis_event_pointer_get_dx (event);
|
||
|
dy = eis_event_pointer_get_dy (event);
|
||
|
|
||
|
clutter_virtual_input_device_notify_relative_motion (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
dx, dy);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_motion_absolute (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
double x, y;
|
||
|
|
||
|
x = eis_event_pointer_get_absolute_x (event);
|
||
|
y = eis_event_pointer_get_absolute_y (event);
|
||
|
|
||
|
clutter_virtual_input_device_notify_absolute_motion (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
x, y);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_scroll (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
double dx, dy;
|
||
|
|
||
|
dx = eis_event_scroll_get_dx (event);
|
||
|
dy = eis_event_scroll_get_dy (event);
|
||
|
|
||
|
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
dx, dy,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN,
|
||
|
CLUTTER_SCROLL_FINISHED_NONE);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_scroll_stop (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
ClutterScrollFinishFlags finish_flags = CLUTTER_SCROLL_FINISHED_NONE;
|
||
|
|
||
|
if (eis_event_scroll_get_stop_x (event))
|
||
|
finish_flags |= CLUTTER_SCROLL_FINISHED_HORIZONTAL;
|
||
|
if (eis_event_scroll_get_stop_y (event))
|
||
|
finish_flags |= CLUTTER_SCROLL_FINISHED_VERTICAL;
|
||
|
|
||
|
if (finish_flags != CLUTTER_SCROLL_FINISHED_NONE)
|
||
|
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
0.0, 0.0,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN,
|
||
|
finish_flags);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_scroll_cancel (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
double dx = 0.0, dy = 0.0;
|
||
|
|
||
|
/* There's no real match for the EIS scroll cancel event, so let's just send a
|
||
|
* really small scroll event that hopefully resets the scroll speed to
|
||
|
* something where kinetic scrolling is not viable.
|
||
|
*/
|
||
|
if (eis_event_scroll_get_stop_x (event))
|
||
|
dx = 0.01;
|
||
|
if (eis_event_scroll_get_stop_y (event))
|
||
|
dy = 0.01;
|
||
|
|
||
|
if (dx != 0.0 || dy != 0.0)
|
||
|
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
dx, dy,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN,
|
||
|
CLUTTER_SCROLL_FINISHED_NONE);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_scroll_discrete (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
int dx, dy;
|
||
|
|
||
|
/* FIXME: need to handle the remainders here for high-resolution scrolling */
|
||
|
dx = eis_event_scroll_get_discrete_dx (event) / 120;
|
||
|
dy = eis_event_scroll_get_discrete_dy (event) / 120;
|
||
|
|
||
|
/* Intentionally interleaved */
|
||
|
while (dx || dy)
|
||
|
{
|
||
|
if (dx > 0)
|
||
|
{
|
||
|
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
CLUTTER_SCROLL_RIGHT,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN);
|
||
|
dx--;
|
||
|
}
|
||
|
else if (dx < 0)
|
||
|
{
|
||
|
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
CLUTTER_SCROLL_LEFT,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN);
|
||
|
dx++;
|
||
|
}
|
||
|
|
||
|
if (dy > 0)
|
||
|
{
|
||
|
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
CLUTTER_SCROLL_DOWN,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN);
|
||
|
dy--;
|
||
|
}
|
||
|
else if (dy < 0)
|
||
|
{
|
||
|
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
|
||
|
g_get_monotonic_time (),
|
||
|
CLUTTER_SCROLL_UP,
|
||
|
CLUTTER_SCROLL_SOURCE_UNKNOWN);
|
||
|
dy++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_button (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
uint32_t button;
|
||
|
gboolean is_press = eis_event_button_get_is_press (event);
|
||
|
|
||
|
button = eis_event_button_get_button (event);
|
||
|
switch (button)
|
||
|
{
|
||
|
case 0x110: /* BTN_LEFT */
|
||
|
button = CLUTTER_BUTTON_PRIMARY;
|
||
|
break;
|
||
|
case 0x111: /* BTN_RIGHT */
|
||
|
button = CLUTTER_BUTTON_SECONDARY;
|
||
|
break;
|
||
|
case 0x112: /* BTN_MIDDLE */
|
||
|
button = CLUTTER_BUTTON_MIDDLE;
|
||
|
break;
|
||
|
default:
|
||
|
if (button > 0x110)
|
||
|
button -= 0x110;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (button > MAX_BUTTON)
|
||
|
return;
|
||
|
|
||
|
if (is_press && !bit_is_set (meta_eis_device->button_state, button))
|
||
|
bit_set (meta_eis_device->button_state, button);
|
||
|
else if (!is_press && bit_is_set (meta_eis_device->button_state, button))
|
||
|
bit_clear (meta_eis_device->button_state, button);
|
||
|
else
|
||
|
return; /* Duplicate press/release, should've been filtered by libeis */
|
||
|
|
||
|
notify_button (meta_eis_device,
|
||
|
button,
|
||
|
eis_event_button_get_is_press (event));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
handle_key (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
struct eis_device *eis_device = eis_event_get_device (event);
|
||
|
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
|
||
|
uint32_t key;
|
||
|
gboolean is_press = eis_event_keyboard_get_key_is_press (event);
|
||
|
|
||
|
key = eis_event_keyboard_get_key (event);
|
||
|
|
||
|
if (key > MAX_KEY)
|
||
|
return;
|
||
|
|
||
|
if (is_press && !bit_is_set (meta_eis_device->key_state, key))
|
||
|
bit_set (meta_eis_device->key_state, key);
|
||
|
else if (!is_press && bit_is_set (meta_eis_device->key_state, key))
|
||
|
bit_clear (meta_eis_device->key_state, key);
|
||
|
else
|
||
|
return; /* Duplicate press/release, should've been filtered by libeis */
|
||
|
|
||
|
notify_key (meta_eis_device, key, is_press);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
drop_kbd_devices (gpointer htkey,
|
||
|
gpointer value,
|
||
|
gpointer data)
|
||
|
{
|
||
|
struct eis_device *eis_device = htkey;
|
||
|
|
||
|
if (!eis_device_has_capability (eis_device, EIS_DEVICE_CAP_KEYBOARD))
|
||
|
return FALSE;
|
||
|
|
||
|
return drop_device (htkey, value, data);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_keymap_changed (MetaBackend *backend,
|
||
|
gpointer data)
|
||
|
{
|
||
|
MetaEisClient *meta_eis_client = data;
|
||
|
|
||
|
/* Changing the keymap means we have to remove our device and recreate it
|
||
|
* with the new keymap.
|
||
|
*/
|
||
|
g_hash_table_foreach_remove (meta_eis_client->eis_devices,
|
||
|
drop_kbd_devices,
|
||
|
meta_eis_client);
|
||
|
|
||
|
add_device (meta_eis_client,
|
||
|
meta_eis_client->eis_seat,
|
||
|
CLUTTER_KEYBOARD_DEVICE,
|
||
|
"virtual keyboard",
|
||
|
configure_keyboard);
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
meta_eis_client_process_event (MetaEisClient *meta_eis_client,
|
||
|
struct eis_event *event)
|
||
|
{
|
||
|
enum eis_event_type type = eis_event_get_type (event);
|
||
|
struct eis_seat *eis_seat;
|
||
|
|
||
|
switch (type)
|
||
|
{
|
||
|
case EIS_EVENT_SEAT_BIND:
|
||
|
eis_seat = eis_event_get_seat (event);
|
||
|
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_POINTER))
|
||
|
add_device (meta_eis_client,
|
||
|
eis_seat,
|
||
|
CLUTTER_POINTER_DEVICE,
|
||
|
"virtual pointer",
|
||
|
configure_rel);
|
||
|
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_KEYBOARD))
|
||
|
{
|
||
|
add_device (meta_eis_client,
|
||
|
eis_seat,
|
||
|
CLUTTER_KEYBOARD_DEVICE,
|
||
|
"virtual keyboard",
|
||
|
configure_keyboard);
|
||
|
|
||
|
g_signal_connect (meta_eis_get_backend (meta_eis_client->meta_eis),
|
||
|
"keymap-changed",
|
||
|
G_CALLBACK (on_keymap_changed),
|
||
|
meta_eis_client);
|
||
|
}
|
||
|
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_POINTER_ABSOLUTE))
|
||
|
add_device (meta_eis_client,
|
||
|
eis_seat,
|
||
|
CLUTTER_POINTER_DEVICE,
|
||
|
"virtual absolute pointer",
|
||
|
configure_abs);
|
||
|
break;
|
||
|
|
||
|
/* We only have one seat, so if the client unbinds from that
|
||
|
* just disconnect it, no point keeping it alive */
|
||
|
case EIS_EVENT_DEVICE_CLOSED:
|
||
|
remove_device (meta_eis_client, eis_event_get_device (event), TRUE);
|
||
|
break;
|
||
|
case EIS_EVENT_POINTER_MOTION:
|
||
|
handle_motion_relative (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_POINTER_MOTION_ABSOLUTE:
|
||
|
handle_motion_absolute (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_BUTTON_BUTTON:
|
||
|
handle_button (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_SCROLL_DELTA:
|
||
|
handle_scroll (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_SCROLL_STOP:
|
||
|
handle_scroll_stop (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_SCROLL_CANCEL:
|
||
|
handle_scroll_cancel (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_SCROLL_DISCRETE:
|
||
|
handle_scroll_discrete (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_KEYBOARD_KEY:
|
||
|
handle_key (meta_eis_client, event);
|
||
|
break;
|
||
|
case EIS_EVENT_FRAME:
|
||
|
/* FIXME: we should be accumulating the above events */
|
||
|
break;
|
||
|
case EIS_EVENT_DEVICE_START_EMULATING:
|
||
|
break;
|
||
|
case EIS_EVENT_DEVICE_STOP_EMULATING:
|
||
|
break;
|
||
|
default:
|
||
|
g_warning ("Unhandled EIS event type %d", type);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
drop_abs_devices (gpointer key,
|
||
|
gpointer value,
|
||
|
gpointer data)
|
||
|
{
|
||
|
struct eis_device *eis_device = key;
|
||
|
|
||
|
if (!eis_device_has_capability (eis_device, EIS_DEVICE_CAP_POINTER_ABSOLUTE))
|
||
|
return FALSE;
|
||
|
|
||
|
return drop_device (key, value, data);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_set_viewports (MetaEisClient *meta_eis_client,
|
||
|
MetaViewportInfo *viewports)
|
||
|
{
|
||
|
/* Updating viewports means we have to recreate our absolute pointer
|
||
|
* devices. */
|
||
|
|
||
|
g_hash_table_foreach_remove (meta_eis_client->eis_devices,
|
||
|
drop_abs_devices,
|
||
|
meta_eis_client);
|
||
|
|
||
|
g_clear_object (&meta_eis_client->viewports);
|
||
|
meta_eis_client->viewports = g_object_ref (viewports);
|
||
|
|
||
|
add_device (meta_eis_client,
|
||
|
meta_eis_client->eis_seat,
|
||
|
CLUTTER_POINTER_DEVICE,
|
||
|
"virtual absolute pointer",
|
||
|
configure_abs);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_monitors_changed (MetaMonitorManager *monitor_manager,
|
||
|
MetaEisClient *meta_eis_client)
|
||
|
{
|
||
|
MetaViewportInfo *viewports;
|
||
|
|
||
|
viewports = meta_monitor_manager_get_viewports (monitor_manager);
|
||
|
meta_eis_client_set_viewports (meta_eis_client, viewports);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_disconnect (MetaEisClient *meta_eis_client)
|
||
|
{
|
||
|
MetaBackend *backend = meta_eis_get_backend (meta_eis_client->meta_eis);
|
||
|
MetaMonitorManager *monitor_manager = meta_backend_get_monitor_manager (backend);
|
||
|
|
||
|
g_signal_handlers_disconnect_by_func (monitor_manager,
|
||
|
on_monitors_changed,
|
||
|
meta_eis_client);
|
||
|
g_hash_table_foreach_remove (meta_eis_client->eis_devices, drop_device, meta_eis_client);
|
||
|
g_clear_pointer (&meta_eis_client->eis_seat, eis_seat_unref);
|
||
|
if (meta_eis_client->eis_client)
|
||
|
eis_client_disconnect (meta_eis_client->eis_client);
|
||
|
g_clear_pointer (&meta_eis_client->eis_client, eis_client_unref);
|
||
|
g_clear_object (&meta_eis_client->viewports);
|
||
|
}
|
||
|
|
||
|
MetaEisClient *
|
||
|
meta_eis_client_new (MetaEis *meta_eis,
|
||
|
struct eis_client *eis_client)
|
||
|
{
|
||
|
MetaEisClient *meta_eis_client;
|
||
|
MetaBackend *backend;
|
||
|
MetaMonitorManager *monitor_manager;
|
||
|
MetaViewportInfo *viewports;
|
||
|
struct eis_seat *eis_seat;
|
||
|
|
||
|
meta_eis_client = g_object_new (META_TYPE_EIS_CLIENT, NULL);
|
||
|
meta_eis_client->meta_eis = meta_eis;
|
||
|
meta_eis_client->eis_client = eis_client_ref (eis_client);
|
||
|
eis_client_set_user_data (meta_eis_client->eis_client, meta_eis_client);
|
||
|
|
||
|
/* We're relying on some third party to filter clients for us */
|
||
|
eis_client_connect (eis_client);
|
||
|
|
||
|
/* We only support one seat for now. libeis keeps the ref for the
|
||
|
* seat and we don't need to care about it.
|
||
|
* The capabilities we define are the maximum capabilities, the client
|
||
|
* may only bind to a subset of those, reducing the capabilities
|
||
|
* of the seat in the process.
|
||
|
*/
|
||
|
eis_seat = eis_client_new_seat (eis_client, "mutter default seat");
|
||
|
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_KEYBOARD);
|
||
|
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_POINTER);
|
||
|
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_POINTER_ABSOLUTE);
|
||
|
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_BUTTON);
|
||
|
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_SCROLL);
|
||
|
|
||
|
eis_seat_add (eis_seat);
|
||
|
eis_seat_unref (eis_seat);
|
||
|
meta_eis_client->eis_seat = eis_seat_ref (eis_seat);
|
||
|
|
||
|
meta_eis_client->eis_devices = g_hash_table_new_full (g_direct_hash, g_direct_equal,
|
||
|
(GDestroyNotify) eis_device_unref,
|
||
|
(GDestroyNotify) meta_eis_device_free);
|
||
|
|
||
|
backend = meta_eis_get_backend (meta_eis);
|
||
|
monitor_manager = meta_backend_get_monitor_manager (backend);
|
||
|
viewports = meta_monitor_manager_get_viewports (monitor_manager);
|
||
|
meta_eis_client_set_viewports (meta_eis_client, viewports);
|
||
|
g_signal_connect (monitor_manager, "monitors-changed",
|
||
|
G_CALLBACK (on_monitors_changed),
|
||
|
meta_eis_client);
|
||
|
|
||
|
return meta_eis_client;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_init (MetaEisClient *meta_eis_client)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_constructed (GObject *object)
|
||
|
{
|
||
|
if (G_OBJECT_CLASS (meta_eis_client_parent_class)->constructed)
|
||
|
G_OBJECT_CLASS (meta_eis_client_parent_class)->constructed (object);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_finalize (GObject *object)
|
||
|
{
|
||
|
MetaEisClient *meta_eis_client = META_EIS_CLIENT (object);
|
||
|
|
||
|
g_signal_handlers_disconnect_by_func (meta_eis_get_backend (meta_eis_client->meta_eis),
|
||
|
on_keymap_changed,
|
||
|
meta_eis_client);
|
||
|
meta_eis_client_disconnect (meta_eis_client);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
meta_eis_client_class_init (MetaEisClientClass *klass)
|
||
|
{
|
||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
|
||
|
object_class->constructed = meta_eis_client_constructed;
|
||
|
object_class->finalize = meta_eis_client_finalize;
|
||
|
}
|