2021-09-24 21:45:48 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2021-03-02 20:37:23 +00:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2020 the sway authors
|
2021-03-03 20:51:19 +00:00
|
|
|
*
|
|
|
|
|
* This file is only needed in support of
|
2021-08-25 20:45:39 +01:00
|
|
|
* - unconstraining XDG popups
|
2022-02-20 13:14:10 +00:00
|
|
|
* - keeping non-layer-shell xdg-popups outside the layers.c code
|
2021-03-02 20:37:23 +00:00
|
|
|
*/
|
|
|
|
|
|
2022-09-16 18:41:02 -04:00
|
|
|
#include "common/mem.h"
|
2021-03-02 20:37:23 +00:00
|
|
|
#include "labwc.h"
|
2022-03-02 21:07:04 +00:00
|
|
|
#include "node.h"
|
2022-11-21 10:10:39 -05:00
|
|
|
#include "view.h"
|
2021-03-02 20:37:23 +00:00
|
|
|
|
2022-02-20 13:14:10 +00:00
|
|
|
struct xdg_popup {
|
2022-09-13 12:51:23 -04:00
|
|
|
struct view *parent_view;
|
2022-02-20 13:14:10 +00:00
|
|
|
struct wlr_xdg_popup *wlr_popup;
|
|
|
|
|
|
2024-02-24 23:16:55 +01:00
|
|
|
struct wl_listener commit;
|
2022-02-20 13:14:10 +00:00
|
|
|
struct wl_listener destroy;
|
|
|
|
|
struct wl_listener new_popup;
|
|
|
|
|
};
|
|
|
|
|
|
2021-03-02 20:37:23 +00:00
|
|
|
static void
|
2024-02-24 23:16:55 +01:00
|
|
|
popup_unconstrain(struct xdg_popup *popup)
|
2021-03-02 20:37:23 +00:00
|
|
|
{
|
2024-02-24 23:16:55 +01:00
|
|
|
struct view *view = popup->parent_view;
|
2021-03-02 20:51:32 +00:00
|
|
|
struct server *server = view->server;
|
2024-02-24 23:16:55 +01:00
|
|
|
struct wlr_box *popup_box = &popup->wlr_popup->current.geometry;
|
2024-03-13 20:01:16 +09:00
|
|
|
struct output *output = output_nearest_to(server,
|
|
|
|
|
view->current.x + popup_box->x,
|
2023-02-08 23:19:14 -05:00
|
|
|
view->current.y + popup_box->y);
|
2024-03-13 20:01:16 +09:00
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
2021-03-02 20:51:32 +00:00
|
|
|
|
|
|
|
|
struct wlr_box output_toplevel_box = {
|
2024-03-13 20:01:16 +09:00
|
|
|
.x = usable.x - view->current.x,
|
|
|
|
|
.y = usable.y - view->current.y,
|
|
|
|
|
.width = usable.width,
|
|
|
|
|
.height = usable.height,
|
2021-03-02 20:51:32 +00:00
|
|
|
};
|
2024-02-24 23:16:55 +01:00
|
|
|
wlr_xdg_popup_unconstrain_from_box(popup->wlr_popup, &output_toplevel_box);
|
2021-03-02 20:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
2022-02-20 13:14:10 +00:00
|
|
|
static void
|
|
|
|
|
handle_xdg_popup_destroy(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct xdg_popup *popup = wl_container_of(listener, popup, destroy);
|
|
|
|
|
wl_list_remove(&popup->destroy.link);
|
|
|
|
|
wl_list_remove(&popup->new_popup.link);
|
2024-02-24 23:16:55 +01:00
|
|
|
|
|
|
|
|
/* Usually already removed unless there was no commit at all */
|
|
|
|
|
if (popup->commit.notify) {
|
|
|
|
|
wl_list_remove(&popup->commit.link);
|
|
|
|
|
}
|
2022-02-20 13:14:10 +00:00
|
|
|
free(popup);
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-24 23:16:55 +01:00
|
|
|
static void
|
|
|
|
|
handle_xdg_popup_commit(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct xdg_popup *popup = wl_container_of(listener, popup, commit);
|
|
|
|
|
|
|
|
|
|
if (popup->wlr_popup->base->initial_commit) {
|
|
|
|
|
popup_unconstrain(popup);
|
|
|
|
|
|
|
|
|
|
/* Prevent getting called over and over again */
|
|
|
|
|
wl_list_remove(&popup->commit.link);
|
|
|
|
|
popup->commit.notify = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-20 13:14:10 +00:00
|
|
|
static void
|
|
|
|
|
popup_handle_new_xdg_popup(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct xdg_popup *popup = wl_container_of(listener, popup, new_popup);
|
|
|
|
|
struct wlr_xdg_popup *wlr_popup = data;
|
2022-09-13 12:51:23 -04:00
|
|
|
xdg_popup_create(popup->parent_view, wlr_popup);
|
2022-02-20 13:14:10 +00:00
|
|
|
}
|
|
|
|
|
|
2021-03-02 20:37:23 +00:00
|
|
|
void
|
|
|
|
|
xdg_popup_create(struct view *view, struct wlr_xdg_popup *wlr_popup)
|
|
|
|
|
{
|
2022-09-13 12:51:23 -04:00
|
|
|
struct wlr_xdg_surface *parent =
|
2023-02-03 14:53:26 -05:00
|
|
|
wlr_xdg_surface_try_from_wlr_surface(wlr_popup->parent);
|
2022-09-13 12:51:23 -04:00
|
|
|
if (!parent) {
|
|
|
|
|
wlr_log(WLR_ERROR, "parent is not a valid XDG surface");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-18 15:22:26 -04:00
|
|
|
struct xdg_popup *popup = znew(*popup);
|
2022-09-13 12:51:23 -04:00
|
|
|
popup->parent_view = view;
|
2022-02-20 13:14:10 +00:00
|
|
|
popup->wlr_popup = wlr_popup;
|
|
|
|
|
|
|
|
|
|
popup->destroy.notify = handle_xdg_popup_destroy;
|
|
|
|
|
wl_signal_add(&wlr_popup->base->events.destroy, &popup->destroy);
|
2024-02-24 23:16:55 +01:00
|
|
|
|
2022-02-20 13:14:10 +00:00
|
|
|
popup->new_popup.notify = popup_handle_new_xdg_popup;
|
|
|
|
|
wl_signal_add(&wlr_popup->base->events.new_popup, &popup->new_popup);
|
|
|
|
|
|
2024-02-24 23:16:55 +01:00
|
|
|
popup->commit.notify = handle_xdg_popup_commit;
|
|
|
|
|
wl_signal_add(&wlr_popup->base->surface->events.commit, &popup->commit);
|
|
|
|
|
|
2022-02-20 13:14:10 +00:00
|
|
|
/*
|
|
|
|
|
* We must add xdg popups to the scene graph so they get rendered. The
|
|
|
|
|
* wlroots scene graph provides a helper for this, but to use it we must
|
|
|
|
|
* provide the proper parent scene node of the xdg popup. To enable
|
|
|
|
|
* this, we always set the user data field of xdg_surfaces to the
|
|
|
|
|
* corresponding scene node.
|
2023-05-20 08:58:42 +01:00
|
|
|
*
|
|
|
|
|
* xdg-popups live in server->xdg_popup_tree so that they can be
|
|
|
|
|
* rendered above always-on-top windows
|
2022-02-20 13:14:10 +00:00
|
|
|
*/
|
2023-05-20 08:58:42 +01:00
|
|
|
struct wlr_scene_tree *parent_tree = NULL;
|
|
|
|
|
if (parent->role == WLR_XDG_SURFACE_ROLE_POPUP) {
|
|
|
|
|
parent_tree = parent->surface->data;
|
|
|
|
|
} else {
|
|
|
|
|
parent_tree = view->server->xdg_popup_tree;
|
|
|
|
|
wlr_scene_node_set_position(&view->server->xdg_popup_tree->node,
|
|
|
|
|
view->current.x, view->current.y);
|
|
|
|
|
}
|
2022-02-20 13:14:10 +00:00
|
|
|
wlr_popup->base->surface->data =
|
2022-06-05 15:17:35 +02:00
|
|
|
wlr_scene_xdg_surface_create(parent_tree, wlr_popup->base);
|
2022-02-25 22:31:24 +00:00
|
|
|
node_descriptor_create(wlr_popup->base->surface->data,
|
|
|
|
|
LAB_NODE_DESC_XDG_POPUP, view);
|
2021-03-02 20:37:23 +00:00
|
|
|
}
|