2014-04-21 19:13:04 -04:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* Written by:
|
|
|
|
* Jasper St. Pierre <jstpierre@mecheye.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2015-01-16 03:50:51 -05:00
|
|
|
#include "meta-backend-native.h"
|
|
|
|
#include "meta-backend-native-private.h"
|
|
|
|
|
2014-04-21 19:47:04 -04:00
|
|
|
#include <meta/main.h>
|
2014-04-21 20:45:24 -04:00
|
|
|
#include <clutter/evdev/clutter-evdev.h>
|
2014-04-21 19:13:04 -04:00
|
|
|
|
2015-01-16 03:50:51 -05:00
|
|
|
#include "meta-barrier-native.h"
|
2014-04-21 19:13:04 -04:00
|
|
|
#include "meta-idle-monitor-native.h"
|
2014-04-21 19:41:11 -04:00
|
|
|
#include "meta-monitor-manager-kms.h"
|
2014-04-22 15:15:11 -04:00
|
|
|
#include "meta-cursor-renderer-native.h"
|
2014-04-22 17:03:22 -04:00
|
|
|
#include "meta-launcher.h"
|
2014-04-21 19:13:04 -04:00
|
|
|
|
2014-04-21 19:47:04 -04:00
|
|
|
struct _MetaBackendNativePrivate
|
|
|
|
{
|
|
|
|
MetaLauncher *launcher;
|
2015-01-16 03:50:51 -05:00
|
|
|
|
|
|
|
MetaBarrierManagerNative *barrier_manager;
|
|
|
|
|
|
|
|
GSettings *keyboard_settings;
|
2014-04-21 19:47:04 -04:00
|
|
|
};
|
|
|
|
typedef struct _MetaBackendNativePrivate MetaBackendNativePrivate;
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (MetaBackendNative, meta_backend_native, META_TYPE_BACKEND);
|
2014-04-21 19:13:04 -04:00
|
|
|
|
2014-09-19 11:25:22 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaBackendNative *native = META_BACKEND_NATIVE (object);
|
|
|
|
MetaBackendNativePrivate *priv = meta_backend_native_get_instance_private (native);
|
|
|
|
|
2014-12-08 06:32:38 -05:00
|
|
|
meta_launcher_free (priv->launcher);
|
2014-09-19 11:25:22 -04:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_backend_native_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
2015-01-16 03:50:51 -05:00
|
|
|
static void
|
|
|
|
constrain_to_barriers (ClutterInputDevice *device,
|
|
|
|
guint32 time,
|
|
|
|
float *new_x,
|
|
|
|
float *new_y)
|
|
|
|
{
|
|
|
|
MetaBackendNative *native = META_BACKEND_NATIVE (meta_get_backend ());
|
|
|
|
MetaBackendNativePrivate *priv =
|
|
|
|
meta_backend_native_get_instance_private (native);
|
|
|
|
|
|
|
|
meta_barrier_manager_native_process (priv->barrier_manager,
|
|
|
|
device,
|
|
|
|
time,
|
|
|
|
new_x, new_y);
|
|
|
|
}
|
|
|
|
|
2014-04-21 20:45:24 -04:00
|
|
|
/*
|
|
|
|
* The pointer constrain code is mostly a rip-off of the XRandR code from Xorg.
|
|
|
|
* (from xserver/randr/rrcrtc.c, RRConstrainCursorHarder)
|
|
|
|
*
|
|
|
|
* Copyright © 2006 Keith Packard
|
|
|
|
* Copyright 2010 Red Hat, Inc
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
check_all_screen_monitors(MetaMonitorInfo *monitors,
|
|
|
|
unsigned n_monitors,
|
|
|
|
float x,
|
|
|
|
float y)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < n_monitors; i++)
|
|
|
|
{
|
|
|
|
MetaMonitorInfo *monitor = &monitors[i];
|
|
|
|
int left, right, top, bottom;
|
|
|
|
|
|
|
|
left = monitor->rect.x;
|
|
|
|
right = left + monitor->rect.width;
|
|
|
|
top = monitor->rect.y;
|
2015-03-02 07:46:16 -05:00
|
|
|
bottom = top + monitor->rect.height;
|
2014-04-21 20:45:24 -04:00
|
|
|
|
|
|
|
if ((x >= left) && (x < right) && (y >= top) && (y < bottom))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
constrain_all_screen_monitors (ClutterInputDevice *device,
|
|
|
|
MetaMonitorInfo *monitors,
|
|
|
|
unsigned n_monitors,
|
|
|
|
float *x,
|
|
|
|
float *y)
|
|
|
|
{
|
|
|
|
ClutterPoint current;
|
|
|
|
unsigned int i;
|
2015-03-02 07:46:16 -05:00
|
|
|
float cx, cy;
|
2014-04-21 20:45:24 -04:00
|
|
|
|
|
|
|
clutter_input_device_get_coords (device, NULL, ¤t);
|
|
|
|
|
2015-03-02 07:46:16 -05:00
|
|
|
cx = current.x;
|
|
|
|
cy = current.y;
|
|
|
|
|
2014-04-21 20:45:24 -04:00
|
|
|
/* if we're trying to escape, clamp to the CRTC we're coming from */
|
|
|
|
for (i = 0; i < n_monitors; i++)
|
|
|
|
{
|
|
|
|
MetaMonitorInfo *monitor = &monitors[i];
|
|
|
|
int left, right, top, bottom;
|
|
|
|
|
|
|
|
left = monitor->rect.x;
|
|
|
|
right = left + monitor->rect.width;
|
|
|
|
top = monitor->rect.y;
|
2015-03-02 07:46:16 -05:00
|
|
|
bottom = top + monitor->rect.height;
|
2014-04-21 20:45:24 -04:00
|
|
|
|
2015-03-02 07:46:16 -05:00
|
|
|
if ((cx >= left) && (cx < right) && (cy >= top) && (cy < bottom))
|
2014-04-21 20:45:24 -04:00
|
|
|
{
|
|
|
|
if (*x < left)
|
|
|
|
*x = left;
|
|
|
|
if (*x >= right)
|
|
|
|
*x = right - 1;
|
|
|
|
if (*y < top)
|
|
|
|
*y = top;
|
|
|
|
if (*y >= bottom)
|
|
|
|
*y = bottom - 1;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
pointer_constrain_callback (ClutterInputDevice *device,
|
|
|
|
guint32 time,
|
|
|
|
float *new_x,
|
|
|
|
float *new_y,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaMonitorManager *monitor_manager;
|
|
|
|
MetaMonitorInfo *monitors;
|
|
|
|
unsigned int n_monitors;
|
|
|
|
gboolean ret;
|
|
|
|
|
2015-01-16 03:50:51 -05:00
|
|
|
/* Constrain to barriers */
|
|
|
|
constrain_to_barriers (device, time, new_x, new_y);
|
|
|
|
|
2014-04-21 20:45:24 -04:00
|
|
|
monitor_manager = meta_monitor_manager_get ();
|
|
|
|
monitors = meta_monitor_manager_get_monitor_infos (monitor_manager, &n_monitors);
|
|
|
|
|
|
|
|
/* if we're moving inside a monitor, we're fine */
|
|
|
|
ret = check_all_screen_monitors(monitors, n_monitors, *new_x, *new_y);
|
|
|
|
if (ret == TRUE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* if we're trying to escape, clamp to the CRTC we're coming from */
|
|
|
|
constrain_all_screen_monitors(device, monitors, n_monitors, new_x, new_y);
|
|
|
|
}
|
|
|
|
|
2015-02-25 11:01:10 -05:00
|
|
|
static void
|
|
|
|
on_monitors_changed (MetaMonitorManager *monitor_manager,
|
|
|
|
MetaBackend *backend)
|
|
|
|
{
|
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
|
|
|
ClutterInputDevice *device = clutter_device_manager_get_core_device (manager, CLUTTER_POINTER_DEVICE);
|
|
|
|
MetaMonitorInfo *monitors, *primary;
|
|
|
|
unsigned int n_monitors;
|
|
|
|
ClutterPoint point;
|
|
|
|
|
|
|
|
if (!clutter_input_device_get_coords (device, NULL, &point))
|
|
|
|
return;
|
|
|
|
|
|
|
|
monitors = meta_monitor_manager_get_monitor_infos (monitor_manager, &n_monitors);
|
|
|
|
|
|
|
|
/* if we're inside a monitor, we're fine */
|
|
|
|
if (check_all_screen_monitors (monitors, n_monitors, point.x, point.y))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* warp the pointer to the primary monitor so it isn't lost */
|
|
|
|
primary = &monitors[meta_monitor_manager_get_primary_index (monitor_manager)];
|
|
|
|
meta_backend_warp_pointer (backend,
|
|
|
|
primary->rect.x + primary->rect.width / 2,
|
|
|
|
primary->rect.y + primary->rect.height / 2);
|
|
|
|
}
|
|
|
|
|
2014-04-22 12:40:51 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_post_init (MetaBackend *backend)
|
|
|
|
{
|
2015-02-25 11:01:10 -05:00
|
|
|
MetaMonitorManager *monitor_manager;
|
2014-04-22 12:40:51 -04:00
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
2014-04-23 09:42:49 -04:00
|
|
|
|
|
|
|
META_BACKEND_CLASS (meta_backend_native_parent_class)->post_init (backend);
|
|
|
|
|
2014-04-22 12:40:51 -04:00
|
|
|
clutter_evdev_set_pointer_constrain_callback (manager, pointer_constrain_callback,
|
|
|
|
NULL, NULL);
|
2015-02-25 11:01:10 -05:00
|
|
|
|
|
|
|
monitor_manager = meta_backend_get_monitor_manager (backend);
|
|
|
|
g_signal_connect_object (monitor_manager, "monitors-changed",
|
|
|
|
G_CALLBACK (on_monitors_changed), backend, G_CONNECT_AFTER);
|
2015-03-09 07:48:10 -04:00
|
|
|
|
|
|
|
/* make sure the pointer is in the visible area after init */
|
|
|
|
on_monitors_changed (monitor_manager, backend);
|
2014-04-22 12:40:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static MetaIdleMonitor *
|
|
|
|
meta_backend_native_create_idle_monitor (MetaBackend *backend,
|
|
|
|
int device_id)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_IDLE_MONITOR_NATIVE,
|
|
|
|
"device-id", device_id,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaMonitorManager *
|
|
|
|
meta_backend_native_create_monitor_manager (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_MONITOR_MANAGER_KMS, NULL);
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:15:11 -04:00
|
|
|
static MetaCursorRenderer *
|
|
|
|
meta_backend_native_create_cursor_renderer (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_CURSOR_RENDERER_NATIVE, NULL);
|
|
|
|
}
|
|
|
|
|
2014-05-27 14:21:54 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_warp_pointer (MetaBackend *backend,
|
|
|
|
int x,
|
|
|
|
int y)
|
|
|
|
{
|
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
|
|
|
ClutterInputDevice *device = clutter_device_manager_get_core_device (manager, CLUTTER_POINTER_DEVICE);
|
|
|
|
|
|
|
|
/* XXX */
|
|
|
|
guint32 time_ = 0;
|
|
|
|
|
|
|
|
clutter_evdev_warp_pointer (device, time_, x, y);
|
|
|
|
}
|
|
|
|
|
2014-08-04 10:47:35 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_set_keymap (MetaBackend *backend,
|
|
|
|
const char *layouts,
|
|
|
|
const char *variants,
|
|
|
|
const char *options)
|
|
|
|
{
|
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
|
|
|
struct xkb_rule_names names;
|
|
|
|
struct xkb_keymap *keymap;
|
|
|
|
struct xkb_context *context;
|
|
|
|
|
|
|
|
names.rules = DEFAULT_XKB_RULES_FILE;
|
|
|
|
names.model = DEFAULT_XKB_MODEL;
|
|
|
|
names.layout = layouts;
|
|
|
|
names.variant = variants;
|
|
|
|
names.options = options;
|
|
|
|
|
|
|
|
context = xkb_context_new (XKB_CONTEXT_NO_FLAGS);
|
|
|
|
keymap = xkb_keymap_new_from_names (context, &names, XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
xkb_context_unref (context);
|
|
|
|
|
|
|
|
clutter_evdev_set_keyboard_map (manager, keymap);
|
|
|
|
|
2014-08-14 17:32:41 -04:00
|
|
|
g_signal_emit_by_name (backend, "keymap-changed", 0);
|
2014-08-04 10:47:35 -04:00
|
|
|
|
|
|
|
xkb_keymap_unref (keymap);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct xkb_keymap *
|
|
|
|
meta_backend_native_get_keymap (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
|
|
|
return clutter_evdev_get_keyboard_map (manager);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_native_lock_layout_group (MetaBackend *backend,
|
|
|
|
guint idx)
|
|
|
|
{
|
|
|
|
ClutterDeviceManager *manager = clutter_device_manager_get_default ();
|
|
|
|
clutter_evdev_set_keyboard_layout_index (manager, idx);
|
2014-09-10 13:16:01 -04:00
|
|
|
g_signal_emit_by_name (backend, "keymap-layout-group-changed", idx, 0);
|
2014-08-04 10:47:35 -04:00
|
|
|
}
|
|
|
|
|
2014-04-22 12:40:51 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_class_init (MetaBackendNativeClass *klass)
|
|
|
|
{
|
|
|
|
MetaBackendClass *backend_class = META_BACKEND_CLASS (klass);
|
2014-09-19 11:25:22 -04:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = meta_backend_native_finalize;
|
2014-04-22 12:40:51 -04:00
|
|
|
|
|
|
|
backend_class->post_init = meta_backend_native_post_init;
|
|
|
|
backend_class->create_idle_monitor = meta_backend_native_create_idle_monitor;
|
|
|
|
backend_class->create_monitor_manager = meta_backend_native_create_monitor_manager;
|
2014-04-22 15:15:11 -04:00
|
|
|
backend_class->create_cursor_renderer = meta_backend_native_create_cursor_renderer;
|
2014-05-27 14:21:54 -04:00
|
|
|
|
|
|
|
backend_class->warp_pointer = meta_backend_native_warp_pointer;
|
2014-08-04 10:47:35 -04:00
|
|
|
backend_class->set_keymap = meta_backend_native_set_keymap;
|
|
|
|
backend_class->get_keymap = meta_backend_native_get_keymap;
|
|
|
|
backend_class->lock_layout_group = meta_backend_native_lock_layout_group;
|
2014-04-22 12:40:51 -04:00
|
|
|
}
|
|
|
|
|
2014-04-21 19:13:04 -04:00
|
|
|
static void
|
|
|
|
meta_backend_native_init (MetaBackendNative *native)
|
|
|
|
{
|
2014-04-21 19:47:04 -04:00
|
|
|
MetaBackendNativePrivate *priv = meta_backend_native_get_instance_private (native);
|
|
|
|
|
|
|
|
/* We're a display server, so start talking to weston-launch. */
|
|
|
|
priv->launcher = meta_launcher_new ();
|
2015-01-16 03:50:51 -05:00
|
|
|
|
|
|
|
priv->barrier_manager = meta_barrier_manager_native_new ();
|
2014-04-21 19:47:04 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_activate_vt (int vt, GError **error)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaBackendNative *native = META_BACKEND_NATIVE (backend);
|
|
|
|
MetaBackendNativePrivate *priv = meta_backend_native_get_instance_private (native);
|
|
|
|
|
|
|
|
return meta_launcher_activate_vt (priv->launcher, vt, error);
|
|
|
|
}
|
|
|
|
|
2015-01-16 03:50:51 -05:00
|
|
|
MetaBarrierManagerNative *
|
|
|
|
meta_backend_native_get_barrier_manager (MetaBackendNative *native)
|
|
|
|
{
|
|
|
|
MetaBackendNativePrivate *priv =
|
|
|
|
meta_backend_native_get_instance_private (native);
|
|
|
|
|
|
|
|
return priv->barrier_manager;
|
|
|
|
}
|
|
|
|
|
2014-04-21 19:47:04 -04:00
|
|
|
/**
|
|
|
|
* meta_activate_session:
|
|
|
|
*
|
|
|
|
* Tells mutter to activate the session. When mutter is a
|
2014-10-12 15:48:18 -04:00
|
|
|
* display server, this tells logind to switch over to
|
2014-04-21 19:47:04 -04:00
|
|
|
* the new session.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
meta_activate_session (void)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
|
|
|
|
/* Do nothing. */
|
|
|
|
if (!META_IS_BACKEND_NATIVE (backend))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
MetaBackendNative *native = META_BACKEND_NATIVE (backend);
|
|
|
|
MetaBackendNativePrivate *priv = meta_backend_native_get_instance_private (native);
|
|
|
|
|
2014-04-22 17:08:57 -04:00
|
|
|
if (!meta_launcher_activate_session (priv->launcher, &error))
|
2014-04-21 19:47:04 -04:00
|
|
|
{
|
|
|
|
g_warning ("Could not activate session: %s\n", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
2014-04-21 19:13:04 -04:00
|
|
|
}
|