2014-03-19 13:26:17 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Red Hat
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* Written by:
|
|
|
|
* Jasper St. Pierre <jstpierre@mecheye.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "window-wayland.h"
|
|
|
|
|
|
|
|
#include "window-private.h"
|
2014-03-19 14:30:12 +00:00
|
|
|
#include "boxes-private.h"
|
2014-03-19 13:36:15 +00:00
|
|
|
#include "stack-tracker.h"
|
|
|
|
#include "meta-wayland-surface.h"
|
2014-03-19 13:26:17 +00:00
|
|
|
|
|
|
|
struct _MetaWindowWayland
|
|
|
|
{
|
|
|
|
MetaWindow parent;
|
2014-04-28 20:19:06 +00:00
|
|
|
|
|
|
|
gboolean has_saved_pos;
|
|
|
|
int saved_x;
|
|
|
|
int saved_y;
|
2014-03-19 13:26:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct _MetaWindowWaylandClass
|
|
|
|
{
|
|
|
|
MetaWindowClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaWindowWayland, meta_window_wayland, META_TYPE_WINDOW)
|
|
|
|
|
2014-03-19 13:36:15 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_manage (MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = window->display;
|
|
|
|
|
|
|
|
meta_display_register_wayland_window (display, window);
|
|
|
|
|
|
|
|
{
|
|
|
|
MetaStackWindow stack_window;
|
|
|
|
stack_window.any.type = META_WINDOW_CLIENT_TYPE_WAYLAND;
|
|
|
|
stack_window.wayland.meta_window = window;
|
|
|
|
meta_stack_tracker_record_add (window->screen->stack_tracker,
|
|
|
|
&stack_window,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_wayland_unmanage (MetaWindow *window)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
MetaStackWindow stack_window;
|
|
|
|
stack_window.any.type = META_WINDOW_CLIENT_TYPE_WAYLAND;
|
|
|
|
stack_window.wayland.meta_window = window;
|
|
|
|
meta_stack_tracker_record_remove (window->screen->stack_tracker,
|
|
|
|
&stack_window,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
|
|
|
|
meta_display_unregister_wayland_window (window->display, window);
|
|
|
|
}
|
|
|
|
|
2014-03-20 19:12:44 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_ping (MetaWindow *window,
|
|
|
|
guint32 serial)
|
|
|
|
{
|
|
|
|
meta_wayland_surface_ping (window->surface, serial);
|
|
|
|
}
|
|
|
|
|
2014-03-20 19:07:44 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_delete (MetaWindow *window,
|
|
|
|
guint32 timestamp)
|
|
|
|
{
|
|
|
|
meta_wayland_surface_delete (window->surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_wayland_kill (MetaWindow *window)
|
|
|
|
{
|
2014-03-25 16:05:21 +00:00
|
|
|
MetaWaylandSurface *surface = window->surface;
|
|
|
|
struct wl_resource *resource = surface->resource;
|
|
|
|
|
|
|
|
/* Send the client an unrecoverable error to kill the client. */
|
|
|
|
wl_resource_post_error (resource,
|
|
|
|
WL_DISPLAY_ERROR_NO_MEMORY,
|
|
|
|
"User requested that we kill you. Sorry. Don't take it too personally.");
|
2014-03-20 19:07:44 +00:00
|
|
|
}
|
|
|
|
|
2014-03-20 19:16:57 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_focus (MetaWindow *window,
|
|
|
|
guint32 timestamp)
|
|
|
|
{
|
|
|
|
meta_display_set_input_focus_window (window->display,
|
|
|
|
window,
|
|
|
|
FALSE,
|
|
|
|
timestamp);
|
|
|
|
}
|
|
|
|
|
2014-03-19 14:30:12 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_move_resize_internal (MetaWindow *window,
|
|
|
|
int gravity,
|
|
|
|
MetaRectangle requested_rect,
|
|
|
|
MetaRectangle constrained_rect,
|
|
|
|
MetaMoveResizeFlags flags,
|
|
|
|
MetaMoveResizeResultFlags *result)
|
|
|
|
{
|
2014-04-28 20:19:06 +00:00
|
|
|
MetaWindowWayland *wl_window = META_WINDOW_WAYLAND (window);
|
2014-04-16 20:28:43 +00:00
|
|
|
gboolean should_move = FALSE;
|
2014-04-28 20:19:06 +00:00
|
|
|
gboolean use_saved_pos = FALSE;
|
2014-04-16 20:28:43 +00:00
|
|
|
|
2014-03-19 14:30:12 +00:00
|
|
|
g_assert (window->frame == NULL);
|
|
|
|
|
|
|
|
/* For wayland clients, the size is completely determined by the client,
|
|
|
|
* and while this allows to avoid some trickery with frames and the resulting
|
|
|
|
* lagging, we also need to insist a bit when the constraints would apply
|
|
|
|
* a different size than the client decides.
|
|
|
|
*
|
|
|
|
* Note that this is not generally a problem for normal toplevel windows (the
|
|
|
|
* constraints don't see the size hints, or just change the position), but
|
|
|
|
* it can be for maximized or fullscreen.
|
|
|
|
*/
|
|
|
|
|
2014-04-28 20:21:06 +00:00
|
|
|
if (flags & META_IS_WAYLAND_RESIZE)
|
2014-03-19 14:30:12 +00:00
|
|
|
{
|
|
|
|
/* This is a call to wl_surface_commit(), ignore the constrained_rect and
|
|
|
|
* update the real client size to match the buffer size.
|
|
|
|
*/
|
|
|
|
|
2014-04-16 20:30:11 +00:00
|
|
|
*result |= META_MOVE_RESIZE_RESULT_RESIZED;
|
2014-03-19 14:30:12 +00:00
|
|
|
window->rect.width = requested_rect.width;
|
|
|
|
window->rect.height = requested_rect.height;
|
2014-04-16 20:28:43 +00:00
|
|
|
|
|
|
|
/* This is a commit of an attach. We should move the window to match the
|
|
|
|
* new position the client wants. */
|
|
|
|
should_move = TRUE;
|
2014-04-28 20:19:06 +00:00
|
|
|
use_saved_pos = TRUE;
|
2014-03-19 14:30:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (constrained_rect.width != window->rect.width ||
|
|
|
|
constrained_rect.height != window->rect.height)
|
|
|
|
{
|
2014-04-28 20:19:06 +00:00
|
|
|
wl_window->has_saved_pos = TRUE;
|
|
|
|
wl_window->saved_x = constrained_rect.x;
|
|
|
|
wl_window->saved_y = constrained_rect.y;
|
|
|
|
|
2014-03-19 14:30:12 +00:00
|
|
|
meta_wayland_surface_configure_notify (window->surface,
|
|
|
|
constrained_rect.width,
|
|
|
|
constrained_rect.height);
|
|
|
|
}
|
2014-04-28 20:21:06 +00:00
|
|
|
else
|
2014-04-16 20:28:43 +00:00
|
|
|
{
|
|
|
|
/* We're just moving the window, so we don't need to wait for a configure
|
|
|
|
* and then ack to simply move the window. */
|
|
|
|
should_move = TRUE;
|
|
|
|
}
|
2014-03-19 14:30:12 +00:00
|
|
|
|
2014-04-28 19:50:40 +00:00
|
|
|
if (should_move)
|
2014-04-16 20:16:01 +00:00
|
|
|
{
|
2014-04-28 20:19:06 +00:00
|
|
|
int new_x, new_y;
|
|
|
|
|
|
|
|
if (use_saved_pos && wl_window->has_saved_pos)
|
|
|
|
{
|
|
|
|
int dx, dy;
|
|
|
|
|
|
|
|
/* The dx/dy that the client asked for. */
|
|
|
|
dx = requested_rect.x - window->rect.x;
|
|
|
|
dy = requested_rect.y - window->rect.y;
|
|
|
|
|
|
|
|
new_x = wl_window->saved_x + dx;
|
|
|
|
new_y = wl_window->saved_y + dy;
|
|
|
|
|
|
|
|
wl_window->has_saved_pos = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
new_x = constrained_rect.x;
|
|
|
|
new_y = constrained_rect.y;
|
|
|
|
}
|
2014-04-28 20:20:53 +00:00
|
|
|
|
2014-04-28 19:51:41 +00:00
|
|
|
if (new_x != window->rect.x || new_y != window->rect.y)
|
2014-04-28 19:50:40 +00:00
|
|
|
{
|
|
|
|
*result |= META_MOVE_RESIZE_RESULT_MOVED;
|
2014-04-28 19:51:41 +00:00
|
|
|
window->rect.x = new_x;
|
|
|
|
window->rect.y = new_y;
|
2014-04-28 19:50:40 +00:00
|
|
|
}
|
2014-03-19 14:30:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-19 13:26:17 +00:00
|
|
|
static void
|
|
|
|
meta_window_wayland_init (MetaWindowWayland *window_wayland)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_window_wayland_class_init (MetaWindowWaylandClass *klass)
|
|
|
|
{
|
2014-03-19 13:36:15 +00:00
|
|
|
MetaWindowClass *window_class = META_WINDOW_CLASS (klass);
|
|
|
|
|
|
|
|
window_class->manage = meta_window_wayland_manage;
|
|
|
|
window_class->unmanage = meta_window_wayland_unmanage;
|
2014-03-20 19:12:44 +00:00
|
|
|
window_class->ping = meta_window_wayland_ping;
|
2014-03-20 19:07:44 +00:00
|
|
|
window_class->delete = meta_window_wayland_delete;
|
|
|
|
window_class->kill = meta_window_wayland_kill;
|
2014-03-20 19:16:57 +00:00
|
|
|
window_class->focus = meta_window_wayland_focus;
|
2014-03-19 14:30:12 +00:00
|
|
|
window_class->move_resize_internal = meta_window_wayland_move_resize_internal;
|
2014-03-19 13:26:17 +00:00
|
|
|
}
|
2014-04-28 21:44:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_window_move_resize_wayland:
|
|
|
|
*
|
|
|
|
* Complete a resize operation from a wayland client.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
meta_window_wayland_move_resize (MetaWindow *window,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int dx,
|
|
|
|
int dy)
|
|
|
|
{
|
|
|
|
int x, y;
|
|
|
|
MetaMoveResizeFlags flags;
|
|
|
|
|
|
|
|
flags = META_IS_WAYLAND_RESIZE;
|
|
|
|
|
|
|
|
meta_window_get_position (window, &x, &y);
|
|
|
|
|
|
|
|
/* dx/dy are ignored during resizing */
|
|
|
|
if (!meta_grab_op_is_resizing (window->display->grab_op))
|
|
|
|
{
|
|
|
|
if (dx != 0 || dy != 0)
|
|
|
|
{
|
|
|
|
x += dx;
|
|
|
|
y += dy;
|
|
|
|
flags |= META_IS_MOVE_ACTION;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (width != window->rect.width || height != window->rect.height)
|
|
|
|
flags |= META_IS_RESIZE_ACTION;
|
|
|
|
|
|
|
|
meta_window_move_resize_internal (window, flags,
|
|
|
|
meta_resize_gravity_from_grab_op (window->display->grab_op),
|
|
|
|
x, y, width, height);
|
|
|
|
meta_window_save_user_window_placement (window);
|
|
|
|
}
|