diff options
Diffstat (limited to 'src/wayland/meta-window-wayland.c')
-rw-r--r-- | src/wayland/meta-window-wayland.c | 1195 |
1 files changed, 0 insertions, 1195 deletions
diff --git a/src/wayland/meta-window-wayland.c b/src/wayland/meta-window-wayland.c deleted file mode 100644 index 12e9567d9..000000000 --- a/src/wayland/meta-window-wayland.c +++ /dev/null @@ -1,1195 +0,0 @@ -/* -*- 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 "wayland/meta-window-wayland.h" - -#include <errno.h> -#include <string.h> - -#include "backends/meta-backend-private.h" -#include "backends/meta-backend-private.h" -#include "backends/meta-logical-monitor.h" -#include "compositor/meta-surface-actor-wayland.h" -#include "compositor/meta-window-actor-private.h" -#include "core/boxes-private.h" -#include "core/stack-tracker.h" -#include "core/window-private.h" -#include "meta/meta-x11-errors.h" -#include "wayland/meta-wayland-actor-surface.h" -#include "wayland/meta-wayland-private.h" -#include "wayland/meta-wayland-surface.h" -#include "wayland/meta-wayland-window-configuration.h" -#include "wayland/meta-wayland-xdg-shell.h" - -struct _MetaWindowWayland -{ - MetaWindow parent; - - int geometry_scale; - - GList *pending_configurations; - gboolean has_pending_state_change; - - gboolean has_last_sent_configuration; - int last_sent_x; - int last_sent_y; - int last_sent_width; - int last_sent_height; - int last_sent_rel_x; - int last_sent_rel_y; - int last_sent_geometry_scale; - MetaGravity last_sent_gravity; - - gboolean has_been_shown; -}; - -struct _MetaWindowWaylandClass -{ - MetaWindowClass parent_class; -}; - -G_DEFINE_TYPE (MetaWindowWayland, meta_window_wayland, META_TYPE_WINDOW) - -static void -set_geometry_scale_for_window (MetaWindowWayland *wl_window, - int geometry_scale) -{ - MetaWindowActor *window_actor; - - wl_window->geometry_scale = geometry_scale; - - window_actor = meta_window_actor_from_window (META_WINDOW (wl_window)); - if (window_actor) - meta_window_actor_set_geometry_scale (window_actor, geometry_scale); -} - -static int -get_window_geometry_scale_for_logical_monitor (MetaLogicalMonitor *logical_monitor) -{ - g_assert (logical_monitor); - - if (meta_is_stage_views_scaled ()) - return 1; - else - return meta_logical_monitor_get_scale (logical_monitor); -} - -static void -meta_window_wayland_manage (MetaWindow *window) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - MetaDisplay *display = window->display; - - wl_window->geometry_scale = meta_window_wayland_get_geometry_scale (window); - - meta_display_register_wayland_window (display, window); - - { - meta_stack_tracker_record_add (window->display->stack_tracker, - window->stamp, - 0); - } - - meta_wayland_surface_window_managed (window->surface, window); -} - -static void -meta_window_wayland_unmanage (MetaWindow *window) -{ - { - meta_stack_tracker_record_remove (window->display->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) -{ - if (meta_window_is_focusable (window)) - { - meta_display_set_input_focus (window->display, - window, - FALSE, - timestamp); - } -} - -static void -meta_window_wayland_configure (MetaWindowWayland *wl_window, - MetaWaylandWindowConfiguration *configuration) -{ - MetaWindow *window = META_WINDOW (wl_window); - - meta_wayland_surface_configure_notify (window->surface, configuration); - - wl_window->pending_configurations = - g_list_prepend (wl_window->pending_configurations, configuration); -} - -static void -surface_state_changed (MetaWindow *window) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - MetaWaylandWindowConfiguration *configuration; - - /* don't send notify when the window is being unmanaged */ - if (window->unmanaging) - return; - - g_return_if_fail (wl_window->has_last_sent_configuration); - - configuration = - meta_wayland_window_configuration_new (window, - wl_window->last_sent_x, - wl_window->last_sent_y, - wl_window->last_sent_width, - wl_window->last_sent_height, - wl_window->last_sent_geometry_scale, - META_MOVE_RESIZE_STATE_CHANGED, - wl_window->last_sent_gravity); - - meta_window_wayland_configure (wl_window, configuration); -} - -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, - MetaGravity gravity, - MetaRectangle unconstrained_rect, - MetaRectangle constrained_rect, - MetaRectangle temporary_rect, - int rel_x, - int rel_y, - MetaMoveResizeFlags flags, - MetaMoveResizeResultFlags *result) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - gboolean can_move_now = FALSE; - int configured_x; - int configured_y; - int configured_width; - int configured_height; - int geometry_scale; - int new_x; - int new_y; - int new_buffer_x; - int new_buffer_y; - - g_assert (window->frame == NULL); - - /* don't do anything if we're dropping the window, see #751847 */ - if (window->unmanaging) - return; - - configured_x = constrained_rect.x; - configured_y = constrained_rect.y; - - /* The scale the window is drawn in might change depending on what monitor it - * is mainly on. Scale the configured rectangle to be in logical pixel - * coordinate space so that we can have a scale independent size to pass - * to the Wayland surface. */ - geometry_scale = meta_window_wayland_get_geometry_scale (window); - - configured_width = constrained_rect.width; - configured_height = constrained_rect.height; - - /* 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_FORCE_MOVE) - { - can_move_now = TRUE; - } - else if (flags & META_MOVE_RESIZE_WAYLAND_FINISH_MOVE_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; - if (window->placement.state == META_PLACEMENT_STATE_CONSTRAINED_CONFIGURED) - window->placement.state = META_PLACEMENT_STATE_CONSTRAINED_FINISHED; - } - else - { - if (window->placement.rule) - { - switch (window->placement.state) - { - case META_PLACEMENT_STATE_UNCONSTRAINED: - case META_PLACEMENT_STATE_CONSTRAINED_CONFIGURED: - case META_PLACEMENT_STATE_INVALIDATED: - can_move_now = FALSE; - break; - case META_PLACEMENT_STATE_CONSTRAINED_PENDING: - { - if (flags & META_MOVE_RESIZE_PLACEMENT_CHANGED || - rel_x != wl_window->last_sent_rel_x || - rel_y != wl_window->last_sent_rel_y || - constrained_rect.width != window->rect.width || - constrained_rect.height != window->rect.height) - { - MetaWaylandWindowConfiguration *configuration; - - configuration = - meta_wayland_window_configuration_new_relative (rel_x, - rel_y, - configured_width, - configured_height, - geometry_scale); - meta_window_wayland_configure (wl_window, configuration); - - wl_window->last_sent_rel_x = rel_x; - wl_window->last_sent_rel_y = rel_y; - - window->placement.state = META_PLACEMENT_STATE_CONSTRAINED_CONFIGURED; - - can_move_now = FALSE; - } - else - { - window->placement.state = - META_PLACEMENT_STATE_CONSTRAINED_FINISHED; - - can_move_now = TRUE; - } - break; - } - case META_PLACEMENT_STATE_CONSTRAINED_FINISHED: - can_move_now = TRUE; - break; - } - } - else if (constrained_rect.width != window->rect.width || - constrained_rect.height != window->rect.height || - flags & META_MOVE_RESIZE_STATE_CHANGED) - { - MetaWaylandWindowConfiguration *configuration; - - if (!meta_wayland_surface_get_buffer (window->surface) && - !META_WINDOW_MAXIMIZED (window) && - window->tile_mode == META_TILE_NONE && - !meta_window_is_fullscreen (window)) - return; - - configuration = - meta_wayland_window_configuration_new (window, - configured_x, - configured_y, - configured_width, - configured_height, - geometry_scale, - flags, - gravity); - meta_window_wayland_configure (wl_window, configuration); - can_move_now = FALSE; - } - else - { - can_move_now = TRUE; - } - } - - wl_window->has_last_sent_configuration = TRUE; - wl_window->last_sent_x = configured_x; - wl_window->last_sent_y = configured_y; - wl_window->last_sent_width = configured_width; - wl_window->last_sent_height = configured_height; - wl_window->last_sent_geometry_scale = geometry_scale; - wl_window->last_sent_gravity = gravity; - - if (can_move_now) - { - new_x = constrained_rect.x; - new_y = constrained_rect.y; - } - else - { - new_x = temporary_rect.x; - new_y = temporary_rect.y; - - wl_window->has_pending_state_change |= - !!(flags & META_MOVE_RESIZE_STATE_CHANGED); - } - - 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; - } - - if (window->placement.rule && - window->placement.state == META_PLACEMENT_STATE_CONSTRAINED_FINISHED) - { - window->placement.current.rel_x = rel_x; - window->placement.current.rel_y = rel_y; - } - - new_buffer_x = new_x - window->custom_frame_extents.left; - 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; - } - - if (can_move_now && - flags & META_MOVE_RESIZE_WAYLAND_STATE_CHANGED) - *result |= META_MOVE_RESIZE_RESULT_STATE_CHANGED; -} - -static void -scale_size (int *width, - int *height, - float scale) -{ - if (*width < G_MAXINT) - { - float new_width = (*width * scale); - *width = (int) MIN (new_width, G_MAXINT); - } - - if (*height < G_MAXINT) - { - float new_height = (*height * scale); - *height = (int) MIN (new_height, G_MAXINT); - } -} - -static void -scale_rect_size (MetaRectangle *rect, - float scale) -{ - scale_size (&rect->width, &rect->height, scale); -} - -static void -meta_window_wayland_update_main_monitor (MetaWindow *window, - MetaWindowUpdateMonitorFlags flags) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - MetaWindow *toplevel_window; - MetaLogicalMonitor *from; - MetaLogicalMonitor *to; - MetaLogicalMonitor *scaled_new; - float from_scale, to_scale; - float scale; - MetaRectangle rect; - - from = window->monitor; - - /* If the window is not a toplevel window (i.e. it's a popup window) just use - * the monitor of the toplevel. */ - toplevel_window = meta_wayland_surface_get_toplevel_window (window->surface); - if (toplevel_window != window) - { - meta_window_update_monitor (toplevel_window, flags); - window->monitor = toplevel_window->monitor; - return; - } - - /* Require both the current and the new monitor would be the new main monitor, - * even given the resulting scale the window would end up having. This is - * needed to avoid jumping back and forth between the new and the old, since - * changing main monitor may cause the window to be resized so that it no - * longer have that same new main monitor. */ - to = meta_window_calculate_main_logical_monitor (window); - - if (from == to) - return; - - if (from == NULL || to == NULL) - { - window->monitor = to; - return; - } - - if (flags & META_WINDOW_UPDATE_MONITOR_FLAGS_FORCE) - { - window->monitor = to; - return; - } - - from_scale = meta_logical_monitor_get_scale (from); - to_scale = meta_logical_monitor_get_scale (to); - - if (from_scale == to_scale) - { - window->monitor = to; - return; - } - - if (meta_is_stage_views_scaled ()) - { - window->monitor = to; - return; - } - - /* To avoid a window alternating between two main monitors because scaling - * changes the main monitor, wait until both the current and the new scale - * will result in the same main monitor. */ - scale = to_scale / from_scale; - rect = window->rect; - scale_rect_size (&rect, scale); - scaled_new = - meta_monitor_manager_get_logical_monitor_from_rect (monitor_manager, &rect); - if (to != scaled_new) - return; - - window->monitor = to; -} - -static void -meta_window_wayland_main_monitor_changed (MetaWindow *window, - const MetaLogicalMonitor *old) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - int old_geometry_scale = wl_window->geometry_scale; - int geometry_scale; - float scale_factor; - MetaWaylandSurface *surface; - - if (!window->monitor) - return; - - geometry_scale = meta_window_wayland_get_geometry_scale (window); - - /* This function makes sure that window geometry, window actor geometry and - * surface actor geometry gets set according the old and current main monitor - * scale. If there either is no past or current main monitor, or if the scale - * didn't change, there is nothing to do. */ - if (old == NULL || - window->monitor == NULL || - old_geometry_scale == geometry_scale) - return; - - /* MetaWindow keeps its rectangles in the physical pixel coordinate space. - * When the main monitor of a window changes, it can cause the corresponding - * window surfaces to be scaled given the monitor scale, so we need to scale - * the rectangles in MetaWindow accordingly. */ - - scale_factor = (float) geometry_scale / old_geometry_scale; - - /* Window size. */ - scale_rect_size (&window->rect, scale_factor); - scale_rect_size (&window->unconstrained_rect, scale_factor); - scale_rect_size (&window->saved_rect, scale_factor); - scale_size (&window->size_hints.min_width, &window->size_hints.min_height, scale_factor); - scale_size (&window->size_hints.max_width, &window->size_hints.max_height, scale_factor); - - /* Window geometry offset (XXX: Need a better place, see - * meta_window_wayland_finish_move_resize). */ - window->custom_frame_extents.left = - (int)(scale_factor * window->custom_frame_extents.left); - window->custom_frame_extents.top = - (int)(scale_factor * window->custom_frame_extents.top); - - /* Buffer rect. */ - scale_rect_size (&window->buffer_rect, scale_factor); - window->buffer_rect.x = - window->rect.x - window->custom_frame_extents.left; - window->buffer_rect.y = - window->rect.y - window->custom_frame_extents.top; - - meta_compositor_sync_window_geometry (window->display->compositor, - window, - TRUE); - - surface = window->surface; - if (surface) - { - MetaWaylandActorSurface *actor_surface = - META_WAYLAND_ACTOR_SURFACE (surface->role); - - meta_wayland_actor_surface_sync_actor_state (actor_surface); - } - - set_geometry_scale_for_window (wl_window, geometry_scale); - meta_window_emit_size_changed (window); -} - -static pid_t -meta_window_wayland_get_client_pid (MetaWindow *window) -{ - MetaWaylandSurface *surface = window->surface; - struct wl_resource *resource = surface->resource; - pid_t pid; - - wl_client_get_credentials (wl_resource_get_client (resource), &pid, NULL, NULL); - return pid; -} - -static void -appears_focused_changed (GObject *object, - GParamSpec *pspec, - gpointer user_data) -{ - MetaWindow *window = META_WINDOW (object); - - if (window->placement.rule) - return; - - surface_state_changed (window); -} - -static void -on_window_shown (MetaWindow *window) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - gboolean has_been_shown; - - has_been_shown = wl_window->has_been_shown; - wl_window->has_been_shown = TRUE; - - if (!has_been_shown) - meta_compositor_sync_updates_frozen (window->display->compositor, window); -} - -static void -meta_window_wayland_init (MetaWindowWayland *wl_window) -{ - MetaWindow *window = META_WINDOW (wl_window); - - wl_window->geometry_scale = 1; - - g_signal_connect (window, "notify::appears-focused", - G_CALLBACK (appears_focused_changed), NULL); - g_signal_connect (window, "shown", - G_CALLBACK (on_window_shown), NULL); -} - -static void -meta_window_wayland_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_wayland_shortcuts_inhibited (MetaWindow *window, - ClutterInputDevice *source) -{ - MetaWaylandCompositor *compositor = meta_wayland_compositor_get_default (); - - return meta_wayland_compositor_is_shortcuts_inhibited (compositor, source); -} - -static gboolean -meta_window_wayland_is_focusable (MetaWindow *window) -{ - return window->input; -} - -static gboolean -meta_window_wayland_can_ping (MetaWindow *window) -{ - return TRUE; -} - -static gboolean -meta_window_wayland_is_stackable (MetaWindow *window) -{ - return meta_wayland_surface_get_buffer (window->surface) != NULL; -} - -static gboolean -meta_window_wayland_are_updates_frozen (MetaWindow *window) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - - return !wl_window->has_been_shown; -} - -static gboolean -meta_window_wayland_is_focus_async (MetaWindow *window) -{ - return FALSE; -} - -static MetaStackLayer -meta_window_wayland_calculate_layer (MetaWindow *window) -{ - return meta_window_get_default_layer (window); -} - -static void -meta_window_wayland_map (MetaWindow *window) -{ -} - -static void -meta_window_wayland_unmap (MetaWindow *window) -{ -} - -static void -meta_window_wayland_finalize (GObject *object) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (object); - - g_list_free_full (wl_window->pending_configurations, - (GDestroyNotify) meta_wayland_window_configuration_free); - - G_OBJECT_CLASS (meta_window_wayland_parent_class)->finalize (object); -} - -static void -meta_window_wayland_class_init (MetaWindowWaylandClass *klass) -{ - GObjectClass *object_class = G_OBJECT_CLASS (klass); - MetaWindowClass *window_class = META_WINDOW_CLASS (klass); - - object_class->finalize = meta_window_wayland_finalize; - - 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->update_main_monitor = meta_window_wayland_update_main_monitor; - window_class->main_monitor_changed = meta_window_wayland_main_monitor_changed; - window_class->get_client_pid = meta_window_wayland_get_client_pid; - window_class->force_restore_shortcuts = meta_window_wayland_force_restore_shortcuts; - window_class->shortcuts_inhibited = meta_window_wayland_shortcuts_inhibited; - window_class->is_focusable = meta_window_wayland_is_focusable; - window_class->is_stackable = meta_window_wayland_is_stackable; - window_class->can_ping = meta_window_wayland_can_ping; - window_class->are_updates_frozen = meta_window_wayland_are_updates_frozen; - window_class->calculate_layer = meta_window_wayland_calculate_layer; - window_class->map = meta_window_wayland_map; - window_class->unmap = meta_window_wayland_unmap; - window_class->is_focus_async = meta_window_wayland_is_focus_async; -} - -MetaWindow * -meta_window_wayland_new (MetaDisplay *display, - MetaWaylandSurface *surface) -{ - XWindowAttributes attrs = { 0 }; - MetaWindowWayland *wl_window; - MetaWindow *window; - - /* - * Set attributes used by _meta_window_shared_new, don't bother trying to fake - * X11 window attributes with the rest, since they'll be ignored anyway. - */ - attrs.x = 0; - attrs.y = 0; - attrs.width = 0; - attrs.height = 0; - attrs.depth = 24; - attrs.visual = NULL; - attrs.map_state = IsUnmapped; - attrs.override_redirect = False; - - window = _meta_window_shared_new (display, - META_WINDOW_CLIENT_TYPE_WAYLAND, - surface, - None, - WithdrawnState, - META_COMP_EFFECT_CREATE, - &attrs); - - wl_window = META_WINDOW_WAYLAND (window); - set_geometry_scale_for_window (wl_window, wl_window->geometry_scale); - - return window; -} - -MetaWaylandWindowConfiguration * -meta_window_wayland_peek_configuration (MetaWindowWayland *wl_window, - uint32_t serial) -{ - GList *l; - - for (l = wl_window->pending_configurations; l; l = l->next) - { - MetaWaylandWindowConfiguration *configuration = l->data; - - if (configuration->serial == serial) - return configuration; - } - - return NULL; -} - -static MetaWaylandWindowConfiguration * -acquire_acked_configuration (MetaWindowWayland *wl_window, - MetaWaylandSurfaceState *pending) -{ - GList *l; - - if (!pending->has_acked_configure_serial) - return NULL; - - for (l = wl_window->pending_configurations; l; l = l->next) - { - MetaWaylandWindowConfiguration *configuration = l->data; - GList *tail; - gboolean is_matching_configuration; - - if (configuration->serial > pending->acked_configure_serial) - continue; - - tail = l; - - if (tail->prev) - { - tail->prev->next = NULL; - tail->prev = NULL; - } - else - { - wl_window->pending_configurations = NULL; - } - - is_matching_configuration = - configuration->serial == pending->acked_configure_serial; - - if (is_matching_configuration) - tail = g_list_delete_link (tail, l); - g_list_free_full (tail, - (GDestroyNotify) meta_wayland_window_configuration_free); - - if (is_matching_configuration) - return configuration; - else - return NULL; - } - - return NULL; -} - -int -meta_window_wayland_get_geometry_scale (MetaWindow *window) -{ - if (!window->monitor) - return 1; - - return get_window_geometry_scale_for_logical_monitor (window->monitor); -} - -static void -calculate_offset (MetaWaylandWindowConfiguration *configuration, - MetaRectangle *geometry, - MetaRectangle *rect) -{ - int offset_x; - int offset_y; - - rect->x = configuration->x; - rect->y = configuration->y; - - offset_x = configuration->width - geometry->width; - offset_y = configuration->height - geometry->height; - switch (configuration->gravity) - { - case META_GRAVITY_SOUTH: - case META_GRAVITY_SOUTH_WEST: - rect->y += offset_y; - break; - case META_GRAVITY_EAST: - case META_GRAVITY_NORTH_EAST: - rect->x += offset_x; - break; - case META_GRAVITY_SOUTH_EAST: - rect->x += offset_x; - rect->y += offset_y; - break; - default: - break; - } -} - -/** - * meta_window_move_resize_wayland: - * - * Complete a resize operation from a wayland client. - */ -void -meta_window_wayland_finish_move_resize (MetaWindow *window, - MetaRectangle new_geom, - MetaWaylandSurfaceState *pending) -{ - MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window); - MetaDisplay *display = window->display; - int dx, dy; - int geometry_scale; - MetaGravity gravity; - MetaRectangle rect; - MetaMoveResizeFlags flags; - MetaWaylandWindowConfiguration *acked_configuration; - gboolean is_window_being_resized; - - /* new_geom is in the logical pixel coordinate space, but MetaWindow wants its - * rects to represent what in turn will end up on the stage, i.e. we need to - * scale new_geom to physical pixels given what buffer scale and texture scale - * is in use. */ - - geometry_scale = meta_window_wayland_get_geometry_scale (window); - new_geom.x *= geometry_scale; - new_geom.y *= geometry_scale; - new_geom.width *= geometry_scale; - new_geom.height *= geometry_scale; - - /* The (dx, dy) offset is also in logical pixel coordinate space and needs - * to be scaled in the same way as new_geom. */ - dx = pending->dx * geometry_scale; - dy = pending->dy * geometry_scale; - - /* 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_FINISH_MOVE_RESIZE; - - acked_configuration = acquire_acked_configuration (wl_window, pending); - - /* x/y are ignored when we're doing interactive resizing */ - is_window_being_resized = (meta_grab_op_is_resizing (display->grab_op) && - display->grab_window == window); - - rect = (MetaRectangle) { - .x = window->rect.x, - .y = window->rect.y, - .width = new_geom.width, - .height = new_geom.height - }; - - if (!is_window_being_resized) - { - if (acked_configuration) - { - if (window->placement.rule) - { - MetaWindow *parent; - - parent = meta_window_get_transient_for (window); - rect.x = parent->rect.x + acked_configuration->rel_x; - rect.y = parent->rect.y + acked_configuration->rel_y; - } - else if (acked_configuration->has_position) - { - calculate_offset (acked_configuration, &new_geom, &rect); - } - } - } - else - { - if (acked_configuration && acked_configuration->has_position) - calculate_offset (acked_configuration, &new_geom, &rect); - } - - rect.x += dx; - rect.y += dy; - - if (rect.x != window->rect.x || rect.y != window->rect.y) - flags |= META_MOVE_RESIZE_MOVE_ACTION; - - if (wl_window->has_pending_state_change && acked_configuration) - { - flags |= META_MOVE_RESIZE_WAYLAND_STATE_CHANGED; - wl_window->has_pending_state_change = FALSE; - } - - if (rect.width != window->rect.width || rect.height != window->rect.height) - flags |= META_MOVE_RESIZE_RESIZE_ACTION; - - if (window->display->grab_window == window) - gravity = meta_resize_gravity_from_grab_op (window->display->grab_op); - else - gravity = META_GRAVITY_STATIC; - meta_window_move_resize_internal (window, flags, gravity, rect); - - g_clear_pointer (&acked_configuration, meta_wayland_window_configuration_free); -} - -void -meta_window_wayland_place_relative_to (MetaWindow *window, - MetaWindow *other, - int x, - int y) -{ - int geometry_scale; - - /* If there is no monitor, we can't position the window reliably. */ - if (!other->monitor) - return; - - geometry_scale = meta_window_wayland_get_geometry_scale (other); - meta_window_move_frame (window, FALSE, - other->buffer_rect.x + (x * geometry_scale), - other->buffer_rect.y + (y * geometry_scale)); - window->placed = TRUE; -} - -void -meta_window_place_with_placement_rule (MetaWindow *window, - MetaPlacementRule *placement_rule) -{ - gboolean first_placement; - - first_placement = !window->placement.rule; - - g_clear_pointer (&window->placement.rule, g_free); - window->placement.rule = g_new0 (MetaPlacementRule, 1); - *window->placement.rule = *placement_rule; - - window->unconstrained_rect.x = window->rect.x; - window->unconstrained_rect.y = window->rect.y; - window->unconstrained_rect.width = placement_rule->width; - window->unconstrained_rect.height = placement_rule->height; - - window->calc_placement = first_placement; - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_PLACEMENT_CHANGED), - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); - window->calc_placement = FALSE; -} - -void -meta_window_update_placement_rule (MetaWindow *window, - MetaPlacementRule *placement_rule) -{ - window->placement.state = META_PLACEMENT_STATE_INVALIDATED; - meta_window_place_with_placement_rule (window, placement_rule); -} - -void -meta_window_wayland_set_min_size (MetaWindow *window, - int width, - int height) -{ - gint64 new_width, new_height; - float scale; - - meta_topic (META_DEBUG_GEOMETRY, "Window %s sets min size %d x %d", - window->desc, width, height); - - if (width == 0 && height == 0) - { - window->size_hints.min_width = 0; - window->size_hints.min_height = 0; - window->size_hints.flags &= ~PMinSize; - - return; - } - - scale = (float) meta_window_wayland_get_geometry_scale (window); - scale_size (&width, &height, scale); - - new_width = width + (window->custom_frame_extents.left + - window->custom_frame_extents.right); - new_height = height + (window->custom_frame_extents.top + - window->custom_frame_extents.bottom); - - window->size_hints.min_width = (int) MIN (new_width, G_MAXINT); - window->size_hints.min_height = (int) MIN (new_height, G_MAXINT); - window->size_hints.flags |= PMinSize; -} - -void -meta_window_wayland_set_max_size (MetaWindow *window, - int width, - int height) - -{ - gint64 new_width, new_height; - float scale; - - meta_topic (META_DEBUG_GEOMETRY, "Window %s sets max size %d x %d", - window->desc, width, height); - - if (width == 0 && height == 0) - { - window->size_hints.max_width = G_MAXINT; - window->size_hints.max_height = G_MAXINT; - window->size_hints.flags &= ~PMaxSize; - - return; - } - - scale = (float) meta_window_wayland_get_geometry_scale (window); - scale_size (&width, &height, scale); - - new_width = width + (window->custom_frame_extents.left + - window->custom_frame_extents.right); - new_height = height + (window->custom_frame_extents.top + - window->custom_frame_extents.bottom); - - window->size_hints.max_width = (int) ((new_width > 0 && new_width < G_MAXINT) ? - new_width : G_MAXINT); - window->size_hints.max_height = (int) ((new_height > 0 && new_height < G_MAXINT) ? - new_height : G_MAXINT); - window->size_hints.flags |= PMaxSize; -} - -void -meta_window_wayland_get_min_size (MetaWindow *window, - int *width, - int *height) -{ - gint64 current_width, current_height; - float scale; - - if (!(window->size_hints.flags & PMinSize)) - { - /* Zero means unlimited */ - *width = 0; - *height = 0; - - return; - } - - current_width = window->size_hints.min_width - - (window->custom_frame_extents.left + - window->custom_frame_extents.right); - current_height = window->size_hints.min_height - - (window->custom_frame_extents.top + - window->custom_frame_extents.bottom); - - *width = MAX (current_width, 0); - *height = MAX (current_height, 0); - - scale = 1.0 / (float) meta_window_wayland_get_geometry_scale (window); - scale_size (width, height, scale); -} - -void -meta_window_wayland_get_max_size (MetaWindow *window, - int *width, - int *height) -{ - gint64 current_width = 0; - gint64 current_height = 0; - float scale; - - if (!(window->size_hints.flags & PMaxSize)) - { - /* Zero means unlimited */ - *width = 0; - *height = 0; - - return; - } - - if (window->size_hints.max_width < G_MAXINT) - current_width = window->size_hints.max_width - - (window->custom_frame_extents.left + - window->custom_frame_extents.right); - - if (window->size_hints.max_height < G_MAXINT) - current_height = window->size_hints.max_height - - (window->custom_frame_extents.top + - window->custom_frame_extents.bottom); - - *width = CLAMP (current_width, 0, G_MAXINT); - *height = CLAMP (current_height, 0, G_MAXINT); - - scale = 1.0 / (float) meta_window_wayland_get_geometry_scale (window); - scale_size (width, height, scale); -} |