diff options
Diffstat (limited to 'src/wayland/meta-window-wayland.c')
-rw-r--r-- | src/wayland/meta-window-wayland.c | 472 |
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); +} |