/* -*- 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 */ #include "config.h" #include "window-wayland.h" #include #include "window-private.h" #include "boxes-private.h" #include "stack-tracker.h" #include "meta-wayland-surface.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 we get a 0x0 size, this means that we're trying to resize * a surface that doesn't have any buffer attached. This can happen * when a client requests an xdg surface before bringing it up. * The constrained_rect will be 1x1 because of how our constraints * code works, and sending that to the window would cause it to * redraw itself, so just don't send anything. */ if (unconstrained_rect.width == 0 && unconstrained_rect.height == 0) 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 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; } 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); }