2017-08-30 07:17:41 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2021-05-10 21:40:49 +00:00
|
|
|
#include <X11/Xatom.h>
|
|
|
|
|
2019-11-19 13:08:45 +00:00
|
|
|
#include "core/frame.h"
|
|
|
|
#include "meta/meta-x11-errors.h"
|
2017-08-30 07:17:41 +00:00
|
|
|
#include "x11/window-x11.h"
|
|
|
|
#include "x11/window-x11-private.h"
|
2019-09-03 14:18:00 +00:00
|
|
|
#include "x11/xprops.h"
|
2017-08-30 07:17:41 +00:00
|
|
|
#include "wayland/meta-window-xwayland.h"
|
|
|
|
#include "wayland/meta-wayland.h"
|
|
|
|
|
2017-10-20 08:47:24 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
|
|
|
|
PROP_XWAYLAND_MAY_GRAB_KEYBOARD,
|
|
|
|
|
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
|
|
|
static GParamSpec *obj_props[PROP_LAST];
|
|
|
|
|
2017-08-30 07:17:41 +00:00
|
|
|
struct _MetaWindowXwayland
|
|
|
|
{
|
|
|
|
MetaWindowX11 parent;
|
2017-10-20 08:47:24 +00:00
|
|
|
|
|
|
|
gboolean xwayland_may_grab_keyboard;
|
2019-11-19 13:08:45 +00:00
|
|
|
int freeze_count;
|
2017-08-30 07:17:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct _MetaWindowXwaylandClass
|
|
|
|
{
|
|
|
|
MetaWindowX11Class parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaWindowXwayland, meta_window_xwayland, META_TYPE_WINDOW_X11)
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_xwayland_init (MetaWindowXwayland *window_xwayland)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-09-03 14:18:00 +00:00
|
|
|
/**
|
|
|
|
* meta_window_xwayland_adjust_fullscreen_monitor_rect:
|
|
|
|
*
|
|
|
|
* This function implements a workaround for X11 apps which use randr to change the
|
|
|
|
* the monitor resolution, followed by setting _NET_WM_FULLSCREEN to make the
|
|
|
|
* window-manager fullscreen them.
|
|
|
|
*
|
|
|
|
* Newer versions of Xwayland support the randr part of this by supporting randr
|
|
|
|
* resolution change emulation in combination with using WPviewport to scale the
|
|
|
|
* app's window (at the emulated resolution) to fill the entire monitor.
|
|
|
|
*
|
|
|
|
* Apps using randr in combination with NET_WM_STATE_FULLSCREEN expect the
|
|
|
|
* fullscreen window to have the size of the emulated randr resolution since
|
|
|
|
* when running on regular Xorg the resolution will actually be changed and
|
|
|
|
* after that going fullscreen through NET_WM_STATE_FULLSCREEN will size
|
|
|
|
* the window to be equal to the new resolution.
|
|
|
|
*
|
|
|
|
* We need to emulate this behavior for these apps to work correctly.
|
|
|
|
*
|
|
|
|
* Xwayland's emulated resolution is a per X11 client setting and Xwayland
|
|
|
|
* will set a special _XWAYLAND_RANDR_EMU_MONITOR_RECTS property on the
|
|
|
|
* toplevel windows of a client (and only those of that client), which has
|
|
|
|
* changed the (emulated) resolution through a randr call.
|
|
|
|
*
|
|
|
|
* Here we check for that property and if it is set we adjust the fullscreen
|
|
|
|
* monitor rect for this window to match the emulated resolution.
|
|
|
|
*
|
|
|
|
* Here is a step-by-step of such an app going fullscreen:
|
|
|
|
* 1. App changes monitor resolution with randr.
|
|
|
|
* 2. Xwayland sets the _XWAYLAND_RANDR_EMU_MONITOR_RECTS property on all the
|
|
|
|
* apps current and future windows. This property contains the origin of the
|
|
|
|
* monitor for which the emulated resolution is set and the emulated
|
|
|
|
* resolution.
|
|
|
|
* 3. App sets _NET_WM_FULLSCREEN.
|
|
|
|
* 4. We check the property and adjust the app's fullscreen size to match
|
|
|
|
* the emulated resolution.
|
|
|
|
* 5. Xwayland sees a Window at monitor origin fully covering the emulated
|
|
|
|
* monitor resolution. Xwayland sets a viewport making the emulated
|
|
|
|
* resolution sized window cover the full actual monitor resolution.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
meta_window_xwayland_adjust_fullscreen_monitor_rect (MetaWindow *window,
|
|
|
|
MetaRectangle *fs_monitor_rect)
|
|
|
|
{
|
|
|
|
MetaX11Display *x11_display = window->display->x11_display;
|
|
|
|
MetaRectangle win_monitor_rect;
|
|
|
|
cairo_rectangle_int_t *rects;
|
|
|
|
uint32_t *list = NULL;
|
|
|
|
int i, n_items = 0;
|
|
|
|
|
|
|
|
if (!window->monitor)
|
|
|
|
{
|
|
|
|
g_warning ("MetaWindow does not have a monitor");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
win_monitor_rect = meta_logical_monitor_get_layout (window->monitor);
|
|
|
|
|
|
|
|
if (!meta_prop_get_cardinal_list (x11_display,
|
|
|
|
window->xwindow,
|
|
|
|
x11_display->atom__XWAYLAND_RANDR_EMU_MONITOR_RECTS,
|
|
|
|
&list, &n_items))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (n_items % 4)
|
|
|
|
{
|
|
|
|
meta_verbose ("_XWAYLAND_RANDR_EMU_MONITOR_RECTS on %s has %d values which is not a multiple of 4",
|
|
|
|
window->desc, n_items);
|
|
|
|
g_free (list);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rects = (cairo_rectangle_int_t *) list;
|
|
|
|
n_items = n_items / 4;
|
|
|
|
for (i = 0; i < n_items; i++)
|
|
|
|
{
|
|
|
|
if (rects[i].x == win_monitor_rect.x && rects[i].y == win_monitor_rect.y)
|
|
|
|
{
|
|
|
|
fs_monitor_rect->width = rects[i].width;
|
|
|
|
fs_monitor_rect->height = rects[i].height;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (list);
|
|
|
|
}
|
|
|
|
|
2017-08-30 07:17:41 +00:00
|
|
|
static void
|
|
|
|
meta_window_xwayland_force_restore_shortcuts (MetaWindow *window,
|
|
|
|
ClutterInputDevice *source)
|
|
|
|
{
|
|
|
|
MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default ();
|
|
|
|
|
|
|
|
meta_wayland_compositor_restore_shortcuts (compositor, source);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_window_xwayland_shortcuts_inhibited (MetaWindow *window,
|
|
|
|
ClutterInputDevice *source)
|
|
|
|
{
|
|
|
|
MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default ();
|
|
|
|
|
|
|
|
return meta_wayland_compositor_is_shortcuts_inhibited (compositor, source);
|
|
|
|
}
|
|
|
|
|
2019-11-19 13:08:45 +00:00
|
|
|
static void
|
|
|
|
apply_allow_commits_x11_property (MetaWindowXwayland *xwayland_window,
|
|
|
|
gboolean allow_commits)
|
|
|
|
{
|
|
|
|
MetaWindow *window = META_WINDOW (xwayland_window);
|
|
|
|
MetaDisplay *display = window->display;
|
|
|
|
MetaX11Display *x11_display = display->x11_display;
|
|
|
|
MetaFrame *frame;
|
|
|
|
Window xwin;
|
|
|
|
guint32 property[1];
|
|
|
|
|
2020-09-21 10:02:46 +00:00
|
|
|
if (!x11_display)
|
|
|
|
return;
|
|
|
|
|
2019-11-19 13:08:45 +00:00
|
|
|
frame = meta_window_get_frame (window);
|
|
|
|
if (!frame)
|
|
|
|
xwin = window->xwindow;
|
|
|
|
else
|
|
|
|
xwin = meta_frame_get_xwindow (frame);
|
|
|
|
|
|
|
|
if (!xwin)
|
|
|
|
return;
|
|
|
|
|
|
|
|
property[0] = !!allow_commits;
|
|
|
|
|
|
|
|
meta_x11_error_trap_push (x11_display);
|
2020-09-21 10:02:46 +00:00
|
|
|
XChangeProperty (x11_display->xdisplay, xwin,
|
2019-11-19 13:08:45 +00:00
|
|
|
x11_display->atom__XWAYLAND_ALLOW_COMMITS,
|
|
|
|
XA_CARDINAL, 32, PropModeReplace,
|
|
|
|
(guchar*) &property, 1);
|
|
|
|
meta_x11_error_trap_pop (x11_display);
|
2020-09-21 10:02:46 +00:00
|
|
|
XFlush (x11_display->xdisplay);
|
2019-11-19 13:08:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_xwayland_freeze_commits (MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaWindowXwayland *xwayland_window = META_WINDOW_XWAYLAND (window);
|
|
|
|
|
|
|
|
if (xwayland_window->freeze_count == 0)
|
|
|
|
apply_allow_commits_x11_property (xwayland_window, FALSE);
|
|
|
|
|
|
|
|
xwayland_window->freeze_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_xwayland_thaw_commits (MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaWindowXwayland *xwayland_window = META_WINDOW_XWAYLAND (window);
|
|
|
|
|
|
|
|
g_return_if_fail (xwayland_window->freeze_count > 0);
|
|
|
|
|
|
|
|
xwayland_window->freeze_count--;
|
|
|
|
if (xwayland_window->freeze_count > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
apply_allow_commits_x11_property (xwayland_window, TRUE);
|
|
|
|
}
|
|
|
|
|
2019-12-06 15:01:35 +00:00
|
|
|
static gboolean
|
|
|
|
meta_window_xwayland_always_update_shape (MetaWindow *window)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* On Xwayland, resizing a window will clear the corresponding Wayland
|
|
|
|
* buffer to plain solid black.
|
|
|
|
*
|
|
|
|
* Therefore, to address the black shadows which sometimes show during
|
|
|
|
* resize with Xwayland, we need to always update the window shape
|
|
|
|
* regardless of the actual frozen state of the window actor.
|
|
|
|
*/
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-10-20 08:47:24 +00:00
|
|
|
static void
|
|
|
|
meta_window_xwayland_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaWindowXwayland *window = META_WINDOW_XWAYLAND (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_XWAYLAND_MAY_GRAB_KEYBOARD:
|
|
|
|
g_value_set_boolean (value, window->xwayland_may_grab_keyboard);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_xwayland_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaWindowXwayland *window = META_WINDOW_XWAYLAND (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_XWAYLAND_MAY_GRAB_KEYBOARD:
|
|
|
|
window->xwayland_may_grab_keyboard = g_value_get_boolean (value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 07:17:41 +00:00
|
|
|
static void
|
|
|
|
meta_window_xwayland_class_init (MetaWindowXwaylandClass *klass)
|
|
|
|
{
|
|
|
|
MetaWindowClass *window_class = META_WINDOW_CLASS (klass);
|
2019-11-19 13:08:45 +00:00
|
|
|
MetaWindowX11Class *window_x11_class = META_WINDOW_X11_CLASS (klass);
|
2017-10-20 08:47:24 +00:00
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
2017-08-30 07:17:41 +00:00
|
|
|
|
2019-09-03 14:18:00 +00:00
|
|
|
window_class->adjust_fullscreen_monitor_rect = meta_window_xwayland_adjust_fullscreen_monitor_rect;
|
2017-08-30 07:17:41 +00:00
|
|
|
window_class->force_restore_shortcuts = meta_window_xwayland_force_restore_shortcuts;
|
|
|
|
window_class->shortcuts_inhibited = meta_window_xwayland_shortcuts_inhibited;
|
2017-10-20 08:47:24 +00:00
|
|
|
|
2019-11-19 13:08:45 +00:00
|
|
|
window_x11_class->freeze_commits = meta_window_xwayland_freeze_commits;
|
|
|
|
window_x11_class->thaw_commits = meta_window_xwayland_thaw_commits;
|
2019-12-06 15:01:35 +00:00
|
|
|
window_x11_class->always_update_shape = meta_window_xwayland_always_update_shape;
|
2019-11-19 13:08:45 +00:00
|
|
|
|
2017-10-20 08:47:24 +00:00
|
|
|
gobject_class->get_property = meta_window_xwayland_get_property;
|
|
|
|
gobject_class->set_property = meta_window_xwayland_set_property;
|
|
|
|
|
|
|
|
obj_props[PROP_XWAYLAND_MAY_GRAB_KEYBOARD] =
|
|
|
|
g_param_spec_boolean ("xwayland-may-grab-keyboard",
|
|
|
|
"Xwayland may use keyboard grabs",
|
|
|
|
"Whether the client may use Xwayland keyboard grabs on this window",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
g_object_class_install_properties (gobject_class, PROP_LAST, obj_props);
|
2017-08-30 07:17:41 +00:00
|
|
|
}
|