6379e54847
Need to deal with surface->resource == NULL and surface->pending_state == NULL in some places. v2: * Avoid expanding conditions to multiple lines. (Georges Basile Stavracas Neto) v3: * Use a single bailout condition in meta_wayland_client_owns_window as well. v4: * Remove spare empty line in meta_wayland_surface_apply_state. (Robert Mader) * Add wl_resource_post_error calls in xdg-shell request handlers. (Robert Mader) * Drop checks in functions which can only be called if there's a valid resource. * Drop more checks which are unnecessary due to leaving the SURFACE_DESTROY signal emission in wl_surface_destructor later. v5: * Move resource = surface->resource assignments to if (!resource) tests. (Jonas Ådahl) v6: * Fix style issue per check-style.py. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1880>
257 lines
8.0 KiB
C
257 lines
8.0 KiB
C
/*
|
|
* Wayland Support
|
|
*
|
|
* Copyright (C) 2018-2019 Robert Mader <robert.mader@posteo.de>
|
|
*
|
|
* 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 "meta-wayland-viewporter.h"
|
|
|
|
#include <glib.h>
|
|
|
|
#include "wayland/meta-wayland-private.h"
|
|
#include "wayland/meta-wayland-subsurface.h"
|
|
#include "wayland/meta-wayland-surface.h"
|
|
#include "wayland/meta-wayland-versions.h"
|
|
|
|
#include "viewporter-server-protocol.h"
|
|
|
|
static void
|
|
wp_viewport_destructor (struct wl_resource *resource)
|
|
{
|
|
MetaWaylandSurface *surface;
|
|
MetaWaylandSurfaceState *pending;
|
|
|
|
surface = wl_resource_get_user_data (resource);
|
|
if (!surface)
|
|
return;
|
|
|
|
g_clear_signal_handler (&surface->viewport.destroy_handler_id, surface);
|
|
|
|
pending = meta_wayland_surface_get_pending_state (surface);
|
|
if (pending)
|
|
{
|
|
pending->viewport_src_rect.size.width = -1;
|
|
pending->viewport_dst_width = -1;
|
|
pending->has_new_viewport_src_rect = TRUE;
|
|
pending->has_new_viewport_dst_size = TRUE;
|
|
}
|
|
|
|
surface->viewport.resource = NULL;
|
|
}
|
|
|
|
static void
|
|
on_surface_destroyed (MetaWaylandSurface *surface)
|
|
{
|
|
wl_resource_set_user_data (surface->viewport.resource, NULL);
|
|
}
|
|
|
|
static void
|
|
wp_viewport_destroy (struct wl_client *client,
|
|
struct wl_resource *resource)
|
|
{
|
|
wl_resource_destroy (resource);
|
|
}
|
|
|
|
static void
|
|
wp_viewport_set_source (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
wl_fixed_t src_x,
|
|
wl_fixed_t src_y,
|
|
wl_fixed_t src_width,
|
|
wl_fixed_t src_height)
|
|
{
|
|
MetaWaylandSurface *surface;
|
|
float new_x;
|
|
float new_y;
|
|
float new_width;
|
|
float new_height;
|
|
|
|
surface = wl_resource_get_user_data (resource);
|
|
if (!surface)
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_NO_SURFACE,
|
|
"wl_surface for this viewport no longer exists");
|
|
return;
|
|
}
|
|
|
|
new_x = wl_fixed_to_double (src_x);
|
|
new_y = wl_fixed_to_double (src_y);
|
|
new_width = wl_fixed_to_double (src_width);
|
|
new_height = wl_fixed_to_double (src_height);
|
|
|
|
if ((new_x >= 0 && new_y >= 0 &&
|
|
new_width > 0 && new_height > 0) ||
|
|
(new_x == -1 && new_y == -1 &&
|
|
new_width == -1 && new_height == -1))
|
|
{
|
|
MetaWaylandSurfaceState *pending;
|
|
|
|
pending = meta_wayland_surface_get_pending_state (surface);
|
|
if (!pending)
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_NO_SURFACE,
|
|
"wl_surface for this viewport no longer exists");
|
|
return;
|
|
}
|
|
|
|
pending->viewport_src_rect.origin.x = new_x;
|
|
pending->viewport_src_rect.origin.y = new_y;
|
|
pending->viewport_src_rect.size.width = new_width;
|
|
pending->viewport_src_rect.size.height = new_height;
|
|
pending->has_new_viewport_src_rect = TRUE;
|
|
}
|
|
else
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_BAD_VALUE,
|
|
"x and y values must be zero or positive and "
|
|
"width and height valuest must be positive or "
|
|
"all values must be -1 to unset the viewport");
|
|
}
|
|
}
|
|
|
|
static void
|
|
wp_viewport_set_destination (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
int dst_width,
|
|
int dst_height)
|
|
{
|
|
MetaWaylandSurface *surface;
|
|
|
|
surface = wl_resource_get_user_data (resource);
|
|
if (!surface)
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_NO_SURFACE,
|
|
"wl_surface for this viewport no longer exists");
|
|
return;
|
|
}
|
|
|
|
if ((dst_width > 0 && dst_height > 0) ||
|
|
(dst_width == -1 && dst_height == -1))
|
|
{
|
|
MetaWaylandSurfaceState *pending;
|
|
|
|
pending = meta_wayland_surface_get_pending_state (surface);
|
|
if (!pending)
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_NO_SURFACE,
|
|
"wl_surface for this viewport no longer exists");
|
|
return;
|
|
}
|
|
|
|
pending->viewport_dst_width = dst_width;
|
|
pending->viewport_dst_height = dst_height;
|
|
pending->has_new_viewport_dst_size = TRUE;
|
|
}
|
|
else
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORT_ERROR_BAD_VALUE,
|
|
"all values must be either positive or -1");
|
|
}
|
|
}
|
|
|
|
static const struct wp_viewport_interface meta_wayland_viewport_interface = {
|
|
wp_viewport_destroy,
|
|
wp_viewport_set_source,
|
|
wp_viewport_set_destination,
|
|
};
|
|
|
|
static void
|
|
wp_viewporter_destroy (struct wl_client *client,
|
|
struct wl_resource *resource)
|
|
{
|
|
wl_resource_destroy (resource);
|
|
}
|
|
|
|
static void
|
|
wp_viewporter_get_viewport (struct wl_client *client,
|
|
struct wl_resource *resource,
|
|
uint32_t viewport_id,
|
|
struct wl_resource *surface_resource)
|
|
{
|
|
MetaWaylandSurface *surface;
|
|
struct wl_resource *viewport_resource;
|
|
|
|
surface = wl_resource_get_user_data (surface_resource);
|
|
if (surface->viewport.resource)
|
|
{
|
|
wl_resource_post_error (resource,
|
|
WP_VIEWPORTER_ERROR_VIEWPORT_EXISTS,
|
|
"viewport already exists on surface");
|
|
return;
|
|
}
|
|
|
|
viewport_resource = wl_resource_create (client,
|
|
&wp_viewport_interface,
|
|
wl_resource_get_version (resource),
|
|
viewport_id);
|
|
wl_resource_set_implementation (viewport_resource,
|
|
&meta_wayland_viewport_interface,
|
|
surface,
|
|
wp_viewport_destructor);
|
|
|
|
surface->viewport.resource = viewport_resource;
|
|
surface->viewport.destroy_handler_id =
|
|
g_signal_connect (surface,
|
|
"destroy",
|
|
G_CALLBACK (on_surface_destroyed),
|
|
NULL);
|
|
}
|
|
|
|
static const struct wp_viewporter_interface meta_wayland_viewporter_interface = {
|
|
wp_viewporter_destroy,
|
|
wp_viewporter_get_viewport,
|
|
};
|
|
|
|
static void
|
|
wp_viewporter_bind (struct wl_client *client,
|
|
void *data,
|
|
uint32_t version,
|
|
uint32_t id)
|
|
{
|
|
struct wl_resource *resource;
|
|
|
|
resource = wl_resource_create (client,
|
|
&wp_viewporter_interface,
|
|
version,
|
|
id);
|
|
wl_resource_set_implementation (resource,
|
|
&meta_wayland_viewporter_interface,
|
|
data,
|
|
NULL);
|
|
}
|
|
|
|
void
|
|
meta_wayland_init_viewporter (MetaWaylandCompositor *compositor)
|
|
{
|
|
if (wl_global_create (compositor->wayland_display,
|
|
&wp_viewporter_interface,
|
|
META_WP_VIEWPORTER_VERSION,
|
|
compositor,
|
|
wp_viewporter_bind) == NULL)
|
|
g_error ("Failed to register a global wl-viewporter object");
|
|
}
|