mirror of
https://github.com/brl/mutter.git
synced 2024-12-26 12:52:14 +00:00
ca5b27baf5
As with xdg-toplevel, a gtk-surface can be unmanaged by the compositor without the client knowing about it, meaning the client may still send updates and make requests. Handle this gracefully by ignoring them. The client needs to reset all the state anyway, if it wants to remap the same surface. https://gitlab.gnome.org/GNOME/mutter/issues/240
392 lines
12 KiB
C
392 lines
12 KiB
C
/*
|
|
* Wayland Support
|
|
*
|
|
* Copyright (C) 2012,2013 Intel Corporation
|
|
* 2013-2016 Red Hat, Inc.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "wayland/meta-wayland-gtk-shell.h"
|
|
|
|
#include "core/bell.h"
|
|
#include "core/window-private.h"
|
|
#include "wayland/meta-wayland-private.h"
|
|
#include "wayland/meta-wayland-surface.h"
|
|
#include "wayland/meta-wayland-versions.h"
|
|
#include "wayland/meta-window-wayland.h"
|
|
|
|
#include "gtk-shell-server-protocol.h"
|
|
|
|
static GQuark quark_gtk_surface_data = 0;
|
|
|
|
typedef struct _MetaWaylandGtkSurface
|
|
{
|
|
struct wl_resource *resource;
|
|
MetaWaylandSurface *surface;
|
|
gboolean is_modal;
|
|
gulong configure_handler_id;
|
|
} MetaWaylandGtkSurface;
|
|
|
|
static void
|
|
gtk_surface_destructor (struct wl_resource *resource)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface = wl_resource_get_user_data (resource);
|
|
|
|
if (gtk_surface->surface)
|
|
{
|
|
g_object_steal_qdata (G_OBJECT (gtk_surface->surface),
|
|
quark_gtk_surface_data);
|
|
g_signal_handler_disconnect (gtk_surface->surface,
|
|
gtk_surface->configure_handler_id);
|
|
}
|
|
|
|
g_free (gtk_surface);
|
|
}
|
|
|
|
static void
|
|
gtk_surface_set_dbus_properties (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
const char *application_id,
|
|
const char *app_menu_path,
|
|
const char *menubar_path,
|
|
const char *window_object_path,
|
|
const char *application_object_path,
|
|
const char *unique_bus_name)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface = wl_resource_get_user_data (resource);
|
|
MetaWaylandSurface *surface = gtk_surface->surface;
|
|
MetaWindow *window;
|
|
|
|
window = surface->window;
|
|
if (!window)
|
|
return;
|
|
|
|
meta_window_set_gtk_dbus_properties (window,
|
|
application_id,
|
|
unique_bus_name,
|
|
app_menu_path,
|
|
menubar_path,
|
|
application_object_path,
|
|
window_object_path);
|
|
}
|
|
|
|
static void
|
|
gtk_surface_set_modal (struct wl_client *client,
|
|
struct wl_resource *resource)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface = wl_resource_get_user_data (resource);
|
|
MetaWaylandSurface *surface = gtk_surface->surface;
|
|
MetaWindow *window;
|
|
|
|
window = surface->window;
|
|
if (!window)
|
|
return;
|
|
|
|
if (gtk_surface->is_modal)
|
|
return;
|
|
|
|
gtk_surface->is_modal = TRUE;
|
|
meta_window_set_type (window, META_WINDOW_MODAL_DIALOG);
|
|
}
|
|
|
|
static void
|
|
gtk_surface_unset_modal (struct wl_client *client,
|
|
struct wl_resource *resource)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface = wl_resource_get_user_data (resource);
|
|
MetaWaylandSurface *surface = gtk_surface->surface;
|
|
MetaWindow *window;
|
|
|
|
window = surface->window;
|
|
if (!window)
|
|
return;
|
|
|
|
if (!gtk_surface->is_modal)
|
|
return;
|
|
|
|
gtk_surface->is_modal = FALSE;
|
|
meta_window_set_type (window, META_WINDOW_NORMAL);
|
|
}
|
|
|
|
static void
|
|
gtk_surface_present (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
uint32_t timestamp)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface = wl_resource_get_user_data (resource);
|
|
MetaWaylandSurface *surface = gtk_surface->surface;
|
|
MetaWindow *window;
|
|
|
|
window = surface->window;
|
|
if (!window)
|
|
return;
|
|
|
|
meta_window_activate_full (window, timestamp,
|
|
META_CLIENT_TYPE_APPLICATION, NULL);
|
|
}
|
|
|
|
static const struct gtk_surface1_interface meta_wayland_gtk_surface_interface = {
|
|
gtk_surface_set_dbus_properties,
|
|
gtk_surface_set_modal,
|
|
gtk_surface_unset_modal,
|
|
gtk_surface_present,
|
|
};
|
|
|
|
static void
|
|
gtk_surface_surface_destroyed (MetaWaylandGtkSurface *gtk_surface)
|
|
{
|
|
wl_resource_set_implementation (gtk_surface->resource,
|
|
NULL, NULL, NULL);
|
|
gtk_surface->surface = NULL;
|
|
}
|
|
|
|
static void
|
|
fill_edge_states (struct wl_array *states,
|
|
MetaWindow *window)
|
|
{
|
|
uint32_t *s;
|
|
|
|
/* Top */
|
|
if (window->edge_constraints[0] != META_EDGE_CONSTRAINT_MONITOR)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_EDGE_CONSTRAINT_RESIZABLE_TOP;
|
|
}
|
|
|
|
/* Right */
|
|
if (window->edge_constraints[1] != META_EDGE_CONSTRAINT_MONITOR)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_EDGE_CONSTRAINT_RESIZABLE_RIGHT;
|
|
}
|
|
|
|
/* Bottom */
|
|
if (window->edge_constraints[2] != META_EDGE_CONSTRAINT_MONITOR)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_EDGE_CONSTRAINT_RESIZABLE_BOTTOM;
|
|
}
|
|
|
|
/* Left */
|
|
if (window->edge_constraints[3] != META_EDGE_CONSTRAINT_MONITOR)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_EDGE_CONSTRAINT_RESIZABLE_LEFT;
|
|
}
|
|
}
|
|
|
|
static void
|
|
send_configure_edges (MetaWaylandGtkSurface *gtk_surface,
|
|
MetaWindow *window)
|
|
{
|
|
struct wl_array edge_states;
|
|
|
|
wl_array_init (&edge_states);
|
|
fill_edge_states (&edge_states, window);
|
|
|
|
gtk_surface1_send_configure_edges (gtk_surface->resource, &edge_states);
|
|
|
|
wl_array_release (&edge_states);
|
|
}
|
|
|
|
static void
|
|
fill_states (struct wl_array *states,
|
|
MetaWindow *window,
|
|
struct wl_resource *resource)
|
|
{
|
|
uint32_t *s;
|
|
guint version;
|
|
|
|
version = wl_resource_get_version (resource);
|
|
|
|
if (version < GTK_SURFACE1_CONFIGURE_EDGES_SINCE_VERSION &&
|
|
(window->tile_mode == META_TILE_LEFT ||
|
|
window->tile_mode == META_TILE_RIGHT))
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_STATE_TILED;
|
|
}
|
|
|
|
if (version >= GTK_SURFACE1_STATE_TILED_TOP_SINCE_VERSION &&
|
|
window->edge_constraints[0] != META_EDGE_CONSTRAINT_NONE)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_STATE_TILED_TOP;
|
|
}
|
|
|
|
if (version >= GTK_SURFACE1_STATE_TILED_RIGHT_SINCE_VERSION &&
|
|
window->edge_constraints[1] != META_EDGE_CONSTRAINT_NONE)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_STATE_TILED_RIGHT;
|
|
}
|
|
|
|
if (version >= GTK_SURFACE1_STATE_TILED_BOTTOM_SINCE_VERSION &&
|
|
window->edge_constraints[2] != META_EDGE_CONSTRAINT_NONE)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_STATE_TILED_BOTTOM;
|
|
}
|
|
|
|
if (version >= GTK_SURFACE1_STATE_TILED_LEFT_SINCE_VERSION &&
|
|
window->edge_constraints[3] != META_EDGE_CONSTRAINT_NONE)
|
|
{
|
|
s = wl_array_add (states, sizeof *s);
|
|
*s = GTK_SURFACE1_STATE_TILED_LEFT;
|
|
}
|
|
}
|
|
|
|
static void
|
|
send_configure (MetaWaylandGtkSurface *gtk_surface,
|
|
MetaWindow *window)
|
|
{
|
|
struct wl_array states;
|
|
|
|
wl_array_init (&states);
|
|
fill_states (&states, window, gtk_surface->resource);
|
|
|
|
gtk_surface1_send_configure (gtk_surface->resource, &states);
|
|
|
|
wl_array_release (&states);
|
|
}
|
|
|
|
static void
|
|
on_configure (MetaWaylandSurface *surface,
|
|
MetaWaylandGtkSurface *gtk_surface)
|
|
{
|
|
send_configure (gtk_surface, surface->window);
|
|
|
|
|
|
if (wl_resource_get_version (gtk_surface->resource) >= GTK_SURFACE1_CONFIGURE_EDGES_SINCE_VERSION)
|
|
send_configure_edges (gtk_surface, surface->window);
|
|
}
|
|
|
|
static void
|
|
gtk_shell_get_gtk_surface (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
guint32 id,
|
|
struct wl_resource *surface_resource)
|
|
{
|
|
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
|
|
MetaWaylandGtkSurface *gtk_surface;
|
|
|
|
gtk_surface = g_object_get_qdata (G_OBJECT (surface), quark_gtk_surface_data);
|
|
if (gtk_surface)
|
|
{
|
|
wl_resource_post_error (surface_resource,
|
|
WL_DISPLAY_ERROR_INVALID_OBJECT,
|
|
"gtk_shell::get_gtk_surface already requested");
|
|
return;
|
|
}
|
|
|
|
gtk_surface = g_new0 (MetaWaylandGtkSurface, 1);
|
|
gtk_surface->surface = surface;
|
|
gtk_surface->resource = wl_resource_create (client,
|
|
>k_surface1_interface,
|
|
wl_resource_get_version (resource),
|
|
id);
|
|
wl_resource_set_implementation (gtk_surface->resource,
|
|
&meta_wayland_gtk_surface_interface,
|
|
gtk_surface, gtk_surface_destructor);
|
|
|
|
gtk_surface->configure_handler_id = g_signal_connect (surface,
|
|
"configure",
|
|
G_CALLBACK (on_configure),
|
|
gtk_surface);
|
|
|
|
g_object_set_qdata_full (G_OBJECT (surface),
|
|
quark_gtk_surface_data,
|
|
gtk_surface,
|
|
(GDestroyNotify) gtk_surface_surface_destroyed);
|
|
}
|
|
|
|
static void
|
|
gtk_shell_set_startup_id (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
const char *startup_id)
|
|
{
|
|
MetaDisplay *display;
|
|
|
|
display = meta_get_display ();
|
|
meta_startup_notification_remove_sequence (display->startup_notification,
|
|
startup_id);
|
|
}
|
|
|
|
static void
|
|
gtk_shell_system_bell (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
struct wl_resource *gtk_surface_resource)
|
|
{
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
if (gtk_surface_resource)
|
|
{
|
|
MetaWaylandGtkSurface *gtk_surface =
|
|
wl_resource_get_user_data (gtk_surface_resource);
|
|
MetaWaylandSurface *surface = gtk_surface->surface;
|
|
|
|
if (!surface->window)
|
|
return;
|
|
|
|
meta_bell_notify (display, surface->window);
|
|
}
|
|
else
|
|
{
|
|
meta_bell_notify (display, NULL);
|
|
}
|
|
}
|
|
|
|
static const struct gtk_shell1_interface meta_wayland_gtk_shell_interface = {
|
|
gtk_shell_get_gtk_surface,
|
|
gtk_shell_set_startup_id,
|
|
gtk_shell_system_bell,
|
|
};
|
|
|
|
static void
|
|
bind_gtk_shell (struct wl_client *client,
|
|
void *data,
|
|
guint32 version,
|
|
guint32 id)
|
|
{
|
|
struct wl_resource *resource;
|
|
uint32_t capabilities = 0;
|
|
|
|
resource = wl_resource_create (client, >k_shell1_interface, version, id);
|
|
wl_resource_set_implementation (resource, &meta_wayland_gtk_shell_interface,
|
|
data, NULL);
|
|
|
|
if (!meta_prefs_get_show_fallback_app_menu ())
|
|
capabilities = GTK_SHELL1_CAPABILITY_GLOBAL_APP_MENU;
|
|
|
|
gtk_shell1_send_capabilities (resource, capabilities);
|
|
}
|
|
|
|
void
|
|
meta_wayland_gtk_shell_init (MetaWaylandCompositor *compositor)
|
|
{
|
|
quark_gtk_surface_data =
|
|
g_quark_from_static_string ("-meta-wayland-gtk-shell-surface-data");
|
|
|
|
if (wl_global_create (compositor->wayland_display,
|
|
>k_shell1_interface,
|
|
META_GTK_SHELL1_VERSION,
|
|
compositor, bind_gtk_shell) == NULL)
|
|
g_error ("Failed to register a global gtk-shell object");
|
|
}
|