mutter/src/wayland/meta-wayland-viewporter.c
Michel Dänzer 6379e54847 wayland/surface: Prepare for decoupled surface & resource lifetimes
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>
2022-12-01 12:41:32 +01:00

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");
}