235 lines
7.7 KiB
C
235 lines
7.7 KiB
C
// Copyright 2020 - 2025, project-repo and the NEDM contributors
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
#include "config.h"
|
|
|
|
#include <X11/Xutil.h>
|
|
#include <stdbool.h>
|
|
#include <wayland-server-core.h>
|
|
#include <wlr/types/wlr_output_layout.h>
|
|
#include <wlr/types/wlr_scene.h>
|
|
#include <wlr/util/box.h>
|
|
#include <wlr/util/log.h>
|
|
#if NEDM_HAS_XWAYLAND
|
|
#include <wlr/xwayland.h>
|
|
#endif
|
|
|
|
#include "output.h"
|
|
#include "server.h"
|
|
#include "view.h"
|
|
#include "workspace.h"
|
|
#include "xwayland.h"
|
|
|
|
struct nedm_xwayland_view *
|
|
xwayland_view_from_view(struct nedm_view *view) {
|
|
return (struct nedm_xwayland_view *)view;
|
|
}
|
|
|
|
static const struct nedm_xwayland_view *
|
|
xwayland_view_from_const_view(const struct nedm_view *view) {
|
|
return (const struct nedm_xwayland_view *)view;
|
|
}
|
|
|
|
bool
|
|
xwayland_view_should_manage(const struct nedm_view *view) {
|
|
const struct nedm_xwayland_view *xwayland_view =
|
|
xwayland_view_from_const_view(view);
|
|
struct wlr_xwayland_surface *xwayland_surface =
|
|
xwayland_view->xwayland_surface;
|
|
return !xwayland_surface->override_redirect;
|
|
}
|
|
|
|
static pid_t
|
|
get_pid(const struct nedm_view *view) {
|
|
struct wlr_xwayland_surface *surf =
|
|
wlr_xwayland_surface_try_from_wlr_surface(view->wlr_surface);
|
|
if(surf == NULL) {
|
|
return -1;
|
|
} else {
|
|
return surf->pid;
|
|
}
|
|
}
|
|
|
|
static char *
|
|
get_title(const struct nedm_view *view) {
|
|
const struct nedm_xwayland_view *xwayland_view =
|
|
xwayland_view_from_const_view(view);
|
|
return xwayland_view->xwayland_surface->title;
|
|
}
|
|
|
|
static bool
|
|
is_primary(const struct nedm_view *view) {
|
|
const struct nedm_xwayland_view *xwayland_view =
|
|
xwayland_view_from_const_view(view);
|
|
struct wlr_xwayland_surface *parent =
|
|
xwayland_view->xwayland_surface->parent;
|
|
return parent == NULL;
|
|
}
|
|
|
|
static void
|
|
activate(struct nedm_view *view, bool activate) {
|
|
struct nedm_xwayland_view *xwayland_view = xwayland_view_from_view(view);
|
|
wlr_xwayland_surface_activate(xwayland_view->xwayland_surface, activate);
|
|
if(activate && !xwayland_view->xwayland_surface->override_redirect) {
|
|
wlr_xwayland_surface_restack(xwayland_view->xwayland_surface, NULL,
|
|
XCB_STACK_MODE_ABOVE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
close(struct nedm_view *view) {
|
|
struct nedm_xwayland_view *xwayland_view = xwayland_view_from_view(view);
|
|
wlr_xwayland_surface_close(xwayland_view->xwayland_surface);
|
|
}
|
|
|
|
static void
|
|
maximize(struct nedm_view *view, int width, int height) {
|
|
struct nedm_xwayland_view *xwayland_view = xwayland_view_from_view(view);
|
|
xcb_size_hints_t *hints = xwayland_view->xwayland_surface->size_hints;
|
|
|
|
if(hints != NULL && hints->flags & PMaxSize) {
|
|
if(width > hints->max_width) {
|
|
width = hints->max_width;
|
|
}
|
|
|
|
if(height > hints->max_height) {
|
|
height = hints->max_height;
|
|
}
|
|
}
|
|
|
|
wlr_xwayland_surface_configure(
|
|
xwayland_view->xwayland_surface,
|
|
view->ox + output_get_layout_box(view->server->curr_output).x,
|
|
view->oy + output_get_layout_box(view->server->curr_output).y, width,
|
|
height);
|
|
wlr_xwayland_surface_set_maximized(xwayland_view->xwayland_surface, true,
|
|
true);
|
|
}
|
|
|
|
static void
|
|
destroy(struct nedm_view *view) {
|
|
struct nedm_xwayland_view *xwayland_view = xwayland_view_from_view(view);
|
|
free(xwayland_view);
|
|
}
|
|
|
|
static void
|
|
handle_xwayland_surface_request_fullscreen(struct wl_listener *listener,
|
|
__attribute__((unused)) void *data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, request_fullscreen);
|
|
struct wlr_xwayland_surface *xwayland_surface =
|
|
xwayland_view->xwayland_surface;
|
|
wlr_xwayland_surface_set_fullscreen(xwayland_view->xwayland_surface,
|
|
xwayland_surface->fullscreen);
|
|
}
|
|
|
|
static void
|
|
handle_xwayland_surface_unmap(struct wl_listener *listener,
|
|
__attribute__((unused)) void *_data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, unmap);
|
|
struct nedm_view *view = &xwayland_view->view;
|
|
|
|
view_unmap(view);
|
|
}
|
|
|
|
static void
|
|
handle_xwayland_surface_map(struct wl_listener *listener,
|
|
__attribute__((unused)) void *_data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, map);
|
|
struct nedm_view *view = &xwayland_view->view;
|
|
if(!xwayland_view_should_manage(view)) {
|
|
view->ox = xwayland_view->xwayland_surface->x -
|
|
view->server->curr_output->layout_box.x;
|
|
view->oy = xwayland_view->xwayland_surface->y -
|
|
view->server->curr_output->layout_box.y;
|
|
}
|
|
|
|
view_map(view, xwayland_view->xwayland_surface->surface,
|
|
view->server->curr_output
|
|
->workspaces[view->server->curr_output->curr_workspace]);
|
|
xwayland_view->scene_tree = wlr_scene_subsurface_tree_create(
|
|
view->scene_tree, xwayland_view->xwayland_surface->surface);
|
|
}
|
|
|
|
static void
|
|
handle_xwayland_surface_destroy(struct wl_listener *listener,
|
|
__attribute__((unused)) void *_data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, destroy);
|
|
struct nedm_view *view = &xwayland_view->view;
|
|
|
|
wl_list_remove(&xwayland_view->associate.link);
|
|
wl_list_remove(&xwayland_view->dissociate.link);
|
|
wl_list_remove(&xwayland_view->destroy.link);
|
|
wl_list_remove(&xwayland_view->request_fullscreen.link);
|
|
xwayland_view->xwayland_surface = NULL;
|
|
|
|
view_destroy(view);
|
|
}
|
|
|
|
static const struct nedm_view_impl xwayland_view_impl = {
|
|
.get_title = get_title,
|
|
.get_pid = get_pid,
|
|
.is_primary = is_primary,
|
|
.activate = activate,
|
|
.close = close,
|
|
.maximize = maximize,
|
|
.destroy = destroy,
|
|
};
|
|
|
|
static void
|
|
handle_xwayland_surface_associate(struct wl_listener *listener,
|
|
__attribute__((unused)) void *data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, associate);
|
|
struct wlr_xwayland_surface *xsurface = xwayland_view->xwayland_surface;
|
|
wl_signal_add(&xsurface->surface->events.unmap, &xwayland_view->unmap);
|
|
xwayland_view->unmap.notify = handle_xwayland_surface_unmap;
|
|
wl_signal_add(&xsurface->surface->events.map, &xwayland_view->map);
|
|
xwayland_view->map.notify = handle_xwayland_surface_map;
|
|
}
|
|
|
|
static void
|
|
handle_xwayland_surface_dissociate(struct wl_listener *listener,
|
|
__attribute__((unused)) void *data) {
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
wl_container_of(listener, xwayland_view, dissociate);
|
|
wl_list_remove(&xwayland_view->map.link);
|
|
wl_list_remove(&xwayland_view->unmap.link);
|
|
}
|
|
|
|
void
|
|
handle_xwayland_surface_new(struct wl_listener *listener, void *data) {
|
|
struct nedm_server *server =
|
|
wl_container_of(listener, server, new_xwayland_surface);
|
|
struct wlr_xwayland_surface *xwayland_surface = data;
|
|
|
|
struct nedm_xwayland_view *xwayland_view =
|
|
calloc(1, sizeof(struct nedm_xwayland_view));
|
|
if(!xwayland_view) {
|
|
wlr_log(WLR_ERROR, "Failed to allocate XWayland view");
|
|
return;
|
|
}
|
|
|
|
xwayland_view->scene_tree = NULL;
|
|
|
|
view_init(&xwayland_view->view, NEDM_XWAYLAND_VIEW, &xwayland_view_impl,
|
|
server);
|
|
xwayland_view->xwayland_surface = xwayland_surface;
|
|
|
|
xwayland_view->associate.notify = handle_xwayland_surface_associate;
|
|
wl_signal_add(&xwayland_surface->events.associate,
|
|
&xwayland_view->associate);
|
|
xwayland_view->dissociate.notify = handle_xwayland_surface_dissociate;
|
|
wl_signal_add(&xwayland_surface->events.dissociate,
|
|
&xwayland_view->dissociate);
|
|
xwayland_view->destroy.notify = handle_xwayland_surface_destroy;
|
|
wl_signal_add(&xwayland_surface->events.destroy, &xwayland_view->destroy);
|
|
xwayland_view->request_fullscreen.notify =
|
|
handle_xwayland_surface_request_fullscreen;
|
|
wl_signal_add(&xwayland_surface->events.request_fullscreen,
|
|
&xwayland_view->request_fullscreen);
|
|
}
|