diff options
Diffstat (limited to 'src/core/window.c')
-rw-r--r-- | src/core/window.c | 8665 |
1 files changed, 0 insertions, 8665 deletions
diff --git a/src/core/window.c b/src/core/window.c deleted file mode 100644 index ea56f3328..000000000 --- a/src/core/window.c +++ /dev/null @@ -1,8665 +0,0 @@ -/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ - -/* - * Copyright (C) 2001 Havoc Pennington, Anders Carlsson - * Copyright (C) 2002, 2003 Red Hat, Inc. - * Copyright (C) 2003 Rob Adams - * Copyright (C) 2004-2006 Elijah Newren - * - * 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 <http://www.gnu.org/licenses/>. - */ - -/** - * SECTION:meta-window - * @title: MetaWindow - * @short_description: A display-agnostic abstraction for a window. - * - * #MetaWindow is the core abstraction in Mutter of a window. It has the - * properties you'd expect, such as a title, an icon, whether it's fullscreen, - * has decorations, etc. - * - * Since a lot of different kinds of windows exist, each window also a - * #MetaWindowType which denotes which kind of window we're exactly dealing - * with. For example, one expects slightly different behaviour from a dialog - * than a "normal" window. The type of a window can be queried with - * meta_window_get_type(). - * - * Common API for windows include: - * - Minimizing: meta_window_minimize() / meta_window_unminimize() - * - Maximizing: meta_window_maximize() / meta_window_unmaximize() - * - Fullscreen: meta_window_make_fullscreen() / meta_window_unmake_fullscreen() - * / meta_window_is_fullscreen() - * - * Each #MetaWindow is part of either one or all #MetaWorkspace<!-- -->s of the - * desktop. You can activate a window on a certain workspace using - * meta_window_activate_with_workspace(), and query on which workspace it is - * located using meta_window_located_on_workspace(). The workspace it is part - * of can be obtained using meta_window_get_workspace(). - * - * Each display protocol should make a subclass to be compatible with that - * protocols' specifics, for example #MetaWindowX11 and #MetaWindowWayland. - * This is independent of the protocol that the client uses, which is modeled - * using the #MetaWindowClientType enum. - * - * To integrate within the Clutter scene graph, which deals with the actual - * rendering, each #MetaWindow will be part of a #MetaWindowActor. - */ - -#include "config.h" - -#include "core/window-private.h" - -#include <math.h> -#include <stdlib.h> -#include <string.h> -#include <X11/Xatom.h> - -#include "backends/meta-backend-private.h" -#include "backends/meta-logical-monitor.h" -#include "cogl/cogl.h" -#include "core/boxes-private.h" -#include "core/constraints.h" -#include "core/edge-resistance.h" -#include "core/frame.h" -#include "core/keybindings-private.h" -#include "core/meta-workspace-manager-private.h" -#include "core/place.h" -#include "core/stack.h" -#include "core/util-private.h" -#include "core/workspace-private.h" -#include "meta/compositor-mutter.h" -#include "meta/group.h" -#include "meta/meta-cursor-tracker.h" -#include "meta/meta-enum-types.h" -#include "meta/meta-x11-errors.h" -#include "meta/prefs.h" -#include "ui/ui.h" -#include "x11/meta-x11-display-private.h" -#include "x11/window-props.h" -#include "x11/window-x11.h" -#include "x11/xprops.h" - -#ifdef HAVE_WAYLAND -#include "wayland/meta-wayland-private.h" -#include "wayland/meta-wayland-surface.h" -#include "wayland/meta-window-wayland.h" -#include "wayland/meta-window-xwayland.h" -#endif - -/* Windows that unmaximize to a size bigger than that fraction of the workarea - * will be scaled down to that size (while maintaining aspect ratio). - * Windows that cover an area greater then this size are automaximized on map. - */ -#define MAX_UNMAXIMIZED_WINDOW_AREA .8 - -#define SNAP_SECURITY_LABEL_PREFIX "snap." - -static int destroying_windows_disallowed = 0; - -/* Each window has a "stamp" which is a non-recycled 64-bit ID. They - * start after the end of the XID space so that, for stacking - * we can keep a guint64 that represents one or the other - */ -static guint64 next_window_stamp = G_GUINT64_CONSTANT(0x100000000); - -static void invalidate_work_areas (MetaWindow *window); -static void set_wm_state (MetaWindow *window); -static void set_net_wm_state (MetaWindow *window); -static void meta_window_set_above (MetaWindow *window, - gboolean new_value); - -static void meta_window_show (MetaWindow *window); -static void meta_window_hide (MetaWindow *window); - -static void meta_window_save_rect (MetaWindow *window); - -static void ensure_mru_position_after (MetaWindow *window, - MetaWindow *after_this_one); - -static void meta_window_move_resize_now (MetaWindow *window); - -static void meta_window_unqueue (MetaWindow *window, guint queuebits); - -static void update_move (MetaWindow *window, - MetaEdgeResistanceFlags flags, - int x, - int y); -static gboolean update_move_timeout (gpointer data); -static void update_resize (MetaWindow *window, - MetaEdgeResistanceFlags flags, - int x, - int y, - gboolean force); -static gboolean update_resize_timeout (gpointer data); -static gboolean should_be_on_all_workspaces (MetaWindow *window); - -static void meta_window_flush_calc_showing (MetaWindow *window); - -static gboolean queue_calc_showing_func (MetaWindow *window, - void *data); - -static void meta_window_move_between_rects (MetaWindow *window, - MetaMoveResizeFlags move_resize_flags, - const MetaRectangle *old_area, - const MetaRectangle *new_area); - -static void unmaximize_window_before_freeing (MetaWindow *window); -static void unminimize_window_and_all_transient_parents (MetaWindow *window); - -static void meta_window_propagate_focus_appearance (MetaWindow *window, - gboolean focused); -static void meta_window_update_icon_now (MetaWindow *window, - gboolean force); - -static void set_workspace_state (MetaWindow *window, - gboolean on_all_workspaces, - MetaWorkspace *workspace); - -static MetaWindow * meta_window_find_tile_match (MetaWindow *window, - MetaTileMode mode); -static void update_edge_constraints (MetaWindow *window); - -/* Idle handlers for the three queues (run with meta_later_add()). The - * "data" parameter in each case will be a GINT_TO_POINTER of the - * index into the queue arrays to use. - * - * TODO: Possibly there is still some code duplication among these, which we - * need to sort out at some point. - */ -static gboolean idle_calc_showing (gpointer data); -static gboolean idle_move_resize (gpointer data); -static gboolean idle_update_icon (gpointer data); - -G_DEFINE_ABSTRACT_TYPE (MetaWindow, meta_window, G_TYPE_OBJECT); - -enum -{ - PROP_0, - - PROP_TITLE, - PROP_ICON, - PROP_MINI_ICON, - PROP_DECORATED, - PROP_FULLSCREEN, - PROP_MAXIMIZED_HORIZONTALLY, - PROP_MAXIMIZED_VERTICALLY, - PROP_MINIMIZED, - PROP_WINDOW_TYPE, - PROP_USER_TIME, - PROP_DEMANDS_ATTENTION, - PROP_URGENT, - PROP_SKIP_TASKBAR, - PROP_MUTTER_HINTS, - PROP_APPEARS_FOCUSED, - PROP_RESIZEABLE, - PROP_ABOVE, - PROP_WM_CLASS, - PROP_GTK_APPLICATION_ID, - PROP_GTK_UNIQUE_BUS_NAME, - PROP_GTK_APPLICATION_OBJECT_PATH, - PROP_GTK_WINDOW_OBJECT_PATH, - PROP_GTK_APP_MENU_OBJECT_PATH, - PROP_GTK_MENUBAR_OBJECT_PATH, - PROP_ON_ALL_WORKSPACES, - - PROP_LAST, -}; - -static GParamSpec *obj_props[PROP_LAST]; - -enum -{ - WORKSPACE_CHANGED, - FOCUS, - RAISED, - UNMANAGING, - UNMANAGED, - SIZE_CHANGED, - POSITION_CHANGED, - SHOWN, - - LAST_SIGNAL -}; - -static guint window_signals[LAST_SIGNAL] = { 0 }; - -static void -prefs_changed_callback (MetaPreference pref, - gpointer data) -{ - MetaWindow *window = data; - - if (pref == META_PREF_WORKSPACES_ONLY_ON_PRIMARY) - { - meta_window_on_all_workspaces_changed (window); - } - else if (pref == META_PREF_ATTACH_MODAL_DIALOGS && - window->type == META_WINDOW_MODAL_DIALOG) - { - window->attached = meta_window_should_attach_to_parent (window); - meta_window_recalc_features (window); - meta_window_queue (window, META_QUEUE_MOVE_RESIZE); - } -} - -static void -meta_window_real_grab_op_began (MetaWindow *window, - MetaGrabOp op) -{ -} - -static void -meta_window_real_grab_op_ended (MetaWindow *window, - MetaGrabOp op) -{ - window->shaken_loose = FALSE; -} - -static void -meta_window_real_current_workspace_changed (MetaWindow *window) -{ -} - -static gboolean -meta_window_real_update_struts (MetaWindow *window) -{ - return FALSE; -} - -static void -meta_window_real_get_default_skip_hints (MetaWindow *window, - gboolean *skip_taskbar_out, - gboolean *skip_pager_out) -{ - *skip_taskbar_out = FALSE; - *skip_pager_out = FALSE; -} - -static gboolean -meta_window_real_update_icon (MetaWindow *window, - cairo_surface_t **icon, - cairo_surface_t **mini_icon) -{ - *icon = NULL; - *mini_icon = NULL; - return FALSE; -} - -static pid_t -meta_window_real_get_client_pid (MetaWindow *window) -{ - return 0; -} - -static void -meta_window_finalize (GObject *object) -{ - MetaWindow *window = META_WINDOW (object); - - if (window->icon) - cairo_surface_destroy (window->icon); - - if (window->mini_icon) - cairo_surface_destroy (window->mini_icon); - - if (window->frame_bounds) - cairo_region_destroy (window->frame_bounds); - - if (window->shape_region) - cairo_region_destroy (window->shape_region); - - if (window->opaque_region) - cairo_region_destroy (window->opaque_region); - - if (window->input_region) - cairo_region_destroy (window->input_region); - - if (window->transient_for) - g_object_unref (window->transient_for); - - g_free (window->sm_client_id); - g_free (window->wm_client_machine); - g_free (window->startup_id); - g_free (window->role); - g_free (window->res_class); - g_free (window->res_name); - g_free (window->title); - g_free (window->desc); - g_free (window->sandboxed_app_id); - g_free (window->gtk_theme_variant); - g_free (window->gtk_application_id); - g_free (window->gtk_unique_bus_name); - g_free (window->gtk_application_object_path); - g_free (window->gtk_window_object_path); - g_free (window->gtk_app_menu_object_path); - g_free (window->gtk_menubar_object_path); - g_free (window->placement.rule); - - G_OBJECT_CLASS (meta_window_parent_class)->finalize (object); -} - -static void -meta_window_get_property(GObject *object, - guint prop_id, - GValue *value, - GParamSpec *pspec) -{ - MetaWindow *win = META_WINDOW (object); - - switch (prop_id) - { - case PROP_TITLE: - g_value_set_string (value, win->title); - break; - case PROP_ICON: - g_value_set_pointer (value, win->icon); - break; - case PROP_MINI_ICON: - g_value_set_pointer (value, win->mini_icon); - break; - case PROP_DECORATED: - g_value_set_boolean (value, win->decorated); - break; - case PROP_FULLSCREEN: - g_value_set_boolean (value, win->fullscreen); - break; - case PROP_MAXIMIZED_HORIZONTALLY: - g_value_set_boolean (value, win->maximized_horizontally); - break; - case PROP_MAXIMIZED_VERTICALLY: - g_value_set_boolean (value, win->maximized_vertically); - break; - case PROP_MINIMIZED: - g_value_set_boolean (value, win->minimized); - break; - case PROP_WINDOW_TYPE: - g_value_set_enum (value, win->type); - break; - case PROP_USER_TIME: - g_value_set_uint (value, win->net_wm_user_time); - break; - case PROP_DEMANDS_ATTENTION: - g_value_set_boolean (value, win->wm_state_demands_attention); - break; - case PROP_URGENT: - g_value_set_boolean (value, win->urgent); - break; - case PROP_SKIP_TASKBAR: - g_value_set_boolean (value, win->skip_taskbar); - break; - case PROP_MUTTER_HINTS: - g_value_set_string (value, win->mutter_hints); - break; - case PROP_APPEARS_FOCUSED: - g_value_set_boolean (value, meta_window_appears_focused (win)); - break; - case PROP_WM_CLASS: - g_value_set_string (value, win->res_class); - break; - case PROP_RESIZEABLE: - g_value_set_boolean (value, win->has_resize_func); - break; - case PROP_ABOVE: - g_value_set_boolean (value, win->wm_state_above); - break; - case PROP_GTK_APPLICATION_ID: - g_value_set_string (value, win->gtk_application_id); - break; - case PROP_GTK_UNIQUE_BUS_NAME: - g_value_set_string (value, win->gtk_unique_bus_name); - break; - case PROP_GTK_APPLICATION_OBJECT_PATH: - g_value_set_string (value, win->gtk_application_object_path); - break; - case PROP_GTK_WINDOW_OBJECT_PATH: - g_value_set_string (value, win->gtk_window_object_path); - break; - case PROP_GTK_APP_MENU_OBJECT_PATH: - g_value_set_string (value, win->gtk_app_menu_object_path); - break; - case PROP_GTK_MENUBAR_OBJECT_PATH: - g_value_set_string (value, win->gtk_menubar_object_path); - break; - case PROP_ON_ALL_WORKSPACES: - g_value_set_boolean (value, win->on_all_workspaces); - break; - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); - break; - } -} - -static void -meta_window_set_property(GObject *object, - guint prop_id, - const GValue *value, - GParamSpec *pspec) -{ - switch (prop_id) - { - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); - break; - } -} - -static void -meta_window_class_init (MetaWindowClass *klass) -{ - GObjectClass *object_class = G_OBJECT_CLASS (klass); - - object_class->finalize = meta_window_finalize; - - object_class->get_property = meta_window_get_property; - object_class->set_property = meta_window_set_property; - - klass->grab_op_began = meta_window_real_grab_op_began; - klass->grab_op_ended = meta_window_real_grab_op_ended; - klass->current_workspace_changed = meta_window_real_current_workspace_changed; - klass->update_struts = meta_window_real_update_struts; - klass->get_default_skip_hints = meta_window_real_get_default_skip_hints; - klass->update_icon = meta_window_real_update_icon; - klass->get_client_pid = meta_window_real_get_client_pid; - - obj_props[PROP_TITLE] = - g_param_spec_string ("title", - "Title", - "The title of the window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_ICON] = - g_param_spec_pointer ("icon", - "Icon", - "Normal icon, usually 96x96 pixels", - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_MINI_ICON] = - g_param_spec_pointer ("mini-icon", - "Mini Icon", - "Mini icon, usually 16x16 pixels", - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_DECORATED] = - g_param_spec_boolean ("decorated", - "Decorated", - "Whether window is decorated", - TRUE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_FULLSCREEN] = - g_param_spec_boolean ("fullscreen", - "Fullscreen", - "Whether window is fullscreened", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_MAXIMIZED_HORIZONTALLY] = - g_param_spec_boolean ("maximized-horizontally", - "Maximized horizontally", - "Whether window is maximized horizontally", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_MAXIMIZED_VERTICALLY] = - g_param_spec_boolean ("maximized-vertically", - "Maximizing vertically", - "Whether window is maximized vertically", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_MINIMIZED] = - g_param_spec_boolean ("minimized", - "Minimizing", - "Whether window is minimized", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_WINDOW_TYPE] = - g_param_spec_enum ("window-type", - "Window Type", - "The type of the window", - META_TYPE_WINDOW_TYPE, - META_WINDOW_NORMAL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_USER_TIME] = - g_param_spec_uint ("user-time", - "User time", - "Timestamp of last user interaction", - 0, - G_MAXUINT, - 0, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_DEMANDS_ATTENTION] = - g_param_spec_boolean ("demands-attention", - "Demands Attention", - "Whether the window has _NET_WM_STATE_DEMANDS_ATTENTION set", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_URGENT] = - g_param_spec_boolean ("urgent", - "Urgent", - "Whether the urgent flag of WM_HINTS is set", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_SKIP_TASKBAR] = - g_param_spec_boolean ("skip-taskbar", - "Skip taskbar", - "Whether the skip-taskbar flag of WM_HINTS is set", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_MUTTER_HINTS] = - g_param_spec_string ("mutter-hints", - "_MUTTER_HINTS", - "Contents of the _MUTTER_HINTS property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_APPEARS_FOCUSED] = - g_param_spec_boolean ("appears-focused", - "Appears focused", - "Whether the window is drawn as being focused", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_RESIZEABLE] = - g_param_spec_boolean ("resizeable", - "Resizeable", - "Whether the window can be resized", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_ABOVE] = - g_param_spec_boolean ("above", - "Above", - "Whether the window is shown as always-on-top", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_WM_CLASS] = - g_param_spec_string ("wm-class", - "WM_CLASS", - "Contents of the WM_CLASS property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_APPLICATION_ID] = - g_param_spec_string ("gtk-application-id", - "_GTK_APPLICATION_ID", - "Contents of the _GTK_APPLICATION_ID property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_UNIQUE_BUS_NAME] = - g_param_spec_string ("gtk-unique-bus-name", - "_GTK_UNIQUE_BUS_NAME", - "Contents of the _GTK_UNIQUE_BUS_NAME property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_APPLICATION_OBJECT_PATH] = - g_param_spec_string ("gtk-application-object-path", - "_GTK_APPLICATION_OBJECT_PATH", - "Contents of the _GTK_APPLICATION_OBJECT_PATH property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_WINDOW_OBJECT_PATH] = - g_param_spec_string ("gtk-window-object-path", - "_GTK_WINDOW_OBJECT_PATH", - "Contents of the _GTK_WINDOW_OBJECT_PATH property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_APP_MENU_OBJECT_PATH] = - g_param_spec_string ("gtk-app-menu-object-path", - "_GTK_APP_MENU_OBJECT_PATH", - "Contents of the _GTK_APP_MENU_OBJECT_PATH property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_GTK_MENUBAR_OBJECT_PATH] = - g_param_spec_string ("gtk-menubar-object-path", - "_GTK_MENUBAR_OBJECT_PATH", - "Contents of the _GTK_MENUBAR_OBJECT_PATH property of this window", - NULL, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - obj_props[PROP_ON_ALL_WORKSPACES] = - g_param_spec_boolean ("on-all-workspaces", - "On all workspaces", - "Whether the window is set to appear on all workspaces", - FALSE, - G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); - - g_object_class_install_properties (object_class, PROP_LAST, obj_props); - - window_signals[WORKSPACE_CHANGED] = - g_signal_new ("workspace-changed", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - window_signals[FOCUS] = - g_signal_new ("focus", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - window_signals[RAISED] = - g_signal_new ("raised", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - window_signals[UNMANAGING] = - g_signal_new ("unmanaging", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - window_signals[UNMANAGED] = - g_signal_new ("unmanaged", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - /** - * MetaWindow::position-changed: - * @window: a #MetaWindow - * - * This is emitted when the position of a window might - * have changed. Specifically, this is emitted when the - * position of the toplevel window has changed, or when - * the position of the client window has changed. - */ - window_signals[POSITION_CHANGED] = - g_signal_new ("position-changed", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - /** - * MetaWindow::shown: - * @window: a #MetaWindow - * - * This is emitted after a window has been shown. - */ - window_signals[SHOWN] = - g_signal_new ("shown", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); - - /** - * MetaWindow::size-changed: - * @window: a #MetaWindow - * - * This is emitted when the size of a window might - * have changed. Specifically, this is emitted when the - * size of the toplevel window has changed, or when the - * size of the client window has changed. - */ - window_signals[SIZE_CHANGED] = - g_signal_new ("size-changed", - G_TYPE_FROM_CLASS (object_class), - G_SIGNAL_RUN_LAST, - 0, - NULL, NULL, NULL, - G_TYPE_NONE, 0); -} - -static void -meta_window_init (MetaWindow *self) -{ - self->stamp = next_window_stamp++; - meta_prefs_add_listener (prefs_changed_callback, self); -} - -static gboolean -is_desktop_or_dock_foreach (MetaWindow *window, - void *data) -{ - gboolean *result = data; - - *result = - window->type == META_WINDOW_DESKTOP || - window->type == META_WINDOW_DOCK || - window->skip_from_window_list; - if (*result) - return FALSE; /* stop as soon as we find one */ - else - return TRUE; -} - -/* window is the window that's newly mapped provoking - * the possible change - */ -static void -maybe_leave_show_desktop_mode (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean is_desktop_or_dock; - - if (!workspace_manager->active_workspace->showing_desktop) - return; - - /* If the window is a transient for the dock or desktop, don't - * leave show desktop mode when the window opens. That's - * so you can e.g. hide all windows, manipulate a file on - * the desktop via a dialog, then unshow windows again. - */ - is_desktop_or_dock = FALSE; - is_desktop_or_dock_foreach (window, - &is_desktop_or_dock); - - meta_window_foreach_ancestor (window, is_desktop_or_dock_foreach, - &is_desktop_or_dock); - - if (!is_desktop_or_dock) - { - meta_workspace_manager_minimize_all_on_active_workspace_except (workspace_manager, - window); - meta_workspace_manager_unshow_desktop (workspace_manager); - } -} - -gboolean -meta_window_should_attach_to_parent (MetaWindow *window) -{ - MetaWindow *parent; - - if (!meta_prefs_get_attach_modal_dialogs () || - window->type != META_WINDOW_MODAL_DIALOG) - return FALSE; - - parent = meta_window_get_transient_for (window); - if (!parent) - return FALSE; - - switch (parent->type) - { - case META_WINDOW_NORMAL: - case META_WINDOW_DIALOG: - case META_WINDOW_MODAL_DIALOG: - return TRUE; - - default: - return FALSE; - } -} - -static gboolean -client_window_should_be_mapped (MetaWindow *window) -{ -#ifdef HAVE_WAYLAND - if (window->client_type == META_WINDOW_CLIENT_TYPE_WAYLAND && - !meta_wayland_surface_get_buffer (window->surface)) - return FALSE; -#endif - - return !window->shaded; -} - -static void -sync_client_window_mapped (MetaWindow *window) -{ - gboolean should_be_mapped = client_window_should_be_mapped (window); - - g_return_if_fail (!window->override_redirect); - - if (window->mapped == should_be_mapped) - return; - - window->mapped = should_be_mapped; - - if (window->mapped) - META_WINDOW_GET_CLASS (window)->map (window); - else - META_WINDOW_GET_CLASS (window)->unmap (window); -} - -static gboolean -meta_window_update_flatpak_id (MetaWindow *window, - uint32_t pid) -{ - g_autoptr(GKeyFile) key_file = NULL; - g_autofree char *info_filename = NULL; - - g_return_val_if_fail (pid != 0, FALSE); - g_return_val_if_fail (window->sandboxed_app_id == NULL, FALSE); - - key_file = g_key_file_new (); - info_filename = g_strdup_printf ("/proc/%u/root/.flatpak-info", pid); - - if (!g_key_file_load_from_file (key_file, info_filename, G_KEY_FILE_NONE, NULL)) - return FALSE; - - window->sandboxed_app_id = g_key_file_get_string (key_file, "Application", "name", NULL); - - return TRUE; -} - -static gboolean -meta_window_update_snap_id (MetaWindow *window, - uint32_t pid) -{ - g_autofree char *security_label_filename = NULL; - g_autofree char *security_label_contents = NULL; - gsize i, security_label_contents_size = 0; - char *contents_start; - char *contents_end; - char *sandboxed_app_id; - - g_return_val_if_fail (pid != 0, FALSE); - g_return_val_if_fail (window->sandboxed_app_id == NULL, FALSE); - - security_label_filename = g_strdup_printf ("/proc/%u/attr/current", pid); - - if (!g_file_get_contents (security_label_filename, - &security_label_contents, - &security_label_contents_size, - NULL)) - return FALSE; - - if (!g_str_has_prefix (security_label_contents, SNAP_SECURITY_LABEL_PREFIX)) - return FALSE; - - /* We need to translate the security profile into the desktop-id. - * The profile is in the form of 'snap.name-space.binary-name (current)' - * while the desktop id will be name-space_binary-name. - */ - security_label_contents_size -= sizeof (SNAP_SECURITY_LABEL_PREFIX) - 1; - contents_start = security_label_contents + sizeof (SNAP_SECURITY_LABEL_PREFIX) - 1; - contents_end = strchr (contents_start, ' '); - - if (contents_end) - security_label_contents_size = contents_end - contents_start; - - for (i = 0; i < security_label_contents_size; ++i) - { - if (contents_start[i] == '.') - contents_start[i] = '_'; - } - - sandboxed_app_id = g_malloc0 (security_label_contents_size + 1); - memcpy (sandboxed_app_id, contents_start, security_label_contents_size); - - window->sandboxed_app_id = sandboxed_app_id; - - return TRUE; -} - -static void -meta_window_update_sandboxed_app_id (MetaWindow *window) -{ - pid_t pid; - - g_clear_pointer (&window->sandboxed_app_id, g_free); - - pid = meta_window_get_pid (window); - - if (pid < 1) - return; - - if (meta_window_update_flatpak_id (window, pid)) - return; - - if (meta_window_update_snap_id (window, pid)) - return; -} - -static void -meta_window_update_desc (MetaWindow *window) -{ - g_clear_pointer (&window->desc, g_free); - - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - window->desc = g_strdup_printf ("0x%lx", window->xwindow); - else - { - guint64 small_stamp = window->stamp - G_GUINT64_CONSTANT(0x100000000); - - window->desc = g_strdup_printf ("W%" G_GUINT64_FORMAT , small_stamp); - } -} - -static void -meta_window_main_monitor_changed (MetaWindow *window, - const MetaLogicalMonitor *old) -{ - META_WINDOW_GET_CLASS (window)->main_monitor_changed (window, old); - - if (old) - g_signal_emit_by_name (window->display, "window-left-monitor", - old->number, window); - if (window->monitor) - g_signal_emit_by_name (window->display, "window-entered-monitor", - window->monitor->number, window); -} - -MetaLogicalMonitor * -meta_window_calculate_main_logical_monitor (MetaWindow *window) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - MetaRectangle window_rect; - - meta_window_get_frame_rect (window, &window_rect); - return meta_monitor_manager_get_logical_monitor_from_rect (monitor_manager, - &window_rect); -} - -static void -meta_window_manage (MetaWindow *window) -{ - COGL_TRACE_BEGIN_SCOPED (MetaWindowManage, - "Window (manage)"); - - META_WINDOW_GET_CLASS (window)->manage (window); -} - -MetaWindow * -_meta_window_shared_new (MetaDisplay *display, - MetaWindowClientType client_type, - MetaWaylandSurface *surface, - Window xwindow, - gulong existing_wm_state, - MetaCompEffect effect, - XWindowAttributes *attrs) -{ - MetaWorkspaceManager *workspace_manager = display->workspace_manager; - MetaWindow *window; - - COGL_TRACE_BEGIN_SCOPED (MetaWindowSharedNew, - "Window (new)"); - - g_assert (attrs != NULL); - - meta_verbose ("attrs->map_state = %d (%s)", - attrs->map_state, - (attrs->map_state == IsUnmapped) ? - "IsUnmapped" : - (attrs->map_state == IsViewable) ? - "IsViewable" : - (attrs->map_state == IsUnviewable) ? - "IsUnviewable" : - "(unknown)"); - - if (client_type == META_WINDOW_CLIENT_TYPE_X11 && !meta_is_wayland_compositor ()) - window = g_object_new (META_TYPE_WINDOW_X11, NULL); -#ifdef HAVE_WAYLAND - else if (client_type == META_WINDOW_CLIENT_TYPE_X11) - window = g_object_new (META_TYPE_WINDOW_XWAYLAND, NULL); - else if (client_type == META_WINDOW_CLIENT_TYPE_WAYLAND) - window = g_object_new (META_TYPE_WINDOW_WAYLAND, NULL); -#endif - else - g_assert_not_reached (); - - window->constructing = TRUE; - - window->client_type = client_type; - window->surface = surface; - window->xwindow = xwindow; - - window->display = display; - meta_display_register_stamp (window->display, &window->stamp, window); - - window->workspace = NULL; - - window->sync_request_counter = None; - window->sync_request_serial = 0; - window->sync_request_timeout_id = 0; - window->sync_request_alarm = None; - - meta_window_update_sandboxed_app_id (window); - meta_window_update_desc (window); - - window->override_redirect = attrs->override_redirect; - - /* avoid tons of stack updates */ - meta_stack_freeze (window->display->stack); - - window->rect.x = attrs->x; - window->rect.y = attrs->y; - window->rect.width = attrs->width; - window->rect.height = attrs->height; - - /* size_hints are the "request" */ - window->size_hints.x = attrs->x; - window->size_hints.y = attrs->y; - window->size_hints.width = attrs->width; - window->size_hints.height = attrs->height; - /* initialize the remaining size_hints as if size_hints.flags were zero */ - meta_set_normal_hints (window, NULL); - - /* And this is our unmaximized size */ - window->saved_rect = window->rect; - window->unconstrained_rect = window->rect; - - window->depth = attrs->depth; - window->xvisual = attrs->visual; - - window->title = NULL; - window->icon = NULL; - window->mini_icon = NULL; - - window->frame = NULL; - window->has_focus = FALSE; - window->attached_focus_window = NULL; - - window->maximized_horizontally = FALSE; - window->maximized_vertically = FALSE; - window->maximize_horizontally_after_placement = FALSE; - window->maximize_vertically_after_placement = FALSE; - window->minimize_after_placement = FALSE; - window->fullscreen = FALSE; - window->require_fully_onscreen = TRUE; - window->require_on_single_monitor = TRUE; - window->require_titlebar_visible = TRUE; - window->on_all_workspaces = FALSE; - window->on_all_workspaces_requested = FALSE; - window->tile_mode = META_TILE_NONE; - window->tile_monitor_number = -1; - window->tile_hfraction = -1.; - window->shaded = FALSE; - window->initially_iconic = FALSE; - window->minimized = FALSE; - window->tab_unminimized = FALSE; - window->iconic = FALSE; - window->mapped = attrs->map_state != IsUnmapped; - window->known_to_compositor = FALSE; - window->visible_to_compositor = FALSE; - window->pending_compositor_effect = effect; - /* if already mapped, no need to worry about focus-on-first-time-showing */ - window->showing_for_first_time = !window->mapped; - /* if already mapped we don't want to do the placement thing; - * override-redirect windows are placed by the app */ - window->placed = ((window->mapped && !window->hidden) || window->override_redirect); - window->denied_focus_and_not_transient = FALSE; - window->unmanaging = FALSE; - window->is_in_queues = 0; - window->keys_grabbed = FALSE; - window->grab_on_frame = FALSE; - window->all_keys_grabbed = FALSE; - window->withdrawn = FALSE; - window->initial_workspace_set = FALSE; - window->initial_timestamp_set = FALSE; - window->net_wm_user_time_set = FALSE; - window->user_time_window = None; - window->input = TRUE; - window->calc_placement = FALSE; - window->shaken_loose = FALSE; - window->have_focus_click_grab = FALSE; - window->disable_sync = FALSE; - - window->unmaps_pending = 0; - window->reparents_pending = 0; - - window->mwm_decorated = TRUE; - window->mwm_border_only = FALSE; - window->mwm_has_close_func = TRUE; - window->mwm_has_minimize_func = TRUE; - window->mwm_has_maximize_func = TRUE; - window->mwm_has_move_func = TRUE; - window->mwm_has_resize_func = TRUE; - - switch (client_type) - { - case META_WINDOW_CLIENT_TYPE_X11: - window->decorated = TRUE; - window->hidden = FALSE; - break; - case META_WINDOW_CLIENT_TYPE_WAYLAND: - window->decorated = FALSE; - window->hidden = TRUE; - break; - } - - window->has_close_func = TRUE; - window->has_minimize_func = TRUE; - window->has_maximize_func = TRUE; - window->has_move_func = TRUE; - window->has_resize_func = TRUE; - - window->has_shade_func = TRUE; - - window->has_fullscreen_func = TRUE; - - window->always_sticky = FALSE; - - window->skip_taskbar = FALSE; - window->skip_pager = FALSE; - window->skip_from_window_list = FALSE; - window->wm_state_above = FALSE; - window->wm_state_below = FALSE; - window->wm_state_demands_attention = FALSE; - - window->res_class = NULL; - window->res_name = NULL; - window->role = NULL; - window->sm_client_id = NULL; - window->wm_client_machine = NULL; - window->is_remote = FALSE; - window->startup_id = NULL; - - window->client_pid = 0; - - window->xtransient_for = None; - window->xclient_leader = None; - - window->type = META_WINDOW_NORMAL; - - window->struts = NULL; - - window->layer = META_LAYER_LAST; /* invalid value */ - window->stack_position = -1; - window->initial_workspace = 0; /* not used */ - window->initial_timestamp = 0; /* not used */ - - window->compositor_private = NULL; - - window->monitor = meta_window_calculate_main_logical_monitor (window); - if (window->monitor) - window->preferred_output_winsys_id = window->monitor->winsys_id; - else - window->preferred_output_winsys_id = UINT_MAX; - - window->tile_match = NULL; - - /* Assign this #MetaWindow a sequence number which can be used - * for sorting. - */ - window->stable_sequence = ++display->window_sequence_counter; - - window->opacity = 0xFF; - - if (window->override_redirect) - { - window->decorated = FALSE; - window->always_sticky = TRUE; - window->has_close_func = FALSE; - window->has_shade_func = FALSE; - window->has_move_func = FALSE; - window->has_resize_func = FALSE; - } - - window->id = meta_display_generate_window_id (display); - - meta_window_manage (window); - - if (!window->override_redirect) - meta_window_update_icon_now (window, TRUE); - - if (window->initially_iconic) - { - /* WM_HINTS said minimized */ - window->minimized = TRUE; - meta_verbose ("Window %s asked to start out minimized", window->desc); - } - - if (existing_wm_state == IconicState) - { - /* WM_STATE said minimized */ - window->minimized = TRUE; - meta_verbose ("Window %s had preexisting WM_STATE = IconicState, minimizing", - window->desc); - - /* Assume window was previously placed, though perhaps it's - * been iconic its whole life, we have no way of knowing. - */ - window->placed = TRUE; - } - - /* Apply any window attributes such as initial workspace - * based on startup notification - */ - meta_display_apply_startup_properties (window->display, window); - - /* Try to get a "launch timestamp" for the window. If the window is - * a transient, we'd like to be able to get a last-usage timestamp - * from the parent window. If the window has no parent, there isn't - * much we can do...except record the current time so that any children - * can use this time as a fallback. - */ - if (!window->override_redirect && !window->net_wm_user_time_set) { - /* First, maybe the app was launched with startup notification using an - * obsolete version of the spec; use that timestamp if it exists. - */ - if (window->initial_timestamp_set) - /* NOTE: Do NOT toggle net_wm_user_time_set to true; this is just - * being recorded as a fallback for potential transients - */ - window->net_wm_user_time = window->initial_timestamp; - else if (window->transient_for != NULL) - meta_window_set_user_time (window, window->transient_for->net_wm_user_time); - else - /* NOTE: Do NOT toggle net_wm_user_time_set to true; this is just - * being recorded as a fallback for potential transients - */ - window->net_wm_user_time = - meta_display_get_current_time_roundtrip (window->display); - } - - window->attached = meta_window_should_attach_to_parent (window); - if (window->attached) - meta_window_recalc_features (window); - - if (window->type == META_WINDOW_DESKTOP || - window->type == META_WINDOW_DOCK) - { - /* Change the default, but don't enforce this if the user - * focuses the dock/desktop and unsticks it using key shortcuts. - * Need to set this before adding to the workspaces so the MRU - * lists will be updated. - */ - window->on_all_workspaces_requested = TRUE; - } - - window->on_all_workspaces = should_be_on_all_workspaces (window); - - /* For the workspace, first honor hints, - * if that fails put transients with parents, - * otherwise put window on active space - */ - - if (window->initial_workspace_set) - { - gboolean on_all_workspaces = window->on_all_workspaces; - MetaWorkspace *workspace = NULL; - - if (window->initial_workspace == (int) 0xFFFFFFFF) - { - meta_topic (META_DEBUG_PLACEMENT, - "Window %s is initially on all spaces", - window->desc); - - /* need to set on_all_workspaces first so that it will be - * added to all the MRU lists - */ - window->on_all_workspaces_requested = TRUE; - - on_all_workspaces = TRUE; - } - else if (!on_all_workspaces) - { - meta_topic (META_DEBUG_PLACEMENT, - "Window %s is initially on space %d", - window->desc, window->initial_workspace); - - workspace = meta_workspace_manager_get_workspace_by_index (workspace_manager, - window->initial_workspace); - } - - /* Ignore when a window requests to be placed on a non-existent workspace - */ - if (on_all_workspaces || workspace != NULL) - set_workspace_state (window, on_all_workspaces, workspace); - } - - /* override-redirect windows are subtly different from other windows - * with window->on_all_workspaces == TRUE. Other windows are part of - * some workspace (so they can return to that if the flag is turned off), - * but appear on other workspaces. override-redirect windows are part - * of no workspace. - */ - if (!window->override_redirect && window->workspace == NULL) - { - if (window->transient_for != NULL) - { - meta_topic (META_DEBUG_PLACEMENT, - "Putting window %s on same workspace as parent %s", - window->desc, window->transient_for->desc); - - g_warn_if_fail (!window->transient_for->override_redirect); - set_workspace_state (window, - window->transient_for->on_all_workspaces, - window->transient_for->workspace); - } - else if (window->on_all_workspaces) - { - meta_topic (META_DEBUG_PLACEMENT, - "Putting window %s on all workspaces", - window->desc); - - set_workspace_state (window, TRUE, NULL); - } - else - { - meta_topic (META_DEBUG_PLACEMENT, - "Putting window %s on active workspace", - window->desc); - - set_workspace_state (window, FALSE, workspace_manager->active_workspace); - } - - meta_window_update_struts (window); - } - - meta_window_main_monitor_changed (window, NULL); - - /* Must add window to stack before doing move/resize, since the - * window might have fullscreen size (i.e. should have been - * fullscreen'd; acrobat is one such braindead case; it withdraws - * and remaps its window whenever trying to become fullscreen...) - * and thus constraints may try to auto-fullscreen it which also - * means restacking it. - */ - if (meta_window_is_stackable (window)) - meta_stack_add (window->display->stack, - window); - else if (window->override_redirect) - window->layer = META_LAYER_OVERRIDE_REDIRECT; /* otherwise set by MetaStack */ - - if (!window->override_redirect) - { - /* FIXME we have a tendency to set this then immediately - * change it again. - */ - set_wm_state (window); - set_net_wm_state (window); - } - - meta_compositor_add_window (window->display->compositor, window); - window->known_to_compositor = TRUE; - - /* Sync stack changes */ - meta_stack_thaw (window->display->stack); - - /* Usually the we'll have queued a stack sync anyways, because we've - * added a new frame window or restacked. But if an undecorated - * window is mapped, already stacked in the right place, then we - * might need to do this explicitly. - */ - meta_stack_tracker_queue_sync_stack (window->display->stack_tracker); - - /* disable show desktop mode unless we're a desktop component */ - maybe_leave_show_desktop_mode (window); - - meta_window_queue (window, META_QUEUE_CALC_SHOWING); - /* See bug 303284; a transient of the given window can already exist, in which - * case we think it should probably be shown. - */ - meta_window_foreach_transient (window, - queue_calc_showing_func, - NULL); - /* See bug 334899; the window may have minimized ancestors - * which need to be shown. - * - * However, we shouldn't unminimize windows here when opening - * a new display because that breaks passing _NET_WM_STATE_HIDDEN - * between window managers when replacing them; see bug 358042. - * - * And we shouldn't unminimize windows if they were initially - * iconic. - */ - if (!window->override_redirect && - !display->display_opening && - !window->initially_iconic) - unminimize_window_and_all_transient_parents (window); - - window->constructing = FALSE; - - meta_display_notify_window_created (display, window); - - if (window->wm_state_demands_attention) - g_signal_emit_by_name (window->display, "window-demands-attention", window); - - return window; -} - -static gboolean -detach_foreach_func (MetaWindow *window, - void *data) -{ - GList **children = data; - MetaWindow *parent; - - if (window->attached) - { - /* Only return the immediate children of the window being unmanaged */ - parent = meta_window_get_transient_for (window); - if (parent->unmanaging) - *children = g_list_prepend (*children, window); - } - - return TRUE; -} - -void -meta_window_unmanage (MetaWindow *window, - guint32 timestamp) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - GList *tmp; - - meta_verbose ("Unmanaging %s", window->desc); - window->unmanaging = TRUE; - - g_clear_handle_id (&window->unmanage_idle_id, g_source_remove); - - g_signal_emit (window, window_signals[UNMANAGING], 0); - - meta_window_free_delete_dialog (window); - - if (window->visible_to_compositor) - { - window->visible_to_compositor = FALSE; - meta_compositor_hide_window (window->display->compositor, window, - META_COMP_EFFECT_DESTROY); - } - - meta_compositor_remove_window (window->display->compositor, window); - window->known_to_compositor = FALSE; - - if (destroying_windows_disallowed > 0) - meta_bug ("Tried to destroy window %s while destruction was not allowed", - window->desc); - - meta_display_unregister_stamp (window->display, window->stamp); - - if (meta_prefs_get_attach_modal_dialogs ()) - { - GList *attached_children = NULL, *iter; - - /* Detach any attached dialogs by unmapping and letting them - * be remapped after @window is destroyed. - */ - meta_window_foreach_transient (window, - detach_foreach_func, - &attached_children); - for (iter = attached_children; iter; iter = iter->next) - meta_window_unmanage (iter->data, timestamp); - g_list_free (attached_children); - } - - /* Make sure to only show window on all workspaces if requested, to - * not confuse other window managers that may take over - */ - if (meta_prefs_get_workspaces_only_on_primary ()) - meta_window_on_all_workspaces_changed (window); - - if (window->fullscreen) - { - MetaGroup *group; - - /* If the window is fullscreen, it may be forcing - * other windows in its group to a higher layer - */ - - meta_stack_freeze (window->display->stack); - group = meta_window_get_group (window); - if (group) - meta_group_update_layers (group); - meta_stack_thaw (window->display->stack); - } - - meta_display_remove_pending_pings_for_window (window->display, window); - - /* safe to do this early as group.c won't re-add to the - * group if window->unmanaging */ - meta_window_shutdown_group (window); - - /* If we have the focus, focus some other window. - * This is done first, so that if the unmap causes - * an EnterNotify the EnterNotify will have final say - * on what gets focused, maintaining sloppy focus - * invariants. - */ - if (meta_window_appears_focused (window)) - meta_window_propagate_focus_appearance (window, FALSE); - if (window->has_focus) - { - meta_topic (META_DEBUG_FOCUS, - "Focusing default window since we're unmanaging %s", - window->desc); - meta_workspace_focus_default_window (workspace_manager->active_workspace, - window, - timestamp); - } - else - { - meta_topic (META_DEBUG_FOCUS, - "Unmanaging window %s which doesn't currently have focus", - window->desc); - } - - g_assert (window->display->focus_window != window); - - if (window->struts) - { - g_slist_free_full (window->struts, g_free); - window->struts = NULL; - - meta_topic (META_DEBUG_WORKAREA, - "Unmanaging window %s which has struts, so invalidating work areas", - window->desc); - invalidate_work_areas (window); - } - - g_clear_handle_id (&window->sync_request_timeout_id, g_source_remove); - - if (window->display->grab_window == window) - meta_display_end_grab_op (window->display, timestamp); - - g_assert (window->display->grab_window != window); - - if (window->maximized_horizontally || window->maximized_vertically) - unmaximize_window_before_freeing (window); - - meta_window_unqueue (window, META_QUEUE_CALC_SHOWING | - META_QUEUE_MOVE_RESIZE | - META_QUEUE_UPDATE_ICON); - - set_workspace_state (window, FALSE, NULL); - - g_assert (window->workspace == NULL); - -#ifndef G_DISABLE_CHECKS - tmp = workspace_manager->workspaces; - while (tmp != NULL) - { - MetaWorkspace *workspace = tmp->data; - - g_assert (g_list_find (workspace->windows, window) == NULL); - g_assert (g_list_find (workspace->mru_list, window) == NULL); - - tmp = tmp->next; - } -#endif - - if (window->monitor) - { - const MetaLogicalMonitor *old = window->monitor; - - window->monitor = NULL; - meta_window_main_monitor_changed (window, old); - } - - if (meta_window_is_in_stack (window)) - meta_stack_remove (window->display->stack, window); - - /* If an undecorated window is being withdrawn, that will change the - * stack as presented to the compositing manager, without actually - * changing the stacking order of X windows. - */ - meta_stack_tracker_queue_sync_stack (window->display->stack_tracker); - - if (window->display->autoraise_window == window) - meta_display_remove_autoraise_callback (window->display); - - META_WINDOW_GET_CLASS (window)->unmanage (window); - - meta_prefs_remove_listener (prefs_changed_callback, window); - meta_display_queue_check_fullscreen (window->display); - - g_signal_emit (window, window_signals[UNMANAGED], 0); - - g_object_unref (window); -} - -static void -set_wm_state (MetaWindow *window) -{ - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - meta_window_x11_set_wm_state (window); -} - -static void -set_net_wm_state (MetaWindow *window) -{ - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - meta_window_x11_set_net_wm_state (window); -} - -static void -set_allowed_actions_hint (MetaWindow *window) -{ - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - meta_window_x11_set_allowed_actions_hint (window); -} - -/** - * meta_window_located_on_workspace: - * @window: a #MetaWindow - * @workspace: a #MetaWorkspace - * - * Returns: whether @window is displayed on @workspace, or whether it - * will be displayed on all workspaces. - */ -gboolean -meta_window_located_on_workspace (MetaWindow *window, - MetaWorkspace *workspace) -{ - return (window->on_all_workspaces) || (window->workspace == workspace); -} - -static gboolean -is_minimized_foreach (MetaWindow *window, - void *data) -{ - gboolean *result = data; - - *result = window->minimized; - if (*result) - return FALSE; /* stop as soon as we find one */ - else - return TRUE; -} - -static gboolean -ancestor_is_minimized (MetaWindow *window) -{ - gboolean is_minimized; - - is_minimized = FALSE; - - meta_window_foreach_ancestor (window, is_minimized_foreach, &is_minimized); - - return is_minimized; -} - -/** - * meta_window_showing_on_its_workspace: - * @window: A #MetaWindow - * - * Returns: %TRUE if window would be visible, if its workspace was current - */ -gboolean -meta_window_showing_on_its_workspace (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean showing; - gboolean is_desktop_or_dock; - MetaWorkspace* workspace_of_window; - - showing = TRUE; - - /* 1. See if we're minimized */ - if (window->minimized) - showing = FALSE; - - /* 2. See if we're in "show desktop" mode */ - is_desktop_or_dock = FALSE; - is_desktop_or_dock_foreach (window, - &is_desktop_or_dock); - - meta_window_foreach_ancestor (window, is_desktop_or_dock_foreach, - &is_desktop_or_dock); - - if (window->on_all_workspaces) - workspace_of_window = workspace_manager->active_workspace; - else if (window->workspace) - workspace_of_window = window->workspace; - else /* This only seems to be needed for startup */ - workspace_of_window = NULL; - - if (showing && - workspace_of_window && workspace_of_window->showing_desktop && - !is_desktop_or_dock) - { - meta_verbose ("We're showing the desktop on the workspace(s) that window %s is on", - window->desc); - showing = FALSE; - } - - /* 3. See if an ancestor is minimized (note that - * ancestor's "mapped" field may not be up to date - * since it's being computed in this same idle queue) - */ - - if (showing) - { - if (ancestor_is_minimized (window)) - showing = FALSE; - } - - return showing; -} - -gboolean -meta_window_should_be_showing (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - -#ifdef HAVE_WAYLAND - if (window->client_type == META_WINDOW_CLIENT_TYPE_WAYLAND && - !meta_wayland_surface_get_buffer (window->surface)) - return FALSE; -#endif - - /* Windows should be showing if they're located on the - * active workspace and they're showing on their own workspace. */ - return (meta_window_located_on_workspace (window, workspace_manager->active_workspace) && - meta_window_showing_on_its_workspace (window)); -} - -static void -implement_showing (MetaWindow *window, - gboolean showing) -{ - /* Actually show/hide the window */ - meta_verbose ("Implement showing = %d for window %s", - showing, window->desc); - - /* Some windows are not stackable until being showed, so add those now. */ - if (meta_window_is_stackable (window) && !meta_window_is_in_stack (window)) - meta_stack_add (window->display->stack, window); - - if (!showing) - { - /* When we manage a new window, we normally delay placing it - * until it is is first shown, but if we're previewing hidden - * windows we might want to know where they are on the screen, - * so we should place the window even if we're hiding it rather - * than showing it. - * Force placing windows only when they should be already mapped, - * see #751887 - */ - if (!window->placed && client_window_should_be_mapped (window)) - meta_window_force_placement (window, FALSE); - - meta_window_hide (window); - } - else - meta_window_show (window); - - if (!window->override_redirect) - sync_client_window_mapped (window); -} - -static void -meta_window_calc_showing (MetaWindow *window) -{ - implement_showing (window, meta_window_should_be_showing (window)); -} - -static guint queue_later[NUMBER_OF_QUEUES] = {0, 0, 0}; -static GSList *queue_pending[NUMBER_OF_QUEUES] = {NULL, NULL, NULL}; - -static int -stackcmp (gconstpointer a, gconstpointer b) -{ - MetaWindow *aw = (gpointer) a; - MetaWindow *bw = (gpointer) b; - - return meta_stack_windows_cmp (aw->display->stack, - aw, bw); -} - -static gboolean -idle_calc_showing (gpointer data) -{ - MetaDisplay *display = meta_get_display (); - GSList *tmp; - GSList *copy; - GSList *should_show; - GSList *should_hide; - GSList *unplaced; - GSList *displays; - guint queue_index = GPOINTER_TO_INT (data); - - COGL_TRACE_BEGIN_SCOPED (MetaWindowCalcShowing, "Window: Calc showing"); - - g_return_val_if_fail (queue_pending[queue_index] != NULL, FALSE); - - meta_topic (META_DEBUG_WINDOW_STATE, - "Clearing the calc_showing queue"); - - /* Work with a copy, for reentrancy. The allowed reentrancy isn't - * complete; destroying a window while we're in here would result in - * badness. But it's OK to queue/unqueue calc_showings. - */ - copy = g_slist_copy (queue_pending[queue_index]); - g_slist_free (queue_pending[queue_index]); - queue_pending[queue_index] = NULL; - queue_later[queue_index] = 0; - - destroying_windows_disallowed += 1; - - /* We map windows from top to bottom and unmap from bottom to - * top, to avoid extra expose events. The exception is - * for unplaced windows, which have to be mapped from bottom to - * top so placement works. - */ - should_show = NULL; - should_hide = NULL; - unplaced = NULL; - displays = NULL; - - COGL_TRACE_BEGIN (MetaWindowCalcShowingCalc, "Window: Calc showing (calc)"); - - tmp = copy; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - if (!window->placed) - unplaced = g_slist_prepend (unplaced, window); - else if (meta_window_should_be_showing (window)) - should_show = g_slist_prepend (should_show, window); - else - should_hide = g_slist_prepend (should_hide, window); - - tmp = tmp->next; - } - - /* bottom to top */ - unplaced = g_slist_sort (unplaced, stackcmp); - should_hide = g_slist_sort (should_hide, stackcmp); - /* top to bottom */ - should_show = g_slist_sort (should_show, stackcmp); - should_show = g_slist_reverse (should_show); - - COGL_TRACE_END (MetaWindowCalcShowingCalc); - - COGL_TRACE_BEGIN (MetaWindowCalcShowingUnplaced, - "Window: Calc showing (calc unplaced)"); - - tmp = unplaced; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - meta_window_calc_showing (window); - - tmp = tmp->next; - } - - COGL_TRACE_END (MetaWindowCalcShowingUnplaced); - - meta_stack_freeze (display->stack); - - COGL_TRACE_BEGIN (MetaWindowCalcShowingShow, "Window: Calc showing (show)"); - tmp = should_show; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - implement_showing (window, TRUE); - - tmp = tmp->next; - } - COGL_TRACE_END (MetaWindowCalcShowingShow); - - COGL_TRACE_BEGIN (MetaWindowCalcShowingHide, "Window: Calc showing (hide)"); - tmp = should_hide; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - implement_showing (window, FALSE); - - tmp = tmp->next; - } - COGL_TRACE_END (MetaWindowCalcShowingHide); - - COGL_TRACE_BEGIN (MetaWindowCalcShowingSync, - "Window: Calc showing (sync stack)"); - meta_stack_thaw (display->stack); - COGL_TRACE_END (MetaWindowCalcShowingSync); - - tmp = copy; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - /* important to set this here for reentrancy - - * if we queue a window again while it's in "copy", - * then queue_calc_showing will just return since - * we are still in the calc_showing queue - */ - window->is_in_queues &= ~META_QUEUE_CALC_SHOWING; - - tmp = tmp->next; - } - - if (meta_prefs_get_focus_mode () != G_DESKTOP_FOCUS_MODE_CLICK) - { - /* When display->mouse_mode is false, we want to ignore - * EnterNotify events unless they come from mouse motion. To do - * that, we set a sentinel property on the root window if we're - * not in mouse_mode. - */ - tmp = should_show; - while (tmp != NULL) - { - MetaWindow *window = tmp->data; - MetaDisplay *display = window->display; - - if (display->x11_display && !display->mouse_mode) - meta_x11_display_increment_focus_sentinel (display->x11_display); - - tmp = tmp->next; - } - } - - g_slist_free (copy); - - g_slist_free (unplaced); - g_slist_free (should_show); - g_slist_free (should_hide); - g_slist_free (displays); - - destroying_windows_disallowed -= 1; - - return FALSE; -} - -#ifdef WITH_VERBOSE_MODE -static const gchar* meta_window_queue_names[NUMBER_OF_QUEUES] = - {"calc_showing", "move_resize", "update_icon"}; -#endif - -static void -meta_window_unqueue (MetaWindow *window, guint queuebits) -{ - gint queuenum; - - for (queuenum=0; queuenum<NUMBER_OF_QUEUES; queuenum++) - { - if ((queuebits & 1<<queuenum) /* they have asked to unqueue */ - && - (window->is_in_queues & 1<<queuenum)) /* it's in the queue */ - { - - meta_topic (META_DEBUG_WINDOW_STATE, - "Removing %s from the %s queue", - window->desc, - meta_window_queue_names[queuenum]); - - /* Note that window may not actually be in the queue - * because it may have been in "copy" inside the idle handler - */ - queue_pending[queuenum] = g_slist_remove (queue_pending[queuenum], window); - window->is_in_queues &= ~(1<<queuenum); - - /* Okay, so maybe we've used up all the entries in the queue. - * In that case, we should kill the function that deals with - * the queue, because there's nothing left for it to do. - */ - if (queue_pending[queuenum] == NULL && queue_later[queuenum] != 0) - { - meta_later_remove (queue_later[queuenum]); - queue_later[queuenum] = 0; - } - } - } -} - -static void -meta_window_flush_calc_showing (MetaWindow *window) -{ - if (window->is_in_queues & META_QUEUE_CALC_SHOWING) - { - meta_window_unqueue (window, META_QUEUE_CALC_SHOWING); - meta_window_calc_showing (window); - } -} - -void -meta_window_queue (MetaWindow *window, guint queuebits) -{ - guint queuenum; - - /* Easier to debug by checking here rather than in the idle */ - g_return_if_fail (!window->override_redirect || (queuebits & META_QUEUE_MOVE_RESIZE) == 0); - - for (queuenum=0; queuenum<NUMBER_OF_QUEUES; queuenum++) - { - if (queuebits & 1<<queuenum) - { - /* Data which varies between queues. - * Yes, these do look a lot like associative arrays: - * I seem to be turning into a Perl programmer. - */ - - const MetaLaterType window_queue_later_when[NUMBER_OF_QUEUES] = - { - META_LATER_CALC_SHOWING, /* CALC_SHOWING */ - META_LATER_RESIZE, /* MOVE_RESIZE */ - META_LATER_BEFORE_REDRAW /* UPDATE_ICON */ - }; - - const GSourceFunc window_queue_later_handler[NUMBER_OF_QUEUES] = - { - idle_calc_showing, - idle_move_resize, - idle_update_icon, - }; - - /* If we're about to drop the window, there's no point in putting - * it on a queue. - */ - if (window->unmanaging) - break; - - /* If the window already claims to be in that queue, there's no - * point putting it in the queue. - */ - if (window->is_in_queues & 1<<queuenum) - break; - - meta_topic (META_DEBUG_WINDOW_STATE, - "Putting %s in the %s queue", - window->desc, - meta_window_queue_names[queuenum]); - - /* So, mark it as being in this queue. */ - window->is_in_queues |= 1<<queuenum; - - /* There's not a lot of point putting things into a queue if - * nobody's on the other end pulling them out. Therefore, - * let's check to see whether an idle handler exists to do - * that. If not, we'll create one. - */ - - if (queue_later[queuenum] == 0) - queue_later[queuenum] = meta_later_add - ( - window_queue_later_when[queuenum], - window_queue_later_handler[queuenum], - GUINT_TO_POINTER(queuenum), - NULL - ); - - /* And now we actually put it on the queue. */ - queue_pending[queuenum] = g_slist_prepend (queue_pending[queuenum], - window); - } - } -} - -static gboolean -intervening_user_event_occurred (MetaWindow *window) -{ - guint32 compare; - MetaWindow *focus_window; - - focus_window = window->display->focus_window; - - meta_topic (META_DEBUG_STARTUP, - "COMPARISON:\n" - " net_wm_user_time_set : %d\n" - " net_wm_user_time : %u\n" - " initial_timestamp_set: %d\n" - " initial_timestamp : %u", - window->net_wm_user_time_set, - window->net_wm_user_time, - window->initial_timestamp_set, - window->initial_timestamp); - if (focus_window != NULL) - { - meta_topic (META_DEBUG_STARTUP, - "COMPARISON (continued):\n" - " focus_window : %s\n" - " fw->net_wm_user_time_set : %d\n" - " fw->net_wm_user_time : %u", - focus_window->desc, - focus_window->net_wm_user_time_set, - focus_window->net_wm_user_time); - } - - /* We expect the most common case for not focusing a new window - * to be when a hint to not focus it has been set. Since we can - * deal with that case rapidly, we use special case it--this is - * merely a preliminary optimization. :) - */ - if ( ((window->net_wm_user_time_set == TRUE) && - (window->net_wm_user_time == 0)) - || - ((window->initial_timestamp_set == TRUE) && - (window->initial_timestamp == 0))) - { - meta_topic (META_DEBUG_STARTUP, - "window %s explicitly requested no focus", - window->desc); - return TRUE; - } - - if (!(window->net_wm_user_time_set) && !(window->initial_timestamp_set)) - { - meta_topic (META_DEBUG_STARTUP, - "no information about window %s found", - window->desc); - return FALSE; - } - - if (focus_window != NULL && - !focus_window->net_wm_user_time_set) - { - meta_topic (META_DEBUG_STARTUP, - "focus window, %s, doesn't have a user time set yet!", - window->desc); - return FALSE; - } - - /* To determine the "launch" time of an application, - * startup-notification can set the TIMESTAMP and the - * application (usually via its toolkit such as gtk or qt) can - * set the _NET_WM_USER_TIME. If both are set, we need to be - * using the newer of the two values. - * - * See http://bugzilla.gnome.org/show_bug.cgi?id=573922 - */ - compare = 0; - if (window->net_wm_user_time_set && - window->initial_timestamp_set) - compare = - XSERVER_TIME_IS_BEFORE (window->net_wm_user_time, - window->initial_timestamp) ? - window->initial_timestamp : window->net_wm_user_time; - else if (window->net_wm_user_time_set) - compare = window->net_wm_user_time; - else if (window->initial_timestamp_set) - compare = window->initial_timestamp; - - if ((focus_window != NULL) && - XSERVER_TIME_IS_BEFORE (compare, focus_window->net_wm_user_time)) - { - meta_topic (META_DEBUG_STARTUP, - "window %s focus prevented by other activity; %u < %u", - window->desc, - compare, - focus_window->net_wm_user_time); - return TRUE; - } - else - { - meta_topic (META_DEBUG_STARTUP, - "new window %s with no intervening events", - window->desc); - return FALSE; - } -} - -/* This function is an ugly hack. It's experimental in nature and ought to be - * replaced by a real hint from the app to the WM if we decide the experimental - * behavior is worthwhile. The basic idea is to get more feedback about how - * usage scenarios of "strict" focus users and what they expect. See #326159. - */ -static gboolean -window_is_terminal (MetaWindow *window) -{ - if (window == NULL || window->res_class == NULL) - return FALSE; - - /* - * Compare res_class, which is not user-settable, and thus theoretically - * a more-reliable indication of term-ness. - */ - - /* gnome-terminal -- if you couldn't guess */ - if (strcmp (window->res_class, "Gnome-terminal") == 0) - return TRUE; - /* xterm, rxvt, aterm */ - else if (strcmp (window->res_class, "XTerm") == 0) - return TRUE; - /* konsole, KDE's terminal program */ - else if (strcmp (window->res_class, "Konsole") == 0) - return TRUE; - /* rxvt-unicode */ - else if (strcmp (window->res_class, "URxvt") == 0) - return TRUE; - /* eterm */ - else if (strcmp (window->res_class, "Eterm") == 0) - return TRUE; - /* KTerm -- some terminal not KDE based; so not like Konsole */ - else if (strcmp (window->res_class, "KTerm") == 0) - return TRUE; - /* Multi-gnome-terminal */ - else if (strcmp (window->res_class, "Multi-gnome-terminal") == 0) - return TRUE; - /* mlterm ("multi lingual terminal emulator on X") */ - else if (strcmp (window->res_class, "mlterm") == 0) - return TRUE; - /* Terminal -- XFCE Terminal */ - else if (strcmp (window->res_class, "Terminal") == 0) - return TRUE; - - return FALSE; -} - -/* This function determines what state the window should have assuming that it - * and the focus_window have no relation - */ -static void -window_state_on_map (MetaWindow *window, - gboolean *takes_focus, - gboolean *places_on_top) -{ - gboolean intervening_events; - - intervening_events = intervening_user_event_occurred (window); - - *takes_focus = !intervening_events; - *places_on_top = *takes_focus; - - /* don't initially focus windows that are intended to not accept - * focus - */ - if (!meta_window_is_focusable (window)) - { - *takes_focus = FALSE; - return; - } - - /* Terminal usage may be different; some users intend to launch - * many apps in quick succession or to just view things in the new - * window while still interacting with the terminal. In that case, - * apps launched from the terminal should not take focus. This - * isn't quite the same as not allowing focus to transfer from - * terminals due to new window map, but the latter is a much easier - * approximation to enforce so we do that. - */ - if (*takes_focus && - meta_prefs_get_focus_new_windows () == G_DESKTOP_FOCUS_NEW_WINDOWS_STRICT && - !window->display->allow_terminal_deactivation && - window_is_terminal (window->display->focus_window) && - !meta_window_is_ancestor_of_transient (window->display->focus_window, - window)) - { - meta_topic (META_DEBUG_FOCUS, - "focus_window is terminal; not focusing new window."); - *takes_focus = FALSE; - *places_on_top = FALSE; - } - - switch (window->type) - { - case META_WINDOW_UTILITY: - case META_WINDOW_TOOLBAR: - *takes_focus = FALSE; - *places_on_top = FALSE; - break; - case META_WINDOW_DOCK: - case META_WINDOW_DESKTOP: - case META_WINDOW_SPLASHSCREEN: - case META_WINDOW_MENU: - /* override redirect types: */ - case META_WINDOW_DROPDOWN_MENU: - case META_WINDOW_POPUP_MENU: - case META_WINDOW_TOOLTIP: - case META_WINDOW_NOTIFICATION: - case META_WINDOW_COMBO: - case META_WINDOW_DND: - case META_WINDOW_OVERRIDE_OTHER: - /* don't focus any of these; places_on_top may be irrelevant for some of - * these (e.g. dock)--but you never know--the focus window might also be - * of the same type in some weird situation... - */ - *takes_focus = FALSE; - break; - case META_WINDOW_NORMAL: - case META_WINDOW_DIALOG: - case META_WINDOW_MODAL_DIALOG: - /* The default is correct for these */ - break; - } -} - -static gboolean -windows_overlap (const MetaWindow *w1, const MetaWindow *w2) -{ - MetaRectangle w1rect, w2rect; - meta_window_get_frame_rect (w1, &w1rect); - meta_window_get_frame_rect (w2, &w2rect); - return meta_rectangle_overlap (&w1rect, &w2rect); -} - -/* Returns whether a new window would be covered by any - * existing window on the same workspace that is set - * to be "above" ("always on top"). A window that is not - * set "above" would be underneath the new window anyway. - * - * We take "covered" to mean even partially covered, but - * some people might prefer entirely covered. I think it - * is more useful to behave this way if any part of the - * window is covered, because a partial coverage could be - * (say) ninety per cent and almost indistinguishable from total. - */ -static gboolean -window_would_be_covered (const MetaWindow *newbie) -{ - MetaWorkspace *workspace = meta_window_get_workspace ((MetaWindow *)newbie); - GList *tmp, *windows; - - windows = meta_workspace_list_windows (workspace); - - tmp = windows; - while (tmp != NULL) - { - MetaWindow *w = tmp->data; - - if (w->wm_state_above && w != newbie) - { - /* We have found a window that is "above". Perhaps it overlaps. */ - if (windows_overlap (w, newbie)) - { - g_list_free (windows); /* clean up... */ - return TRUE; /* yes, it does */ - } - } - - tmp = tmp->next; - } - - g_list_free (windows); - return FALSE; /* none found */ -} - -void -meta_window_force_placement (MetaWindow *window, - gboolean force_move) -{ - MetaMoveResizeFlags flags; - - if (window->placed) - return; - - /* We have to recalc the placement here since other windows may - * have been mapped/placed since we last did constrain_position - */ - - /* calc_placement is an efficiency hack to avoid - * multiple placement calculations before we finally - * show the window. - */ - window->calc_placement = TRUE; - - flags = META_MOVE_RESIZE_MOVE_ACTION | META_MOVE_RESIZE_RESIZE_ACTION; - if (force_move) - flags |= META_MOVE_RESIZE_FORCE_MOVE; - - meta_window_move_resize_internal (window, - flags, - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); - window->calc_placement = FALSE; - - /* don't ever do the initial position constraint thing again. - * This is toggled here so that initially-iconified windows - * still get placed when they are ultimately shown. - */ - window->placed = TRUE; - - /* Don't want to accidentally reuse the fact that we had been denied - * focus in any future constraints unless we're denied focus again. - */ - window->denied_focus_and_not_transient = FALSE; -} - -static void -meta_window_show (MetaWindow *window) -{ - gboolean did_show; - gboolean takes_focus_on_map; - gboolean place_on_top_on_map; - gboolean needs_stacking_adjustment; - MetaWindow *focus_window; - gboolean notify_demands_attention = FALSE; - MetaDisplay *display = window->display; - - meta_topic (META_DEBUG_WINDOW_STATE, - "Showing window %s, shaded: %d iconic: %d placed: %d", - window->desc, window->shaded, window->iconic, window->placed); - - focus_window = window->display->focus_window; /* May be NULL! */ - did_show = FALSE; - window_state_on_map (window, &takes_focus_on_map, &place_on_top_on_map); - needs_stacking_adjustment = FALSE; - - meta_topic (META_DEBUG_WINDOW_STATE, - "Window %s %s focus on map, and %s place on top on map.", - window->desc, - takes_focus_on_map ? "does" : "does not", - place_on_top_on_map ? "does" : "does not"); - - /* Now, in some rare cases we should *not* put a new window on top. - * These cases include certain types of windows showing for the first - * time, and any window which would be covered because of another window - * being set "above" ("always on top"). - * - * FIXME: Although "place_on_top_on_map" and "takes_focus_on_map" are - * generally based on the window type, there is a special case when the - * focus window is a terminal for them both to be false; this should - * probably rather be a term in the "if" condition below. - */ - - if ( focus_window != NULL && window->showing_for_first_time && - ( (!place_on_top_on_map && !takes_focus_on_map) || - window_would_be_covered (window) ) - ) { - if (!meta_window_is_ancestor_of_transient (focus_window, window)) - { - needs_stacking_adjustment = TRUE; - if (!window->placed) - window->denied_focus_and_not_transient = TRUE; - } - } - - if (!window->placed) - { - if (window->monitor && - meta_prefs_get_auto_maximize() && - window->showing_for_first_time && - window->has_maximize_func) - { - MetaRectangle work_area; - meta_window_get_work_area_for_monitor (window, window->monitor->number, &work_area); - /* Automaximize windows that map with a size > MAX_UNMAXIMIZED_WINDOW_AREA of the work area */ - if (window->rect.width * window->rect.height > work_area.width * work_area.height * MAX_UNMAXIMIZED_WINDOW_AREA) - { - window->maximize_horizontally_after_placement = TRUE; - window->maximize_vertically_after_placement = TRUE; - } - } - meta_window_force_placement (window, FALSE); - } - - if (needs_stacking_adjustment) - { - gboolean overlap; - - /* This window isn't getting focus on map. We may need to do some - * special handing with it in regards to - * - the stacking of the window - * - the MRU position of the window - * - the demands attention setting of the window - * - * Firstly, set the flag so we don't give the window focus anyway - * and confuse people. - */ - - takes_focus_on_map = FALSE; - - overlap = windows_overlap (window, focus_window); - - /* We want alt tab to go to the denied-focus window */ - ensure_mru_position_after (window, focus_window); - - /* We don't want the denied-focus window to obscure the focus - * window, and if we're in both click-to-focus mode and - * raise-on-click mode then we want to maintain the invariant - * that MRU order == stacking order. The need for this if - * comes from the fact that in sloppy/mouse focus the focus - * window may not overlap other windows and also can be - * considered "below" them; this combination means that - * placing the denied-focus window "below" the focus window - * in the stack when it doesn't overlap it confusingly places - * that new window below a lot of other windows. - */ - if (overlap || - (meta_prefs_get_focus_mode () == G_DESKTOP_FOCUS_MODE_CLICK && - meta_prefs_get_raise_on_click ())) - meta_window_stack_just_below (window, focus_window); - - /* If the window will be obscured by the focus window, then the - * user might not notice the window appearing so set the - * demands attention hint. - * - * We set the hint ourselves rather than calling - * meta_window_set_demands_attention() because that would cause - * a recalculation of overlap, and a call to set_net_wm_state() - * which we are going to call ourselves here a few lines down. - */ - if (overlap) - { - if (!window->wm_state_demands_attention) - { - window->wm_state_demands_attention = TRUE; - notify_demands_attention = TRUE; - } - } - } - - if (window->hidden) - { - meta_stack_freeze (window->display->stack); - window->hidden = FALSE; - meta_stack_thaw (window->display->stack); - did_show = TRUE; - } - - if (window->iconic) - { - window->iconic = FALSE; - set_wm_state (window); - } - - if (!window->visible_to_compositor) - { - MetaCompEffect effect = META_COMP_EFFECT_NONE; - - window->visible_to_compositor = TRUE; - - switch (window->pending_compositor_effect) - { - case META_COMP_EFFECT_CREATE: - case META_COMP_EFFECT_UNMINIMIZE: - effect = window->pending_compositor_effect; - break; - case META_COMP_EFFECT_NONE: - case META_COMP_EFFECT_DESTROY: - case META_COMP_EFFECT_MINIMIZE: - break; - } - - meta_compositor_show_window (window->display->compositor, - window, effect); - window->pending_compositor_effect = META_COMP_EFFECT_NONE; - } - - /* We don't want to worry about all cases from inside - * implement_showing(); we only want to worry about focus if this - * window has not been shown before. - */ - if (window->showing_for_first_time) - { - window->showing_for_first_time = FALSE; - if (takes_focus_on_map) - { - guint32 timestamp; - - timestamp = meta_display_get_current_time_roundtrip (window->display); - - meta_window_focus (window, timestamp); - } - else if (display->x11_display) - { - /* Prevent EnterNotify events in sloppy/mouse focus from - * erroneously focusing the window that had been denied - * focus. FIXME: This introduces a race; I have a couple - * ideas for a better way to accomplish the same thing, but - * they're more involved so do it this way for now. - */ - meta_x11_display_increment_focus_sentinel (display->x11_display); - } - } - - set_net_wm_state (window); - - if (did_show && window->struts) - { - meta_topic (META_DEBUG_WORKAREA, - "Mapped window %s with struts, so invalidating work areas", - window->desc); - invalidate_work_areas (window); - } - - if (did_show) - meta_display_queue_check_fullscreen (window->display); - - /* - * Now that we have shown the window, we no longer want to consider the - * initial timestamp in any subsequent deliberations whether to focus this - * window or not, so clear the flag. - * - * See http://bugzilla.gnome.org/show_bug.cgi?id=573922 - */ - window->initial_timestamp_set = FALSE; - - if (notify_demands_attention) - { - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_DEMANDS_ATTENTION]); - g_signal_emit_by_name (window->display, "window-demands-attention", - window); - } - - if (did_show) - g_signal_emit (window, window_signals[SHOWN], 0); -} - -static void -meta_window_hide (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean did_hide; - - meta_topic (META_DEBUG_WINDOW_STATE, - "Hiding window %s", window->desc); - - if (window->visible_to_compositor) - { - MetaCompEffect effect = META_COMP_EFFECT_NONE; - - window->visible_to_compositor = FALSE; - - switch (window->pending_compositor_effect) - { - case META_COMP_EFFECT_CREATE: - case META_COMP_EFFECT_UNMINIMIZE: - case META_COMP_EFFECT_NONE: - break; - case META_COMP_EFFECT_DESTROY: - case META_COMP_EFFECT_MINIMIZE: - effect = window->pending_compositor_effect; - break; - } - - meta_compositor_hide_window (window->display->compositor, window, effect); - window->pending_compositor_effect = META_COMP_EFFECT_NONE; - } - - did_hide = FALSE; - - if (!window->hidden) - { - meta_stack_freeze (window->display->stack); - window->hidden = TRUE; - meta_stack_thaw (window->display->stack); - - did_hide = TRUE; - } - - if (!window->iconic) - { - window->iconic = TRUE; - set_wm_state (window); - } - - set_net_wm_state (window); - - if (did_hide && window->struts) - { - meta_topic (META_DEBUG_WORKAREA, - "Unmapped window %s with struts, so invalidating work areas", - window->desc); - invalidate_work_areas (window); - } - - if (window->has_focus) - { - MetaWindow *not_this_one = NULL; - MetaWorkspace *my_workspace = meta_window_get_workspace (window); - guint32 timestamp = meta_display_get_current_time_roundtrip (window->display); - - /* - * If this window is modal, passing the not_this_one window to - * _focus_default_window() makes the focus to be given to this window's - * ancestor. This can only be the case if the window is on the currently - * active workspace; when it is not, we need to pass in NULL, so as to - * focus the default window for the active workspace (this scenario - * arises when we are switching workspaces). - * We also pass in NULL if we are in the process of hiding all non-desktop - * windows to avoid unexpected changes to the stacking order. - */ - if (my_workspace == workspace_manager->active_workspace && - !my_workspace->showing_desktop) - not_this_one = window; - - meta_workspace_focus_default_window (workspace_manager->active_workspace, - not_this_one, - timestamp); - } - - if (did_hide) - meta_display_queue_check_fullscreen (window->display); -} - -static gboolean -queue_calc_showing_func (MetaWindow *window, - void *data) -{ - meta_window_queue(window, META_QUEUE_CALC_SHOWING); - return TRUE; -} - -void -meta_window_minimize (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - if (!window->minimized) - { - window->minimized = TRUE; - window->pending_compositor_effect = META_COMP_EFFECT_MINIMIZE; - meta_window_queue(window, META_QUEUE_CALC_SHOWING); - - meta_window_foreach_transient (window, - queue_calc_showing_func, - NULL); - - if (window->has_focus) - { - meta_topic (META_DEBUG_FOCUS, - "Focusing default window due to minimization of focus window %s", - window->desc); - } - else - { - meta_topic (META_DEBUG_FOCUS, - "Minimizing window %s which doesn't have the focus", - window->desc); - } - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MINIMIZED]); - } -} - -void -meta_window_unminimize (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - if (window->minimized) - { - window->minimized = FALSE; - window->pending_compositor_effect = META_COMP_EFFECT_UNMINIMIZE; - meta_window_queue(window, META_QUEUE_CALC_SHOWING); - - meta_window_foreach_transient (window, - queue_calc_showing_func, - NULL); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MINIMIZED]); - } -} - -static void -ensure_size_hints_satisfied (MetaRectangle *rect, - const XSizeHints *size_hints) -{ - int minw, minh, maxw, maxh; /* min/max width/height */ - int basew, baseh, winc, hinc; /* base width/height, width/height increment */ - int extra_width, extra_height; - - minw = size_hints->min_width; minh = size_hints->min_height; - maxw = size_hints->max_width; maxh = size_hints->max_height; - basew = size_hints->base_width; baseh = size_hints->base_height; - winc = size_hints->width_inc; hinc = size_hints->height_inc; - - /* First, enforce min/max size constraints */ - rect->width = CLAMP (rect->width, minw, maxw); - rect->height = CLAMP (rect->height, minh, maxh); - - /* Now, verify size increment constraints are satisfied, or make them be */ - extra_width = (rect->width - basew) % winc; - extra_height = (rect->height - baseh) % hinc; - - rect->width -= extra_width; - rect->height -= extra_height; - - /* Adjusting width/height down, as done above, may violate minimum size - * constraints, so one last fix. - */ - if (rect->width < minw) - rect->width += ((minw - rect->width)/winc + 1)*winc; - if (rect->height < minh) - rect->height += ((minh - rect->height)/hinc + 1)*hinc; -} - -static void -meta_window_save_rect (MetaWindow *window) -{ - if (!(META_WINDOW_MAXIMIZED (window) || META_WINDOW_TILED_SIDE_BY_SIDE (window) || window->fullscreen)) - { - /* save size/pos as appropriate args for move_resize */ - if (!window->maximized_horizontally) - { - window->saved_rect.x = window->rect.x; - window->saved_rect.width = window->rect.width; - } - if (!window->maximized_vertically) - { - window->saved_rect.y = window->rect.y; - window->saved_rect.height = window->rect.height; - } - } -} - -void -meta_window_maximize_internal (MetaWindow *window, - MetaMaximizeFlags directions, - MetaRectangle *saved_rect) -{ - /* At least one of the two directions ought to be set */ - gboolean maximize_horizontally, maximize_vertically; - maximize_horizontally = directions & META_MAXIMIZE_HORIZONTAL; - maximize_vertically = directions & META_MAXIMIZE_VERTICAL; - g_assert (maximize_horizontally || maximize_vertically); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Maximizing %s%s", - window->desc, - maximize_horizontally && maximize_vertically ? "" : - maximize_horizontally ? " horizontally" : - maximize_vertically ? " vertically" : "BUGGGGG"); - - if (saved_rect != NULL) - window->saved_rect = *saved_rect; - else - meta_window_save_rect (window); - - if (maximize_horizontally && maximize_vertically) - window->saved_maximize = TRUE; - - window->maximized_horizontally = - window->maximized_horizontally || maximize_horizontally; - window->maximized_vertically = - window->maximized_vertically || maximize_vertically; - - /* Update the edge constraints */ - update_edge_constraints (window); - - meta_window_recalc_features (window); - set_net_wm_state (window); - - if (window->monitor && window->monitor->in_fullscreen) - meta_display_queue_check_fullscreen (window->display); - - g_object_freeze_notify (G_OBJECT (window)); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MAXIMIZED_HORIZONTALLY]); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MAXIMIZED_VERTICALLY]); - g_object_thaw_notify (G_OBJECT (window)); -} - -void -meta_window_maximize (MetaWindow *window, - MetaMaximizeFlags directions) -{ - MetaRectangle *saved_rect = NULL; - gboolean maximize_horizontally, maximize_vertically; - - g_return_if_fail (!window->override_redirect); - - /* At least one of the two directions ought to be set */ - maximize_horizontally = directions & META_MAXIMIZE_HORIZONTAL; - maximize_vertically = directions & META_MAXIMIZE_VERTICAL; - g_assert (maximize_horizontally || maximize_vertically); - - /* Only do something if the window isn't already maximized in the - * given direction(s). - */ - if ((maximize_horizontally && !window->maximized_horizontally) || - (maximize_vertically && !window->maximized_vertically)) - { - if (window->shaded && maximize_vertically) - { - /* Shading sucks anyway; I'm not adding a timestamp argument - * to this function just for this niche usage & corner case. - */ - guint32 timestamp = - meta_display_get_current_time_roundtrip (window->display); - meta_window_unshade (window, timestamp); - } - - /* if the window hasn't been placed yet, we'll maximize it then - */ - if (!window->placed) - { - window->maximize_horizontally_after_placement = - window->maximize_horizontally_after_placement || - maximize_horizontally; - window->maximize_vertically_after_placement = - window->maximize_vertically_after_placement || - maximize_vertically; - return; - } - - if (window->tile_mode != META_TILE_NONE) - { - saved_rect = &window->saved_rect; - - window->maximized_vertically = FALSE; - window->tile_mode = META_TILE_NONE; - } - - meta_window_maximize_internal (window, - directions, - saved_rect); - - MetaRectangle old_frame_rect, old_buffer_rect; - - meta_window_get_frame_rect (window, &old_frame_rect); - meta_window_get_buffer_rect (window, &old_buffer_rect); - - meta_compositor_size_change_window (window->display->compositor, window, - META_SIZE_CHANGE_MAXIMIZE, - &old_frame_rect, &old_buffer_rect); - - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_STATE_CHANGED), - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); - } -} - -/** - * meta_window_get_maximized: - * @window: a #MetaWindow - * - * Gets the current maximization state of the window, as combination - * of the %META_MAXIMIZE_HORIZONTAL and %META_MAXIMIZE_VERTICAL flags; - * - * Return value: current maximization state - */ -MetaMaximizeFlags -meta_window_get_maximized (MetaWindow *window) -{ - return ((window->maximized_horizontally ? META_MAXIMIZE_HORIZONTAL : 0) | - (window->maximized_vertically ? META_MAXIMIZE_VERTICAL : 0)); -} - -/** - * meta_window_is_fullscreen: - * @window: a #MetaWindow - * - * Return value: %TRUE if the window is currently fullscreen - */ -gboolean -meta_window_is_fullscreen (MetaWindow *window) -{ - return window->fullscreen; -} - -/** - * meta_window_is_screen_sized: - * @window: A #MetaWindow - * - * Return value: %TRUE if the window is occupies the - * the whole screen (all monitors). - */ -gboolean -meta_window_is_screen_sized (MetaWindow *window) -{ - MetaRectangle window_rect; - int screen_width, screen_height; - - meta_display_get_size (window->display, &screen_width, &screen_height); - meta_window_get_frame_rect (window, &window_rect); - - if (window_rect.x == 0 && window_rect.y == 0 && - window_rect.width == screen_width && window_rect.height == screen_height) - return TRUE; - - return FALSE; -} - -/** - * meta_window_is_monitor_sized: - * @window: a #MetaWindow - * - * Return value: %TRUE if the window is occupies an entire monitor or - * the whole screen. - */ -gboolean -meta_window_is_monitor_sized (MetaWindow *window) -{ - if (!window->monitor) - return FALSE; - - if (window->fullscreen) - return TRUE; - - if (meta_window_is_screen_sized (window)) - return TRUE; - - if (window->override_redirect) - { - MetaRectangle window_rect, monitor_rect; - - meta_window_get_frame_rect (window, &window_rect); - meta_display_get_monitor_geometry (window->display, window->monitor->number, &monitor_rect); - - if (meta_rectangle_equal (&window_rect, &monitor_rect)) - return TRUE; - } - - return FALSE; -} - -/** - * meta_window_is_on_primary_monitor: - * @window: a #MetaWindow - * - * Return value: %TRUE if the window is on the primary monitor - */ -gboolean -meta_window_is_on_primary_monitor (MetaWindow *window) -{ - g_return_val_if_fail (window->monitor, FALSE); - - return window->monitor->is_primary; -} - -static void -meta_window_get_tile_fraction (MetaWindow *window, - MetaTileMode tile_mode, - double *fraction) -{ - MetaWindow *tile_match; - - /* Make sure the tile match is up-to-date and matches the - * passed in mode rather than the current state - */ - tile_match = meta_window_find_tile_match (window, tile_mode); - - if (tile_mode == META_TILE_NONE) - *fraction = -1.; - else if (tile_mode == META_TILE_MAXIMIZED) - *fraction = 1.; - else if (tile_match) - *fraction = 1. - tile_match->tile_hfraction; - else if (META_WINDOW_TILED_SIDE_BY_SIDE (window)) - { - if (window->tile_mode != tile_mode) - *fraction = 1. - window->tile_hfraction; - else - *fraction = window->tile_hfraction; - } - else - *fraction = .5; -} - -static void -meta_window_update_tile_fraction (MetaWindow *window, - int new_w, - int new_h) -{ - MetaWindow *tile_match = window->tile_match; - MetaRectangle work_area; - - if (!META_WINDOW_TILED_SIDE_BY_SIDE (window)) - return; - - meta_window_get_work_area_for_monitor (window, - window->tile_monitor_number, - &work_area); - window->tile_hfraction = (double)new_w / work_area.width; - - if (tile_match && window->display->grab_window == window) - meta_window_tile (tile_match, tile_match->tile_mode); -} - -static void -update_edge_constraints (MetaWindow *window) -{ - switch (window->tile_mode) - { - case META_TILE_NONE: - window->edge_constraints.top = META_EDGE_CONSTRAINT_NONE; - window->edge_constraints.right = META_EDGE_CONSTRAINT_NONE; - window->edge_constraints.bottom = META_EDGE_CONSTRAINT_NONE; - window->edge_constraints.left = META_EDGE_CONSTRAINT_NONE; - break; - - case META_TILE_MAXIMIZED: - window->edge_constraints.top = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.right = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.bottom = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.left = META_EDGE_CONSTRAINT_MONITOR; - break; - - case META_TILE_LEFT: - window->edge_constraints.top = META_EDGE_CONSTRAINT_MONITOR; - - if (window->tile_match) - window->edge_constraints.right = META_EDGE_CONSTRAINT_WINDOW; - else - window->edge_constraints.right = META_EDGE_CONSTRAINT_NONE; - - window->edge_constraints.bottom = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.left = META_EDGE_CONSTRAINT_MONITOR; - break; - - case META_TILE_RIGHT: - window->edge_constraints.top = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.right = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.bottom = META_EDGE_CONSTRAINT_MONITOR; - - if (window->tile_match) - window->edge_constraints.left = META_EDGE_CONSTRAINT_WINDOW; - else - window->edge_constraints.left = META_EDGE_CONSTRAINT_NONE; - break; - } - - /* h/vmaximize also modify the edge constraints */ - if (window->maximized_vertically) - { - window->edge_constraints.top = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.bottom = META_EDGE_CONSTRAINT_MONITOR; - } - - if (window->maximized_horizontally) - { - window->edge_constraints.right = META_EDGE_CONSTRAINT_MONITOR; - window->edge_constraints.left = META_EDGE_CONSTRAINT_MONITOR; - } -} - -void -meta_window_untile (MetaWindow *window) -{ - window->tile_monitor_number = - window->saved_maximize ? window->monitor->number - : -1; - window->tile_mode = - window->saved_maximize ? META_TILE_MAXIMIZED - : META_TILE_NONE; - - if (window->saved_maximize) - meta_window_maximize (window, META_MAXIMIZE_BOTH); - else - meta_window_unmaximize (window, META_MAXIMIZE_BOTH); -} - -void -meta_window_tile (MetaWindow *window, - MetaTileMode tile_mode) -{ - MetaMaximizeFlags directions; - MetaRectangle old_frame_rect, old_buffer_rect; - - meta_window_get_tile_fraction (window, tile_mode, &window->tile_hfraction); - window->tile_mode = tile_mode; - - /* Don't do anything if no tiling is requested */ - if (window->tile_mode == META_TILE_NONE) - { - window->tile_monitor_number = -1; - return; - } - else if (window->tile_monitor_number < 0) - { - window->tile_monitor_number = window->monitor->number; - } - - if (window->tile_mode == META_TILE_MAXIMIZED) - directions = META_MAXIMIZE_BOTH; - else - directions = META_MAXIMIZE_VERTICAL; - - meta_window_maximize_internal (window, directions, NULL); - meta_display_update_tile_preview (window->display, FALSE); - - /* Setup the edge constraints */ - update_edge_constraints (window); - - meta_window_get_frame_rect (window, &old_frame_rect); - meta_window_get_buffer_rect (window, &old_buffer_rect); - - meta_compositor_size_change_window (window->display->compositor, window, - META_SIZE_CHANGE_MAXIMIZE, - &old_frame_rect, &old_buffer_rect); - - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_STATE_CHANGED), - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); - - if (window->frame) - meta_frame_queue_draw (window->frame); -} - -MetaTileMode -meta_window_get_tile_mode (MetaWindow *window) -{ - return window->tile_mode; -} - -void -meta_window_restore_tile (MetaWindow *window, - MetaTileMode mode, - int width, - int height) -{ - meta_window_update_tile_fraction (window, width, height); - meta_window_tile (window, mode); -} - -static gboolean -meta_window_can_tile_maximized (MetaWindow *window) -{ - return window->has_maximize_func; -} - -gboolean -meta_window_can_tile_side_by_side (MetaWindow *window) -{ - int monitor; - MetaRectangle tile_area; - MetaRectangle client_rect; - - if (!meta_window_can_tile_maximized (window)) - return FALSE; - - monitor = meta_display_get_current_monitor (window->display); - meta_window_get_work_area_for_monitor (window, monitor, &tile_area); - - /* Do not allow tiling in portrait orientation */ - if (tile_area.height > tile_area.width) - return FALSE; - - tile_area.width /= 2; - - meta_window_frame_rect_to_client_rect (window, &tile_area, &client_rect); - - return client_rect.width >= window->size_hints.min_width && - client_rect.height >= window->size_hints.min_height; -} - -static void -unmaximize_window_before_freeing (MetaWindow *window) -{ - meta_topic (META_DEBUG_WINDOW_OPS, - "Unmaximizing %s just before freeing", - window->desc); - - window->maximized_horizontally = FALSE; - window->maximized_vertically = FALSE; - - if (window->withdrawn) /* See bug #137185 */ - { - window->rect = window->saved_rect; - set_net_wm_state (window); - } -#ifdef HAVE_WAYLAND - else if (!meta_is_wayland_compositor ()) - { - /* Do NOT update net_wm_state: this screen is closing, - * it likely will be managed by another window manager - * that will need the current _NET_WM_STATE atoms. - * Moreover, it will need to know the unmaximized geometry, - * therefore move_resize the window to saved_rect here - * before closing it. */ - meta_window_move_resize_frame (window, - FALSE, - window->saved_rect.x, - window->saved_rect.y, - window->saved_rect.width, - window->saved_rect.height); - } -#endif -} - -static void -meta_window_maybe_apply_size_hints (MetaWindow *window, - MetaRectangle *target_rect) -{ - meta_window_frame_rect_to_client_rect (window, target_rect, target_rect); - ensure_size_hints_satisfied (target_rect, &window->size_hints); - meta_window_client_rect_to_frame_rect (window, target_rect, target_rect); -} - -void -meta_window_unmaximize (MetaWindow *window, - MetaMaximizeFlags directions) -{ - gboolean unmaximize_horizontally, unmaximize_vertically; - - g_return_if_fail (!window->override_redirect); - - /* At least one of the two directions ought to be set */ - unmaximize_horizontally = directions & META_MAXIMIZE_HORIZONTAL; - unmaximize_vertically = directions & META_MAXIMIZE_VERTICAL; - g_assert (unmaximize_horizontally || unmaximize_vertically); - - if (unmaximize_horizontally && unmaximize_vertically) - window->saved_maximize = FALSE; - - /* Only do something if the window isn't already maximized in the - * given direction(s). - */ - if ((unmaximize_horizontally && window->maximized_horizontally) || - (unmaximize_vertically && window->maximized_vertically)) - { - MetaRectangle *desired_rect; - MetaRectangle target_rect; - MetaRectangle work_area; - MetaRectangle old_frame_rect, old_buffer_rect; - gboolean has_target_size; - - meta_window_get_work_area_for_monitor (window, window->monitor->number, &work_area); - meta_window_get_frame_rect (window, &old_frame_rect); - meta_window_get_buffer_rect (window, &old_buffer_rect); - - if (unmaximize_vertically) - window->tile_mode = META_TILE_NONE; - - meta_topic (META_DEBUG_WINDOW_OPS, - "Unmaximizing %s%s", - window->desc, - unmaximize_horizontally && unmaximize_vertically ? "" : - unmaximize_horizontally ? " horizontally" : - unmaximize_vertically ? " vertically" : "BUGGGGG"); - - window->maximized_horizontally = - window->maximized_horizontally && !unmaximize_horizontally; - window->maximized_vertically = - window->maximized_vertically && !unmaximize_vertically; - - /* Update the edge constraints */ - update_edge_constraints (window); - - /* recalc_features() will eventually clear the cached frame - * extents, but we need the correct frame extents in the code below, - * so invalidate the old frame extents manually up front. - */ - meta_window_frame_size_changed (window); - - desired_rect = &window->saved_rect; - - /* Unmaximize to the saved_rect position in the direction(s) - * being unmaximized. - */ - target_rect = old_frame_rect; - - /* Avoid unmaximizing to "almost maximized" size when the previous size - * is greater then 80% of the work area use MAX_UNMAXIMIZED_WINDOW_AREA of the work area as upper limit - * while maintaining the aspect ratio. - */ - if (unmaximize_horizontally && unmaximize_vertically && - desired_rect->width * desired_rect->height > work_area.width * work_area.height * MAX_UNMAXIMIZED_WINDOW_AREA) - { - if (desired_rect->width > desired_rect->height) - { - float aspect = (float)desired_rect->height / (float)desired_rect->width; - desired_rect->width = MAX (work_area.width * sqrt (MAX_UNMAXIMIZED_WINDOW_AREA), window->size_hints.min_width); - desired_rect->height = MAX (desired_rect->width * aspect, window->size_hints.min_height); - } - else - { - float aspect = (float)desired_rect->width / (float)desired_rect->height; - desired_rect->height = MAX (work_area.height * sqrt (MAX_UNMAXIMIZED_WINDOW_AREA), window->size_hints.min_height); - desired_rect->width = MAX (desired_rect->height * aspect, window->size_hints.min_width); - } - } - - if (unmaximize_horizontally) - { - target_rect.x = desired_rect->x; - target_rect.width = desired_rect->width; - } - if (unmaximize_vertically) - { - target_rect.y = desired_rect->y; - target_rect.height = desired_rect->height; - } - - /* Window's size hints may have changed while maximized, making - * saved_rect invalid. #329152 - * Do not enforce limits, if no previous 'saved_rect' has been stored. - */ - has_target_size = (target_rect.width > 0 && target_rect.height > 0); - if (has_target_size) - meta_window_maybe_apply_size_hints (window, &target_rect); - - meta_compositor_size_change_window (window->display->compositor, window, - META_SIZE_CHANGE_UNMAXIMIZE, - &old_frame_rect, &old_buffer_rect); - - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_STATE_CHANGED | - META_MOVE_RESIZE_UNMAXIMIZE), - META_GRAVITY_NORTH_WEST, - target_rect); - - /* When we unmaximize, if we're doing a mouse move also we could - * get the window suddenly jumping to the upper left corner of - * the workspace, since that's where it was when the grab op - * started. So we need to update the grab anchor position. - */ - if (meta_grab_op_is_moving (window->display->grab_op) && - window->display->grab_window == window) - { - window->display->grab_anchor_window_pos = target_rect; - } - - meta_window_recalc_features (window); - set_net_wm_state (window); - if (!window->monitor->in_fullscreen) - meta_display_queue_check_fullscreen (window->display); - } - - g_object_freeze_notify (G_OBJECT (window)); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MAXIMIZED_HORIZONTALLY]); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MAXIMIZED_VERTICALLY]); - g_object_thaw_notify (G_OBJECT (window)); -} - -void -meta_window_make_above (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - meta_window_set_above (window, TRUE); - meta_window_raise (window); -} - -void -meta_window_unmake_above (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - meta_window_set_above (window, FALSE); - meta_window_raise (window); -} - -static void -meta_window_set_above (MetaWindow *window, - gboolean new_value) -{ - new_value = new_value != FALSE; - if (new_value == window->wm_state_above) - return; - - window->wm_state_above = new_value; - meta_window_update_layer (window); - set_net_wm_state (window); - meta_window_frame_size_changed (window); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_ABOVE]); -} - -void -meta_window_make_fullscreen_internal (MetaWindow *window) -{ - if (!window->fullscreen) - { - meta_topic (META_DEBUG_WINDOW_OPS, - "Fullscreening %s", window->desc); - - if (window->shaded) - { - /* Shading sucks anyway; I'm not adding a timestamp argument - * to this function just for this niche usage & corner case. - */ - guint32 timestamp = - meta_display_get_current_time_roundtrip (window->display); - meta_window_unshade (window, timestamp); - } - - window->saved_rect_fullscreen = window->rect; - - window->fullscreen = TRUE; - - meta_stack_freeze (window->display->stack); - - meta_window_raise (window); - meta_stack_thaw (window->display->stack); - - meta_window_recalc_features (window); - set_net_wm_state (window); - - /* For the auto-minimize feature, if we fail to get focus */ - meta_display_queue_check_fullscreen (window->display); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_FULLSCREEN]); - } -} - -void -meta_window_make_fullscreen (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - if (!window->fullscreen) - { - MetaRectangle old_frame_rect, old_buffer_rect; - - meta_window_get_frame_rect (window, &old_frame_rect); - meta_window_get_buffer_rect (window, &old_buffer_rect); - - meta_compositor_size_change_window (window->display->compositor, - window, META_SIZE_CHANGE_FULLSCREEN, - &old_frame_rect, &old_buffer_rect); - - meta_window_make_fullscreen_internal (window); - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_STATE_CHANGED), - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); - } -} - -void -meta_window_unmake_fullscreen (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - if (window->fullscreen) - { - MetaRectangle old_frame_rect, old_buffer_rect, target_rect; - gboolean has_target_size; - - meta_topic (META_DEBUG_WINDOW_OPS, - "Unfullscreening %s", window->desc); - - window->fullscreen = FALSE; - target_rect = window->saved_rect_fullscreen; - - meta_window_frame_size_changed (window); - meta_window_get_frame_rect (window, &old_frame_rect); - meta_window_get_buffer_rect (window, &old_buffer_rect); - - /* Window's size hints may have changed while maximized, making - * saved_rect invalid. #329152 - * Do not enforce limits, if no previous 'saved_rect' has been stored. - */ - has_target_size = (target_rect.width > 0 && target_rect.height > 0); - if (has_target_size) - meta_window_maybe_apply_size_hints (window, &target_rect); - - /* Need to update window->has_resize_func before we move_resize() - */ - meta_window_recalc_features (window); - set_net_wm_state (window); - - meta_compositor_size_change_window (window->display->compositor, - window, META_SIZE_CHANGE_UNFULLSCREEN, - &old_frame_rect, &old_buffer_rect); - - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_STATE_CHANGED | - META_MOVE_RESIZE_UNFULLSCREEN), - META_GRAVITY_NORTH_WEST, - target_rect); - - meta_display_queue_check_fullscreen (window->display); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_FULLSCREEN]); - } -} - -static void -meta_window_clear_fullscreen_monitors (MetaWindow *window) -{ - window->fullscreen_monitors.top = NULL; - window->fullscreen_monitors.bottom = NULL; - window->fullscreen_monitors.left = NULL; - window->fullscreen_monitors.right = NULL; -} - -void -meta_window_update_fullscreen_monitors (MetaWindow *window, - MetaLogicalMonitor *top, - MetaLogicalMonitor *bottom, - MetaLogicalMonitor *left, - MetaLogicalMonitor *right) -{ - if (top && bottom && left && right) - { - window->fullscreen_monitors.top = top; - window->fullscreen_monitors.bottom = bottom; - window->fullscreen_monitors.left = left; - window->fullscreen_monitors.right = right; - } - else - { - meta_window_clear_fullscreen_monitors (window); - } - - if (window->fullscreen) - { - meta_window_queue(window, META_QUEUE_MOVE_RESIZE); - } -} - -gboolean -meta_window_has_fullscreen_monitors (MetaWindow *window) -{ - return window->fullscreen_monitors.top != NULL; -} - -void -meta_window_adjust_fullscreen_monitor_rect (MetaWindow *window, - MetaRectangle *monitor_rect) -{ - MetaWindowClass *window_class = META_WINDOW_GET_CLASS (window); - - if (window_class->adjust_fullscreen_monitor_rect) - window_class->adjust_fullscreen_monitor_rect (window, monitor_rect); -} - -void -meta_window_shade (MetaWindow *window, - guint32 timestamp) -{ - g_return_if_fail (!window->override_redirect); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Shading %s", window->desc); - if (!window->shaded) - { - window->shaded = TRUE; - - meta_window_queue(window, META_QUEUE_MOVE_RESIZE | META_QUEUE_CALC_SHOWING); - meta_window_frame_size_changed (window); - - /* After queuing the calc showing, since _focus flushes it, - * and we need to focus the frame - */ - meta_topic (META_DEBUG_FOCUS, - "Re-focusing window %s after shading it", - window->desc); - meta_window_focus (window, timestamp); - - set_net_wm_state (window); - } -} - -void -meta_window_unshade (MetaWindow *window, - guint32 timestamp) -{ - g_return_if_fail (!window->override_redirect); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Unshading %s", window->desc); - if (window->shaded) - { - window->shaded = FALSE; - meta_window_queue(window, META_QUEUE_MOVE_RESIZE | META_QUEUE_CALC_SHOWING); - meta_window_frame_size_changed (window); - - /* focus the window */ - meta_topic (META_DEBUG_FOCUS, - "Focusing window %s after unshading it", - window->desc); - meta_window_focus (window, timestamp); - - set_net_wm_state (window); - } -} - -static gboolean -unminimize_func (MetaWindow *window, - void *data) -{ - meta_window_unminimize (window); - return TRUE; -} - -static void -unminimize_window_and_all_transient_parents (MetaWindow *window) -{ - meta_window_unminimize (window); - meta_window_foreach_ancestor (window, unminimize_func, NULL); -} - -void -meta_window_activate_full (MetaWindow *window, - guint32 timestamp, - MetaClientType source_indication, - MetaWorkspace *workspace) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean allow_workspace_switch; - - if (window->unmanaging) - { - g_warning ("Trying to activate unmanaged window '%s'", window->desc); - return; - } - - meta_topic (META_DEBUG_FOCUS, - "_NET_ACTIVE_WINDOW message sent for %s at time %u " - "by client type %u.", - window->desc, timestamp, source_indication); - - allow_workspace_switch = (timestamp != 0); - if (timestamp != 0 && - XSERVER_TIME_IS_BEFORE (timestamp, window->display->last_user_time)) - { - meta_topic (META_DEBUG_FOCUS, - "last_user_time (%u) is more recent; ignoring " - " _NET_ACTIVE_WINDOW message.", - window->display->last_user_time); - meta_window_set_demands_attention(window); - return; - } - - if (timestamp == 0) - timestamp = meta_display_get_current_time_roundtrip (window->display); - - meta_window_set_user_time (window, timestamp); - - /* disable show desktop mode unless we're a desktop component */ - maybe_leave_show_desktop_mode (window); - - /* Get window on current or given workspace */ - if (workspace == NULL) - workspace = workspace_manager->active_workspace; - - /* For non-transient windows, we just set up a pulsing indicator, - rather than move windows or workspaces. - See http://bugzilla.gnome.org/show_bug.cgi?id=482354 */ - if (window->transient_for == NULL && - !allow_workspace_switch && - !meta_window_located_on_workspace (window, workspace)) - { - meta_window_set_demands_attention (window); - /* We've marked it as demanding, don't need to do anything else. */ - return; - } - else if (window->transient_for != NULL) - { - /* Move transients to current workspace - preference dialogs should appear over - the source window. */ - meta_window_change_workspace (window, workspace); - } - - if (window->shaded) - meta_window_unshade (window, timestamp); - - unminimize_window_and_all_transient_parents (window); - - if (meta_prefs_get_raise_on_click () || - source_indication == META_CLIENT_TYPE_PAGER) - meta_window_raise (window); - - meta_topic (META_DEBUG_FOCUS, - "Focusing window %s due to activation", - window->desc); - - if (meta_window_located_on_workspace (window, workspace)) - meta_window_focus (window, timestamp); - else - meta_workspace_activate_with_focus (window->workspace, window, timestamp); - - meta_window_check_alive (window, timestamp); -} - -/* This function exists since most of the functionality in window_activate - * is useful for Mutter, but Mutter shouldn't need to specify a client - * type for itself. ;-) - */ -void -meta_window_activate (MetaWindow *window, - guint32 timestamp) -{ - g_return_if_fail (!window->override_redirect); - - /* We're not really a pager, but the behavior we want is the same as if - * we were such. If we change the pager behavior later, we could revisit - * this and just add extra flags to window_activate. - */ - meta_window_activate_full (window, timestamp, META_CLIENT_TYPE_PAGER, NULL); -} - -void -meta_window_activate_with_workspace (MetaWindow *window, - guint32 timestamp, - MetaWorkspace *workspace) -{ - g_return_if_fail (!window->override_redirect); - - meta_window_activate_full (window, timestamp, META_CLIENT_TYPE_APPLICATION, workspace); -} - -/** - * meta_window_updates_are_frozen: - * @window: a #MetaWindow - * - * Gets whether the compositor should be updating the window contents; - * window content updates may be frozen at client request by setting - * an odd value in the extended _NET_WM_SYNC_REQUEST_COUNTER counter - * by the window manager during a resize operation while waiting for - * the client to redraw. - * - * Return value: %TRUE if updates are currently frozen - */ -gboolean -meta_window_updates_are_frozen (MetaWindow *window) -{ - return META_WINDOW_GET_CLASS (window)->are_updates_frozen (window); -} - -static void -meta_window_reposition (MetaWindow *window) -{ - meta_window_move_resize_internal (window, - (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION), - META_GRAVITY_NORTH_WEST, - window->rect); -} - -static gboolean -maybe_move_attached_window (MetaWindow *window, - void *data) -{ - if (window->hidden) - return G_SOURCE_CONTINUE; - - if (meta_window_is_attached_dialog (window) || - meta_window_get_placement_rule (window)) - meta_window_reposition (window); - - return G_SOURCE_CONTINUE; -} - -/** - * meta_window_get_monitor: - * @window: a #MetaWindow - * - * Gets index of the monitor that this window is on. - * - * Return Value: The index of the monitor in the screens monitor list, or -1 - * if the window has been recently unmanaged and does not have a monitor. - */ -int -meta_window_get_monitor (MetaWindow *window) -{ - if (!window->monitor) - return -1; - - return window->monitor->number; -} - -MetaLogicalMonitor * -meta_window_get_main_logical_monitor (MetaWindow *window) -{ - return window->monitor; -} - -static MetaLogicalMonitor * -find_monitor_by_winsys_id (MetaWindow *window, - uint64_t winsys_id) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - GList *logical_monitors, *l; - - logical_monitors = - meta_monitor_manager_get_logical_monitors (monitor_manager); - - for (l = logical_monitors; l; l = l->next) - { - MetaLogicalMonitor *logical_monitor = l->data; - - if (logical_monitor->winsys_id == winsys_id) - return logical_monitor; - } - - return NULL; -} - -/* This is called when the monitor setup has changed. The window->monitor - * reference is still "valid", but refer to the previous monitor setup */ -void -meta_window_update_for_monitors_changed (MetaWindow *window) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - const MetaLogicalMonitor *old, *new; - - if (meta_window_has_fullscreen_monitors (window)) - meta_window_clear_fullscreen_monitors (window); - - if (window->override_redirect || window->type == META_WINDOW_DESKTOP) - { - meta_window_update_monitor (window, - META_WINDOW_UPDATE_MONITOR_FLAGS_FORCE); - goto out; - } - - old = window->monitor; - - /* Try the preferred output first */ - new = find_monitor_by_winsys_id (window, window->preferred_output_winsys_id); - - /* Otherwise, try to find the old output on a new monitor */ - if (old && !new) - new = find_monitor_by_winsys_id (window, old->winsys_id); - - /* Fall back to primary if everything else failed */ - if (!new) - new = meta_monitor_manager_get_primary_logical_monitor (monitor_manager); - - if (window->tile_mode != META_TILE_NONE) - { - if (new) - window->tile_monitor_number = new->number; - else - window->tile_monitor_number = -1; - } - - if (new && old) - { - /* This will eventually reach meta_window_update_monitor that - * will send leave/enter-monitor events. The old != new monitor - * check will always fail (due to the new logical_monitors set) so - * we will always send the events, even if the new and old monitor - * index is the same. That is right, since the enumeration of the - * monitors changed and the same index could be refereing - * to a different monitor. */ - meta_window_move_between_rects (window, - META_MOVE_RESIZE_FORCE_UPDATE_MONITOR, - &old->rect, - &new->rect); - } - else - { - meta_window_update_monitor (window, - META_WINDOW_UPDATE_MONITOR_FLAGS_FORCE); - } - -out: - g_assert (!window->monitor || - g_list_find (meta_monitor_manager_get_logical_monitors (monitor_manager), - window->monitor)); -} - -void -meta_window_update_monitor (MetaWindow *window, - MetaWindowUpdateMonitorFlags flags) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - const MetaLogicalMonitor *old; - - old = window->monitor; - META_WINDOW_GET_CLASS (window)->update_main_monitor (window, flags); - if (old != window->monitor) - { - meta_window_on_all_workspaces_changed (window); - - /* If workspaces only on primary and we moved back to primary due to a user action, - * ensure that the window is now in that workspace. We do this because while - * the window is on a non-primary monitor it is always visible, so it would be - * very jarring if it disappeared when it crossed the monitor border. - * The one time we want it to both change to the primary monitor and a non-active - * workspace is when dropping the window on some other workspace thumbnail directly. - * That should be handled by explicitly moving the window before changing the - * workspace. - */ - if (meta_prefs_get_workspaces_only_on_primary () && - flags & META_WINDOW_UPDATE_MONITOR_FLAGS_USER_OP && - meta_window_is_on_primary_monitor (window) && - workspace_manager->active_workspace != window->workspace) - meta_window_change_workspace (window, workspace_manager->active_workspace); - - meta_window_main_monitor_changed (window, old); - - /* If we're changing monitors, we need to update the has_maximize_func flag, - * as the working area has changed. */ - meta_window_recalc_features (window); - } -} - -void -meta_window_move_resize_internal (MetaWindow *window, - MetaMoveResizeFlags flags, - MetaGravity gravity, - MetaRectangle frame_rect) -{ - /* The rectangle here that's passed in *always* in "frame rect" - * coordinates. That means the position of the frame's visible bounds, - * with x and y being absolute (root window) coordinates. - * - * For an X11 framed window, the client window's server rectangle is - * inset from this rectangle by the frame's visible borders, and the - * frame window's server rectangle is outset by the invisible borders. - * - * For an X11 unframed window, the rectangle here directly matches - * the server's rectangle, since the visible and invisible borders - * are both 0. - * - * For an X11 CSD window, the client window's server rectangle is - * outset from this rectagle by the client-specified frame extents. - * - * For a Wayland window, this rectangle can simply be sent directly - * to the client. - */ - - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean did_placement; - MetaRectangle unconstrained_rect; - MetaRectangle constrained_rect; - MetaRectangle temporary_rect; - int rel_x = 0; - int rel_y = 0; - MetaMoveResizeResultFlags result = 0; - gboolean moved_or_resized = FALSE; - MetaWindowUpdateMonitorFlags update_monitor_flags; - - g_return_if_fail (!window->override_redirect); - - /* The action has to be a move, a resize or the wayland client - * acking our choice of size. - */ - g_assert (flags & (META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION | - META_MOVE_RESIZE_WAYLAND_FINISH_MOVE_RESIZE)); - - did_placement = !window->placed && window->calc_placement; - - /* We don't need it in the idle queue anymore. */ - meta_window_unqueue (window, META_QUEUE_MOVE_RESIZE); - - if ((flags & META_MOVE_RESIZE_RESIZE_ACTION) && (flags & META_MOVE_RESIZE_MOVE_ACTION)) - { - /* We're both moving and resizing. Just use the passed in rect. */ - unconstrained_rect = frame_rect; - } - else if ((flags & META_MOVE_RESIZE_RESIZE_ACTION)) - { - /* If this is only a resize, then ignore the position given in - * the parameters and instead calculate the new position from - * resizing the old rectangle with the given gravity. */ - meta_rectangle_resize_with_gravity (&window->rect, - &unconstrained_rect, - gravity, - frame_rect.width, - frame_rect.height); - } - else if ((flags & META_MOVE_RESIZE_MOVE_ACTION)) - { - /* If this is only a move, then ignore the passed in size and - * just use the existing size of the window. */ - unconstrained_rect.x = frame_rect.x; - unconstrained_rect.y = frame_rect.y; - unconstrained_rect.width = window->rect.width; - unconstrained_rect.height = window->rect.height; - } - else if ((flags & META_MOVE_RESIZE_WAYLAND_FINISH_MOVE_RESIZE)) - { - /* This is a Wayland buffer acking our size. The new rect is - * just the existing one we have. Ignore the passed-in rect - * completely. */ - unconstrained_rect = window->rect; - } - else - g_assert_not_reached (); - - constrained_rect = unconstrained_rect; - temporary_rect = window->rect; - if (flags & (META_MOVE_RESIZE_MOVE_ACTION | META_MOVE_RESIZE_RESIZE_ACTION) && - !(flags & META_MOVE_RESIZE_WAYLAND_FINISH_MOVE_RESIZE) && - !(flags & (META_MOVE_RESIZE_UNMAXIMIZE | META_MOVE_RESIZE_UNFULLSCREEN)) && - window->monitor) - { - MetaRectangle old_rect; - meta_window_get_frame_rect (window, &old_rect); - - meta_window_constrain (window, - flags, - gravity, - &old_rect, - &constrained_rect, - &temporary_rect, - &rel_x, - &rel_y); - } - else if (window->placement.rule) - { - rel_x = window->placement.pending.rel_x; - rel_y = window->placement.pending.rel_y; - } - - /* If we did placement, then we need to save the position that the window - * was placed at to make sure that meta_window_move_resize_now places the - * window correctly. - */ - if (did_placement) - { - unconstrained_rect.x = constrained_rect.x; - unconstrained_rect.y = constrained_rect.y; - } - - /* Do the protocol-specific move/resize logic */ - META_WINDOW_GET_CLASS (window)->move_resize_internal (window, - gravity, - unconstrained_rect, - constrained_rect, - temporary_rect, - rel_x, - rel_y, - flags, &result); - - if (result & META_MOVE_RESIZE_RESULT_MOVED) - { - moved_or_resized = TRUE; - g_signal_emit (window, window_signals[POSITION_CHANGED], 0); - } - - if (result & META_MOVE_RESIZE_RESULT_RESIZED) - { - moved_or_resized = TRUE; - g_signal_emit (window, window_signals[SIZE_CHANGED], 0); - } - - if (moved_or_resized || did_placement) - window->unconstrained_rect = unconstrained_rect; - - if ((moved_or_resized || - did_placement || - (result & META_MOVE_RESIZE_RESULT_STATE_CHANGED) != 0) && - window->known_to_compositor) - { - meta_compositor_sync_window_geometry (window->display->compositor, - window, - did_placement); - } - - update_monitor_flags = META_WINDOW_UPDATE_MONITOR_FLAGS_NONE; - if (flags & META_MOVE_RESIZE_USER_ACTION) - update_monitor_flags |= META_WINDOW_UPDATE_MONITOR_FLAGS_USER_OP; - if (flags & META_MOVE_RESIZE_FORCE_UPDATE_MONITOR) - update_monitor_flags |= META_WINDOW_UPDATE_MONITOR_FLAGS_FORCE; - - if (window->monitor) - { - uint64_t old_output_winsys_id; - - old_output_winsys_id = window->monitor->winsys_id; - - meta_window_update_monitor (window, update_monitor_flags); - - if (old_output_winsys_id != window->monitor->winsys_id && - flags & META_MOVE_RESIZE_MOVE_ACTION && flags & META_MOVE_RESIZE_USER_ACTION) - window->preferred_output_winsys_id = window->monitor->winsys_id; - } - else - { - meta_window_update_monitor (window, update_monitor_flags); - } - - if ((result & META_MOVE_RESIZE_RESULT_FRAME_SHAPE_CHANGED) && window->frame_bounds) - { - cairo_region_destroy (window->frame_bounds); - window->frame_bounds = NULL; - } - - meta_window_foreach_transient (window, maybe_move_attached_window, NULL); - - meta_stack_update_window_tile_matches (window->display->stack, - workspace_manager->active_workspace); -} - -/** - * meta_window_move_frame: - * @window: a #MetaWindow - * @user_op: bool to indicate whether or not this is a user operation - * @root_x_nw: desired x pos - * @root_y_nw: desired y pos - * - * Moves the window to the desired location on window's assigned - * workspace, using the northwest edge of the frame as the reference, - * instead of the actual window's origin, but only if a frame is present. - * Otherwise, acts identically to meta_window_move(). - */ -void -meta_window_move_frame (MetaWindow *window, - gboolean user_op, - int root_x_nw, - int root_y_nw) -{ - MetaMoveResizeFlags flags; - MetaRectangle rect = { root_x_nw, root_y_nw, 0, 0 }; - - g_return_if_fail (!window->override_redirect); - - flags = (user_op ? META_MOVE_RESIZE_USER_ACTION : 0) | META_MOVE_RESIZE_MOVE_ACTION; - meta_window_move_resize_internal (window, flags, META_GRAVITY_NORTH_WEST, rect); -} - -static void -meta_window_move_between_rects (MetaWindow *window, - MetaMoveResizeFlags move_resize_flags, - const MetaRectangle *old_area, - const MetaRectangle *new_area) -{ - int rel_x, rel_y; - double scale_x, scale_y; - - if (old_area) - { - rel_x = window->unconstrained_rect.x - old_area->x; - rel_y = window->unconstrained_rect.y - old_area->y; - scale_x = (double)new_area->width / old_area->width; - scale_y = (double)new_area->height / old_area->height; - } - else - { - rel_x = rel_y = scale_x = scale_y = 0; - } - - window->unconstrained_rect.x = new_area->x + rel_x * scale_x; - window->unconstrained_rect.y = new_area->y + rel_y * scale_y; - window->saved_rect.x = window->unconstrained_rect.x; - window->saved_rect.y = window->unconstrained_rect.y; - - meta_window_move_resize_internal (window, - move_resize_flags | - META_MOVE_RESIZE_MOVE_ACTION | - META_MOVE_RESIZE_RESIZE_ACTION, - META_GRAVITY_NORTH_WEST, - window->unconstrained_rect); -} - -/** - * meta_window_move_resize_frame: - * @window: a #MetaWindow - * @user_op: bool to indicate whether or not this is a user operation - * @root_x_nw: new x - * @root_y_nw: new y - * @w: desired width - * @h: desired height - * - * Resizes the window so that its outer bounds (including frame) - * fit within the given rect - */ -void -meta_window_move_resize_frame (MetaWindow *window, - gboolean user_op, - int root_x_nw, - int root_y_nw, - int w, - int h) -{ - MetaMoveResizeFlags flags; - MetaRectangle rect = { root_x_nw, root_y_nw, w, h }; - - g_return_if_fail (!window->override_redirect); - - flags = (user_op ? META_MOVE_RESIZE_USER_ACTION : 0) | META_MOVE_RESIZE_MOVE_ACTION | META_MOVE_RESIZE_RESIZE_ACTION; - - meta_window_move_resize_internal (window, flags, META_GRAVITY_NORTH_WEST, rect); -} - -/** - * meta_window_move_to_monitor: - * @window: a #MetaWindow - * @monitor: desired monitor index - * - * Moves the window to the monitor with index @monitor, keeping - * the relative position of the window's top left corner. - */ -void -meta_window_move_to_monitor (MetaWindow *window, - int monitor) -{ - MetaRectangle old_area, new_area; - - if (window->tile_mode != META_TILE_NONE) - window->tile_monitor_number = monitor; - - meta_window_get_work_area_for_monitor (window, - window->monitor->number, - &old_area); - meta_window_get_work_area_for_monitor (window, - monitor, - &new_area); - - if (window->unconstrained_rect.width == 0 || - window->unconstrained_rect.height == 0 || - !meta_rectangle_overlap (&window->unconstrained_rect, &old_area)) - { - meta_window_move_between_rects (window, 0, NULL, &new_area); - } - else - { - if (monitor == window->monitor->number) - return; - - meta_window_move_between_rects (window, 0, &old_area, &new_area); - } - - window->preferred_output_winsys_id = window->monitor->winsys_id; - - if (window->fullscreen || window->override_redirect) - meta_display_queue_check_fullscreen (window->display); -} - -static void -adjust_size_for_tile_match (MetaWindow *window, - int *new_w, - int *new_h) -{ - MetaRectangle work_area, rect; - MetaWindow *tile_match = window->tile_match; - - if (!META_WINDOW_TILED_SIDE_BY_SIDE (window) || !tile_match) - return; - - meta_window_get_work_area_for_monitor (window, window->tile_monitor_number, &work_area); - - /* Make sure the resize does not break minimum sizes */ - rect = work_area; - rect.width = *new_w; - - meta_window_frame_rect_to_client_rect (window, &rect, &rect); - *new_w += MAX(0, window->size_hints.min_width - rect.width); - - /* Make sure we're not resizing the tile match below its min width */ - rect = work_area; - rect.width = work_area.width - *new_w; - - meta_window_frame_rect_to_client_rect (tile_match, &rect, &rect); - *new_w -= MAX(0, tile_match->size_hints.min_width - rect.width); -} - -void -meta_window_resize_frame_with_gravity (MetaWindow *window, - gboolean user_op, - int w, - int h, - MetaGravity gravity) -{ - MetaMoveResizeFlags flags; - MetaRectangle rect; - - rect.width = w; - rect.height = h; - - if (user_op) - { - /* When resizing in-tandem with a tile match, we need to respect - * its minimum width - */ - if (window->display->grab_window == window) - adjust_size_for_tile_match (window, &w, &h); - meta_window_update_tile_fraction (window, w, h); - } - - flags = (user_op ? META_MOVE_RESIZE_USER_ACTION : 0) | META_MOVE_RESIZE_RESIZE_ACTION; - meta_window_move_resize_internal (window, flags, gravity, rect); -} - -static void -meta_window_move_resize_now (MetaWindow *window) -{ - meta_window_move_resize_frame (window, FALSE, - window->unconstrained_rect.x, - window->unconstrained_rect.y, - window->unconstrained_rect.width, - window->unconstrained_rect.height); -} - -static gboolean -idle_move_resize (gpointer data) -{ - GSList *tmp; - GSList *copy; - guint queue_index = GPOINTER_TO_INT (data); - - meta_topic (META_DEBUG_GEOMETRY, "Clearing the move_resize queue"); - - /* Work with a copy, for reentrancy. The allowed reentrancy isn't - * complete; destroying a window while we're in here would result in - * badness. But it's OK to queue/unqueue move_resizes. - */ - copy = g_slist_copy (queue_pending[queue_index]); - g_slist_free (queue_pending[queue_index]); - queue_pending[queue_index] = NULL; - queue_later[queue_index] = 0; - - destroying_windows_disallowed += 1; - - tmp = copy; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - /* As a side effect, sets window->move_resize_queued = FALSE */ - meta_window_move_resize_now (window); - - tmp = tmp->next; - } - - g_slist_free (copy); - - destroying_windows_disallowed -= 1; - - return FALSE; -} - -void -meta_window_get_gravity_position (MetaWindow *window, - MetaGravity gravity, - int *root_x, - int *root_y) -{ - MetaRectangle frame_extents; - int w, h; - int x, y; - - w = window->rect.width; - h = window->rect.height; - - if (gravity == META_GRAVITY_STATIC) - { - frame_extents = window->rect; - if (window->frame) - { - frame_extents.x = window->frame->rect.x + window->frame->child_x; - frame_extents.y = window->frame->rect.y + window->frame->child_y; - } - } - else - { - if (window->frame == NULL) - frame_extents = window->rect; - else - frame_extents = window->frame->rect; - } - - x = frame_extents.x; - y = frame_extents.y; - - switch (gravity) - { - case META_GRAVITY_NORTH: - case META_GRAVITY_CENTER: - case META_GRAVITY_SOUTH: - /* Find center of frame. */ - x += frame_extents.width / 2; - /* Center client window on that point. */ - x -= w / 2; - break; - - case META_GRAVITY_SOUTH_EAST: - case META_GRAVITY_EAST: - case META_GRAVITY_NORTH_EAST: - /* Find right edge of frame */ - x += frame_extents.width; - /* Align left edge of client at that point. */ - x -= w; - break; - default: - break; - } - - switch (gravity) - { - case META_GRAVITY_WEST: - case META_GRAVITY_CENTER: - case META_GRAVITY_EAST: - /* Find center of frame. */ - y += frame_extents.height / 2; - /* Center client window there. */ - y -= h / 2; - break; - case META_GRAVITY_SOUTH_WEST: - case META_GRAVITY_SOUTH: - case META_GRAVITY_SOUTH_EAST: - /* Find south edge of frame */ - y += frame_extents.height; - /* Place bottom edge of client there */ - y -= h; - break; - default: - break; - } - - if (root_x) - *root_x = x; - if (root_y) - *root_y = y; -} - -void -meta_window_get_session_geometry (MetaWindow *window, - int *x, - int *y, - int *width, - int *height) -{ - meta_window_get_gravity_position (window, - window->size_hints.win_gravity, - x, y); - - *width = (window->rect.width - window->size_hints.base_width) / - window->size_hints.width_inc; - *height = (window->rect.height - window->size_hints.base_height) / - window->size_hints.height_inc; -} - -/** - * meta_window_get_buffer_rect: - * @window: a #MetaWindow - * @rect: (out): pointer to an allocated #MetaRectangle - * - * Gets the rectangle that the pixmap or buffer of @window occupies. - * - * For X11 windows, this is the server-side geometry of the toplevel - * window. - * - * For Wayland windows, this is the bounding rectangle of the attached - * buffer. - */ -void -meta_window_get_buffer_rect (const MetaWindow *window, - MetaRectangle *rect) -{ - *rect = window->buffer_rect; -} - -/** - * meta_window_client_rect_to_frame_rect: - * @window: a #MetaWindow - * @client_rect: client rectangle in root coordinates - * @frame_rect: (out): location to store the computed corresponding frame bounds. - * - * Converts a desired bounds of the client window into the corresponding bounds - * of the window frame (excluding invisible borders and client side shadows.) - */ -void -meta_window_client_rect_to_frame_rect (MetaWindow *window, - MetaRectangle *client_rect, - MetaRectangle *frame_rect) -{ - if (!frame_rect) - return; - - *frame_rect = *client_rect; - - /* The support for G_MAXINT here to mean infinity is a convenience for - * constraints.c:get_size_limits() and not something that we provide - * in other locations or document. - */ - if (window->frame) - { - MetaFrameBorders borders; - meta_frame_calc_borders (window->frame, &borders); - - frame_rect->x -= borders.visible.left; - frame_rect->y -= borders.visible.top; - if (frame_rect->width != G_MAXINT) - frame_rect->width += borders.visible.left + borders.visible.right; - if (frame_rect->height != G_MAXINT) - frame_rect->height += borders.visible.top + borders.visible.bottom; - } - else - { - const GtkBorder *extents = &window->custom_frame_extents; - frame_rect->x += extents->left; - frame_rect->y += extents->top; - if (frame_rect->width != G_MAXINT) - frame_rect->width -= extents->left + extents->right; - if (frame_rect->height != G_MAXINT) - frame_rect->height -= extents->top + extents->bottom; - } -} - -/** - * meta_window_frame_rect_to_client_rect: - * @window: a #MetaWindow - * @frame_rect: desired frame bounds for the window - * @client_rect: (out): location to store the computed corresponding client rectangle. - * - * Converts a desired frame bounds for a window into the bounds of the client - * window. - */ -void -meta_window_frame_rect_to_client_rect (MetaWindow *window, - MetaRectangle *frame_rect, - MetaRectangle *client_rect) -{ - if (!client_rect) - return; - - *client_rect = *frame_rect; - - if (window->frame) - { - MetaFrameBorders borders; - meta_frame_calc_borders (window->frame, &borders); - - client_rect->x += borders.visible.left; - client_rect->y += borders.visible.top; - client_rect->width -= borders.visible.left + borders.visible.right; - client_rect->height -= borders.visible.top + borders.visible.bottom; - } - else - { - const GtkBorder *extents = &window->custom_frame_extents; - client_rect->x -= extents->left; - client_rect->y -= extents->top; - client_rect->width += extents->left + extents->right; - client_rect->height += extents->top + extents->bottom; - } -} - -/** - * meta_window_get_frame_rect: - * @window: a #MetaWindow - * @rect: (out): pointer to an allocated #MetaRectangle - * - * Gets the rectangle that bounds @window that is what the user thinks of - * as the edge of the window. This doesn't include any extra reactive - * area that we or the client adds to the window, or any area that the - * client adds to draw a client-side shadow. - */ -void -meta_window_get_frame_rect (const MetaWindow *window, - MetaRectangle *rect) -{ - *rect = window->rect; -} - -/** - * meta_window_get_client_area_rect: - * @window: a #MetaWindow - * @rect: (out): pointer to a cairo rectangle - * - * Gets the rectangle for the boundaries of the client area, relative - * to the buffer rect. If the window is shaded, the height of the - * rectangle is 0. - */ -void -meta_window_get_client_area_rect (const MetaWindow *window, - cairo_rectangle_int_t *rect) -{ - MetaFrameBorders borders; - - meta_frame_calc_borders (window->frame, &borders); - - rect->x = borders.total.left; - rect->y = borders.total.top; - - rect->width = window->buffer_rect.width - borders.total.left - borders.total.right; - if (window->shaded) - rect->height = 0; - else - rect->height = window->buffer_rect.height - borders.total.top - borders.total.bottom; -} - -void -meta_window_get_titlebar_rect (MetaWindow *window, - MetaRectangle *rect) -{ - meta_window_get_frame_rect (window, rect); - - /* The returned rectangle is relative to the frame rect. */ - rect->x = 0; - rect->y = 0; - - if (window->frame) - { - rect->height = window->frame->child_y; - } - else - { - /* Pick an arbitrary height for a titlebar. We might want to - * eventually have CSD windows expose their borders to us. */ - rect->height = 50; - } -} - -const char* -meta_window_get_startup_id (MetaWindow *window) -{ - if (window->startup_id == NULL) - { - MetaGroup *group; - - group = meta_window_get_group (window); - - if (group != NULL) - return meta_group_get_startup_id (group); - } - - return window->startup_id; -} - -static MetaWindow* -get_modal_transient (MetaWindow *window) -{ - GSList *windows; - GSList *tmp; - MetaWindow *modal_transient; - - /* A window can't be the transient of itself, but this is just for - * convenience in the loop below; we manually fix things up at the - * end if no real modal transient was found. - */ - modal_transient = window; - - windows = meta_display_list_windows (window->display, META_LIST_DEFAULT); - tmp = windows; - while (tmp != NULL) - { - MetaWindow *transient = tmp->data; - - if (transient->transient_for == modal_transient && - transient->type == META_WINDOW_MODAL_DIALOG) - { - modal_transient = transient; - tmp = windows; - continue; - } - - tmp = tmp->next; - } - - g_slist_free (windows); - - if (window == modal_transient) - modal_transient = NULL; - - return modal_transient; -} - -static gboolean -meta_window_transient_can_focus (MetaWindow *window) -{ -#ifdef HAVE_WAYLAND - if (window->client_type == META_WINDOW_CLIENT_TYPE_WAYLAND) - return meta_wayland_surface_get_buffer (window->surface) != NULL; -#endif - - return TRUE; -} - -/* XXX META_EFFECT_FOCUS */ -void -meta_window_focus (MetaWindow *window, - guint32 timestamp) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - MetaWindow *modal_transient; - - g_return_if_fail (!window->override_redirect); - - /* This is a oneshot flag */ - window->restore_focus_on_map = FALSE; - - meta_topic (META_DEBUG_FOCUS, - "Setting input focus to window %s, input: %d focusable: %d", - window->desc, window->input, meta_window_is_focusable (window)); - - if (window->display->grab_window && - window->display->grab_window != window && - window->display->grab_window->all_keys_grabbed && - !window->display->grab_window->unmanaging) - { - meta_topic (META_DEBUG_FOCUS, - "Current focus window %s has global keygrab, not focusing window %s after all", - window->display->grab_window->desc, window->desc); - return; - } - - modal_transient = get_modal_transient (window); - if (modal_transient != NULL && - !modal_transient->unmanaging && - meta_window_transient_can_focus (modal_transient)) - { - meta_topic (META_DEBUG_FOCUS, - "%s has %s as a modal transient, so focusing it instead.", - window->desc, modal_transient->desc); - if (!meta_window_located_on_workspace (modal_transient, workspace_manager->active_workspace)) - meta_window_change_workspace (modal_transient, workspace_manager->active_workspace); - window = modal_transient; - } - - meta_window_flush_calc_showing (window); - - if ((!window->mapped || window->hidden) && !window->shaded) - { - meta_topic (META_DEBUG_FOCUS, - "Window %s is not showing, not focusing after all", - window->desc); - return; - } - - META_WINDOW_GET_CLASS (window)->focus (window, timestamp); - - if (window->display->event_route == META_EVENT_ROUTE_NORMAL) - { - MetaBackend *backend = meta_get_backend (); - ClutterStage *stage = CLUTTER_STAGE (meta_backend_get_stage (backend)); - clutter_stage_set_key_focus (stage, NULL); - } - - if (window->close_dialog && - meta_close_dialog_is_visible (window->close_dialog)) - meta_close_dialog_focus (window->close_dialog); - - if (window->wm_state_demands_attention) - meta_window_unset_demands_attention(window); - -/* meta_effect_run_focus(window, NULL, NULL); */ -} - -/* Workspace management. Invariants: - * - * - window->workspace describes the workspace the window is on. - * - * - workspace->windows is a list of windows that is located on - * that workspace. - * - * - If the window is on_all_workspaces, then - * window->workspace == NULL, but workspace->windows contains - * the window. - */ - -static void -set_workspace_state (MetaWindow *window, - gboolean on_all_workspaces, - MetaWorkspace *workspace) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - - /* If we're on all workspaces, then our new workspace must be NULL, - * otherwise it must be set, unless we're unmanaging. */ - if (on_all_workspaces) - g_assert_null (workspace); - else - g_assert_true (window->unmanaging || workspace != NULL); - - /* If this is an override-redirect window, ensure that the only - * times we're setting the workspace state is either during construction - * to mark as on_all_workspaces, or when unmanaging to remove all the - * workspaces. */ - if (window->override_redirect) - g_return_if_fail ((window->constructing && on_all_workspaces) || window->unmanaging); - - if (on_all_workspaces == window->on_all_workspaces && - workspace == window->workspace && - !window->constructing) - return; - - if (window->workspace) - meta_workspace_remove_window (window->workspace, window); - else if (window->on_all_workspaces) - { - GList *l; - for (l = workspace_manager->workspaces; l != NULL; l = l->next) - { - MetaWorkspace *ws = l->data; - meta_workspace_remove_window (ws, window); - } - } - - window->on_all_workspaces = on_all_workspaces; - window->workspace = workspace; - - if (window->workspace) - meta_workspace_add_window (window->workspace, window); - else if (window->on_all_workspaces) - { - GList *l; - for (l = workspace_manager->workspaces; l != NULL; l = l->next) - { - MetaWorkspace *ws = l->data; - meta_workspace_add_window (ws, window); - } - } - - /* queue a move_resize since changing workspaces may change - * the relevant struts - */ - if (!window->override_redirect) - meta_window_queue (window, META_QUEUE_MOVE_RESIZE); - meta_window_queue (window, META_QUEUE_CALC_SHOWING); - meta_window_current_workspace_changed (window); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_ON_ALL_WORKSPACES]); - g_signal_emit (window, window_signals[WORKSPACE_CHANGED], 0); -} - -static gboolean -should_be_on_all_workspaces (MetaWindow *window) -{ - if (window->always_sticky) - return TRUE; - - if (window->on_all_workspaces_requested) - return TRUE; - - if (window->override_redirect) - return TRUE; - - if (meta_prefs_get_workspaces_only_on_primary () && - !window->unmanaging && - window->monitor && - !meta_window_is_on_primary_monitor (window)) - return TRUE; - - return FALSE; -} - -void -meta_window_on_all_workspaces_changed (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - gboolean on_all_workspaces = should_be_on_all_workspaces (window); - - if (window->on_all_workspaces == on_all_workspaces) - return; - - MetaWorkspace *workspace; - - if (on_all_workspaces) - { - workspace = NULL; - } - else - { - /* We're coming out of the sticky state. Put the window on - * the currently active workspace. */ - workspace = workspace_manager->active_workspace; - } - - set_workspace_state (window, on_all_workspaces, workspace); -} - -static void -meta_window_change_workspace_without_transients (MetaWindow *window, - MetaWorkspace *workspace) -{ - /* Try to unstick the window if it's stuck. This doesn't - * have any guarantee that we'll actually unstick the - * window, since it could be stuck for other reasons. */ - if (window->on_all_workspaces_requested) - meta_window_unstick (window); - - /* We failed to unstick the window. */ - if (window->on_all_workspaces) - return; - - if (window->workspace == workspace) - return; - - set_workspace_state (window, FALSE, workspace); -} - -static gboolean -change_workspace_foreach (MetaWindow *window, - void *data) -{ - meta_window_change_workspace_without_transients (window, data); - return TRUE; -} - -void -meta_window_change_workspace (MetaWindow *window, - MetaWorkspace *workspace) -{ - g_return_if_fail (!window->override_redirect); - - meta_window_change_workspace_without_transients (window, workspace); - - meta_window_foreach_transient (window, change_workspace_foreach, - workspace); - meta_window_foreach_ancestor (window, change_workspace_foreach, - workspace); -} - -static void -window_stick_impl (MetaWindow *window) -{ - meta_verbose ("Sticking window %s current on_all_workspaces = %d", - window->desc, window->on_all_workspaces); - - if (window->on_all_workspaces_requested) - return; - - /* We don't change window->workspaces, because we revert - * to that original workspace list if on_all_workspaces is - * toggled back off. - */ - window->on_all_workspaces_requested = TRUE; - meta_window_on_all_workspaces_changed (window); -} - -static void -window_unstick_impl (MetaWindow *window) -{ - if (!window->on_all_workspaces_requested) - return; - - /* Revert to window->workspaces */ - - window->on_all_workspaces_requested = FALSE; - meta_window_on_all_workspaces_changed (window); -} - -static gboolean -stick_foreach_func (MetaWindow *window, - void *data) -{ - gboolean stick; - - stick = *(gboolean*)data; - if (stick) - window_stick_impl (window); - else - window_unstick_impl (window); - return TRUE; -} - -void -meta_window_stick (MetaWindow *window) -{ - gboolean stick = TRUE; - - g_return_if_fail (!window->override_redirect); - - window_stick_impl (window); - meta_window_foreach_transient (window, - stick_foreach_func, - &stick); -} - -void -meta_window_unstick (MetaWindow *window) -{ - gboolean stick = FALSE; - - g_return_if_fail (!window->override_redirect); - - window_unstick_impl (window); - meta_window_foreach_transient (window, - stick_foreach_func, - &stick); -} - -void -meta_window_current_workspace_changed (MetaWindow *window) -{ - META_WINDOW_GET_CLASS (window)->current_workspace_changed (window); -} - -static gboolean -find_root_ancestor (MetaWindow *window, - void *data) -{ - MetaWindow **ancestor = data; - - /* Overwrite the previously "most-root" ancestor with the new one found */ - *ancestor = window; - - /* We want this to continue until meta_window_foreach_ancestor quits because - * there are no more valid ancestors. - */ - return TRUE; -} - -/** - * meta_window_find_root_ancestor: - * @window: a #MetaWindow - * - * Follow the chain of parents of @window, skipping transient windows, - * and return the "root" window which has no non-transient parent. - * - * Returns: (transfer none): The root ancestor window - */ -MetaWindow * -meta_window_find_root_ancestor (MetaWindow *window) -{ - MetaWindow *ancestor; - ancestor = window; - meta_window_foreach_ancestor (window, find_root_ancestor, &ancestor); - return ancestor; -} - -void -meta_window_raise (MetaWindow *window) -{ - MetaWindow *ancestor; - - g_return_if_fail (!window->override_redirect); - - ancestor = meta_window_find_root_ancestor (window); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Raising window %s, ancestor of %s", - ancestor->desc, window->desc); - - /* Raise the ancestor of the window (if the window has no ancestor, - * then ancestor will be set to the window itself); do this because - * it's weird to see windows from other apps stacked between a child - * and parent window of the currently active app. The stacking - * constraints in stack.c then magically take care of raising all - * the child windows appropriately. - */ - if (window->display->stack == ancestor->display->stack) - { - meta_stack_raise (window->display->stack, ancestor); - } - else - { - meta_warning ( - "Either stacks aren't per screen or some window has a weird " - "transient_for hint; window->display->stack != " - "ancestor->screen->stack. window = %s, ancestor = %s.", - window->desc, ancestor->desc); - /* We could raise the window here, but don't want to do that twice and - * so we let the case below handle that. - */ - } - - /* Okay, so stacking constraints misses one case: If a window has - * two children and we want to raise one of those children, then - * raising the ancestor isn't enough; we need to also raise the - * correct child. See bug 307875. - */ - if (window != ancestor) - meta_stack_raise (window->display->stack, window); - - g_signal_emit (window, window_signals[RAISED], 0); -} - -void -meta_window_lower (MetaWindow *window) -{ - g_return_if_fail (!window->override_redirect); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Lowering window %s", window->desc); - - meta_stack_lower (window->display->stack, window); -} - -/* - * Move window to the requested workspace; append controls whether new WS - * should be created if one does not exist. - */ -void -meta_window_change_workspace_by_index (MetaWindow *window, - gint space_index, - gboolean append) -{ - MetaWorkspaceManager *workspace_manager; - MetaWorkspace *workspace; - MetaDisplay *display; - - g_return_if_fail (!window->override_redirect); - - if (space_index == -1) - { - meta_window_stick (window); - return; - } - - display = window->display; - workspace_manager = display->workspace_manager; - - workspace = - meta_workspace_manager_get_workspace_by_index (workspace_manager, space_index); - - if (!workspace && append) - workspace = meta_workspace_manager_append_new_workspace (workspace_manager, FALSE, META_CURRENT_TIME); - - if (workspace) - meta_window_change_workspace (window, workspace); -} - -static void -meta_window_appears_focused_changed (MetaWindow *window) -{ - set_net_wm_state (window); - meta_window_frame_size_changed (window); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_APPEARS_FOCUSED]); - - if (window->frame) - meta_frame_queue_draw (window->frame); -} - -static gboolean -should_propagate_focus_appearance (MetaWindow *window) -{ - /* Parents of attached modal dialogs should appear focused. */ - if (meta_window_is_attached_dialog (window)) - return TRUE; - - /* Parents of these sorts of override-redirect windows should - * appear focused. */ - switch (window->type) - { - case META_WINDOW_DROPDOWN_MENU: - case META_WINDOW_POPUP_MENU: - case META_WINDOW_COMBO: - case META_WINDOW_TOOLTIP: - case META_WINDOW_NOTIFICATION: - case META_WINDOW_DND: - case META_WINDOW_OVERRIDE_OTHER: - return TRUE; - default: - break; - } - - return FALSE; -} - -/** - * meta_window_propagate_focus_appearance: - * @window: the window to start propagating from - * @focused: %TRUE if @window's ancestors should appear focused, - * %FALSE if they should not. - * - * Adjusts the value of #MetaWindow:appears-focused on @window's - * ancestors (but not @window itself). If @focused is %TRUE, each of - * @window's ancestors will have its %attached_focus_window field set - * to the current %focus_window. If @focused if %FALSE, each of - * @window's ancestors will have its %attached_focus_window field - * cleared if it is currently %focus_window. - */ -static void -meta_window_propagate_focus_appearance (MetaWindow *window, - gboolean focused) -{ - MetaWindow *child, *parent, *focus_window; - - focus_window = window->display->focus_window; - - child = window; - parent = meta_window_get_transient_for (child); - while (parent && (!focused || should_propagate_focus_appearance (child))) - { - gboolean child_focus_state_changed; - - if (focused) - { - if (parent->attached_focus_window == focus_window) - break; - child_focus_state_changed = (parent->attached_focus_window == NULL); - parent->attached_focus_window = focus_window; - } - else - { - if (parent->attached_focus_window != focus_window) - break; - child_focus_state_changed = (parent->attached_focus_window != NULL); - parent->attached_focus_window = NULL; - } - - if (child_focus_state_changed && !parent->has_focus) - { - meta_window_appears_focused_changed (parent); - } - - child = parent; - parent = meta_window_get_transient_for (child); - } -} - -void -meta_window_set_focused_internal (MetaWindow *window, - gboolean focused) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - - if (focused) - { - window->has_focus = TRUE; - if (window->override_redirect) - return; - - /* Move to the front of the focusing workspace's MRU list. - * We should only be "removing" it from the MRU list if it's - * not already there. Note that it's possible that we might - * be processing this FocusIn after we've changed to a - * different workspace; we should therefore update the MRU - * list only if the window is actually on the active - * workspace. - */ - if (workspace_manager->active_workspace && - meta_window_located_on_workspace (window, - workspace_manager->active_workspace)) - { - GList* link; - link = g_list_find (workspace_manager->active_workspace->mru_list, - window); - g_assert (link); - - workspace_manager->active_workspace->mru_list = - g_list_remove_link (workspace_manager->active_workspace->mru_list, - link); - g_list_free (link); - - workspace_manager->active_workspace->mru_list = - g_list_prepend (workspace_manager->active_workspace->mru_list, - window); - } - - if (window->frame) - meta_frame_queue_draw (window->frame); - - /* Ungrab click to focus button since the sync grab can interfere - * with some things you might do inside the focused window, by - * causing the client to get funky enter/leave events. - * - * The reason we usually have a passive grab on the window is - * so that we can intercept clicks and raise the window in - * response. For click-to-focus we don't need that since the - * focused window is already raised. When raise_on_click is - * FALSE we also don't need that since we don't do anything - * when the window is clicked. - * - * There is dicussion in bugs 102209, 115072, and 461577 - */ - if (meta_prefs_get_focus_mode () == G_DESKTOP_FOCUS_MODE_CLICK || - !meta_prefs_get_raise_on_click()) - { - meta_display_ungrab_focus_window_button (window->display, window); - /* Since we ungrab with XIAnyModifier above, all button - grabs go way so we need to re-grab the window buttons. */ - meta_display_grab_window_buttons (window->display, window->xwindow); - } - - g_signal_emit (window, window_signals[FOCUS], 0); - - if (!window->attached_focus_window) - meta_window_appears_focused_changed (window); - - meta_window_propagate_focus_appearance (window, TRUE); - } - else - { - window->has_focus = FALSE; - if (window->override_redirect) - return; - - meta_window_propagate_focus_appearance (window, FALSE); - - if (!window->attached_focus_window) - meta_window_appears_focused_changed (window); - - /* Re-grab for click to focus and raise-on-click, if necessary */ - if (meta_prefs_get_focus_mode () == G_DESKTOP_FOCUS_MODE_CLICK || - !meta_prefs_get_raise_on_click ()) - meta_display_grab_focus_window_button (window->display, window); - } -} - -/** - * meta_window_get_icon_geometry: - * @window: a #MetaWindow - * @rect: (out): rectangle into which to store the returned geometry. - * - * Gets the location of the icon corresponding to the window. The location - * will be provided set by the task bar or other user interface element - * displaying the icon, and is relative to the root window. - * - * Return value: %TRUE if the icon geometry was successfully retrieved. - */ -gboolean -meta_window_get_icon_geometry (MetaWindow *window, - MetaRectangle *rect) -{ - g_return_val_if_fail (!window->override_redirect, FALSE); - - if (window->icon_geometry_set) - { - if (rect) - *rect = window->icon_geometry; - - return TRUE; - } - - return FALSE; -} - -/** - * meta_window_set_icon_geometry: - * @window: a #MetaWindow - * @rect: (nullable): rectangle with the desired geometry or %NULL. - * - * Sets or unsets the location of the icon corresponding to the window. If - * set, the location should correspond to a dock, task bar or other user - * interface element displaying the icon, and is relative to the root window. - */ -void -meta_window_set_icon_geometry (MetaWindow *window, - MetaRectangle *rect) -{ - if (rect) - { - window->icon_geometry = *rect; - window->icon_geometry_set = TRUE; - } - else - { - window->icon_geometry_set = FALSE; - } -} - -static void -redraw_icon (MetaWindow *window) -{ - /* We could probably be smart and just redraw the icon here, - * instead of the whole frame. - */ - if (window->frame) - meta_frame_queue_draw (window->frame); -} - -static void -meta_window_update_icon_now (MetaWindow *window, - gboolean force) -{ - gboolean changed; - cairo_surface_t *icon = NULL; - cairo_surface_t *mini_icon; - - g_return_if_fail (!window->override_redirect); - - changed = META_WINDOW_GET_CLASS (window)->update_icon (window, &icon, &mini_icon); - - if (changed || force) - { - if (window->icon) - cairo_surface_destroy (window->icon); - window->icon = icon; - - if (window->mini_icon) - cairo_surface_destroy (window->mini_icon); - window->mini_icon = mini_icon; - - g_object_freeze_notify (G_OBJECT (window)); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_ICON]); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_MINI_ICON]); - g_object_thaw_notify (G_OBJECT (window)); - - redraw_icon (window); - } -} - -static gboolean -idle_update_icon (gpointer data) -{ - GSList *tmp; - GSList *copy; - guint queue_index = GPOINTER_TO_INT (data); - - meta_topic (META_DEBUG_GEOMETRY, "Clearing the update_icon queue"); - - /* Work with a copy, for reentrancy. The allowed reentrancy isn't - * complete; destroying a window while we're in here would result in - * badness. But it's OK to queue/unqueue update_icons. - */ - copy = g_slist_copy (queue_pending[queue_index]); - g_slist_free (queue_pending[queue_index]); - queue_pending[queue_index] = NULL; - queue_later[queue_index] = 0; - - destroying_windows_disallowed += 1; - - tmp = copy; - while (tmp != NULL) - { - MetaWindow *window; - - window = tmp->data; - - meta_window_update_icon_now (window, FALSE); - window->is_in_queues &= ~META_QUEUE_UPDATE_ICON; - - tmp = tmp->next; - } - - g_slist_free (copy); - - destroying_windows_disallowed -= 1; - - return FALSE; -} - -GList* -meta_window_get_workspaces (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - - if (window->on_all_workspaces) - return workspace_manager->workspaces; - else if (window->workspace != NULL) - return window->workspace->list_containing_self; - else if (window->constructing) - return NULL; - else - g_assert_not_reached (); - return NULL; -} - -static void -invalidate_work_areas (MetaWindow *window) -{ - GList *tmp; - - tmp = meta_window_get_workspaces (window); - - while (tmp != NULL) - { - meta_workspace_invalidate_work_area (tmp->data); - tmp = tmp->next; - } -} - -void -meta_window_update_struts (MetaWindow *window) -{ - if (META_WINDOW_GET_CLASS (window)->update_struts (window)) - invalidate_work_areas (window); -} - -static void -meta_window_type_changed (MetaWindow *window) -{ - gboolean old_decorated = window->decorated; - GObject *object = G_OBJECT (window); - - window->attached = meta_window_should_attach_to_parent (window); - meta_window_recalc_features (window); - - if (!window->override_redirect) - set_net_wm_state (window); - - /* Update frame */ - if (window->decorated) - meta_window_ensure_frame (window); - else - meta_window_destroy_frame (window); - - /* update stacking constraints */ - meta_window_update_layer (window); - - meta_window_grab_keys (window); - - g_object_freeze_notify (object); - - if (old_decorated != window->decorated) - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_DECORATED]); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_WINDOW_TYPE]); - - g_object_thaw_notify (object); -} - -void -meta_window_set_type (MetaWindow *window, - MetaWindowType type) -{ - if (window->type == type) - return; - - window->type = type; - meta_window_type_changed (window); -} - -void -meta_window_frame_size_changed (MetaWindow *window) -{ - if (window->frame) - meta_frame_clear_cached_borders (window->frame); -} - -static void -meta_window_get_default_skip_hints (MetaWindow *window, - gboolean *skip_taskbar_out, - gboolean *skip_pager_out) -{ - META_WINDOW_GET_CLASS (window)->get_default_skip_hints (window, skip_taskbar_out, skip_pager_out); -} - -static void -meta_window_recalc_skip_features (MetaWindow *window) -{ - switch (window->type) - { - /* Force skip taskbar/pager on these window types */ - case META_WINDOW_DESKTOP: - case META_WINDOW_DOCK: - case META_WINDOW_TOOLBAR: - case META_WINDOW_MENU: - case META_WINDOW_UTILITY: - case META_WINDOW_SPLASHSCREEN: - case META_WINDOW_DROPDOWN_MENU: - case META_WINDOW_POPUP_MENU: - case META_WINDOW_TOOLTIP: - case META_WINDOW_NOTIFICATION: - case META_WINDOW_COMBO: - case META_WINDOW_DND: - case META_WINDOW_OVERRIDE_OTHER: - window->skip_taskbar = TRUE; - window->skip_pager = TRUE; - break; - - case META_WINDOW_DIALOG: - case META_WINDOW_MODAL_DIALOG: - /* only skip taskbar if we have a real transient parent - (and ignore the application hints) */ - if (window->transient_for != NULL) - window->skip_taskbar = TRUE; - else - window->skip_taskbar = window->skip_from_window_list; - break; - - case META_WINDOW_NORMAL: - { - gboolean skip_taskbar_hint, skip_pager_hint; - meta_window_get_default_skip_hints (window, &skip_taskbar_hint, &skip_pager_hint); - window->skip_taskbar = skip_taskbar_hint | window->skip_from_window_list; - window->skip_pager = skip_pager_hint | window->skip_from_window_list; - } - break; - } -} - -void -meta_window_recalc_features (MetaWindow *window) -{ - gboolean old_has_close_func; - gboolean old_has_minimize_func; - gboolean old_has_move_func; - gboolean old_has_resize_func; - gboolean old_has_shade_func; - gboolean old_always_sticky; - gboolean old_skip_taskbar; - - old_has_close_func = window->has_close_func; - old_has_minimize_func = window->has_minimize_func; - old_has_move_func = window->has_move_func; - old_has_resize_func = window->has_resize_func; - old_has_shade_func = window->has_shade_func; - old_always_sticky = window->always_sticky; - old_skip_taskbar = window->skip_taskbar; - - /* Use MWM hints initially */ - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - window->decorated = window->mwm_decorated; - else - window->decorated = FALSE; - window->border_only = window->mwm_border_only; - window->has_close_func = window->mwm_has_close_func; - window->has_minimize_func = window->mwm_has_minimize_func; - window->has_maximize_func = window->mwm_has_maximize_func; - window->has_move_func = window->mwm_has_move_func; - - window->has_resize_func = TRUE; - - /* If min_size == max_size, then don't allow resize */ - if (window->size_hints.min_width == window->size_hints.max_width && - window->size_hints.min_height == window->size_hints.max_height) - window->has_resize_func = FALSE; - else if (!window->mwm_has_resize_func) - { - /* We ignore mwm_has_resize_func because WM_NORMAL_HINTS is the - * authoritative source for that info. Some apps such as mplayer or - * xine disable resize via MWM but not WM_NORMAL_HINTS, but that - * leads to e.g. us not fullscreening their windows. Apps that set - * MWM but not WM_NORMAL_HINTS are basically broken. We complain - * about these apps but make them work. - */ - - meta_warning ("Window %s sets an MWM hint indicating it isn't resizable, but sets min size %d x %d and max size %d x %d; this doesn't make much sense.", - window->desc, - window->size_hints.min_width, - window->size_hints.min_height, - window->size_hints.max_width, - window->size_hints.max_height); - } - - window->has_shade_func = TRUE; - window->has_fullscreen_func = TRUE; - - window->always_sticky = FALSE; - - /* Semantic category overrides the MWM hints */ - if (window->type == META_WINDOW_TOOLBAR) - window->decorated = FALSE; - - if (window->type == META_WINDOW_DESKTOP || - window->type == META_WINDOW_DOCK || - window->override_redirect) - window->always_sticky = TRUE; - - if (window->override_redirect || - meta_window_get_frame_type (window) == META_FRAME_TYPE_LAST) - { - window->decorated = FALSE; - window->has_close_func = FALSE; - window->has_shade_func = FALSE; - - /* FIXME this keeps panels and things from using - * NET_WM_MOVERESIZE; the problem is that some - * panels (edge panels) have fixed possible locations, - * and others ("floating panels") do not. - * - * Perhaps we should require edge panels to explicitly - * disable movement? - */ - window->has_move_func = FALSE; - window->has_resize_func = FALSE; - } - - if (window->type != META_WINDOW_NORMAL) - { - window->has_minimize_func = FALSE; - window->has_maximize_func = FALSE; - window->has_fullscreen_func = FALSE; - } - - if (!window->has_resize_func) - { - window->has_maximize_func = FALSE; - MetaRectangle display_rect = { 0 }; - - meta_display_get_size (window->display, &display_rect.width, - &display_rect.height); - - /* don't allow fullscreen if we can't resize, unless the size - * is entire screen size (kind of broken, because we - * actually fullscreen to monitor size not screen size) - */ - if (window->size_hints.min_width == display_rect.width && - window->size_hints.min_height == display_rect.height) - ; /* leave fullscreen available */ - else - window->has_fullscreen_func = FALSE; - } - - /* We leave fullscreen windows decorated, just push the frame outside - * the screen. This avoids flickering to unparent them. - * - * Note that setting has_resize_func = FALSE here must come after the - * above code that may disable fullscreen, because if the window - * is not resizable purely due to fullscreen, we don't want to - * disable fullscreen mode. - */ - if (window->fullscreen) - { - window->has_shade_func = FALSE; - window->has_move_func = FALSE; - window->has_resize_func = FALSE; - window->has_maximize_func = FALSE; - } - - if (window->has_maximize_func && window->monitor) - { - MetaRectangle work_area, client_rect; - - meta_window_get_work_area_current_monitor (window, &work_area); - meta_window_frame_rect_to_client_rect (window, &work_area, &client_rect); - - if (window->size_hints.min_width >= client_rect.width || - window->size_hints.min_height >= client_rect.height) - window->has_maximize_func = FALSE; - } - - meta_topic (META_DEBUG_WINDOW_OPS, - "Window %s fullscreen = %d not resizable, maximizable = %d fullscreenable = %d min size %dx%d max size %dx%d", - window->desc, - window->fullscreen, - window->has_maximize_func, window->has_fullscreen_func, - window->size_hints.min_width, - window->size_hints.min_height, - window->size_hints.max_width, - window->size_hints.max_height); - - /* no shading if not decorated */ - if (!window->decorated || window->border_only) - window->has_shade_func = FALSE; - - meta_window_recalc_skip_features (window); - - /* To prevent users from losing windows, let's prevent users from - * minimizing skip-taskbar windows through the window decorations. */ - if (window->skip_taskbar) - window->has_minimize_func = FALSE; - - meta_topic (META_DEBUG_WINDOW_OPS, - "Window %s decorated = %d border_only = %d has_close = %d has_minimize = %d has_maximize = %d has_move = %d has_shade = %d skip_taskbar = %d skip_pager = %d", - window->desc, - window->decorated, - window->border_only, - window->has_close_func, - window->has_minimize_func, - window->has_maximize_func, - window->has_move_func, - window->has_shade_func, - window->skip_taskbar, - window->skip_pager); - - if (old_skip_taskbar != window->skip_taskbar) - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_SKIP_TASKBAR]); - - /* FIXME: - * Lame workaround for recalc_features being used overzealously. - * The fix is to only recalc_features when something has - * actually changed. - */ - if (window->constructing || - old_has_close_func != window->has_close_func || - old_has_minimize_func != window->has_minimize_func || - old_has_move_func != window->has_move_func || - old_has_resize_func != window->has_resize_func || - old_has_shade_func != window->has_shade_func || - old_always_sticky != window->always_sticky) - set_allowed_actions_hint (window); - - if (window->has_resize_func != old_has_resize_func) - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_RESIZEABLE]); - - meta_window_frame_size_changed (window); - - /* FIXME perhaps should ensure if we don't have a shade func, - * we aren't shaded, etc. - */ -} - -void -meta_window_show_menu (MetaWindow *window, - MetaWindowMenuType menu, - int x, - int y) -{ - g_return_if_fail (!window->override_redirect); - meta_compositor_show_window_menu (window->display->compositor, window, menu, x, y); -} - -void -meta_window_show_menu_for_rect (MetaWindow *window, - MetaWindowMenuType menu, - MetaRectangle *rect) -{ - g_return_if_fail (!window->override_redirect); - meta_compositor_show_window_menu_for_rect (window->display->compositor, window, menu, rect); -} - -void -meta_window_shove_titlebar_onscreen (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - MetaRectangle frame_rect; - GList *onscreen_region; - int horiz_amount, vert_amount; - - g_return_if_fail (!window->override_redirect); - - /* If there's no titlebar, don't bother */ - if (!window->frame) - return; - - /* Get the basic info we need */ - meta_window_get_frame_rect (window, &frame_rect); - onscreen_region = workspace_manager->active_workspace->screen_region; - - /* Extend the region (just in case the window is too big to fit on the - * screen), then shove the window on screen, then return the region to - * normal. - */ - horiz_amount = frame_rect.width; - vert_amount = frame_rect.height; - meta_rectangle_expand_region (onscreen_region, - horiz_amount, - horiz_amount, - 0, - vert_amount); - meta_rectangle_shove_into_region(onscreen_region, - FIXED_DIRECTION_X, - &frame_rect); - meta_rectangle_expand_region (onscreen_region, - -horiz_amount, - -horiz_amount, - 0, - -vert_amount); - - meta_window_move_frame (window, FALSE, frame_rect.x, frame_rect.y); -} - -gboolean -meta_window_titlebar_is_onscreen (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - MetaRectangle titlebar_rect, frame_rect; - GList *onscreen_region; - gboolean is_onscreen; - - const int min_height_needed = 8; - const float min_width_percent = 0.5; - const int min_width_absolute = 50; - - /* Titlebar can't be offscreen if there is no titlebar... */ - if (!window->frame) - return TRUE; - - /* Get the rectangle corresponding to the titlebar */ - meta_window_get_titlebar_rect (window, &titlebar_rect); - - /* Translate into screen coordinates */ - meta_window_get_frame_rect (window, &frame_rect); - titlebar_rect.x = frame_rect.x; - titlebar_rect.y = frame_rect.y; - - /* Run through the spanning rectangles for the screen and see if one of - * them overlaps with the titlebar sufficiently to consider it onscreen. - */ - is_onscreen = FALSE; - onscreen_region = workspace_manager->active_workspace->screen_region; - while (onscreen_region) - { - MetaRectangle *spanning_rect = onscreen_region->data; - MetaRectangle overlap; - - meta_rectangle_intersect (&titlebar_rect, spanning_rect, &overlap); - if (overlap.height > MIN (titlebar_rect.height, min_height_needed) && - overlap.width > MIN (titlebar_rect.width * min_width_percent, - min_width_absolute)) - { - is_onscreen = TRUE; - break; - } - - onscreen_region = onscreen_region->next; - } - - return is_onscreen; -} - -static gboolean -check_moveresize_frequency (MetaWindow *window, - gdouble *remaining) -{ - int64_t current_time; - const double max_resizes_per_second = 25.0; - const double ms_between_resizes = 1000.0 / max_resizes_per_second; - double elapsed; - - current_time = g_get_real_time (); - - /* If we are throttling via _NET_WM_SYNC_REQUEST, we don't need - * an artificial timeout-based throttled */ - if (!window->disable_sync && - window->sync_request_alarm != None) - return TRUE; - - elapsed = (current_time - window->display->grab_last_moveresize_time) / 1000; - - if (elapsed >= 0.0 && elapsed < ms_between_resizes) - { - meta_topic (META_DEBUG_RESIZING, - "Delaying move/resize as only %g of %g ms elapsed", - elapsed, ms_between_resizes); - - if (remaining) - *remaining = (ms_between_resizes - elapsed); - - return FALSE; - } - - meta_topic (META_DEBUG_RESIZING, - " Checked moveresize freq, allowing move/resize now (%g of %g seconds elapsed)", - elapsed / 1000.0, 1.0 / max_resizes_per_second); - - return TRUE; -} - -static gboolean -update_move_timeout (gpointer data) -{ - MetaWindow *window = data; - - update_move (window, - window->display->grab_last_edge_resistance_flags, - window->display->grab_latest_motion_x, - window->display->grab_latest_motion_y); - - return FALSE; -} - -static void -update_move_maybe_tile (MetaWindow *window, - int shake_threshold, - int x, - int y) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - MetaLogicalMonitor *logical_monitor; - MetaDisplay *display = window->display; - MetaRectangle work_area; - - /* For side-by-side tiling we are interested in the inside vertical - * edges of the work area of the monitor where the pointer is located, - * and in the outside top edge for maximized tiling. - * - * For maximized tiling we use the outside edge instead of the - * inside edge, because we don't want to force users to maximize - * windows they are placing near the top of their screens. - * - * The "current" idea of meta_window_get_work_area_current_monitor() and - * meta_screen_get_current_monitor() is slightly different: the former - * refers to the monitor which contains the largest part of the window, - * the latter to the one where the pointer is located. - */ - logical_monitor = - meta_monitor_manager_get_logical_monitor_at (monitor_manager, x, y); - if (!logical_monitor) - return; - - meta_window_get_work_area_for_monitor (window, - logical_monitor->number, - &work_area); - - /* Check if the cursor is in a position which triggers tiling - * and set tile_mode accordingly. - */ - if (meta_window_can_tile_side_by_side (window) && - x >= logical_monitor->rect.x && x < (work_area.x + shake_threshold)) - display->preview_tile_mode = META_TILE_LEFT; - else if (meta_window_can_tile_side_by_side (window) && - x >= work_area.x + work_area.width - shake_threshold && - x < (logical_monitor->rect.x + logical_monitor->rect.width)) - display->preview_tile_mode = META_TILE_RIGHT; - else if (meta_window_can_tile_maximized (window) && - y >= logical_monitor->rect.y && y <= work_area.y) - display->preview_tile_mode = META_TILE_MAXIMIZED; - else - display->preview_tile_mode = META_TILE_NONE; - - if (display->preview_tile_mode != META_TILE_NONE) - window->tile_monitor_number = logical_monitor->number; -} - -static void -update_move (MetaWindow *window, - MetaEdgeResistanceFlags flags, - int x, - int y) -{ - int dx, dy; - int new_x, new_y; - MetaRectangle old; - int shake_threshold; - MetaDisplay *display = window->display; - - display->grab_latest_motion_x = x; - display->grab_latest_motion_y = y; - - dx = x - display->grab_anchor_root_x; - dy = y - display->grab_anchor_root_y; - - new_x = display->grab_anchor_window_pos.x + dx; - new_y = display->grab_anchor_window_pos.y + dy; - - meta_verbose ("x,y = %d,%d anchor ptr %d,%d anchor pos %d,%d dx,dy %d,%d", - x, y, - display->grab_anchor_root_x, - display->grab_anchor_root_y, - display->grab_anchor_window_pos.x, - display->grab_anchor_window_pos.y, - dx, dy); - - /* Don't bother doing anything if no move has been specified. (This - * happens often, even in keyboard moving, due to the warping of the - * pointer. - */ - if (dx == 0 && dy == 0) - return; - - /* Originally for detaching maximized windows, but we use this - * for the zones at the sides of the monitor where trigger tiling - * because it's about the right size - */ -#define DRAG_THRESHOLD_TO_SHAKE_THRESHOLD_FACTOR 6 - shake_threshold = meta_prefs_get_drag_threshold () * - DRAG_THRESHOLD_TO_SHAKE_THRESHOLD_FACTOR; - - if (flags & META_EDGE_RESISTANCE_SNAP) - { - /* We don't want to tile while snapping. Also, clear any previous tile - request. */ - display->preview_tile_mode = META_TILE_NONE; - window->tile_monitor_number = -1; - } - else if (meta_prefs_get_edge_tiling () && - !META_WINDOW_MAXIMIZED (window) && - !META_WINDOW_TILED_SIDE_BY_SIDE (window)) - { - update_move_maybe_tile (window, shake_threshold, x, y); - } - - /* shake loose (unmaximize) maximized or tiled window if dragged beyond - * the threshold in the Y direction. Tiled windows can also be pulled - * loose via X motion. - */ - - if ((META_WINDOW_MAXIMIZED (window) && ABS (dy) >= shake_threshold) || - (META_WINDOW_TILED_SIDE_BY_SIDE (window) && (MAX (ABS (dx), ABS (dy)) >= shake_threshold))) - { - double prop; - - /* Shake loose, so that the window snaps back to maximized - * when dragged near the top; do not snap back if tiling - * is enabled, as top edge tiling can be used in that case - */ - window->shaken_loose = !meta_prefs_get_edge_tiling (); - window->tile_mode = META_TILE_NONE; - - /* move the unmaximized window to the cursor */ - prop = - ((double)(x - display->grab_initial_window_pos.x)) / - ((double)display->grab_initial_window_pos.width); - - display->grab_initial_window_pos.x = x - window->saved_rect.width * prop; - - /* If we started dragging the window from above the top of the window, - * pretend like we started dragging from the middle of the titlebar - * instead, as the "correct" anchoring looks wrong. */ - if (display->grab_anchor_root_y < display->grab_initial_window_pos.y) - { - MetaRectangle titlebar_rect; - meta_window_get_titlebar_rect (window, &titlebar_rect); - display->grab_anchor_root_y = display->grab_initial_window_pos.y + titlebar_rect.height / 2; - } - - window->saved_rect.x = display->grab_initial_window_pos.x; - window->saved_rect.y = display->grab_initial_window_pos.y; - - meta_window_unmaximize (window, META_MAXIMIZE_BOTH); - return; - } - - /* remaximize window on another monitor if window has been shaken - * loose or it is still maximized (then move straight) - */ - else if ((window->shaken_loose || META_WINDOW_MAXIMIZED (window)) && - window->tile_mode != META_TILE_LEFT && window->tile_mode != META_TILE_RIGHT) - { - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = - meta_backend_get_monitor_manager (backend); - int n_logical_monitors; - const MetaLogicalMonitor *wmonitor; - MetaRectangle work_area; - int monitor; - - window->tile_mode = META_TILE_NONE; - wmonitor = window->monitor; - n_logical_monitors = - meta_monitor_manager_get_num_logical_monitors (monitor_manager); - - for (monitor = 0; monitor < n_logical_monitors; monitor++) - { - meta_window_get_work_area_for_monitor (window, monitor, &work_area); - - /* check if cursor is near the top of a monitor work area */ - if (x >= work_area.x && - x < (work_area.x + work_area.width) && - y >= work_area.y && - y < (work_area.y + shake_threshold)) - { - /* move the saved rect if window will become maximized on an - * other monitor so user isn't surprised on a later unmaximize - */ - if (wmonitor->number != monitor) - { - window->saved_rect.x = work_area.x; - window->saved_rect.y = work_area.y; - - if (window->frame) - { - window->saved_rect.x += window->frame->child_x; - window->saved_rect.y += window->frame->child_y; - } - - window->unconstrained_rect.x = window->saved_rect.x; - window->unconstrained_rect.y = window->saved_rect.y; - - meta_window_unmaximize (window, META_MAXIMIZE_BOTH); - - display->grab_initial_window_pos = work_area; - display->grab_anchor_root_x = x; - display->grab_anchor_root_y = y; - window->shaken_loose = FALSE; - - meta_window_maximize (window, META_MAXIMIZE_BOTH); - } - - return; - } - } - } - - /* Delay showing the tile preview slightly to make it more unlikely to - * trigger it unwittingly, e.g. when shaking loose the window or moving - * it to another monitor. - */ - meta_display_update_tile_preview (window->display, - window->tile_mode != META_TILE_NONE); - - meta_window_get_frame_rect (window, &old); - - /* Don't allow movement in the maximized directions or while tiled */ - if (window->maximized_horizontally || META_WINDOW_TILED_SIDE_BY_SIDE (window)) - new_x = old.x; - if (window->maximized_vertically) - new_y = old.y; - - /* Do any edge resistance/snapping */ - meta_window_edge_resistance_for_move (window, - &new_x, - &new_y, - update_move_timeout, - flags); - - meta_window_move_frame (window, TRUE, new_x, new_y); -} - -static gboolean -update_resize_timeout (gpointer data) -{ - MetaWindow *window = data; - - update_resize (window, - window->display->grab_last_edge_resistance_flags, - window->display->grab_latest_motion_x, - window->display->grab_latest_motion_y, - TRUE); - return FALSE; -} - -static void -update_resize (MetaWindow *window, - MetaEdgeResistanceFlags flags, - int x, - int y, - gboolean force) -{ - int dx, dy; - MetaGravity gravity; - MetaRectangle new_rect; - MetaRectangle old_rect; - double remaining = 0; - - window->display->grab_latest_motion_x = x; - window->display->grab_latest_motion_y = y; - - dx = x - window->display->grab_anchor_root_x; - dy = y - window->display->grab_anchor_root_y; - - /* Attached modal dialogs are special in that size - * changes apply to both sides, so that the dialog - * remains centered to the parent. - */ - if (meta_window_is_attached_dialog (window)) - { - dx *= 2; - dy *= 2; - } - - new_rect.width = window->display->grab_anchor_window_pos.width; - new_rect.height = window->display->grab_anchor_window_pos.height; - - /* Don't bother doing anything if no move has been specified. (This - * happens often, even in keyboard resizing, due to the warping of the - * pointer. - */ - if (dx == 0 && dy == 0) - return; - - if (window->display->grab_op == META_GRAB_OP_KEYBOARD_RESIZING_UNKNOWN) - { - MetaGrabOp op = META_GRAB_OP_WINDOW_BASE | META_GRAB_OP_WINDOW_FLAG_KEYBOARD; - - if (dx > 0) - op |= META_GRAB_OP_WINDOW_DIR_EAST; - else if (dx < 0) - op |= META_GRAB_OP_WINDOW_DIR_WEST; - - if (dy > 0) - op |= META_GRAB_OP_WINDOW_DIR_SOUTH; - else if (dy < 0) - op |= META_GRAB_OP_WINDOW_DIR_NORTH; - - window->display->grab_op = op; - - meta_window_update_keyboard_resize (window, TRUE); - } - - if (window->display->grab_op & META_GRAB_OP_WINDOW_DIR_EAST) - new_rect.width += dx; - else if (window->display->grab_op & META_GRAB_OP_WINDOW_DIR_WEST) - new_rect.width -= dx; - - if (window->display->grab_op & META_GRAB_OP_WINDOW_DIR_SOUTH) - new_rect.height += dy; - else if (window->display->grab_op & META_GRAB_OP_WINDOW_DIR_NORTH) - new_rect.height -= dy; - - meta_window_maybe_apply_size_hints (window, &new_rect); - - /* If we're waiting for a request for _NET_WM_SYNC_REQUEST, we'll - * resize the window when the window responds, or when we time - * the response out. - */ - if (window->sync_request_timeout_id != 0) - return; - - if (!check_moveresize_frequency (window, &remaining) && !force) - { - /* we are ignoring an event here, so we schedule a - * compensation event when we would otherwise not ignore - * an event. Otherwise we can become stuck if the user never - * generates another event. - */ - if (!window->display->grab_resize_timeout_id) - { - window->display->grab_resize_timeout_id = - g_timeout_add ((int)remaining, update_resize_timeout, window); - g_source_set_name_by_id (window->display->grab_resize_timeout_id, - "[mutter] update_resize_timeout"); - } - - return; - } - - /* Remove any scheduled compensation events */ - g_clear_handle_id (&window->display->grab_resize_timeout_id, g_source_remove); - - meta_window_get_frame_rect (window, &old_rect); - - /* One sided resizing ought to actually be one-sided, despite the fact that - * aspect ratio windows don't interact nicely with the above stuff. So, - * to avoid some nasty flicker, we enforce that. - */ - - if ((window->display->grab_op & (META_GRAB_OP_WINDOW_DIR_WEST | META_GRAB_OP_WINDOW_DIR_EAST)) == 0) - new_rect.width = old_rect.width; - - if ((window->display->grab_op & (META_GRAB_OP_WINDOW_DIR_NORTH | META_GRAB_OP_WINDOW_DIR_SOUTH)) == 0) - new_rect.height = old_rect.height; - - /* compute gravity of client during operation */ - gravity = meta_resize_gravity_from_grab_op (window->display->grab_op); - g_assert (gravity >= 0); - - /* Do any edge resistance/snapping */ - meta_window_edge_resistance_for_resize (window, - &new_rect.width, - &new_rect.height, - gravity, - update_resize_timeout, - flags); - - meta_window_resize_frame_with_gravity (window, TRUE, - new_rect.width, new_rect.height, - gravity); - - /* Store the latest resize time, if we actually resized. */ - if (window->rect.width != old_rect.width || - window->rect.height != old_rect.height) - window->display->grab_last_moveresize_time = g_get_real_time (); -} - -static void -maybe_maximize_tiled_window (MetaWindow *window) -{ - MetaRectangle work_area; - gint shake_threshold; - - if (!META_WINDOW_TILED_SIDE_BY_SIDE (window)) - return; - - shake_threshold = meta_prefs_get_drag_threshold (); - - meta_window_get_work_area_for_monitor (window, - window->tile_monitor_number, - &work_area); - if (window->rect.width >= work_area.width - shake_threshold) - meta_window_maximize (window, META_MAXIMIZE_BOTH); -} - -void -meta_window_update_resize (MetaWindow *window, - MetaEdgeResistanceFlags flags, - int x, int y, - gboolean force) -{ - update_resize (window, flags, x, y, force); -} - -static void -end_grab_op (MetaWindow *window, - const ClutterEvent *event) -{ - ClutterModifierType modifiers; - MetaEdgeResistanceFlags last_flags; - gfloat x, y; - - clutter_event_get_coords (event, &x, &y); - modifiers = clutter_event_get_state (event); - meta_display_check_threshold_reached (window->display, x, y); - - /* If the user was snap moving then ignore the button - * release because they may have let go of shift before - * releasing the mouse button and they almost certainly do - * not want a non-snapped movement to occur from the button - * release. - */ - last_flags = window->display->grab_last_edge_resistance_flags; - if ((last_flags & META_EDGE_RESISTANCE_SNAP) == 0) - { - MetaEdgeResistanceFlags flags = META_EDGE_RESISTANCE_DEFAULT; - - if (modifiers & CLUTTER_SHIFT_MASK) - flags |= META_EDGE_RESISTANCE_SNAP; - - if (modifiers & CLUTTER_CONTROL_MASK) - flags |= META_EDGE_RESISTANCE_WINDOWS; - - if (meta_grab_op_is_moving (window->display->grab_op)) - { - if (window->display->preview_tile_mode != META_TILE_NONE) - meta_window_tile (window, window->display->preview_tile_mode); - else - update_move (window, flags, x, y); - } - else if (meta_grab_op_is_resizing (window->display->grab_op)) - { - if (window->tile_match != NULL) - flags |= (META_EDGE_RESISTANCE_SNAP | META_EDGE_RESISTANCE_WINDOWS); - - update_resize (window, flags, x, y, TRUE); - maybe_maximize_tiled_window (window); - } - } - window->display->preview_tile_mode = META_TILE_NONE; - meta_display_end_grab_op (window->display, clutter_event_get_time (event)); -} - -gboolean -meta_window_handle_mouse_grab_op_event (MetaWindow *window, - const ClutterEvent *event) -{ - ClutterEventSequence *sequence = clutter_event_get_event_sequence (event); - ClutterModifierType modifier_state; - MetaEdgeResistanceFlags flags; - gfloat x, y; - - switch (event->type) - { - case CLUTTER_TOUCH_BEGIN: - if (!meta_display_is_pointer_emulating_sequence (window->display, sequence)) - return FALSE; - - return TRUE; - - case CLUTTER_BUTTON_PRESS: - { - ClutterModifierType grab_mods = meta_display_get_compositor_modifiers (window->display); - - /* This is the keybinding or menu case where we've - * been dragging around the window without the button - * pressed. */ - - if ((meta_grab_op_is_mouse (window->display->grab_op) && - (event->button.modifier_state & grab_mods) == grab_mods && - window->display->grab_button != (int) event->button.button) || - meta_grab_op_is_keyboard (window->display->grab_op)) - { - end_grab_op (window, event); - return FALSE; - } - return TRUE; - } - - case CLUTTER_TOUCH_END: - if (!meta_display_is_pointer_emulating_sequence (window->display, sequence)) - return FALSE; - - end_grab_op (window, event); - return TRUE; - - case CLUTTER_BUTTON_RELEASE: - if (event->button.button == 1 || - event->button.button == (unsigned int) meta_prefs_get_mouse_button_resize ()) - end_grab_op (window, event); - - return TRUE; - - case CLUTTER_TOUCH_UPDATE: - if (!meta_display_is_pointer_emulating_sequence (window->display, sequence)) - return FALSE; - - /* Fall through */ - case CLUTTER_MOTION: - modifier_state = clutter_event_get_state (event); - clutter_event_get_coords (event, &x, &y); - flags = META_EDGE_RESISTANCE_DEFAULT; - - if (modifier_state & CLUTTER_SHIFT_MASK) - flags |= META_EDGE_RESISTANCE_SNAP; - - if (modifier_state & CLUTTER_CONTROL_MASK) - flags |= META_EDGE_RESISTANCE_WINDOWS; - - meta_display_check_threshold_reached (window->display, x, y); - if (meta_grab_op_is_moving (window->display->grab_op)) - { - update_move (window, flags, x, y); - } - else if (meta_grab_op_is_resizing (window->display->grab_op)) - { - if (window->tile_match != NULL) - flags |= (META_EDGE_RESISTANCE_SNAP | META_EDGE_RESISTANCE_WINDOWS); - - update_resize (window, flags, x, y, FALSE); - } - return TRUE; - - case CLUTTER_TOUCH_CANCEL: - end_grab_op (window, event); - return FALSE; - - default: - return FALSE; - } -} - -void -meta_window_get_work_area_for_logical_monitor (MetaWindow *window, - MetaLogicalMonitor *logical_monitor, - MetaRectangle *area) -{ - GList *tmp; - - g_assert (logical_monitor); - - /* Initialize to the whole monitor */ - *area = logical_monitor->rect; - - tmp = meta_window_get_workspaces (window); - while (tmp != NULL) - { - MetaRectangle workspace_work_area; - meta_workspace_get_work_area_for_logical_monitor (tmp->data, - logical_monitor, - &workspace_work_area); - meta_rectangle_intersect (area, - &workspace_work_area, - area); - tmp = tmp->next; - } - - meta_topic (META_DEBUG_WORKAREA, - "Window %s monitor %d has work area %d,%d %d x %d", - window->desc, logical_monitor->number, - area->x, area->y, area->width, area->height); -} - -/** - * meta_window_get_work_area_current_monitor: - * @window: a #MetaWindow - * @area: (out): a location to store the work area - * - * Get the work area for the monitor @window is currently on. - */ -void -meta_window_get_work_area_current_monitor (MetaWindow *window, - MetaRectangle *area) -{ - meta_window_get_work_area_for_monitor (window, - window->monitor->number, - area); -} - -/** - * meta_window_get_work_area_for_monitor: - * @window: a #MetaWindow - * @which_monitor: a moniotr to get the work area for - * @area: (out): a location to store the work area - * - * Get the work area for @window, given the monitor index - * @which_monitor. - */ -void -meta_window_get_work_area_for_monitor (MetaWindow *window, - int which_monitor, - MetaRectangle *area) -{ - MetaBackend *backend = meta_get_backend (); - MetaMonitorManager *monitor_manager = meta_backend_get_monitor_manager (backend); - MetaLogicalMonitor *logical_monitor; - - g_return_if_fail (which_monitor >= 0); - - logical_monitor = - meta_monitor_manager_get_logical_monitor_from_number (monitor_manager, - which_monitor); - - meta_window_get_work_area_for_logical_monitor (window, logical_monitor, area); -} - -/** - * meta_window_get_work_area_all_monitors: - * @window: a #MetaWindow - * @area: (out): a location to store the work area - * - * Get the work area for all monitors for @window. - */ -void -meta_window_get_work_area_all_monitors (MetaWindow *window, - MetaRectangle *area) -{ - GList *tmp; - MetaRectangle display_rect = { 0 }; - - meta_display_get_size (window->display, - &display_rect.width, - &display_rect.height); - - /* Initialize to the whole display */ - *area = display_rect; - - tmp = meta_window_get_workspaces (window); - while (tmp != NULL) - { - MetaRectangle workspace_work_area; - meta_workspace_get_work_area_all_monitors (tmp->data, - &workspace_work_area); - meta_rectangle_intersect (area, - &workspace_work_area, - area); - tmp = tmp->next; - } - - meta_topic (META_DEBUG_WORKAREA, - "Window %s has whole-screen work area %d,%d %d x %d", - window->desc, area->x, area->y, area->width, area->height); -} - -int -meta_window_get_current_tile_monitor_number (MetaWindow *window) -{ - int tile_monitor_number = window->tile_monitor_number; - - if (tile_monitor_number < 0) - { - meta_warning ("%s called with an invalid monitor number; using 0 instead", G_STRFUNC); - tile_monitor_number = 0; - } - - return tile_monitor_number; -} - -void -meta_window_get_tile_area (MetaWindow *window, - MetaTileMode tile_mode, - MetaRectangle *tile_area) -{ - MetaRectangle work_area; - int tile_monitor_number; - double fraction; - - g_return_if_fail (tile_mode != META_TILE_NONE); - - tile_monitor_number = meta_window_get_current_tile_monitor_number (window); - - meta_window_get_work_area_for_monitor (window, tile_monitor_number, &work_area); - meta_window_get_tile_fraction (window, tile_mode, &fraction); - - *tile_area = work_area; - tile_area->width = round (tile_area->width * fraction); - - if (tile_mode == META_TILE_RIGHT) - tile_area->x += work_area.width - tile_area->width; -} - -gboolean -meta_window_same_application (MetaWindow *window, - MetaWindow *other_window) -{ - MetaGroup *group = meta_window_get_group (window); - MetaGroup *other_group = meta_window_get_group (other_window); - - return - group!=NULL && - other_group!=NULL && - group==other_group; -} - -/** - * meta_window_is_client_decorated: - * - * Check if if the window has decorations drawn by the client. - * (window->decorated refers only to whether we should add decorations) - */ -gboolean -meta_window_is_client_decorated (MetaWindow *window) -{ - if (window->client_type == META_WINDOW_CLIENT_TYPE_WAYLAND) - { - /* Assume all Wayland clients draw decorations - not strictly - * true but good enough for current purposes. - */ - return TRUE; - } - else - { - /* Currently the implementation here is hackish - - * has_custom_frame_extents() is set if _GTK_FRAME_EXTENTS is set - * to any value even 0. GTK+ always sets _GTK_FRAME_EXTENTS for - * client-side-decorated window, even if the value is 0 because - * the window is maxized and has no invisible borders or shadows. - */ - return window->has_custom_frame_extents; - } -} - -/** - * meta_window_foreach_transient: - * @window: a #MetaWindow - * @func: (scope call) (closure user_data): Called for each window which is a transient of @window (transitively) - * @user_data: User data - * - * Call @func for every window which is either transient for @window, or is - * a transient of a window which is in turn transient for @window. - * The order of window enumeration is not defined. - * - * Iteration will stop if @func at any point returns %FALSE. - */ -void -meta_window_foreach_transient (MetaWindow *window, - MetaWindowForeachFunc func, - void *user_data) -{ - GSList *windows; - GSList *tmp; - - windows = meta_display_list_windows (window->display, META_LIST_DEFAULT); - - tmp = windows; - while (tmp != NULL) - { - MetaWindow *transient = tmp->data; - - if (meta_window_is_ancestor_of_transient (window, transient)) - { - if (!(* func) (transient, user_data)) - break; - } - - tmp = tmp->next; - } - - g_slist_free (windows); -} - -/** - * meta_window_foreach_ancestor: - * @window: a #MetaWindow - * @func: (scope call) (closure user_data): Called for each window which is a transient parent of @window - * @user_data: User data - * - * If @window is transient, call @func with the window for which it's transient, - * repeatedly until either we find a non-transient window, or @func returns %FALSE. - */ -void -meta_window_foreach_ancestor (MetaWindow *window, - MetaWindowForeachFunc func, - void *user_data) -{ - MetaWindow *w; - - w = window; - do - { - if (w->transient_for == NULL) - break; - - w = w->transient_for; - } - while (w && (* func) (w, user_data)); -} - -typedef struct -{ - MetaWindow *ancestor; - gboolean found; -} FindAncestorData; - -static gboolean -find_ancestor_func (MetaWindow *window, - void *data) -{ - FindAncestorData *d = data; - - if (window == d->ancestor) - { - d->found = TRUE; - return FALSE; - } - - return TRUE; -} - -/** - * meta_window_is_ancestor_of_transient: - * @window: a #MetaWindow - * @transient: a #MetaWindow - * - * The function determines whether @window is an ancestor of @transient; it does - * so by traversing the @transient's ancestors until it either locates @window - * or reaches an ancestor that is not transient. - * - * Return Value: %TRUE if window is an ancestor of transient. - */ -gboolean -meta_window_is_ancestor_of_transient (MetaWindow *window, - MetaWindow *transient) -{ - FindAncestorData d; - - d.ancestor = window; - d.found = FALSE; - - meta_window_foreach_ancestor (transient, find_ancestor_func, &d); - - return d.found; -} - -/* Warp pointer to location appropriate for grab, - * return root coordinates where pointer ended up. - */ -static gboolean -warp_grab_pointer (MetaWindow *window, - MetaGrabOp grab_op, - int *x, - int *y) -{ - MetaRectangle rect; - MetaRectangle display_rect = { 0 }; - MetaDisplay *display; - ClutterSeat *seat; - - display = window->display; - meta_display_get_size (display, - &display_rect.width, - &display_rect.height); - - /* We may not have done begin_grab_op yet, i.e. may not be in a grab - */ - - meta_window_get_frame_rect (window, &rect); - - if (grab_op & META_GRAB_OP_WINDOW_DIR_WEST) - *x = 0; - else if (grab_op & META_GRAB_OP_WINDOW_DIR_EAST) - *x = rect.width - 1; - else - *x = rect.width / 2; - - if (grab_op & META_GRAB_OP_WINDOW_DIR_NORTH) - *y = 0; - else if (grab_op & META_GRAB_OP_WINDOW_DIR_SOUTH) - *y = rect.height - 1; - else - *y = rect.height / 2; - - *x += rect.x; - *y += rect.y; - - /* Avoid weird bouncing at the screen edge; see bug 154706 */ - *x = CLAMP (*x, 0, display_rect.width - 1); - *y = CLAMP (*y, 0, display_rect.height - 1); - - meta_topic (META_DEBUG_WINDOW_OPS, - "Warping pointer to %d,%d with window at %d,%d", - *x, *y, rect.x, rect.y); - - /* Need to update the grab positions so that the MotionNotify and other - * events generated by the XWarpPointer() call below don't cause complete - * funkiness. See bug 124582 and bug 122670. - */ - display->grab_anchor_root_x = *x; - display->grab_anchor_root_y = *y; - display->grab_latest_motion_x = *x; - display->grab_latest_motion_y = *y; - meta_window_get_frame_rect (window, - &display->grab_anchor_window_pos); - - seat = clutter_backend_get_default_seat (clutter_get_default_backend ()); - clutter_seat_warp_pointer (seat, *x, *y); - - return TRUE; -} - -void -meta_window_begin_grab_op (MetaWindow *window, - MetaGrabOp op, - gboolean frame_action, - guint32 timestamp) -{ - int x, y; - - warp_grab_pointer (window, - op, &x, &y); - - meta_display_begin_grab_op (window->display, - window, - op, - FALSE, - frame_action, - 0 /* button */, - 0, - timestamp, - x, y); -} - -void -meta_window_update_keyboard_resize (MetaWindow *window, - gboolean update_cursor) -{ - int x, y; - - warp_grab_pointer (window, - window->display->grab_op, - &x, &y); - - if (update_cursor) - meta_display_update_cursor (window->display); -} - -void -meta_window_update_keyboard_move (MetaWindow *window) -{ - int x, y; - - warp_grab_pointer (window, - window->display->grab_op, - &x, &y); -} - -MetaStackLayer -meta_window_get_default_layer (MetaWindow *window) -{ - if (window->wm_state_below) - return META_LAYER_BOTTOM; - else if (window->wm_state_above && !META_WINDOW_MAXIMIZED (window)) - return META_LAYER_TOP; - else - return META_LAYER_NORMAL; -} - -void -meta_window_update_layer (MetaWindow *window) -{ - MetaGroup *group; - - meta_stack_freeze (window->display->stack); - group = meta_window_get_group (window); - if (group) - meta_group_update_layers (group); - else - meta_stack_update_layer (window->display->stack, window); - meta_stack_thaw (window->display->stack); -} - -/* ensure_mru_position_after ensures that window appears after - * below_this_one in the active_workspace's mru_list (i.e. it treats - * window as having been less recently used than below_this_one) - */ -static void -ensure_mru_position_after (MetaWindow *window, - MetaWindow *after_this_one) -{ - /* This is sort of slow since it runs through the entire list more - * than once (especially considering the fact that we expect the - * windows of interest to be the first two elements in the list), - * but it doesn't matter while we're only using it on new window - * map. - */ - - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - GList* active_mru_list; - GList* window_position; - GList* after_this_one_position; - - active_mru_list = workspace_manager->active_workspace->mru_list; - window_position = g_list_find (active_mru_list, window); - after_this_one_position = g_list_find (active_mru_list, after_this_one); - - /* after_this_one_position is NULL when we switch workspaces, but in - * that case we don't need to do any MRU shuffling so we can simply - * return. - */ - if (after_this_one_position == NULL) - return; - - if (g_list_length (window_position) > g_list_length (after_this_one_position)) - { - workspace_manager->active_workspace->mru_list = - g_list_delete_link (workspace_manager->active_workspace->mru_list, - window_position); - - workspace_manager->active_workspace->mru_list = - g_list_insert_before (workspace_manager->active_workspace->mru_list, - after_this_one_position->next, - window); - } -} - -gboolean -meta_window_is_in_stack (MetaWindow *window) -{ - return window->stack_position >= 0; -} - -void -meta_window_stack_just_below (MetaWindow *window, - MetaWindow *below_this_one) -{ - g_return_if_fail (window != NULL); - g_return_if_fail (below_this_one != NULL); - - if (window->stack_position > below_this_one->stack_position) - { - meta_topic (META_DEBUG_STACK, - "Setting stack position of window %s to %d (making it below window %s).", - window->desc, - below_this_one->stack_position, - below_this_one->desc); - meta_window_set_stack_position (window, below_this_one->stack_position); - } - else - { - meta_topic (META_DEBUG_STACK, - "Window %s was already below window %s.", - window->desc, below_this_one->desc); - } -} - -void -meta_window_stack_just_above (MetaWindow *window, - MetaWindow *above_this_one) -{ - g_return_if_fail (window != NULL); - g_return_if_fail (above_this_one != NULL); - - if (window->stack_position < above_this_one->stack_position) - { - meta_topic (META_DEBUG_STACK, - "Setting stack position of window %s to %d (making it above window %s).", - window->desc, - above_this_one->stack_position, - above_this_one->desc); - meta_window_set_stack_position (window, above_this_one->stack_position); - } - else - { - meta_topic (META_DEBUG_STACK, - "Window %s was already above window %s.", - window->desc, above_this_one->desc); - } -} - -/** - * meta_window_get_user_time: - * @window: a #MetaWindow - * - * The user time represents a timestamp for the last time the user - * interacted with this window. Note this property is only available - * for non-override-redirect windows. - * - * The property is set by Mutter initially upon window creation, - * and updated thereafter on input events (key and button presses) seen by Mutter, - * client updates to the _NET_WM_USER_TIME property (if later than the current time) - * and when focusing the window. - * - * Returns: The last time the user interacted with this window. - */ -guint32 -meta_window_get_user_time (MetaWindow *window) -{ - return window->net_wm_user_time; -} - -void -meta_window_set_user_time (MetaWindow *window, - guint32 timestamp) -{ - /* FIXME: If Soeren's suggestion in bug 151984 is implemented, it will allow - * us to sanity check the timestamp here and ensure it doesn't correspond to - * a future time. - */ - - g_return_if_fail (!window->override_redirect); - - /* Only update the time if this timestamp is newer... */ - if (window->net_wm_user_time_set && - XSERVER_TIME_IS_BEFORE (timestamp, window->net_wm_user_time)) - { - meta_topic (META_DEBUG_STARTUP, - "Window %s _NET_WM_USER_TIME not updated to %u, because it " - "is less than %u", - window->desc, timestamp, window->net_wm_user_time); - } - else - { - meta_topic (META_DEBUG_STARTUP, - "Window %s has _NET_WM_USER_TIME of %u", - window->desc, timestamp); - window->net_wm_user_time_set = TRUE; - window->net_wm_user_time = timestamp; - if (XSERVER_TIME_IS_BEFORE (window->display->last_user_time, timestamp)) - window->display->last_user_time = timestamp; - - /* If this is a terminal, user interaction with it means the user likely - * doesn't want to have focus transferred for now due to new windows. - */ - if (meta_prefs_get_focus_new_windows () == G_DESKTOP_FOCUS_NEW_WINDOWS_STRICT && - window_is_terminal (window)) - window->display->allow_terminal_deactivation = FALSE; - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_USER_TIME]); - } -} - -/** - * meta_window_get_stable_sequence: - * @window: A #MetaWindow - * - * The stable sequence number is a monotonicially increasing - * unique integer assigned to each #MetaWindow upon creation. - * - * This number can be useful for sorting windows in a stable - * fashion. - * - * Returns: Internal sequence number for this window - */ -guint32 -meta_window_get_stable_sequence (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), 0); - - return window->stable_sequence; -} - -/* Sets the demands_attention hint on a window, but only - * if it's at least partially obscured (see #305882). - */ -void -meta_window_set_demands_attention (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - MetaRectangle candidate_rect, other_rect; - GList *stack = window->display->stack->sorted; - MetaWindow *other_window; - gboolean obscured = FALSE; - - MetaWorkspace *workspace = workspace_manager->active_workspace; - - if (window->wm_state_demands_attention) - return; - - if (!meta_window_located_on_workspace (window, workspace)) - { - /* windows on other workspaces are necessarily obscured */ - obscured = TRUE; - } - else if (window->minimized) - { - obscured = TRUE; - } - else - { - meta_window_get_frame_rect (window, &candidate_rect); - - /* The stack is sorted with the top windows first. */ - - while (stack != NULL && stack->data != window) - { - other_window = stack->data; - stack = stack->next; - - if (meta_window_located_on_workspace (other_window, workspace)) - { - meta_window_get_frame_rect (other_window, &other_rect); - - if (meta_rectangle_overlap (&candidate_rect, &other_rect)) - { - obscured = TRUE; - break; - } - } - } - } - - if (obscured) - { - meta_topic (META_DEBUG_WINDOW_OPS, - "Marking %s as needing attention", - window->desc); - - window->wm_state_demands_attention = TRUE; - set_net_wm_state (window); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_DEMANDS_ATTENTION]); - g_signal_emit_by_name (window->display, "window-demands-attention", - window); - } - else - { - /* If the window's in full view, there's no point setting the flag. */ - - meta_topic (META_DEBUG_WINDOW_OPS, - "Not marking %s as needing attention because " - "it's in full view", - window->desc); - } -} - -void -meta_window_unset_demands_attention (MetaWindow *window) -{ - meta_topic (META_DEBUG_WINDOW_OPS, - "Marking %s as not needing attention", window->desc); - - if (window->wm_state_demands_attention) - { - window->wm_state_demands_attention = FALSE; - set_net_wm_state (window); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_DEMANDS_ATTENTION]); - } -} - -/** - * meta_window_get_frame: (skip) - * @window: a #MetaWindow - * - */ -MetaFrame * -meta_window_get_frame (MetaWindow *window) -{ - return window->frame; -} - -/** - * meta_window_appears_focused: - * @window: a #MetaWindow - * - * Determines if the window should be drawn with a focused appearance. This is - * true for focused windows but also true for windows with a focused modal - * dialog attached. - * - * Return value: %TRUE if the window should be drawn with a focused frame - */ -gboolean -meta_window_appears_focused (MetaWindow *window) -{ - return window->has_focus || (window->attached_focus_window != NULL); -} - -gboolean -meta_window_has_focus (MetaWindow *window) -{ - return window->has_focus; -} - -gboolean -meta_window_is_shaded (MetaWindow *window) -{ - return window->shaded; -} - -/** - * meta_window_is_override_redirect: - * @window: A #MetaWindow - * - * Returns: %TRUE if this window isn't managed by mutter; it will - * control its own positioning and mutter won't draw decorations - * among other things. In X terminology this is "override redirect". - */ -gboolean -meta_window_is_override_redirect (MetaWindow *window) -{ - return window->override_redirect; -} - -/** - * meta_window_is_skip_taskbar: - * @window: A #MetaWindow - * - * Gets whether this window should be ignored by task lists. - * - * Return value: %TRUE if the skip bar hint is set. - */ -gboolean -meta_window_is_skip_taskbar (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), FALSE); - - return window->skip_taskbar; -} - -/** - * meta_window_get_display: - * @window: A #MetaWindow - * - * Returns: (transfer none): The display for @window - */ -MetaDisplay * -meta_window_get_display (MetaWindow *window) -{ - return window->display; -} - -/** - * meta_window_get_xwindow: (skip) - * @window: a #MetaWindow - * - */ -Window -meta_window_get_xwindow (MetaWindow *window) -{ - return window->xwindow; -} - -MetaWindowType -meta_window_get_window_type (MetaWindow *window) -{ - return window->type; -} - -/** - * meta_window_get_workspace: - * @window: a #MetaWindow - * - * Gets the #MetaWorkspace that the window is currently displayed on. - * If the window is on all workspaces, returns the currently active - * workspace. - * - * Return value: (transfer none): the #MetaWorkspace for the window - */ -MetaWorkspace * -meta_window_get_workspace (MetaWindow *window) -{ - MetaWorkspaceManager *workspace_manager = window->display->workspace_manager; - - if (window->on_all_workspaces) - return workspace_manager->active_workspace; - else - return window->workspace; -} - -gboolean -meta_window_is_on_all_workspaces (MetaWindow *window) -{ - return window->on_all_workspaces; -} - -gboolean -meta_window_is_hidden (MetaWindow *window) -{ - return window->hidden; -} - -const char * -meta_window_get_description (MetaWindow *window) -{ - if (!window) - return NULL; - - return window->desc; -} - -/** - * meta_window_get_wm_class: - * @window: a #MetaWindow - * - * Return the current value of the name part of WM_CLASS X property. - */ -const char * -meta_window_get_wm_class (MetaWindow *window) -{ - if (!window) - return NULL; - - return window->res_class; -} - -/** - * meta_window_get_wm_class_instance: - * @window: a #MetaWindow - * - * Return the current value of the instance part of WM_CLASS X property. - */ -const char * -meta_window_get_wm_class_instance (MetaWindow *window) -{ - if (!window) - return NULL; - - return window->res_name; -} - -/** - * meta_window_get_sandboxed_app_id: - * @window: a #MetaWindow - * - * Gets an unique id for a sandboxed app (currently flatpaks and snaps are - * supported). - * - * Return value: (transfer none): the sandboxed application ID or %NULL - **/ -const char * -meta_window_get_sandboxed_app_id (MetaWindow *window) -{ - /* We're abusing this API here not to break the gnome shell assumptions - * or adding a new function, to be renamed to generic names in new versions */ - return window->sandboxed_app_id; -} - -/** - * meta_window_get_gtk_theme_variant: - * @window: a #MetaWindow - * - * Return value: (transfer none): the theme variant or %NULL - **/ -const char * -meta_window_get_gtk_theme_variant (MetaWindow *window) -{ - return window->gtk_theme_variant; -} - -/** - * meta_window_get_gtk_application_id: - * @window: a #MetaWindow - * - * Return value: (transfer none): the application ID - **/ -const char * -meta_window_get_gtk_application_id (MetaWindow *window) -{ - return window->gtk_application_id; -} - -/** - * meta_window_get_gtk_unique_bus_name: - * @window: a #MetaWindow - * - * Return value: (transfer none): the unique name - **/ -const char * -meta_window_get_gtk_unique_bus_name (MetaWindow *window) -{ - return window->gtk_unique_bus_name; -} - -/** - * meta_window_get_gtk_application_object_path: - * @window: a #MetaWindow - * - * Return value: (transfer none): the object path - **/ -const char * -meta_window_get_gtk_application_object_path (MetaWindow *window) -{ - return window->gtk_application_object_path; -} - -/** - * meta_window_get_gtk_window_object_path: - * @window: a #MetaWindow - * - * Return value: (transfer none): the object path - **/ -const char * -meta_window_get_gtk_window_object_path (MetaWindow *window) -{ - return window->gtk_window_object_path; -} - -/** - * meta_window_get_gtk_app_menu_object_path: - * @window: a #MetaWindow - * - * Return value: (transfer none): the object path - **/ -const char * -meta_window_get_gtk_app_menu_object_path (MetaWindow *window) -{ - return window->gtk_app_menu_object_path; -} - -/** - * meta_window_get_gtk_menubar_object_path: - * @window: a #MetaWindow - * - * Return value: (transfer none): the object path - **/ -const char * -meta_window_get_gtk_menubar_object_path (MetaWindow *window) -{ - return window->gtk_menubar_object_path; -} - -/** - * meta_window_get_compositor_private: - * @window: a #MetaWindow - * - * Gets the compositor's wrapper object for @window. - * - * Return value: (transfer none): the wrapper object. - **/ -GObject * -meta_window_get_compositor_private (MetaWindow *window) -{ - if (!window) - return NULL; - return window->compositor_private; -} - -void -meta_window_set_compositor_private (MetaWindow *window, GObject *priv) -{ - if (!window) - return; - window->compositor_private = priv; -} - -const char * -meta_window_get_role (MetaWindow *window) -{ - if (!window) - return NULL; - - return window->role; -} - -/** - * meta_window_get_title: - * @window: a #MetaWindow - * - * Returns: the current title of the window. - */ -const char * -meta_window_get_title (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), NULL); - - return window->title; -} - -MetaStackLayer -meta_window_get_layer (MetaWindow *window) -{ - return window->layer; -} - -/** - * meta_window_get_transient_for: - * @window: a #MetaWindow - * - * Returns the #MetaWindow for the window that is pointed to by the - * WM_TRANSIENT_FOR hint on this window (see XGetTransientForHint() - * or XSetTransientForHint()). Metacity keeps transient windows above their - * parents. A typical usage of this hint is for a dialog that wants to stay - * above its associated window. - * - * Return value: (transfer none): the window this window is transient for, or - * %NULL if the WM_TRANSIENT_FOR hint is unset or does not point to a toplevel - * window that Metacity knows about. - */ -MetaWindow * -meta_window_get_transient_for (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), NULL); - - if (window->transient_for) - return window->transient_for; - else if (window->xtransient_for) - return meta_x11_display_lookup_x_window (window->display->x11_display, - window->xtransient_for); - else - return NULL; -} - -/** - * meta_window_get_pid: - * @window: a #MetaWindow - * - * Returns the pid of the process that created this window, if available - * to the windowing system. - * - * Note that the value returned by this is vulnerable to spoofing attacks - * by the client. - * - * Return value: the pid, or 0 if not known. - */ -pid_t -meta_window_get_pid (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), 0); - - if (window->client_pid == 0) - window->client_pid = META_WINDOW_GET_CLASS (window)->get_client_pid (window); - - return window->client_pid; -} - -/** - * meta_window_get_client_machine: - * @window: a #MetaWindow - * - * Returns name of the client machine from which this windows was created, - * if known (obtained from the WM_CLIENT_MACHINE property). - * - * Return value: (transfer none): the machine name, or NULL; the string is - * owned by the window manager and should not be freed or modified by the - * caller. - */ -const char * -meta_window_get_client_machine (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), NULL); - - return window->wm_client_machine; -} - -/** - * meta_window_is_remote: - * @window: a #MetaWindow - * - * Returns: %TRUE if this window originates from a host - * different from the one running mutter. - */ -gboolean -meta_window_is_remote (MetaWindow *window) -{ - return window->is_remote; -} - -/** - * meta_window_get_mutter_hints: - * @window: a #MetaWindow - * - * Gets the current value of the _MUTTER_HINTS property. - * - * The purpose of the hints is to allow fine-tuning of the Window Manager and - * Compositor behaviour on per-window basis, and is intended primarily for - * hints that are plugin-specific. - * - * The property is a list of colon-separated key=value pairs. The key names for - * any plugin-specific hints must be suitably namespaced to allow for shared - * use; 'mutter-' key prefix is reserved for internal use, and must not be used - * by plugins. - * - * Return value: (transfer none): the _MUTTER_HINTS string, or %NULL if no hints - * are set. - */ -const char * -meta_window_get_mutter_hints (MetaWindow *window) -{ - g_return_val_if_fail (META_IS_WINDOW (window), NULL); - - return window->mutter_hints; -} - -/** - * meta_window_get_frame_type: - * @window: a #MetaWindow - * - * Gets the type of window decorations that should be used for this window. - * - * Return value: the frame type - */ -MetaFrameType -meta_window_get_frame_type (MetaWindow *window) -{ - MetaFrameType base_type = META_FRAME_TYPE_LAST; - - switch (window->type) - { - case META_WINDOW_NORMAL: - base_type = META_FRAME_TYPE_NORMAL; - break; - - case META_WINDOW_DIALOG: - base_type = META_FRAME_TYPE_DIALOG; - break; - - case META_WINDOW_MODAL_DIALOG: - if (meta_window_is_attached_dialog (window)) - base_type = META_FRAME_TYPE_ATTACHED; - else - base_type = META_FRAME_TYPE_MODAL_DIALOG; - break; - - case META_WINDOW_MENU: - base_type = META_FRAME_TYPE_MENU; - break; - - case META_WINDOW_UTILITY: - base_type = META_FRAME_TYPE_UTILITY; - break; - - case META_WINDOW_DESKTOP: - case META_WINDOW_DOCK: - case META_WINDOW_TOOLBAR: - case META_WINDOW_SPLASHSCREEN: - case META_WINDOW_DROPDOWN_MENU: - case META_WINDOW_POPUP_MENU: - case META_WINDOW_TOOLTIP: - case META_WINDOW_NOTIFICATION: - case META_WINDOW_COMBO: - case META_WINDOW_DND: - case META_WINDOW_OVERRIDE_OTHER: - /* No frame */ - base_type = META_FRAME_TYPE_LAST; - break; - } - - if (base_type == META_FRAME_TYPE_LAST) - { - /* can't add border if undecorated */ - return META_FRAME_TYPE_LAST; - } - else if (window->border_only) - { - /* override base frame type */ - return META_FRAME_TYPE_BORDER; - } - else - { - return base_type; - } -} - -/** - * meta_window_get_frame_bounds: - * @window: a #MetaWindow - * - * Gets a region representing the outer bounds of the window's frame. - * - * Return value: (transfer none) (nullable): a #cairo_region_t - * holding the outer bounds of the window, or %NULL if the window - * doesn't have a frame. - */ -cairo_region_t * -meta_window_get_frame_bounds (MetaWindow *window) -{ - if (!window->frame_bounds) - { - if (window->frame) - window->frame_bounds = meta_frame_get_frame_bounds (window->frame); - } - - return window->frame_bounds; -} - -/** - * meta_window_is_attached_dialog: - * @window: a #MetaWindow - * - * Tests if @window is should be attached to its parent window. - * (If the "attach_modal_dialogs" option is not enabled, this will - * always return %FALSE.) - * - * Return value: whether @window should be attached to its parent - */ -gboolean -meta_window_is_attached_dialog (MetaWindow *window) -{ - return window->attached; -} - -/** - * meta_window_get_tile_match: - * @window: a #MetaWindow - * - * Returns the matching tiled window on the same monitor as @window. This is - * the topmost tiled window in a complementary tile mode that is: - * - * - on the same monitor; - * - on the same workspace; - * - spanning the remaining monitor width; - * - there is no 3rd window stacked between both tiled windows that's - * partially visible in the common edge. - * - * Return value: (transfer none) (nullable): the matching tiled window or - * %NULL if it doesn't exist. - */ -MetaWindow * -meta_window_get_tile_match (MetaWindow *window) -{ - return window->tile_match; -} - -void -meta_window_compute_tile_match (MetaWindow *window) -{ - window->tile_match = meta_window_find_tile_match (window, window->tile_mode); -} - -static MetaWindow * -meta_window_find_tile_match (MetaWindow *window, - MetaTileMode current_mode) -{ - MetaWindow *match; - MetaStack *stack; - MetaTileMode match_tile_mode = META_TILE_NONE; - - if (window->shaded || window->minimized) - return NULL; - - if (current_mode == META_TILE_LEFT) - match_tile_mode = META_TILE_RIGHT; - else if (current_mode == META_TILE_RIGHT) - match_tile_mode = META_TILE_LEFT; - else - return NULL; - - stack = window->display->stack; - - for (match = meta_stack_get_top (stack); - match; - match = meta_stack_get_below (stack, match, FALSE)) - { - if (!match->shaded && - !match->minimized && - match->tile_mode == match_tile_mode && - match->tile_monitor_number == window->tile_monitor_number && - meta_window_get_workspace (match) == meta_window_get_workspace (window)) - break; - } - - if (match) - { - MetaWindow *above, *bottommost, *topmost; - MetaRectangle above_rect, bottommost_rect, topmost_rect; - - if (meta_stack_windows_cmp (window->display->stack, match, window) > 0) - { - topmost = match; - bottommost = window; - } - else - { - topmost = window; - bottommost = match; - } - - meta_window_get_frame_rect (bottommost, &bottommost_rect); - meta_window_get_frame_rect (topmost, &topmost_rect); - - /* - * If we are looking for a tile match while actually being tiled, - * rather than a match for a potential tile mode, then discard - * windows with too much gap or overlap - */ - if (window->tile_mode == current_mode && - !(meta_grab_op_is_resizing (window->display->grab_op) && - window->display->grab_window == window && - window->tile_match != NULL)) - { - int threshold = meta_prefs_get_drag_threshold (); - if (ABS (topmost_rect.x - bottommost_rect.x - bottommost_rect.width) > threshold && - ABS (bottommost_rect.x - topmost_rect.x - topmost_rect.width) > threshold) - return NULL; - } - - /* - * If there's a window stacked in between which is partially visible - * behind the topmost tile we don't consider the tiles to match. - */ - for (above = meta_stack_get_above (stack, bottommost, FALSE); - above && above != topmost; - above = meta_stack_get_above (stack, above, FALSE)) - { - if (above->minimized || - above->monitor != window->monitor || - meta_window_get_workspace (above) != meta_window_get_workspace (window)) - continue; - - meta_window_get_frame_rect (above, &above_rect); - - if (meta_rectangle_overlap (&above_rect, &bottommost_rect) && - meta_rectangle_overlap (&above_rect, &topmost_rect)) - return NULL; - } - } - - return match; -} - -void -meta_window_set_title (MetaWindow *window, - const char *title) -{ - g_free (window->title); - window->title = g_strdup (title); - - if (window->frame) - meta_frame_update_title (window->frame); - - meta_window_update_desc (window); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_TITLE]); -} - -void -meta_window_set_wm_class (MetaWindow *window, - const char *wm_class, - const char *wm_instance) -{ - g_free (window->res_class); - g_free (window->res_name); - - window->res_name = g_strdup (wm_instance); - window->res_class = g_strdup (wm_class); - - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_WM_CLASS]); -} - -void -meta_window_set_gtk_dbus_properties (MetaWindow *window, - const char *application_id, - const char *unique_bus_name, - const char *appmenu_path, - const char *menubar_path, - const char *application_object_path, - const char *window_object_path) -{ - g_object_freeze_notify (G_OBJECT (window)); - - g_free (window->gtk_application_id); - window->gtk_application_id = g_strdup (application_id); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_APPLICATION_ID]); - - g_free (window->gtk_unique_bus_name); - window->gtk_unique_bus_name = g_strdup (unique_bus_name); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_UNIQUE_BUS_NAME]); - - g_free (window->gtk_app_menu_object_path); - window->gtk_app_menu_object_path = g_strdup (appmenu_path); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_APP_MENU_OBJECT_PATH]); - - g_free (window->gtk_menubar_object_path); - window->gtk_menubar_object_path = g_strdup (menubar_path); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_MENUBAR_OBJECT_PATH]); - - g_free (window->gtk_application_object_path); - window->gtk_application_object_path = g_strdup (application_object_path); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_APPLICATION_OBJECT_PATH]); - - g_free (window->gtk_window_object_path); - window->gtk_window_object_path = g_strdup (window_object_path); - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_GTK_WINDOW_OBJECT_PATH]); - - g_object_thaw_notify (G_OBJECT (window)); -} - -static gboolean -check_transient_for_loop (MetaWindow *window, - MetaWindow *parent) -{ - while (parent) - { - if (parent == window) - return TRUE; - parent = parent->transient_for; - } - - return FALSE; -} - -gboolean -meta_window_has_transient_type (MetaWindow *window) -{ - return (window->type == META_WINDOW_DIALOG || - window->type == META_WINDOW_MODAL_DIALOG || - window->type == META_WINDOW_TOOLBAR || - window->type == META_WINDOW_MENU || - window->type == META_WINDOW_UTILITY); -} - -void -meta_window_set_transient_for (MetaWindow *window, - MetaWindow *parent) -{ - if (check_transient_for_loop (window, parent)) - { - meta_warning ("Setting %s transient for %s would create a loop.", - window->desc, parent->desc); - return; - } - - if (meta_window_appears_focused (window) && window->transient_for != NULL) - meta_window_propagate_focus_appearance (window, FALSE); - - /* may now be a dialog */ - if (window->client_type == META_WINDOW_CLIENT_TYPE_X11) - { - meta_window_x11_recalc_window_type (window); - - if (!window->constructing) - { - /* If the window attaches, detaches, or changes attached - * parents, we need to destroy the MetaWindow and let a new one - * be created (which happens as a side effect of - * meta_window_unmanage()). The condition below is correct - * because we know window->transient_for has changed. - */ - if (window->attached || meta_window_should_attach_to_parent (window)) - { - guint32 timestamp; - - timestamp = - meta_display_get_current_time_roundtrip (window->display); - meta_window_unmanage (window, timestamp); - return; - } - } - } - else if (window->attached && parent == NULL) - { - guint32 timestamp; - - timestamp = - meta_display_get_current_time_roundtrip (window->display); - meta_window_unmanage (window, timestamp); - return; - } - /* We know this won't create a reference cycle because we check for loops */ - g_clear_object (&window->transient_for); - window->transient_for = parent ? g_object_ref (parent) : NULL; - - /* update stacking constraints */ - if (!window->override_redirect) - meta_stack_update_transient (window->display->stack, window); - - /* possibly change its group. We treat being a window's transient as - * equivalent to making it your group leader, to work around shortcomings - * in programs such as xmms-- see #328211. - */ - if (window->xtransient_for != None && - window->xgroup_leader != None && - window->xtransient_for != window->xgroup_leader) - meta_window_group_leader_changed (window); - - if (!window->constructing && !window->override_redirect) - meta_window_queue (window, META_QUEUE_MOVE_RESIZE | META_QUEUE_CALC_SHOWING); - - if (meta_window_appears_focused (window) && window->transient_for != NULL) - meta_window_propagate_focus_appearance (window, TRUE); -} - -void -meta_window_set_opacity (MetaWindow *window, - guint8 opacity) -{ - window->opacity = opacity; - - meta_compositor_window_opacity_changed (window->display->compositor, window); -} - -static void -reset_ignored_crossing_serials (MetaDisplay *display) -{ - int i; - - i = 0; - while (i < N_IGNORED_CROSSING_SERIALS) - { - display->ignored_crossing_serials[i] = 0; - ++i; - } -} - -typedef struct -{ - MetaWindow *window; - int pointer_x; - int pointer_y; -} MetaFocusData; - -static void -mouse_mode_focus (MetaWindow *window, - guint32 timestamp) -{ - MetaDisplay *display = window->display; - - if (window->override_redirect) - return; - - if (window->type != META_WINDOW_DESKTOP) - { - meta_topic (META_DEBUG_FOCUS, - "Focusing %s at time %u.", window->desc, timestamp); - - meta_window_focus (window, timestamp); - - if (meta_prefs_get_auto_raise ()) - meta_display_queue_autoraise_callback (display, window); - else - meta_topic (META_DEBUG_FOCUS, "Auto raise is disabled"); - } - else - { - /* In mouse focus mode, we defocus when the mouse *enters* - * the DESKTOP window, instead of defocusing on LeaveNotify. - * This is because having the mouse enter override-redirect - * child windows unfortunately causes LeaveNotify events that - * we can't distinguish from the mouse actually leaving the - * toplevel window as we expect. But, since we filter out - * EnterNotify events on override-redirect windows, this - * alternative mechanism works great. - */ - if (meta_prefs_get_focus_mode() == G_DESKTOP_FOCUS_MODE_MOUSE && - display->focus_window != NULL) - { - meta_topic (META_DEBUG_FOCUS, - "Unsetting focus from %s due to mouse entering " - "the DESKTOP window", - display->focus_window->desc); - meta_display_unset_input_focus (display, timestamp); - } - } -} - -static gboolean -window_has_pointer_wayland (MetaWindow *window) -{ - ClutterSeat *seat; - ClutterInputDevice *dev; - ClutterStage *stage; - ClutterActor *pointer_actor, *window_actor; - - seat = clutter_backend_get_default_seat (clutter_get_default_backend ()); - dev = clutter_seat_get_pointer (seat); - stage = CLUTTER_STAGE (meta_backend_get_stage (meta_get_backend ())); - pointer_actor = clutter_stage_get_device_actor (stage, dev, NULL); - window_actor = CLUTTER_ACTOR (meta_window_get_compositor_private (window)); - - return pointer_actor && clutter_actor_contains (window_actor, pointer_actor); -} - -static gboolean -window_has_pointer_x11 (MetaWindow *window) -{ - MetaX11Display *x11_display = window->display->x11_display; - Window root, child; - double root_x, root_y, x, y; - XIButtonState buttons; - XIModifierState mods; - XIGroupState group; - - meta_x11_error_trap_push (x11_display); - XIQueryPointer (x11_display->xdisplay, - META_VIRTUAL_CORE_POINTER_ID, - x11_display->xroot, - &root, &child, - &root_x, &root_y, &x, &y, - &buttons, &mods, &group); - meta_x11_error_trap_pop (x11_display); - free (buttons.mask); - - return meta_x11_display_lookup_x_window (x11_display, child) == window; -} - -gboolean -meta_window_has_pointer (MetaWindow *window) -{ - if (meta_is_wayland_compositor ()) - return window_has_pointer_wayland (window); - else - return window_has_pointer_x11 (window); -} - -static gboolean -window_focus_on_pointer_rest_callback (gpointer data) -{ - MetaFocusData *focus_data = data; - MetaWindow *window = focus_data->window; - MetaDisplay *display = window->display; - MetaBackend *backend = meta_get_backend (); - MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend); - graphene_point_t point; - guint32 timestamp; - - if (meta_prefs_get_focus_mode () == G_DESKTOP_FOCUS_MODE_CLICK) - goto out; - - meta_cursor_tracker_get_pointer (cursor_tracker, &point, NULL); - - if ((int) point.x != focus_data->pointer_x || - (int) point.y != focus_data->pointer_y) - { - focus_data->pointer_x = point.x; - focus_data->pointer_y = point.y; - return G_SOURCE_CONTINUE; - } - - if (!meta_window_has_pointer (window)) - goto out; - - timestamp = meta_display_get_current_time_roundtrip (display); - mouse_mode_focus (window, timestamp); - - out: - display->focus_timeout_id = 0; - return G_SOURCE_REMOVE; -} - -/* The interval, in milliseconds, we use in focus-follows-mouse - * mode to check whether the pointer has stopped moving after a - * crossing event. - */ -#define FOCUS_TIMEOUT_DELAY 25 - -static void -queue_focus_callback (MetaDisplay *display, - MetaWindow *window, - int pointer_x, - int pointer_y) -{ - MetaFocusData *focus_data; - - focus_data = g_new (MetaFocusData, 1); - focus_data->window = window; - focus_data->pointer_x = pointer_x; - focus_data->pointer_y = pointer_y; - - g_clear_handle_id (&display->focus_timeout_id, g_source_remove); - - display->focus_timeout_id = - g_timeout_add_full (G_PRIORITY_DEFAULT, - FOCUS_TIMEOUT_DELAY, - window_focus_on_pointer_rest_callback, - focus_data, - g_free); - g_source_set_name_by_id (display->focus_timeout_id, - "[mutter] window_focus_on_pointer_rest_callback"); -} - -void -meta_window_handle_enter (MetaWindow *window, - guint32 timestamp, - guint root_x, - guint root_y) -{ - MetaDisplay *display = window->display; - - switch (meta_prefs_get_focus_mode ()) - { - case G_DESKTOP_FOCUS_MODE_SLOPPY: - case G_DESKTOP_FOCUS_MODE_MOUSE: - display->mouse_mode = TRUE; - if (window->type != META_WINDOW_DOCK) - { - if (meta_prefs_get_focus_change_on_pointer_rest()) - queue_focus_callback (display, window, root_x, root_y); - else - mouse_mode_focus (window, timestamp); - - /* stop ignoring stuff */ - reset_ignored_crossing_serials (display); - } - break; - case G_DESKTOP_FOCUS_MODE_CLICK: - break; - } - - if (window->type == META_WINDOW_DOCK) - meta_window_raise (window); -} - -void -meta_window_handle_leave (MetaWindow *window) -{ - if (window->type == META_WINDOW_DOCK && !window->has_focus) - meta_window_lower (window); -} - -gboolean -meta_window_handle_ui_frame_event (MetaWindow *window, - const ClutterEvent *event) -{ - if (!window->frame) - return FALSE; - - return meta_ui_frame_handle_event (window->frame->ui_frame, event); -} - -void -meta_window_handle_ungrabbed_event (MetaWindow *window, - const ClutterEvent *event) -{ - MetaDisplay *display = window->display; - gboolean unmodified; - gboolean is_window_grab; - gboolean is_window_button_grab_allowed; - ClutterModifierType grab_mods, event_mods; - ClutterInputDevice *source; - gfloat x, y; - guint button; - - if (window->unmanaging) - return; - - if (event->type != CLUTTER_BUTTON_PRESS && - event->type != CLUTTER_TOUCH_BEGIN) - return; - - if (event->type == CLUTTER_TOUCH_BEGIN) - { - ClutterEventSequence *sequence; - - button = 1; - sequence = clutter_event_get_event_sequence (event); - if (!meta_display_is_pointer_emulating_sequence (window->display, sequence)) - return; - } - else - button = clutter_event_get_button (event); - - if (display->grab_op != META_GRAB_OP_NONE) - return; - - /* Some windows might not ask for input, in which case we might be here - * because we selected for ButtonPress on the root window. In that case, - * we have to take special care not to act for an override-redirect window. - */ - if (window->override_redirect) - return; - - /* Don't focus panels--they must explicitly request focus. - * See bug 160470 - */ - if (window->type != META_WINDOW_DOCK) - { - meta_topic (META_DEBUG_FOCUS, - "Focusing %s due to button %u press (display.c)", - window->desc, button); - meta_window_focus (window, event->any.time); - meta_window_check_alive (window, event->any.time); - } - else - /* However, do allow terminals to lose focus due to new - * window mappings after the user clicks on a panel. - */ - display->allow_terminal_deactivation = TRUE; - - /* We have three passive button grabs: - * - on any button, without modifiers => focuses and maybe raises the window - * - on resize button, with modifiers => start an interactive resizing - * (normally <Super>middle) - * - on move button, with modifiers => start an interactive move - * (normally <Super>left) - * - on menu button, with modifiers => show the window menu - * (normally <Super>right) - * - * We may get here because we actually have a button - * grab on the window, or because we're a wayland - * compositor and thus we see all the events, so we - * need to check if the event is interesting. - * We want an event that is not modified for a window. - * - * We may have other events on the window, for example - * a click on a frame button, but that's not for us to - * care about. Just let the event through. - */ - - grab_mods = meta_display_get_compositor_modifiers (display); - event_mods = clutter_event_get_state (event); - unmodified = (event_mods & grab_mods) == 0; - source = clutter_event_get_source_device (event); - is_window_button_grab_allowed = !display->focus_window || - !meta_window_shortcuts_inhibited (display->focus_window, source); - is_window_grab = (is_window_button_grab_allowed && - ((event_mods & grab_mods) == grab_mods)); - - clutter_event_get_coords (event, &x, &y); - - if (unmodified) - { - if (meta_prefs_get_raise_on_click ()) - meta_window_raise (window); - else - meta_topic (META_DEBUG_FOCUS, - "Not raising window on click due to don't-raise-on-click option"); - } - else if (is_window_grab && (int) button == meta_prefs_get_mouse_button_resize ()) - { - if (window->has_resize_func) - { - gboolean north, south; - gboolean west, east; - MetaRectangle frame_rect; - MetaGrabOp op = META_GRAB_OP_WINDOW_BASE; - - meta_window_get_frame_rect (window, &frame_rect); - - west = x < (frame_rect.x + 1 * frame_rect.width / 3); - east = x > (frame_rect.x + 2 * frame_rect.width / 3); - north = y < (frame_rect.y + 1 * frame_rect.height / 3); - south = y > (frame_rect.y + 2 * frame_rect.height / 3); - - if (west) - op |= META_GRAB_OP_WINDOW_DIR_WEST; - if (east) - op |= META_GRAB_OP_WINDOW_DIR_EAST; - if (north) - op |= META_GRAB_OP_WINDOW_DIR_NORTH; - if (south) - op |= META_GRAB_OP_WINDOW_DIR_SOUTH; - - if (op != META_GRAB_OP_WINDOW_BASE) - meta_display_begin_grab_op (display, - window, - op, - TRUE, - FALSE, - button, - 0, - event->any.time, - x, y); - } - } - else if (is_window_grab && (int) button == meta_prefs_get_mouse_button_menu ()) - { - if (meta_prefs_get_raise_on_click ()) - meta_window_raise (window); - meta_window_show_menu (window, - META_WINDOW_MENU_WM, - x, y); - } - else if (is_window_grab && (int) button == 1) - { - if (window->has_move_func) - { - meta_display_begin_grab_op (display, - window, - META_GRAB_OP_MOVING, - TRUE, - FALSE, - button, - 0, - event->any.time, - x, y); - } - } -} - -gboolean -meta_window_can_maximize (MetaWindow *window) -{ - return window->has_maximize_func; -} - -gboolean -meta_window_can_minimize (MetaWindow *window) -{ - return window->has_minimize_func; -} - -gboolean -meta_window_can_shade (MetaWindow *window) -{ - return window->has_shade_func; -} - -gboolean -meta_window_can_close (MetaWindow *window) -{ - return window->has_close_func; -} - -gboolean -meta_window_is_always_on_all_workspaces (MetaWindow *window) -{ - return window->always_sticky; -} - -gboolean -meta_window_is_above (MetaWindow *window) -{ - return window->wm_state_above; -} - -gboolean -meta_window_allows_move (MetaWindow *window) -{ - return META_WINDOW_ALLOWS_MOVE (window); -} - -gboolean -meta_window_allows_resize (MetaWindow *window) -{ - return META_WINDOW_ALLOWS_RESIZE (window); -} - -void -meta_window_set_urgent (MetaWindow *window, - gboolean urgent) -{ - if (window->urgent == urgent) - return; - - window->urgent = urgent; - g_object_notify_by_pspec (G_OBJECT (window), obj_props[PROP_URGENT]); - - if (urgent) - g_signal_emit_by_name (window->display, "window-marked-urgent", window); -} - -void -meta_window_grab_op_began (MetaWindow *window, - MetaGrabOp op) -{ - META_WINDOW_GET_CLASS (window)->grab_op_began (window, op); -} - -void -meta_window_grab_op_ended (MetaWindow *window, - MetaGrabOp op) -{ - META_WINDOW_GET_CLASS (window)->grab_op_ended (window, op); -} - -void -meta_window_emit_size_changed (MetaWindow *window) -{ - g_signal_emit (window, window_signals[SIZE_CHANGED], 0); -} - -MetaPlacementRule * -meta_window_get_placement_rule (MetaWindow *window) -{ - return window->placement.rule; -} - -void -meta_window_force_restore_shortcuts (MetaWindow *window, - ClutterInputDevice *source) -{ - META_WINDOW_GET_CLASS (window)->force_restore_shortcuts (window, source); -} - -gboolean -meta_window_shortcuts_inhibited (MetaWindow *window, - ClutterInputDevice *source) -{ - return META_WINDOW_GET_CLASS (window)->shortcuts_inhibited (window, source); -} - -gboolean -meta_window_is_focusable (MetaWindow *window) -{ - g_return_val_if_fail (!window->unmanaging, FALSE); - - return META_WINDOW_GET_CLASS (window)->is_focusable (window); -} - -gboolean -meta_window_can_ping (MetaWindow *window) -{ - g_return_val_if_fail (!window->unmanaging, FALSE); - - return META_WINDOW_GET_CLASS (window)->can_ping (window); -} - -gboolean -meta_window_is_stackable (MetaWindow *window) -{ - return META_WINDOW_GET_CLASS (window)->is_stackable (window); -} - -gboolean -meta_window_is_focus_async (MetaWindow *window) -{ - return META_WINDOW_GET_CLASS (window)->is_focus_async (window); -} - -MetaStackLayer -meta_window_calculate_layer (MetaWindow *window) -{ - return META_WINDOW_GET_CLASS (window)->calculate_layer (window); -} - -/** - * meta_window_get_id: - * @window: a #MetaWindow - * - * Returns the window id associated with window. - * - * Returns: The window id - */ -uint64_t -meta_window_get_id (MetaWindow *window) -{ - return window->id; -} - -/** - * meta_window_get_client_type: - * @window: a #MetaWindow - * - * Returns the #MetaWindowClientType of the window. - * - * Returns: The root ancestor window - */ -MetaWindowClientType -meta_window_get_client_type (MetaWindow *window) -{ - return window->client_type; -} |