NEDM/xdg_shell.c

392 lines
13 KiB
C

// Copyright 2020 - 2025, project-repo and the NEDM contributors
// SPDX-License-Identifier: MIT
#define _POSIX_C_SOURCE 200809L
#include <assert.h>
#include <stdbool.h>
#include <wayland-server-core.h>
#include <wlr/types/wlr_output_layout.h>
#include <wlr/types/wlr_scene.h>
#include <wlr/types/wlr_xdg_decoration_v1.h>
#include <wlr/types/wlr_xdg_shell.h>
#include <wlr/util/box.h>
#include <wlr/util/edges.h>
#include <wlr/util/log.h>
#include "output.h"
#include "server.h"
#include "view.h"
#include "workspace.h"
#include "xdg_shell.h"
static void
xdg_decoration_handle_destroy(struct wl_listener *listener,
__attribute__((unused)) void *data) {
struct nedm_xdg_decoration *xdg_decoration =
wl_container_of(listener, xdg_decoration, destroy);
wl_list_remove(&xdg_decoration->destroy.link);
wl_list_remove(&xdg_decoration->request_mode.link);
wl_list_remove(&xdg_decoration->link);
free(xdg_decoration);
}
static void
xdg_decoration_handle_request_mode(struct wl_listener *listener,
__attribute__((unused)) void *_data) {
struct nedm_xdg_decoration *xdg_decoration =
wl_container_of(listener, xdg_decoration, request_mode);
if(xdg_decoration->wlr_decoration->toplevel->base->initialized) {
wlr_xdg_toplevel_decoration_v1_set_mode(
xdg_decoration->wlr_decoration,
WLR_XDG_TOPLEVEL_DECORATION_V1_MODE_SERVER_SIDE);
}
}
static void
popup_unconstrain(struct nedm_xdg_shell_popup *popup) {
struct nedm_view *view = popup->view;
struct wlr_box output_toplevel_box = {
.x = -view->ox,
.y = -view->oy,
.width = view->workspace->output->wlr_output->width,
.height = view->workspace->output->wlr_output->height};
wlr_xdg_popup_unconstrain_from_box(popup->wlr_popup, &output_toplevel_box);
}
static struct nedm_xdg_shell_view *
xdg_shell_view_from_view(struct nedm_view *view) {
return (struct nedm_xdg_shell_view *)view;
}
static const struct nedm_xdg_shell_view *
xdg_shell_view_from_const_view(const struct nedm_view *view) {
return (const struct nedm_xdg_shell_view *)view;
}
static pid_t
get_pid(const struct nedm_view *view) {
pid_t pid;
struct wl_client *client =
wl_resource_get_client(view->wlr_surface->resource);
wl_client_get_credentials(client, &pid, NULL, NULL);
return pid;
}
static char *
get_title(const struct nedm_view *view) {
const struct nedm_xdg_shell_view *xdg_shell_view =
xdg_shell_view_from_const_view(view);
return xdg_shell_view->toplevel->title;
}
static bool
is_primary(const struct nedm_view *view) {
const struct nedm_xdg_shell_view *xdg_shell_view =
xdg_shell_view_from_const_view(view);
struct wlr_xdg_toplevel *parent = xdg_shell_view->toplevel->parent;
return parent == NULL;
}
static void
activate(struct nedm_view *view, bool activate) {
struct nedm_xdg_shell_view *xdg_shell_view = xdg_shell_view_from_view(view);
wlr_xdg_toplevel_set_activated(xdg_shell_view->toplevel, activate);
}
static void
close(struct nedm_view *view) {
struct nedm_xdg_shell_view *xdg_shell_view = xdg_shell_view_from_view(view);
if(view->type == NEDM_XDG_SHELL_VIEW) {
wlr_xdg_toplevel_send_close(xdg_shell_view->toplevel);
}
}
static void
maximize(struct nedm_view *view, int width, int height) {
struct nedm_xdg_shell_view *xdg_shell_view = xdg_shell_view_from_view(view);
wlr_xdg_toplevel_set_size(xdg_shell_view->toplevel, width, height);
enum wlr_edges edges =
WLR_EDGE_LEFT | WLR_EDGE_RIGHT | WLR_EDGE_TOP | WLR_EDGE_BOTTOM;
wlr_xdg_toplevel_set_tiled(xdg_shell_view->toplevel, edges);
}
static void
destroy(struct nedm_view *view) {
struct nedm_xdg_shell_view *xdg_shell_view = xdg_shell_view_from_view(view);
free(xdg_shell_view);
}
static void
handle_xdg_shell_surface_request_fullscreen(
struct wl_listener *listener, __attribute__((unused)) void *data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, request_fullscreen);
/**
* Certain clients do not like figuring out their own window geometry if
* they display in fullscreen mode, so we set it here (if the view was
* already mapped).
*/
if(xdg_shell_view->view.workspace != NULL) {
struct wlr_box layout_box;
wlr_output_layout_get_box(
xdg_shell_view->view.server->output_layout,
xdg_shell_view->view.workspace->output->wlr_output, &layout_box);
wlr_xdg_toplevel_set_size(xdg_shell_view->toplevel, layout_box.width,
layout_box.height);
}
wlr_xdg_toplevel_set_fullscreen(
xdg_shell_view->toplevel,
xdg_shell_view->toplevel->requested.fullscreen);
}
static void
handle_xdg_shell_surface_unmap(struct wl_listener *listener,
__attribute__((unused)) void *_data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, unmap);
struct nedm_view *view = &xdg_shell_view->view;
wl_list_remove(&xdg_shell_view->new_popup.link);
wl_list_remove(&xdg_shell_view->request_fullscreen.link);
view_unmap(view);
}
struct nedm_xdg_decoration *
xdg_decoration_from_surface(struct wlr_surface *surface,
struct nedm_server *server) {
struct nedm_xdg_decoration *deco;
wl_list_for_each(deco, &server->xdg_decorations, link) {
if(deco->wlr_decoration->toplevel->base->surface == surface) {
return deco;
}
}
return NULL;
}
static void
handle_xdg_shell_surface_map(struct wl_listener *listener,
__attribute__((unused)) void *_data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, map);
struct nedm_view *view = &xdg_shell_view->view;
xdg_shell_view->new_popup.notify = handle_xdg_shell_popup_new;
wl_signal_add(&xdg_shell_view->toplevel->base->events.new_popup,
&xdg_shell_view->new_popup);
xdg_shell_view->request_fullscreen.notify =
handle_xdg_shell_surface_request_fullscreen;
wl_signal_add(&xdg_shell_view->toplevel->events.request_fullscreen,
&xdg_shell_view->request_fullscreen);
view_map(view, xdg_shell_view->toplevel->base->surface,
view->server->curr_output
->workspaces[view->server->curr_output->curr_workspace]);
}
static void
handle_xdg_shell_surface_destroy(struct wl_listener *listener,
__attribute__((unused)) void *_data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, destroy);
struct nedm_view *view = &xdg_shell_view->view;
wl_list_remove(&xdg_shell_view->map.link);
wl_list_remove(&xdg_shell_view->unmap.link);
wl_list_remove(&xdg_shell_view->destroy.link);
wl_list_remove(&xdg_shell_view->commit.link);
xdg_shell_view->toplevel = NULL;
view_destroy(view);
}
static const struct nedm_view_impl xdg_shell_view_impl = {.get_pid = get_pid,
.get_title = get_title,
.is_primary =
is_primary,
.activate = activate,
.close = close,
.maximize = maximize,
.destroy = destroy};
void
handle_xdg_shell_toplevel_commit(struct wl_listener *listener,
__attribute__((unused)) void *data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, commit);
struct nedm_view *view = &xdg_shell_view->view;
struct wlr_xdg_surface *xdg_surface = xdg_shell_view->toplevel->base;
if(xdg_surface->initial_commit) {
struct nedm_xdg_decoration *decoration =
xdg_decoration_from_surface(xdg_surface->surface, view->server);
if(decoration != NULL) {
wlr_xdg_toplevel_decoration_v1_set_mode(
decoration->wlr_decoration,
WLR_XDG_TOPLEVEL_DECORATION_V1_MODE_SERVER_SIDE);
}
wlr_xdg_surface_schedule_configure(xdg_surface);
wlr_xdg_toplevel_set_wm_capabilities(
xdg_shell_view->toplevel, XDG_TOPLEVEL_WM_CAPABILITIES_FULLSCREEN);
}
}
void
handle_xdg_shell_toplevel_new(struct wl_listener *listener, void *data) {
struct nedm_server *server =
wl_container_of(listener, server, new_xdg_shell_toplevel);
struct wlr_xdg_toplevel *xdg_toplevel = data;
wlr_xdg_surface_ping(xdg_toplevel->base);
struct nedm_xdg_shell_view *xdg_shell_view =
calloc(1, sizeof(struct nedm_xdg_shell_view));
if(!xdg_shell_view) {
wlr_log(WLR_ERROR, "Failed to allocate XDG Shell view");
return;
}
view_init(&xdg_shell_view->view, NEDM_XDG_SHELL_VIEW, &xdg_shell_view_impl,
server);
xdg_shell_view->toplevel = xdg_toplevel;
xdg_shell_view->commit.notify = handle_xdg_shell_toplevel_commit;
wl_signal_add(&xdg_toplevel->base->surface->events.commit,
&xdg_shell_view->commit);
xdg_shell_view->map.notify = handle_xdg_shell_surface_map;
wl_signal_add(&xdg_toplevel->base->surface->events.map,
&xdg_shell_view->map);
xdg_shell_view->unmap.notify = handle_xdg_shell_surface_unmap;
wl_signal_add(&xdg_toplevel->base->surface->events.unmap,
&xdg_shell_view->unmap);
xdg_shell_view->destroy.notify = handle_xdg_shell_surface_destroy;
wl_signal_add(&xdg_toplevel->base->events.destroy,
&xdg_shell_view->destroy);
wlr_scene_xdg_surface_create(xdg_shell_view->view.scene_tree,
xdg_toplevel->base);
xdg_toplevel->base->data = xdg_shell_view;
return;
}
static void
handle_xdg_shell_popup_destroy(struct wl_listener *listener,
__attribute__((unused)) void *data) {
struct nedm_xdg_shell_popup *popup =
wl_container_of(listener, popup, destroy);
wl_list_remove(&popup->new_popup.link);
wl_list_remove(&popup->destroy.link);
wl_list_remove(&popup->commit.link);
wl_list_remove(&popup->reposition.link);
wlr_scene_node_destroy(&popup->scene_tree->node);
free(popup);
}
static void
handle_xdg_shell_popup_commit(struct wl_listener *listener,
__attribute__((unused)) void *data) {
struct nedm_xdg_shell_popup *popup = wl_container_of(listener, popup, commit);
if(popup->wlr_popup->base->initial_commit) {
popup_unconstrain(popup);
}
}
static void
handle_xdg_shell_popup_reposition(struct wl_listener *listener,
__attribute__((unused)) void *data) {
struct nedm_xdg_shell_popup *popup =
wl_container_of(listener, popup, reposition);
popup_unconstrain(popup);
}
static struct nedm_xdg_shell_popup *
create_xdg_popup(struct wlr_xdg_popup *wlr_popup, struct nedm_view *view,
struct wlr_scene_tree *parent_tree);
void
handle_xdg_shell_popup_new_popup(struct wl_listener *listener, void *data) {
struct nedm_xdg_shell_popup *popup =
wl_container_of(listener, popup, new_popup);
struct wlr_xdg_popup *wlr_popup = data;
create_xdg_popup(wlr_popup, popup->view, popup->xdg_surface_tree);
}
static struct nedm_xdg_shell_popup *
create_xdg_popup(struct wlr_xdg_popup *wlr_popup, struct nedm_view *view,
struct wlr_scene_tree *parent_tree) {
struct wlr_xdg_surface *xdg_surface = wlr_popup->base;
struct nedm_xdg_shell_popup *popup = calloc(1, sizeof(*popup));
if(!popup) {
return NULL;
}
popup->wlr_popup = wlr_popup;
popup->view = view;
struct nedm_xdg_shell_view *xdg_view = wl_container_of(view, xdg_view, view);
xdg_surface->data = xdg_view;
popup->scene_tree = wlr_scene_tree_create(parent_tree);
if(!popup->scene_tree) {
free(popup);
return NULL;
}
popup->xdg_surface_tree =
wlr_scene_xdg_surface_create(popup->scene_tree, xdg_surface);
if(!popup->xdg_surface_tree) {
wlr_scene_node_destroy(&popup->scene_tree->node);
free(popup);
return NULL;
}
wl_signal_add(&xdg_surface->surface->events.commit, &popup->commit);
popup->commit.notify = handle_xdg_shell_popup_commit;
wl_signal_add(&xdg_surface->events.new_popup, &popup->new_popup);
popup->new_popup.notify = handle_xdg_shell_popup_new_popup;
wl_signal_add(&wlr_popup->events.reposition, &popup->reposition);
popup->reposition.notify = handle_xdg_shell_popup_reposition;
wl_signal_add(&wlr_popup->events.destroy, &popup->destroy);
popup->destroy.notify = handle_xdg_shell_popup_destroy;
return popup;
}
void
handle_xdg_shell_popup_new(struct wl_listener *listener, void *data) {
struct nedm_xdg_shell_view *xdg_shell_view =
wl_container_of(listener, xdg_shell_view, new_popup);
struct wlr_xdg_popup *xdg_popup = data;
create_xdg_popup(xdg_popup, &xdg_shell_view->view,
xdg_shell_view->view.scene_tree);
return;
}
void
handle_xdg_toplevel_decoration(struct wl_listener *listener, void *data) {
struct nedm_server *server =
wl_container_of(listener, server, xdg_toplevel_decoration);
struct wlr_xdg_toplevel_decoration_v1 *wlr_decoration = data;
struct nedm_xdg_decoration *xdg_decoration =
calloc(1, sizeof(struct nedm_xdg_decoration));
if(!xdg_decoration) {
return;
}
wl_list_insert(&server->xdg_decorations, &xdg_decoration->link);
xdg_decoration->wlr_decoration = wlr_decoration;
xdg_decoration->server = server;
xdg_decoration->destroy.notify = xdg_decoration_handle_destroy;
wl_signal_add(&wlr_decoration->events.destroy, &xdg_decoration->destroy);
xdg_decoration->request_mode.notify = xdg_decoration_handle_request_mode;
wl_signal_add(&wlr_decoration->events.request_mode,
&xdg_decoration->request_mode);
}