labwc/src/desktop.c

358 lines
8 KiB
C
Raw Normal View History

2021-09-24 21:45:48 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2020-12-30 10:29:21 +00:00
#include "config.h"
2020-09-28 20:53:59 +01:00
#include <assert.h>
#include "labwc.h"
#include "layers.h"
#include "node.h"
2021-03-21 20:54:55 +00:00
#include "ssd.h"
#include "common/scene-helpers.h"
2022-06-15 02:02:50 +02:00
#include "workspaces.h"
static void
move_to_front(struct view *view)
{
wl_list_remove(&view->link);
wl_list_insert(&view->server->views, &view->link);
wlr_scene_node_raise_to_top(&view->scene_tree->node);
}
2020-12-30 10:29:21 +00:00
#if HAVE_XWAYLAND
static struct wlr_xwayland_surface *
top_parent_of(struct view *view)
{
struct wlr_xwayland_surface *s = view->xwayland_surface;
while (s->parent) {
s = s->parent;
}
return s;
}
static void
move_xwayland_sub_views_to_front(struct view *parent)
{
if (!parent || parent->type != LAB_XWAYLAND_VIEW) {
return;
}
struct view *view, *next;
wl_list_for_each_reverse_safe(view, next, &parent->server->views, link)
{
/* need to stop here, otherwise loops keeps going forever */
if (view == parent) {
break;
}
if (view->type != LAB_XWAYLAND_VIEW) {
continue;
}
if (!view->mapped && !view->minimized) {
continue;
}
if (top_parent_of(view) != parent->xwayland_surface) {
continue;
}
move_to_front(view);
/* TODO: we should probably focus on these too here */
}
}
2020-12-30 10:29:21 +00:00
#endif
void
desktop_move_to_front(struct view *view)
{
if (!view) {
return;
}
move_to_front(view);
#if HAVE_XWAYLAND
move_xwayland_sub_views_to_front(view);
#endif
cursor_update_focus(view->server);
}
static void
wl_list_insert_tail(struct wl_list *list, struct wl_list *elm)
{
elm->prev = list->prev;
elm->next = list;
list->prev = elm;
elm->prev->next = elm;
}
void
desktop_move_to_back(struct view *view)
{
if (!view) {
return;
}
wl_list_remove(&view->link);
wl_list_insert_tail(&view->server->views, &view->link);
}
static void
deactivate_all_views(struct server *server)
{
struct view *view;
wl_list_for_each (view, &server->views, link) {
if (!view->mapped) {
continue;
}
view_set_activated(view, false);
}
}
void
desktop_focus_and_activate_view(struct seat *seat, struct view *view)
2020-09-14 17:35:44 +01:00
{
if (!view) {
2020-10-08 20:22:52 +01:00
seat_focus_surface(seat, NULL);
2020-09-14 17:35:44 +01:00
return;
}
/*
* Guard against views with no mapped surfaces when handling
* 'request_activate' and 'request_minimize'.
* See notes by isfocusable()
*/
if (!view->surface) {
return;
}
if (input_inhibit_blocks_surface(seat, view->surface->resource)) {
2021-08-25 19:59:49 +01:00
return;
}
if (view->minimized) {
/*
* Unminimizing will map the view which triggers a call to this
* function again.
*/
2021-08-05 13:00:34 +01:00
view_minimize(view, false);
2020-10-08 20:08:41 +01:00
return;
}
if (!view->mapped) {
return;
}
struct wlr_surface *prev_surface;
prev_surface = seat->seat->keyboard_state.focused_surface;
/* Do not re-focus an already focused surface. */
if (prev_surface == view->surface) {
return;
}
deactivate_all_views(view->server);
view_set_activated(view, true);
seat_focus_surface(seat, view->surface);
2020-09-14 17:35:44 +01:00
}
/*
* Some xwayland apps produce unmapped surfaces on startup and also leave
2021-08-25 19:59:49 +01:00
* some unmapped surfaces kicking around on 'close' (for example leafpad's
* "about" dialogue). Whilst this is not normally a problem, we have to be
2020-09-14 17:35:44 +01:00
* careful when cycling between views. The only views we should focus are
* those that are already mapped and those that have been minimized.
*/
bool
isfocusable(struct view *view)
{
/* filter out those xwayland surfaces that have never been mapped */
if (!view->surface) {
return false;
}
return (view->mapped || view->minimized);
}
2022-06-15 02:02:50 +02:00
static struct wl_list *
get_prev_item(struct wl_list *item)
{
2022-06-15 02:02:50 +02:00
return item->prev;
}
static struct wl_list *
get_next_item(struct wl_list *item)
{
return item->next;
}
static struct view *
first_view(struct server *server)
{
2022-06-15 02:02:50 +02:00
struct wlr_scene_node *node;
struct wl_list *list_head =
&server->workspace_current->tree->children;
wl_list_for_each_reverse(node, list_head, link) {
return node_view_from_node(node);
}
return NULL;
}
struct view *
2022-06-15 02:02:50 +02:00
desktop_cycle_view(struct server *server, struct view *start_view,
2021-12-26 23:29:01 +00:00
enum lab_cycle_dir dir)
{
2022-06-15 02:02:50 +02:00
struct view *view = start_view ? start_view : first_view(server);
if (!view) {
return NULL;
2020-10-31 14:46:33 +00:00
}
2022-06-15 02:02:50 +02:00
start_view = view;
struct wlr_scene_node *node = &view->scene_tree->node;
2022-06-15 02:02:50 +02:00
assert(node->parent);
struct wl_list *list_head = &node->parent->children;
struct wl_list *list_item = &node->link;
struct wl_list *(*iter)(struct wl_list *);
2022-06-15 02:02:50 +02:00
/* Scene nodes are ordered like last node == displayed topmost */
iter = dir == LAB_CYCLE_DIR_FORWARD ? get_prev_item : get_next_item;
do {
list_item = iter(list_item);
if (list_item == list_head) {
/* Start / End of list reached. Roll over */
list_item = iter(list_item);
}
2022-06-15 02:02:50 +02:00
node = wl_container_of(list_item, node, link);
view = node_view_from_node(node);
if (isfocusable(view)) {
return view;
}
} while (view != start_view);
/* No focusable views found, including the one we started with */
return NULL;
}
static struct view *
topmost_mapped_view(struct server *server)
{
2022-06-15 02:02:50 +02:00
struct view *view;
struct wl_list *node_list;
struct wlr_scene_node *node;
node_list = &server->workspace_current->tree->children;
wl_list_for_each_reverse(node, node_list, link) {
view = node_view_from_node(node);
if (view->mapped) {
return view;
}
}
2022-06-15 02:02:50 +02:00
return NULL;
}
struct view *
desktop_focused_view(struct server *server)
{
struct seat *seat = &server->seat;
struct wlr_surface *focused_surface;
focused_surface = seat->seat->keyboard_state.focused_surface;
if (!focused_surface) {
return NULL;
}
struct view *view;
wl_list_for_each (view, &server->views, link) {
if (view->surface == focused_surface) {
return view;
}
}
return NULL;
}
void
desktop_focus_topmost_mapped_view(struct server *server)
{
struct view *view = topmost_mapped_view(server);
desktop_focus_and_activate_view(&server->seat, view);
desktop_move_to_front(view);
}
struct view *
desktop_node_and_view_at(struct server *server, double lx, double ly,
struct wlr_scene_node **scene_node, double *sx, double *sy,
enum ssd_part_type *view_area)
{
struct wlr_scene_node *node =
wlr_scene_node_at(&server->scene->tree.node, lx, ly, sx, sy);
*scene_node = node;
if (!node) {
*view_area = LAB_SSD_ROOT;
2021-07-19 07:06:36 +01:00
return NULL;
}
if (node->type == WLR_SCENE_NODE_BUFFER) {
struct wlr_surface *surface = lab_wlr_surface_from_node(node);
if (surface && wlr_surface_is_layer_surface(surface)) {
*view_area = LAB_SSD_LAYER_SURFACE;
return NULL;
}
#if HAVE_XWAYLAND
if (node->parent == server->unmanaged_tree) {
*view_area = LAB_SSD_UNMANAGED;
return NULL;
}
#endif
}
while (node) {
struct node_descriptor *desc = node->data;
2022-03-03 04:33:33 +01:00
/* TODO: convert to switch() */
if (desc) {
if (desc->type == LAB_NODE_DESC_VIEW) {
goto has_view_data;
}
if (desc->type == LAB_NODE_DESC_XDG_POPUP) {
goto has_view_data;
}
if (desc->type == LAB_NODE_DESC_SSD_BUTTON) {
/* Always return the top scene node for SSD buttons */
struct ssd_button *button = node_ssd_button_from_node(node);
*scene_node = node;
*view_area = button->type;
return button->view;
}
if (desc->type == LAB_NODE_DESC_LAYER_SURFACE) {
/* FIXME: we shouldn't have to set *view_area */
*view_area = LAB_SSD_CLIENT;
return NULL;
}
if (desc->type == LAB_NODE_DESC_LAYER_POPUP) {
/* FIXME: we shouldn't have to set *view_area */
*view_area = LAB_SSD_CLIENT;
return NULL;
}
2022-03-03 04:33:33 +01:00
if (desc->type == LAB_NODE_DESC_MENUITEM) {
/* Always return the top scene node for menu items */
*scene_node = node;
*view_area = LAB_SSD_MENU;
return NULL;
}
}
/* node->parent is always a *wlr_scene_tree */
node = node->parent ? &node->parent->node : NULL;
}
if (!node) {
wlr_log(WLR_ERROR, "Unknown node detected");
}
*view_area = LAB_SSD_NONE;
return NULL;
struct view *view;
struct node_descriptor *desc;
has_view_data:
desc = node->data;
view = desc->data;
2022-02-21 03:18:38 +01:00
*view_area = ssd_get_part_type(view, *scene_node);
return view;
}
struct view *
2021-11-26 19:27:50 +00:00
desktop_view_at_cursor(struct server *server)
{
double sx, sy;
struct wlr_scene_node *node;
enum ssd_part_type view_area = LAB_SSD_NONE;
return desktop_node_and_view_at(server,
server->seat.cursor->x, server->seat.cursor->y,
&node, &sx, &sy, &view_area);
}