summaryrefslogtreecommitdiff
path: root/src/wayland/meta-window-wayland.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/wayland/meta-window-wayland.c')
-rw-r--r--src/wayland/meta-window-wayland.c472
1 files changed, 472 insertions, 0 deletions
diff --git a/src/wayland/meta-window-wayland.c b/src/wayland/meta-window-wayland.c
new file mode 100644
index 00000000..8c6ca581
--- /dev/null
+++ b/src/wayland/meta-window-wayland.c
@@ -0,0 +1,472 @@
+/* -*- 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"
+
+#include "meta-window-wayland.h"
+
+#include <meta/errors.h>
+#include "window-private.h"
+#include "boxes-private.h"
+#include "stack-tracker.h"
+#include "meta-wayland-surface.h"
+#include "compositor/meta-surface-actor-wayland.h"
+
+struct _MetaWindowWayland
+{
+ MetaWindow parent;
+
+ MetaWaylandSerial pending_configure_serial;
+ gboolean has_pending_move;
+ int pending_move_x;
+ int pending_move_y;
+
+ int last_sent_width;
+ int last_sent_height;
+};
+
+struct _MetaWindowWaylandClass
+{
+ MetaWindowClass parent_class;
+};
+
+G_DEFINE_TYPE (MetaWindowWayland, meta_window_wayland, META_TYPE_WINDOW)
+
+static void
+meta_window_wayland_manage (MetaWindow *window)
+{
+ MetaDisplay *display = window->display;
+
+ meta_display_register_wayland_window (display, window);
+
+ {
+ meta_stack_tracker_record_add (window->screen->stack_tracker,
+ window->stamp,
+ 0);
+ }
+
+ if (window->surface->role == META_WAYLAND_SURFACE_ROLE_XDG_POPUP)
+ {
+ MetaWaylandSurface *parent = window->surface->popup.parent;
+
+ g_assert (parent);
+
+ meta_window_set_transient_for (window, parent->window);
+ meta_window_set_type (window, META_WINDOW_DROPDOWN_MENU);
+ }
+}
+
+static void
+meta_window_wayland_unmanage (MetaWindow *window)
+{
+ {
+ meta_stack_tracker_record_remove (window->screen->stack_tracker,
+ window->stamp,
+ 0);
+ }
+
+ meta_display_unregister_wayland_window (window->display, window);
+}
+
+static void
+meta_window_wayland_ping (MetaWindow *window,
+ guint32 serial)
+{
+ meta_wayland_surface_ping (window->surface, serial);
+}
+
+static void
+meta_window_wayland_delete (MetaWindow *window,
+ guint32 timestamp)
+{
+ meta_wayland_surface_delete (window->surface);
+}
+
+static void
+meta_window_wayland_kill (MetaWindow *window)
+{
+ MetaWaylandSurface *surface = window->surface;
+ struct wl_resource *resource = surface->resource;
+
+ /* Send the client an unrecoverable error to kill the client. */
+ wl_resource_post_error (resource,
+ WL_DISPLAY_ERROR_NO_MEMORY,
+ "User requested that we kill you. Sorry. Don't take it too personally.");
+}
+
+static void
+meta_window_wayland_focus (MetaWindow *window,
+ guint32 timestamp)
+{
+ meta_display_set_input_focus_window (window->display,
+ window,
+ FALSE,
+ timestamp);
+}
+
+static void
+surface_state_changed (MetaWindow *window)
+{
+ MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window);
+
+ meta_wayland_surface_configure_notify (window->surface,
+ wl_window->last_sent_width,
+ wl_window->last_sent_height,
+ &wl_window->pending_configure_serial);
+}
+
+static void
+meta_window_wayland_grab_op_began (MetaWindow *window,
+ MetaGrabOp op)
+{
+ if (meta_grab_op_is_resizing (op))
+ surface_state_changed (window);
+
+ META_WINDOW_CLASS (meta_window_wayland_parent_class)->grab_op_began (window, op);
+}
+
+static void
+meta_window_wayland_grab_op_ended (MetaWindow *window,
+ MetaGrabOp op)
+{
+ if (meta_grab_op_is_resizing (op))
+ surface_state_changed (window);
+
+ META_WINDOW_CLASS (meta_window_wayland_parent_class)->grab_op_ended (window, op);
+}
+
+static void
+meta_window_wayland_move_resize_internal (MetaWindow *window,
+ int gravity,
+ MetaRectangle unconstrained_rect,
+ MetaRectangle constrained_rect,
+ MetaMoveResizeFlags flags,
+ MetaMoveResizeResultFlags *result)
+{
+ MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window);
+ gboolean can_move_now;
+
+ g_assert (window->frame == NULL);
+
+ /* For wayland clients, the size is completely determined by the client,
+ * and while this allows to avoid some trickery with frames and the resulting
+ * lagging, we also need to insist a bit when the constraints would apply
+ * a different size than the client decides.
+ *
+ * Note that this is not generally a problem for normal toplevel windows (the
+ * constraints don't see the size hints, or just change the position), but
+ * it can be for maximized or fullscreen.
+ */
+
+ if (flags & META_MOVE_RESIZE_WAYLAND_RESIZE)
+ {
+ /* This is a call to wl_surface_commit(), ignore the constrained_rect and
+ * update the real client size to match the buffer size.
+ */
+
+ if (window->rect.width != unconstrained_rect.width ||
+ window->rect.height != unconstrained_rect.height)
+ {
+ *result |= META_MOVE_RESIZE_RESULT_RESIZED;
+ window->rect.width = unconstrained_rect.width;
+ window->rect.height = unconstrained_rect.height;
+ }
+
+ /* This is a commit of an attach. We should move the window to match the
+ * new position the client wants. */
+ can_move_now = TRUE;
+ }
+ else
+ {
+ /* If the size changed, or the state changed, then we have to wait until
+ * the client acks our configure before moving the window. */
+ if (constrained_rect.width != window->rect.width ||
+ constrained_rect.height != window->rect.height ||
+ (flags & META_MOVE_RESIZE_STATE_CHANGED))
+ {
+ /* If the constrained size is 1x1 and the unconstrained size is 0x0
+ * it means that we are trying to resize a window where the client has
+ * not yet committed a buffer. The 1x1 constrained size is a result of
+ * how the constraints code works. Lets avoid trying to have the
+ * client configure itself to draw on a 1x1 surface.
+ *
+ * We cannot guard against only an empty unconstrained_rect here,
+ * because the client may have created a xdg surface without a buffer
+ * attached and asked it to be maximized. In such case we should let
+ * it know about the expected window geometry of a maximized window,
+ * even though there is currently no buffer attached. */
+ if (unconstrained_rect.width == 0 &&
+ unconstrained_rect.height == 0 &&
+ constrained_rect.width == 1 &&
+ constrained_rect.height == 1)
+ return;
+
+ meta_wayland_surface_configure_notify (window->surface,
+ constrained_rect.width,
+ constrained_rect.height,
+ &wl_window->pending_configure_serial);
+
+ /* We need to wait until the resize completes before we can move */
+ can_move_now = FALSE;
+ }
+ else
+ {
+ /* We're just moving the window, so we don't need to wait for a configure
+ * and then ack to simply move the window. */
+ can_move_now = TRUE;
+ }
+ }
+
+ wl_window->last_sent_width = constrained_rect.width;
+ wl_window->last_sent_height = constrained_rect.height;
+
+ if (can_move_now)
+ {
+ int new_x = constrained_rect.x;
+ int new_y = constrained_rect.y;
+
+ if (new_x != window->rect.x || new_y != window->rect.y)
+ {
+ *result |= META_MOVE_RESIZE_RESULT_MOVED;
+ window->rect.x = new_x;
+ window->rect.y = new_y;
+ }
+
+ int new_buffer_x = new_x - window->custom_frame_extents.left;
+ int new_buffer_y = new_y - window->custom_frame_extents.top;
+
+ if (new_buffer_x != window->buffer_rect.x || new_buffer_y != window->buffer_rect.y)
+ {
+ *result |= META_MOVE_RESIZE_RESULT_MOVED;
+ window->buffer_rect.x = new_buffer_x;
+ window->buffer_rect.y = new_buffer_y;
+ }
+ }
+ else
+ {
+ int new_x = constrained_rect.x;
+ int new_y = constrained_rect.y;
+
+ if (new_x != window->rect.x || new_y != window->rect.y)
+ {
+ wl_window->has_pending_move = TRUE;
+ wl_window->pending_move_x = new_x;
+ wl_window->pending_move_y = new_y;
+ }
+ }
+}
+
+static void
+meta_window_wayland_main_monitor_changed (MetaWindow *window,
+ const MetaMonitorInfo *old)
+{
+ MetaWaylandSurface *surface = window->surface;
+
+ if (surface)
+ {
+ MetaSurfaceActorWayland *actor =
+ META_SURFACE_ACTOR_WAYLAND (surface->surface_actor);
+
+ meta_surface_actor_wayland_sync_state_recursive (actor);
+ }
+}
+
+static void
+appears_focused_changed (GObject *object,
+ GParamSpec *pspec,
+ gpointer user_data)
+{
+ MetaWindow *window = META_WINDOW (object);
+
+ /* When we're unmanaging, we remove focus from the window,
+ * causing this to fire. Don't do anything in that case. */
+ if (window->unmanaging)
+ return;
+
+ surface_state_changed (window);
+}
+
+static void
+meta_window_wayland_init (MetaWindowWayland *wl_window)
+{
+ MetaWindow *window = META_WINDOW (wl_window);
+
+ g_signal_connect (window, "notify::appears-focused",
+ G_CALLBACK (appears_focused_changed), NULL);
+}
+
+static void
+meta_window_wayland_class_init (MetaWindowWaylandClass *klass)
+{
+ MetaWindowClass *window_class = META_WINDOW_CLASS (klass);
+
+ window_class->manage = meta_window_wayland_manage;
+ window_class->unmanage = meta_window_wayland_unmanage;
+ window_class->ping = meta_window_wayland_ping;
+ window_class->delete = meta_window_wayland_delete;
+ window_class->kill = meta_window_wayland_kill;
+ window_class->focus = meta_window_wayland_focus;
+ window_class->grab_op_began = meta_window_wayland_grab_op_began;
+ window_class->grab_op_ended = meta_window_wayland_grab_op_ended;
+ window_class->move_resize_internal = meta_window_wayland_move_resize_internal;
+ window_class->main_monitor_changed = meta_window_wayland_main_monitor_changed;
+}
+
+MetaWindow *
+meta_window_wayland_new (MetaDisplay *display,
+ MetaWaylandSurface *surface)
+{
+ XWindowAttributes attrs;
+ MetaScreen *scr = display->screen;
+ MetaWindow *window;
+
+ attrs.x = 0;
+ attrs.y = 0;
+ attrs.width = 0;
+ attrs.height = 0;
+ attrs.border_width = 0;
+ attrs.depth = 24;
+ attrs.visual = NULL;
+ attrs.root = scr->xroot;
+ attrs.class = InputOutput;
+ attrs.bit_gravity = NorthWestGravity;
+ attrs.win_gravity = NorthWestGravity;
+ attrs.backing_store = 0;
+ attrs.backing_planes = ~0;
+ attrs.backing_pixel = 0;
+ attrs.save_under = 0;
+ attrs.colormap = 0;
+ attrs.map_installed = 1;
+ attrs.map_state = IsUnmapped;
+ attrs.all_event_masks = ~0;
+ attrs.your_event_mask = 0;
+ attrs.do_not_propagate_mask = 0;
+ attrs.override_redirect = 0;
+ attrs.screen = scr->xscreen;
+
+ /* XXX: Note: In the Wayland case we currently still trap X errors while
+ * creating a MetaWindow because we will still be making various redundant
+ * X requests (passing a window xid of None) until we thoroughly audit all
+ * the code to make sure it knows about non X based clients...
+ */
+ meta_error_trap_push (display); /* Push a trap over all of window
+ * creation, to reduce XSync() calls
+ */
+
+ window = _meta_window_shared_new (display,
+ scr,
+ META_WINDOW_CLIENT_TYPE_WAYLAND,
+ surface,
+ None,
+ WithdrawnState,
+ META_COMP_EFFECT_CREATE,
+ &attrs);
+ window->can_ping = TRUE;
+
+ meta_error_trap_pop (display); /* pop the XSync()-reducing trap */
+
+ return window;
+}
+
+static gboolean
+should_do_pending_move (MetaWindowWayland *wl_window,
+ MetaWaylandSerial *acked_configure_serial)
+{
+ if (!wl_window->has_pending_move)
+ return FALSE;
+
+ if (wl_window->pending_configure_serial.set)
+ {
+ /* If we're waiting for a configure and this isn't an ACK for
+ * any configure, then fizzle it out. */
+ if (!acked_configure_serial->set)
+ return FALSE;
+
+ /* If we're waiting for a configure and this isn't an ACK for
+ * the configure we're waiting for, then fizzle it out. */
+ if (acked_configure_serial->value != wl_window->pending_configure_serial.value)
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+/**
+ * meta_window_move_resize_wayland:
+ *
+ * Complete a resize operation from a wayland client.
+ */
+void
+meta_window_wayland_move_resize (MetaWindow *window,
+ MetaWaylandSerial *acked_configure_serial,
+ MetaRectangle new_geom,
+ int dx,
+ int dy)
+{
+ MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window);
+ int gravity;
+ MetaRectangle rect;
+ MetaMoveResizeFlags flags;
+
+ /* XXX: Find a better place to store the window geometry offsets. */
+ window->custom_frame_extents.left = new_geom.x;
+ window->custom_frame_extents.top = new_geom.y;
+
+ flags = META_MOVE_RESIZE_WAYLAND_RESIZE;
+
+ /* x/y are ignored when we're doing interactive resizing */
+ if (!meta_grab_op_is_resizing (window->display->grab_op))
+ {
+ if (wl_window->has_pending_move && should_do_pending_move (wl_window, acked_configure_serial))
+ {
+ rect.x = wl_window->pending_move_x;
+ rect.y = wl_window->pending_move_y;
+ wl_window->has_pending_move = FALSE;
+ flags |= META_MOVE_RESIZE_MOVE_ACTION;
+ }
+ else
+ {
+ rect.x = window->rect.x;
+ rect.y = window->rect.y;
+ }
+
+ if (dx != 0 || dy != 0)
+ {
+ rect.x += dx;
+ rect.y += dy;
+ flags |= META_MOVE_RESIZE_MOVE_ACTION;
+ }
+ }
+
+ wl_window->pending_configure_serial.set = FALSE;
+
+ rect.width = new_geom.width;
+ rect.height = new_geom.height;
+
+ if (rect.width != window->rect.width || rect.height != window->rect.height)
+ flags |= META_MOVE_RESIZE_RESIZE_ACTION;
+
+ gravity = meta_resize_gravity_from_grab_op (window->display->grab_op);
+ meta_window_move_resize_internal (window, flags, gravity, rect);
+}