wlroots/rootston/wl_shell.c

128 lines
4.3 KiB
C
Raw Normal View History

#include <assert.h>
#include <stdlib.h>
#include <stdbool.h>
#include <wayland-server.h>
#include <wlr/types/wlr_box.h>
#include <wlr/types/wlr_surface.h>
#include <wlr/types/wlr_wl_shell.h>
#include <wlr/util/log.h>
#include "rootston/desktop.h"
#include "rootston/server.h"
#include "rootston/input.h"
2017-09-30 14:43:48 +00:00
static void resize(struct roots_view *view, uint32_t width, uint32_t height) {
assert(view->type == ROOTS_WL_SHELL_VIEW);
struct wlr_wl_shell_surface *surf = view->wl_shell_surface;
wlr_wl_shell_surface_configure(surf, WL_SHELL_SURFACE_RESIZE_NONE, width,
height);
}
static void close(struct roots_view *view) {
assert(view->type == ROOTS_WL_SHELL_VIEW);
struct wlr_wl_shell_surface *surf = view->wl_shell_surface;
wl_client_destroy(surf->client);
}
2017-09-30 08:39:06 +00:00
static void handle_request_move(struct wl_listener *listener, void *data) {
2017-09-27 22:29:37 +00:00
struct roots_wl_shell_surface *roots_surface =
wl_container_of(listener, roots_surface, request_move);
struct roots_view *view = roots_surface->view;
struct roots_input *input = view->desktop->server->input;
struct wlr_wl_shell_surface_move_event *e = data;
2017-09-28 23:48:55 +00:00
const struct roots_input_event *event = get_input_event(input, e->serial);
2017-09-27 22:29:37 +00:00
if (!event || input->mode != ROOTS_CURSOR_PASSTHROUGH) {
return;
}
2017-09-28 23:48:55 +00:00
view_begin_move(input, event->cursor, view);
2017-09-27 22:29:37 +00:00
}
2017-09-30 14:43:48 +00:00
static void handle_request_resize(struct wl_listener *listener, void *data) {
struct roots_wl_shell_surface *roots_surface =
2017-09-30 17:38:31 +00:00
wl_container_of(listener, roots_surface, request_resize);
2017-09-30 14:43:48 +00:00
struct roots_view *view = roots_surface->view;
struct roots_input *input = view->desktop->server->input;
struct wlr_wl_shell_surface_resize_event *e = data;
const struct roots_input_event *event = get_input_event(input, e->serial);
if (!event || input->mode != ROOTS_CURSOR_PASSTHROUGH) {
return;
}
view_begin_resize(input, event->cursor, view, e->edges);
}
2017-10-08 15:59:38 +00:00
static void handle_surface_commit(struct wl_listener *listener, void *data) {
2017-10-16 18:35:16 +00:00
// TODO do we need to do anything here?
2017-10-08 15:59:38 +00:00
}
static void handle_destroy(struct wl_listener *listener, void *data) {
struct roots_wl_shell_surface *roots_surface =
wl_container_of(listener, roots_surface, destroy);
wl_list_remove(&roots_surface->destroy.link);
2017-09-27 22:29:37 +00:00
wl_list_remove(&roots_surface->request_move.link);
wl_list_remove(&roots_surface->request_resize.link);
view_destroy(roots_surface->view);
free(roots_surface);
}
static int shell_surface_compare_equals(const void *item, const void *cmp_to) {
const struct roots_view *view = item;
if (view->type == ROOTS_WL_SHELL_VIEW && view->wl_shell_surface == cmp_to) {
return 0;
}
return -1;
}
void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
struct roots_desktop *desktop =
wl_container_of(listener, desktop, wl_shell_surface);
struct wlr_wl_shell_surface *surface = data;
wlr_log(L_DEBUG, "new shell surface: title=%s, class=%s",
surface->title, surface->class);
wlr_wl_shell_surface_ping(surface);
2017-09-27 12:48:53 +00:00
struct roots_wl_shell_surface *roots_surface =
calloc(1, sizeof(struct roots_wl_shell_surface));
2017-10-08 15:59:38 +00:00
if (!roots_surface) {
return;
}
roots_surface->destroy.notify = handle_destroy;
2017-09-27 22:29:37 +00:00
wl_signal_add(&surface->events.destroy, &roots_surface->destroy);
2017-09-30 08:39:06 +00:00
roots_surface->request_move.notify = handle_request_move;
2017-09-27 22:29:37 +00:00
wl_signal_add(&surface->events.request_move, &roots_surface->request_move);
2017-09-30 14:43:48 +00:00
roots_surface->request_resize.notify = handle_request_resize;
2017-10-08 15:59:38 +00:00
wl_signal_add(&surface->events.request_resize,
&roots_surface->request_resize);
roots_surface->surface_commit.notify = handle_surface_commit;
2017-10-08 17:12:28 +00:00
wl_signal_add(&surface->surface->events.commit,
2017-10-08 15:59:38 +00:00
&roots_surface->surface_commit);
2017-09-27 12:48:53 +00:00
struct roots_view *view = calloc(1, sizeof(struct roots_view));
if (!view) {
free(roots_surface);
return;
}
2017-09-27 12:48:53 +00:00
view->type = ROOTS_WL_SHELL_VIEW;
2017-09-27 12:48:53 +00:00
view->wl_shell_surface = surface;
view->roots_wl_shell_surface = roots_surface;
2017-09-27 13:03:35 +00:00
view->wlr_surface = surface->surface;
2017-09-30 14:43:48 +00:00
view->resize = resize;
view->close = close;
2017-09-27 12:48:53 +00:00
view->desktop = desktop;
roots_surface->view = view;
2017-09-28 23:48:55 +00:00
list_add(desktop->views, view);
2017-10-16 18:35:16 +00:00
view_initialize(view);
2017-10-19 16:33:02 +00:00
if (surface->state == WLR_WL_SHELL_SURFACE_STATE_TRANSIENT) {
// we need to map it relative to the parent
int i = list_seq_find(desktop->views, shell_surface_compare_equals,
surface->parent);
2017-10-19 16:33:02 +00:00
if (i != -1) {
struct roots_view *parent = desktop->views->items[i];
view_set_position(view,
parent->x + surface->transient_state->x,
parent->y + surface->transient_state->y);
}
}
}