2015-06-17 04:10:52 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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
|
2023-08-07 09:50:23 +00:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2015-06-17 04:10:52 +00:00
|
|
|
*
|
|
|
|
* Written by:
|
|
|
|
* Jonas Ådahl <jadahl@gmail.com>
|
|
|
|
*/
|
|
|
|
|
2018-10-19 07:15:54 +00:00
|
|
|
/**
|
2023-03-28 14:35:11 +00:00
|
|
|
* MetaPointerConfinementWayland:
|
|
|
|
*
|
|
|
|
* A #MetaPointerConstraint implementing pointer confinement
|
2018-10-19 07:15:54 +00:00
|
|
|
*
|
|
|
|
* A MetaPointerConfinementConstraint implements the client pointer constraint
|
|
|
|
* "pointer confinement": the cursor should not be able to "break out" of a
|
|
|
|
* certain area defined by the client requesting it.
|
|
|
|
*/
|
|
|
|
|
2015-06-17 04:10:52 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "wayland/meta-pointer-confinement-wayland.h"
|
|
|
|
|
|
|
|
#include <glib-object.h>
|
|
|
|
|
|
|
|
#include "backends/meta-backend-private.h"
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "backends/meta-pointer-constraint.h"
|
2022-06-13 08:25:02 +00:00
|
|
|
#include "compositor/region-utils.h"
|
2015-06-17 04:10:52 +00:00
|
|
|
#include "wayland/meta-wayland-pointer-constraints.h"
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "wayland/meta-wayland-pointer.h"
|
|
|
|
#include "wayland/meta-wayland-seat.h"
|
2023-10-23 21:59:19 +00:00
|
|
|
#include "wayland/meta-wayland-surface-private.h"
|
2022-05-30 21:48:44 +00:00
|
|
|
#include "wayland/meta-wayland.h"
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
typedef struct _MetaPointerConfinementWaylandPrivate MetaPointerConfinementWaylandPrivate;
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
struct _MetaPointerConfinementWaylandPrivate
|
|
|
|
{
|
2015-06-17 04:10:52 +00:00
|
|
|
MetaWaylandPointerConstraint *constraint;
|
2020-07-08 16:17:13 +00:00
|
|
|
gboolean enabled;
|
2015-06-17 04:10:52 +00:00
|
|
|
};
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
enum
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
PROP_0,
|
|
|
|
PROP_WAYLAND_POINTER_CONSTRAINT,
|
|
|
|
N_PROPS,
|
|
|
|
};
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
static GParamSpec *props[N_PROPS] = { 0 };
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (MetaPointerConfinementWayland,
|
|
|
|
meta_pointer_confinement_wayland,
|
|
|
|
G_TYPE_OBJECT)
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2022-05-30 21:48:44 +00:00
|
|
|
static MetaBackend *
|
|
|
|
backend_from_confinement (MetaPointerConfinementWayland *confinement)
|
|
|
|
{
|
|
|
|
MetaPointerConfinementWaylandPrivate *priv =
|
|
|
|
meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
MetaWaylandCompositor *compositor =
|
|
|
|
meta_wayland_pointer_constraint_get_compositor (priv->constraint);
|
|
|
|
MetaContext *context = meta_wayland_compositor_get_context (compositor);
|
|
|
|
|
|
|
|
return meta_context_get_backend (context);
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:22:07 +00:00
|
|
|
static void
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_update (MetaPointerConfinementWayland *self)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConstraint *constraint;
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
constraint =
|
|
|
|
META_POINTER_CONFINEMENT_WAYLAND_GET_CLASS (self)->create_constraint (self);
|
2022-05-30 21:48:44 +00:00
|
|
|
meta_backend_set_client_pointer_constraint (backend_from_confinement (self),
|
|
|
|
constraint);
|
2020-07-08 16:17:13 +00:00
|
|
|
g_object_unref (constraint);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-07-08 16:17:13 +00:00
|
|
|
surface_geometry_changed (MetaWaylandSurface *surface,
|
|
|
|
MetaPointerConfinementWayland *self)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_update (self);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-07-08 16:17:13 +00:00
|
|
|
window_position_changed (MetaWindow *window,
|
|
|
|
MetaPointerConfinementWayland *self)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_update (self);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
void
|
|
|
|
meta_pointer_confinement_wayland_enable (MetaPointerConfinementWayland *confinement)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
|
|
|
MetaWaylandPointerConstraint *constraint;
|
|
|
|
MetaWaylandSurface *surface;
|
|
|
|
MetaWindow *window;
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
g_assert (!priv->enabled);
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
priv->enabled = TRUE;
|
|
|
|
constraint = priv->constraint;
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
surface = meta_wayland_pointer_constraint_get_surface (constraint);
|
|
|
|
g_signal_connect_object (surface,
|
|
|
|
"geometry-changed",
|
|
|
|
G_CALLBACK (surface_geometry_changed),
|
|
|
|
confinement,
|
|
|
|
0);
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
window = meta_wayland_surface_get_window (surface);
|
|
|
|
if (window)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
g_signal_connect_object (window,
|
|
|
|
"position-changed",
|
|
|
|
G_CALLBACK (window_position_changed),
|
|
|
|
confinement,
|
|
|
|
0);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_update (confinement);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
void
|
|
|
|
meta_pointer_confinement_wayland_disable (MetaPointerConfinementWayland *confinement)
|
2015-06-17 04:10:52 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
|
|
|
MetaWaylandPointerConstraint *constraint;
|
2015-06-17 04:10:52 +00:00
|
|
|
MetaWaylandSurface *surface;
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaWindow *window;
|
2022-05-30 21:48:44 +00:00
|
|
|
MetaBackend *backend;
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
constraint = priv->constraint;
|
|
|
|
g_assert (priv->enabled);
|
|
|
|
|
|
|
|
priv->enabled = FALSE;
|
|
|
|
surface = meta_wayland_pointer_constraint_get_surface (constraint);
|
|
|
|
g_signal_handlers_disconnect_by_func (surface, surface_geometry_changed,
|
|
|
|
confinement);
|
|
|
|
|
|
|
|
window = meta_wayland_surface_get_window (surface);
|
|
|
|
if (window)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
g_signal_handlers_disconnect_by_func (window, window_position_changed,
|
|
|
|
confinement);
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
|
2022-05-30 21:48:44 +00:00
|
|
|
backend = backend_from_confinement (confinement);
|
|
|
|
meta_backend_set_client_pointer_constraint (backend, NULL);
|
2020-07-08 16:17:13 +00:00
|
|
|
}
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
static void
|
|
|
|
meta_pointer_confinement_wayland_init (MetaPointerConfinementWayland *confinement_wayland)
|
|
|
|
{
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
static void
|
|
|
|
meta_pointer_confinement_wayland_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWayland *confinement;
|
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
confinement = META_POINTER_CONFINEMENT_WAYLAND (object);
|
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
|
|
|
|
switch (prop_id)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
case PROP_WAYLAND_POINTER_CONSTRAINT:
|
|
|
|
g_value_set_object (value, priv->constraint);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
2015-07-07 07:22:07 +00:00
|
|
|
}
|
|
|
|
}
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2015-07-07 07:22:07 +00:00
|
|
|
static void
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWayland *confinement;
|
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
|
|
|
|
|
|
|
confinement = META_POINTER_CONFINEMENT_WAYLAND (object);
|
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
|
|
|
|
switch (prop_id)
|
2015-07-07 07:22:07 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
case PROP_WAYLAND_POINTER_CONSTRAINT:
|
|
|
|
priv->constraint = g_value_get_object (value);
|
2015-07-07 07:22:07 +00:00
|
|
|
break;
|
2020-07-08 16:17:13 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
2015-07-07 07:22:07 +00:00
|
|
|
break;
|
|
|
|
}
|
2015-06-17 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
static MetaPointerConstraint *
|
|
|
|
meta_pointer_confinement_wayland_create_constraint (MetaPointerConfinementWayland *confinement)
|
2015-06-17 04:10:52 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
|
|
|
MetaPointerConstraint *constraint;
|
2015-06-17 04:10:52 +00:00
|
|
|
MetaWaylandSurface *surface;
|
2023-09-04 14:30:38 +00:00
|
|
|
g_autoptr (MtkRegion) region = NULL;
|
2022-06-13 08:25:02 +00:00
|
|
|
int geometry_scale;
|
2020-07-08 16:17:13 +00:00
|
|
|
float dx, dy;
|
2022-06-13 08:23:40 +00:00
|
|
|
double min_edge_distance;
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
surface = meta_wayland_pointer_constraint_get_surface (priv->constraint);
|
2015-06-17 04:10:52 +00:00
|
|
|
region =
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_wayland_pointer_constraint_calculate_effective_region (priv->constraint);
|
2015-07-07 07:22:07 +00:00
|
|
|
|
2022-06-13 08:25:02 +00:00
|
|
|
geometry_scale = meta_wayland_surface_get_geometry_scale (surface);
|
|
|
|
if (geometry_scale != 1)
|
|
|
|
{
|
2023-09-04 14:30:38 +00:00
|
|
|
g_autoptr (MtkRegion) scaled_region = NULL;
|
2022-06-13 08:25:02 +00:00
|
|
|
|
|
|
|
scaled_region = meta_region_scale (region, geometry_scale);
|
2023-09-04 14:30:38 +00:00
|
|
|
g_clear_pointer (®ion, mtk_region_unref);
|
|
|
|
region = g_steal_pointer (&scaled_region);
|
2022-06-13 08:25:02 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_wayland_surface_get_absolute_coordinates (surface, 0, 0, &dx, &dy);
|
2023-09-04 14:30:38 +00:00
|
|
|
mtk_region_translate (region, dx, dy);
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2022-06-13 08:25:02 +00:00
|
|
|
min_edge_distance = wl_fixed_to_double (1) * geometry_scale;
|
2023-09-04 14:30:38 +00:00
|
|
|
constraint = meta_pointer_constraint_new (g_steal_pointer (®ion),
|
|
|
|
min_edge_distance);
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
return constraint;
|
2017-10-09 12:49:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-07-08 16:17:13 +00:00
|
|
|
meta_pointer_confinement_wayland_class_init (MetaPointerConfinementWaylandClass *klass)
|
2017-10-09 12:49:10 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->set_property = meta_pointer_confinement_wayland_set_property;
|
|
|
|
object_class->get_property = meta_pointer_confinement_wayland_get_property;
|
|
|
|
|
|
|
|
klass->create_constraint = meta_pointer_confinement_wayland_create_constraint;
|
|
|
|
|
|
|
|
props[PROP_WAYLAND_POINTER_CONSTRAINT] =
|
2023-06-28 12:02:43 +00:00
|
|
|
g_param_spec_object ("wayland-pointer-constraint", NULL, NULL,
|
2020-07-08 16:17:13 +00:00
|
|
|
META_TYPE_WAYLAND_POINTER_CONSTRAINT,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
|
|
G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (object_class, N_PROPS, props);
|
2017-10-09 12:49:10 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWayland *
|
2015-06-17 04:10:52 +00:00
|
|
|
meta_pointer_confinement_wayland_new (MetaWaylandPointerConstraint *constraint)
|
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
return g_object_new (META_TYPE_POINTER_CONFINEMENT_WAYLAND,
|
|
|
|
"wayland-pointer-constraint", constraint,
|
|
|
|
NULL);
|
2015-06-17 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaWaylandPointerConstraint *
|
|
|
|
meta_pointer_confinement_wayland_get_wayland_pointer_constraint (MetaPointerConfinementWayland *confinement)
|
2015-06-17 04:10:52 +00:00
|
|
|
{
|
2020-07-08 16:17:13 +00:00
|
|
|
MetaPointerConfinementWaylandPrivate *priv;
|
2015-06-17 04:10:52 +00:00
|
|
|
|
2020-07-08 16:17:13 +00:00
|
|
|
priv = meta_pointer_confinement_wayland_get_instance_private (confinement);
|
|
|
|
return priv->constraint;
|
2015-06-17 04:10:52 +00:00
|
|
|
}
|