mirror of
https://github.com/hyprwm/wlroots-hyprland.git
synced 2025-01-24 07:29:50 +01:00
e479dc1ef0
A launchee notifies with a "remove"¹ message when done starting up. Catch these and forward to the compositor. This allows the compositor to end the startup sequence that might have been started by another protocol like xdg-activation. We don't handle other messages since we expect the launcher to use a wayland protocol like xdg-activation. While `_NET_STARTUP_ID` helps to associate toplevels with startup-ids this signals the end of the startup sequence. 1) https://specifications.freedesktop.org/startup-notification-spec/startup-notification-latest.txt
156 lines
4 KiB
C
156 lines
4 KiB
C
#define _POSIX_C_SOURCE 200809L
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdlib.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
#include <wayland-server-core.h>
|
|
#include <wlr/util/log.h>
|
|
#include <wlr/xwayland.h>
|
|
#include "sockets.h"
|
|
#include "util/signal.h"
|
|
#include "xwayland/xwm.h"
|
|
|
|
struct wlr_xwayland_cursor {
|
|
uint8_t *pixels;
|
|
uint32_t stride;
|
|
uint32_t width;
|
|
uint32_t height;
|
|
int32_t hotspot_x;
|
|
int32_t hotspot_y;
|
|
};
|
|
|
|
static void handle_server_destroy(struct wl_listener *listener, void *data) {
|
|
struct wlr_xwayland *xwayland =
|
|
wl_container_of(listener, xwayland, server_destroy);
|
|
wlr_xwayland_destroy(xwayland);
|
|
}
|
|
|
|
static void handle_server_ready(struct wl_listener *listener, void *data) {
|
|
struct wlr_xwayland *xwayland =
|
|
wl_container_of(listener, xwayland, server_ready);
|
|
struct wlr_xwayland_server_ready_event *event = data;
|
|
|
|
xwayland->xwm = xwm_create(xwayland, event->wm_fd);
|
|
if (!xwayland->xwm) {
|
|
return;
|
|
}
|
|
|
|
if (xwayland->seat) {
|
|
xwm_set_seat(xwayland->xwm, xwayland->seat);
|
|
}
|
|
|
|
if (xwayland->cursor != NULL) {
|
|
struct wlr_xwayland_cursor *cur = xwayland->cursor;
|
|
xwm_set_cursor(xwayland->xwm, cur->pixels, cur->stride, cur->width,
|
|
cur->height, cur->hotspot_x, cur->hotspot_y);
|
|
}
|
|
|
|
wlr_signal_emit_safe(&xwayland->events.ready, NULL);
|
|
}
|
|
|
|
void wlr_xwayland_destroy(struct wlr_xwayland *xwayland) {
|
|
if (!xwayland) {
|
|
return;
|
|
}
|
|
|
|
wl_list_remove(&xwayland->server_destroy.link);
|
|
wl_list_remove(&xwayland->server_ready.link);
|
|
free(xwayland->cursor);
|
|
|
|
wlr_xwayland_set_seat(xwayland, NULL);
|
|
wlr_xwayland_server_destroy(xwayland->server);
|
|
free(xwayland);
|
|
}
|
|
|
|
struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display,
|
|
struct wlr_compositor *compositor, bool lazy) {
|
|
struct wlr_xwayland *xwayland = calloc(1, sizeof(struct wlr_xwayland));
|
|
if (!xwayland) {
|
|
return NULL;
|
|
}
|
|
|
|
xwayland->wl_display = wl_display;
|
|
xwayland->compositor = compositor;
|
|
|
|
wl_signal_init(&xwayland->events.new_surface);
|
|
wl_signal_init(&xwayland->events.ready);
|
|
wl_signal_init(&xwayland->events.remove_startup_info);
|
|
|
|
struct wlr_xwayland_server_options options = {
|
|
.lazy = lazy,
|
|
.enable_wm = true,
|
|
};
|
|
xwayland->server = wlr_xwayland_server_create(wl_display, &options);
|
|
if (xwayland->server == NULL) {
|
|
free(xwayland);
|
|
return NULL;
|
|
}
|
|
|
|
xwayland->display_name = xwayland->server->display_name;
|
|
|
|
xwayland->server_destroy.notify = handle_server_destroy;
|
|
wl_signal_add(&xwayland->server->events.destroy, &xwayland->server_destroy);
|
|
|
|
xwayland->server_ready.notify = handle_server_ready;
|
|
wl_signal_add(&xwayland->server->events.ready, &xwayland->server_ready);
|
|
|
|
return xwayland;
|
|
}
|
|
|
|
void wlr_xwayland_set_cursor(struct wlr_xwayland *xwayland,
|
|
uint8_t *pixels, uint32_t stride, uint32_t width, uint32_t height,
|
|
int32_t hotspot_x, int32_t hotspot_y) {
|
|
if (xwayland->xwm != NULL) {
|
|
xwm_set_cursor(xwayland->xwm, pixels, stride, width, height,
|
|
hotspot_x, hotspot_y);
|
|
return;
|
|
}
|
|
|
|
free(xwayland->cursor);
|
|
|
|
xwayland->cursor = calloc(1, sizeof(struct wlr_xwayland_cursor));
|
|
if (xwayland->cursor == NULL) {
|
|
return;
|
|
}
|
|
xwayland->cursor->pixels = pixels;
|
|
xwayland->cursor->stride = stride;
|
|
xwayland->cursor->width = width;
|
|
xwayland->cursor->height = height;
|
|
xwayland->cursor->hotspot_x = hotspot_x;
|
|
xwayland->cursor->hotspot_y = hotspot_y;
|
|
}
|
|
|
|
static void xwayland_handle_seat_destroy(struct wl_listener *listener,
|
|
void *data) {
|
|
struct wlr_xwayland *xwayland =
|
|
wl_container_of(listener, xwayland, seat_destroy);
|
|
|
|
wlr_xwayland_set_seat(xwayland, NULL);
|
|
}
|
|
|
|
void wlr_xwayland_set_seat(struct wlr_xwayland *xwayland,
|
|
struct wlr_seat *seat) {
|
|
if (xwayland->seat) {
|
|
wl_list_remove(&xwayland->seat_destroy.link);
|
|
}
|
|
|
|
xwayland->seat = seat;
|
|
|
|
if (xwayland->xwm) {
|
|
xwm_set_seat(xwayland->xwm, seat);
|
|
}
|
|
|
|
if (seat == NULL) {
|
|
return;
|
|
}
|
|
|
|
xwayland->seat_destroy.notify = xwayland_handle_seat_destroy;
|
|
wl_signal_add(&seat->events.destroy, &xwayland->seat_destroy);
|
|
}
|