/* * 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 . * */ #include "config.h" #include #include "core/frame.h" #include "meta/meta-x11-errors.h" #include "x11/window-x11.h" #include "x11/window-x11-private.h" #include "x11/xprops.h" #include "wayland/meta-window-xwayland.h" #include "wayland/meta-wayland.h" #include "wayland/meta-wayland-surface.h" enum { PROP_0, PROP_XWAYLAND_MAY_GRAB_KEYBOARD, PROP_SURFACE, PROP_LAST }; static GParamSpec *obj_props[PROP_LAST]; struct _MetaWindowXwayland { MetaWindowX11 parent; MetaWaylandSurface *surface; gboolean xwayland_may_grab_keyboard; int freeze_count; }; 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) { } /** * 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); } static void meta_window_xwayland_force_restore_shortcuts (MetaWindow *window, ClutterInputDevice *source) { MetaDisplay *display = meta_window_get_display (window); MetaContext *context = meta_display_get_context (display); MetaWaylandCompositor *compositor = meta_context_get_wayland_compositor (context); meta_wayland_compositor_restore_shortcuts (compositor, source); } static gboolean meta_window_xwayland_shortcuts_inhibited (MetaWindow *window, ClutterInputDevice *source) { MetaDisplay *display = meta_window_get_display (window); MetaContext *context = meta_display_get_context (display); MetaWaylandCompositor *compositor = meta_context_get_wayland_compositor (context); return meta_wayland_compositor_is_shortcuts_inhibited (compositor, source); } static MetaWaylandSurface * meta_window_xwayland_get_wayland_surface (MetaWindow *window) { MetaWindowXwayland *xwayland_window = META_WINDOW_XWAYLAND (window); return xwayland_window->surface; } 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]; if (!x11_display) return; 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); XChangeProperty (x11_display->xdisplay, xwin, x11_display->atom__XWAYLAND_ALLOW_COMMITS, XA_CARDINAL, 32, PropModeReplace, (guchar*) &property, 1); meta_x11_error_trap_pop (x11_display); XFlush (x11_display->xdisplay); } 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); } 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; } 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; case PROP_SURFACE: g_value_set_object (value, window->surface); 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; case PROP_SURFACE: window->surface = g_value_get_object (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void meta_window_xwayland_class_init (MetaWindowXwaylandClass *klass) { MetaWindowClass *window_class = META_WINDOW_CLASS (klass); MetaWindowX11Class *window_x11_class = META_WINDOW_X11_CLASS (klass); GObjectClass *gobject_class = G_OBJECT_CLASS (klass); window_class->adjust_fullscreen_monitor_rect = meta_window_xwayland_adjust_fullscreen_monitor_rect; window_class->force_restore_shortcuts = meta_window_xwayland_force_restore_shortcuts; window_class->shortcuts_inhibited = meta_window_xwayland_shortcuts_inhibited; window_class->get_wayland_surface = meta_window_xwayland_get_wayland_surface; window_x11_class->freeze_commits = meta_window_xwayland_freeze_commits; window_x11_class->thaw_commits = meta_window_xwayland_thaw_commits; window_x11_class->always_update_shape = meta_window_xwayland_always_update_shape; 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); obj_props[PROP_SURFACE] = g_param_spec_object ("surface", "Surface", "The corresponding Wayland surface", META_TYPE_WAYLAND_SURFACE, G_PARAM_CONSTRUCT | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties (gobject_class, PROP_LAST, obj_props); } void meta_window_xwayland_set_surface (MetaWindowXwayland *window, MetaWaylandSurface *surface) { window->surface = surface; }