2021-09-24 21:45:48 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2022-04-19 23:01:23 +01:00
|
|
|
#include <assert.h>
|
2021-02-27 17:10:53 -05:00
|
|
|
#include <stdio.h>
|
2021-07-20 19:54:57 +01:00
|
|
|
#include <strings.h>
|
2022-04-20 18:48:49 +01:00
|
|
|
#include <xcb/xcb_icccm.h>
|
2021-03-20 14:36:40 +00:00
|
|
|
#include "labwc.h"
|
2021-03-21 20:54:55 +00:00
|
|
|
#include "ssd.h"
|
2022-06-09 17:10:36 +02:00
|
|
|
#include "menu/menu.h"
|
2022-06-15 02:02:50 +02:00
|
|
|
#include "workspaces.h"
|
2021-03-02 20:37:23 +00:00
|
|
|
|
2022-04-02 21:34:51 -04:00
|
|
|
#define MAX(a, b) (((a) > (b)) ? (a) : (b))
|
|
|
|
|
|
2022-07-01 02:07:40 +02:00
|
|
|
/**
|
|
|
|
|
* All view_apply_xxx_geometry() functions must *not* modify
|
|
|
|
|
* any state besides repositioning or resizing the view.
|
|
|
|
|
*
|
|
|
|
|
* They may be called repeatably during output layout changes.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
enum view_edge {
|
|
|
|
|
VIEW_EDGE_INVALID = 0,
|
|
|
|
|
|
|
|
|
|
VIEW_EDGE_LEFT,
|
|
|
|
|
VIEW_EDGE_RIGHT,
|
|
|
|
|
VIEW_EDGE_UP,
|
|
|
|
|
VIEW_EDGE_DOWN,
|
|
|
|
|
VIEW_EDGE_CENTER,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static enum view_edge
|
|
|
|
|
view_edge_invert(enum view_edge edge)
|
|
|
|
|
{
|
|
|
|
|
switch (edge) {
|
|
|
|
|
case VIEW_EDGE_LEFT:
|
|
|
|
|
return VIEW_EDGE_RIGHT;
|
|
|
|
|
case VIEW_EDGE_RIGHT:
|
|
|
|
|
return VIEW_EDGE_LEFT;
|
|
|
|
|
case VIEW_EDGE_UP:
|
|
|
|
|
return VIEW_EDGE_DOWN;
|
|
|
|
|
case VIEW_EDGE_DOWN:
|
|
|
|
|
return VIEW_EDGE_UP;
|
|
|
|
|
case VIEW_EDGE_CENTER:
|
|
|
|
|
case VIEW_EDGE_INVALID:
|
|
|
|
|
default:
|
|
|
|
|
return VIEW_EDGE_INVALID;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct wlr_box
|
|
|
|
|
view_get_edge_snap_box(struct view *view, struct output *output,
|
|
|
|
|
enum view_edge edge)
|
|
|
|
|
{
|
|
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
|
|
|
|
if (usable.height == output->wlr_output->height
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
|
|
|
|
usable.height /= output->wlr_output->scale;
|
|
|
|
|
}
|
|
|
|
|
if (usable.width == output->wlr_output->width
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
|
|
|
|
usable.width /= output->wlr_output->scale;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int x_offset = edge == VIEW_EDGE_RIGHT
|
|
|
|
|
? (usable.width + rc.gap) / 2 : rc.gap;
|
|
|
|
|
int y_offset = edge == VIEW_EDGE_DOWN
|
|
|
|
|
? (usable.height + rc.gap) / 2 : rc.gap;
|
|
|
|
|
|
|
|
|
|
int base_width, base_height;
|
|
|
|
|
switch (edge) {
|
|
|
|
|
case VIEW_EDGE_LEFT:
|
|
|
|
|
case VIEW_EDGE_RIGHT:
|
|
|
|
|
base_width = (usable.width - 3 * rc.gap) / 2;
|
|
|
|
|
base_height = usable.height - 2 * rc.gap;
|
|
|
|
|
break;
|
|
|
|
|
case VIEW_EDGE_UP:
|
|
|
|
|
case VIEW_EDGE_DOWN:
|
|
|
|
|
base_width = usable.width - 2 * rc.gap;
|
|
|
|
|
base_height = (usable.height - 3 * rc.gap) / 2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
case VIEW_EDGE_CENTER:
|
|
|
|
|
base_width = usable.width - 2 * rc.gap;
|
|
|
|
|
base_height = usable.height - 2 * rc.gap;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
struct wlr_box dst = {
|
|
|
|
|
.x = x_offset + usable.x + view->margin.left,
|
|
|
|
|
.y = y_offset + usable.y + view->margin.top,
|
|
|
|
|
.width = base_width - view->margin.left - view->margin.right,
|
|
|
|
|
.height = base_height - view->margin.top - view->margin.bottom,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return dst;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-16 19:24:26 +01:00
|
|
|
void
|
|
|
|
|
view_set_activated(struct view *view, bool activated)
|
|
|
|
|
{
|
2022-02-21 03:18:38 +01:00
|
|
|
if (view->ssd.enabled) {
|
|
|
|
|
ssd_set_active(view);
|
|
|
|
|
}
|
2021-10-16 19:24:26 +01:00
|
|
|
if (view->impl->set_activated) {
|
|
|
|
|
view->impl->set_activated(view, activated);
|
|
|
|
|
}
|
|
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_activated(
|
|
|
|
|
view->toplevel_handle, activated);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-23 12:24:24 +01:00
|
|
|
void
|
|
|
|
|
view_close(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
if (view->impl->close) {
|
|
|
|
|
view->impl->close(view);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
void
|
2022-01-07 20:53:48 +00:00
|
|
|
view_move(struct view *view, double x, double y)
|
2020-05-26 12:56:33 +01:00
|
|
|
{
|
2022-01-07 20:53:48 +00:00
|
|
|
if (view->impl->move) {
|
|
|
|
|
view->impl->move(view, x, y);
|
2021-10-18 19:35:41 +01:00
|
|
|
}
|
2021-11-15 14:59:03 -05:00
|
|
|
view_discover_output(view);
|
2022-02-18 00:07:37 +01:00
|
|
|
wlr_scene_node_set_position(&view->scene_tree->node, view->x, view->y);
|
2022-03-10 06:50:42 +01:00
|
|
|
ssd_update_geometry(view);
|
2020-05-26 12:56:33 +01:00
|
|
|
}
|
|
|
|
|
|
2022-03-07 20:59:54 +00:00
|
|
|
/* N.B. Use view_move() if not resizing. */
|
2020-12-23 18:52:46 +00:00
|
|
|
void
|
2022-01-07 20:53:48 +00:00
|
|
|
view_move_resize(struct view *view, struct wlr_box geo)
|
2020-12-23 18:52:46 +00:00
|
|
|
{
|
2022-01-07 20:53:48 +00:00
|
|
|
if (view->impl->configure) {
|
|
|
|
|
view->impl->configure(view, geo);
|
2021-10-18 19:35:41 +01:00
|
|
|
}
|
2022-01-07 20:53:48 +00:00
|
|
|
ssd_update_title(view);
|
2021-11-15 14:59:03 -05:00
|
|
|
view_discover_output(view);
|
2020-12-23 18:52:46 +00:00
|
|
|
}
|
|
|
|
|
|
2021-10-17 18:31:53 +00:00
|
|
|
#define MIN_VIEW_WIDTH (100)
|
|
|
|
|
#define MIN_VIEW_HEIGHT (60)
|
|
|
|
|
|
2022-04-02 21:34:51 -04:00
|
|
|
#if HAVE_XWAYLAND
|
|
|
|
|
static int
|
|
|
|
|
round_to_increment(int val, int base, int inc)
|
|
|
|
|
{
|
|
|
|
|
if (base < 0 || inc <= 0)
|
|
|
|
|
return val;
|
|
|
|
|
return base + (val - base + inc / 2) / inc * inc;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-10-17 18:31:53 +00:00
|
|
|
void
|
2022-04-02 21:34:51 -04:00
|
|
|
view_adjust_size(struct view *view, int *w, int *h)
|
2021-10-17 18:31:53 +00:00
|
|
|
{
|
|
|
|
|
int min_width = MIN_VIEW_WIDTH;
|
|
|
|
|
int min_height = MIN_VIEW_HEIGHT;
|
|
|
|
|
#if HAVE_XWAYLAND
|
2022-04-02 21:34:51 -04:00
|
|
|
if (view->type == LAB_XWAYLAND_VIEW) {
|
2022-04-20 18:48:49 +01:00
|
|
|
xcb_size_hints_t *hints = view->xwayland_surface->size_hints;
|
|
|
|
|
|
2022-04-02 21:34:51 -04:00
|
|
|
/*
|
|
|
|
|
* Honor size increments from WM_SIZE_HINTS. Typically, X11
|
|
|
|
|
* terminal emulators will use WM_SIZE_HINTS to make sure that
|
|
|
|
|
* the terminal is resized to a width/height evenly divisible by
|
|
|
|
|
* the cell (character) size.
|
|
|
|
|
*/
|
|
|
|
|
if (hints) {
|
|
|
|
|
*w = round_to_increment(*w, hints->base_width,
|
|
|
|
|
hints->width_inc);
|
|
|
|
|
*h = round_to_increment(*h, hints->base_height,
|
|
|
|
|
hints->height_inc);
|
2021-10-17 18:31:53 +00:00
|
|
|
|
2022-04-02 21:34:51 -04:00
|
|
|
min_width = MAX(1, hints->min_width);
|
|
|
|
|
min_height = MAX(1, hints->min_height);
|
|
|
|
|
}
|
2021-10-17 18:31:53 +00:00
|
|
|
}
|
2022-04-02 21:34:51 -04:00
|
|
|
#endif
|
|
|
|
|
*w = MAX(*w, min_width);
|
|
|
|
|
*h = MAX(*h, min_height);
|
2021-10-17 18:31:53 +00:00
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
void
|
2021-08-05 13:00:34 +01:00
|
|
|
view_minimize(struct view *view, bool minimized)
|
2020-09-08 20:51:33 +01:00
|
|
|
{
|
2021-08-05 13:00:34 +01:00
|
|
|
if (view->minimized == minimized) {
|
2020-09-08 20:51:33 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2021-08-05 13:00:34 +01:00
|
|
|
if (view->toplevel_handle) {
|
2021-11-26 18:49:44 +00:00
|
|
|
wlr_foreign_toplevel_handle_v1_set_minimized(
|
|
|
|
|
view->toplevel_handle, minimized);
|
2021-08-05 13:00:34 +01:00
|
|
|
}
|
|
|
|
|
view->minimized = minimized;
|
|
|
|
|
if (minimized) {
|
|
|
|
|
view->impl->unmap(view);
|
2021-12-06 21:23:49 +00:00
|
|
|
desktop_move_to_back(view);
|
2021-12-27 23:41:33 +01:00
|
|
|
view_set_activated(view, false);
|
2021-08-05 13:00:34 +01:00
|
|
|
} else {
|
|
|
|
|
view->impl->map(view);
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-09-08 20:51:33 +01:00
|
|
|
}
|
2020-09-29 20:48:50 +01:00
|
|
|
|
2021-11-13 12:09:40 -05:00
|
|
|
/* view_wlr_output - return the output that a view is mostly on */
|
|
|
|
|
struct wlr_output *
|
|
|
|
|
view_wlr_output(struct view *view)
|
2021-10-18 19:59:11 -04:00
|
|
|
{
|
2021-10-19 18:15:56 -04:00
|
|
|
double closest_x, closest_y;
|
|
|
|
|
struct wlr_output *wlr_output = NULL;
|
|
|
|
|
wlr_output_layout_closest_point(view->server->output_layout, wlr_output,
|
|
|
|
|
view->x + view->w / 2, view->y + view->h / 2, &closest_x,
|
|
|
|
|
&closest_y);
|
|
|
|
|
wlr_output = wlr_output_layout_output_at(view->server->output_layout,
|
|
|
|
|
closest_x, closest_y);
|
|
|
|
|
return wlr_output;
|
2021-10-18 19:59:11 -04:00
|
|
|
}
|
|
|
|
|
|
2021-07-20 19:40:37 +01:00
|
|
|
static struct output *
|
|
|
|
|
view_output(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
struct wlr_output *wlr_output = view_wlr_output(view);
|
|
|
|
|
return output_from_wlr_output(view->server, wlr_output);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
static bool
|
|
|
|
|
view_compute_centered_position(struct view *view, int w, int h, int *x, int *y)
|
2021-07-09 21:39:20 +01:00
|
|
|
{
|
2022-04-19 23:01:23 +01:00
|
|
|
struct output *output = view_output(view);
|
2022-04-26 21:55:04 +01:00
|
|
|
if (!output) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2022-04-19 23:01:23 +01:00
|
|
|
struct wlr_output *wlr_output = output->wlr_output;
|
2021-07-20 19:40:37 +01:00
|
|
|
if (!wlr_output) {
|
2021-12-31 17:30:55 -05:00
|
|
|
return false;
|
2021-07-09 21:39:20 +01:00
|
|
|
}
|
|
|
|
|
|
2022-04-19 23:01:23 +01:00
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
|
|
|
|
int width = w + view->margin.left + view->margin.right;
|
|
|
|
|
int height = h + view->margin.top + view->margin.bottom;
|
2022-06-06 18:31:18 +01:00
|
|
|
*x = usable.x + (usable.width - width) / 2;
|
|
|
|
|
*y = usable.y + (usable.height - height) / 2;
|
2022-06-06 18:33:40 +01:00
|
|
|
|
2022-06-06 18:38:04 +01:00
|
|
|
/* If view is bigger than usable area, just top/left align it */
|
|
|
|
|
if (*x < rc.gap) {
|
|
|
|
|
*x = rc.gap;
|
|
|
|
|
}
|
|
|
|
|
if (*y < rc.gap) {
|
|
|
|
|
*y = rc.gap;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-06 18:33:40 +01:00
|
|
|
#if HAVE_XWAYLAND
|
|
|
|
|
/* TODO: refactor xwayland.c functions to get rid of this */
|
|
|
|
|
if (view->type == LAB_XWAYLAND_VIEW) {
|
|
|
|
|
*x += view->margin.left;
|
|
|
|
|
*y += view->margin.top;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
view_center(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
int x, y;
|
|
|
|
|
if (view_compute_centered_position(view, view->w, view->h, &x, &y)) {
|
|
|
|
|
view_move(view, x, y);
|
2021-12-24 15:53:49 -05:00
|
|
|
}
|
2021-12-31 17:30:55 -05:00
|
|
|
}
|
|
|
|
|
|
2022-07-01 02:07:40 +02:00
|
|
|
static void
|
|
|
|
|
view_apply_tiled_geometry(struct view *view, struct output *output)
|
|
|
|
|
{
|
|
|
|
|
assert(view->tiled);
|
|
|
|
|
if (!output) {
|
|
|
|
|
output = view_output(view);
|
|
|
|
|
}
|
|
|
|
|
if (!output) {
|
|
|
|
|
wlr_log(WLR_ERROR, "Can't tile: no output");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct wlr_box dst = view_get_edge_snap_box(view, output, view->tiled);
|
|
|
|
|
if (view->w == dst.width && view->h == dst.height) {
|
|
|
|
|
/* move horizontally/vertically without changing size */
|
|
|
|
|
view_move(view, dst.x, dst.y);
|
|
|
|
|
} else {
|
|
|
|
|
view_move_resize(view, dst);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
static void
|
|
|
|
|
view_apply_fullscreen_geometry(struct view *view, struct wlr_output *wlr_output)
|
|
|
|
|
{
|
|
|
|
|
struct output *output =
|
|
|
|
|
output_from_wlr_output(view->server, wlr_output);
|
|
|
|
|
struct wlr_box box = { 0 };
|
|
|
|
|
wlr_output_effective_resolution(wlr_output, &box.width, &box.height);
|
|
|
|
|
double ox = 0, oy = 0;
|
|
|
|
|
wlr_output_layout_output_coords(output->server->output_layout,
|
|
|
|
|
output->wlr_output, &ox, &oy);
|
|
|
|
|
box.x -= ox;
|
|
|
|
|
box.y -= oy;
|
|
|
|
|
view_move_resize(view, box);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
view_apply_maximized_geometry(struct view *view)
|
|
|
|
|
{
|
2022-01-08 11:46:46 -05:00
|
|
|
/*
|
|
|
|
|
* The same code handles both initial maximize and re-maximize
|
|
|
|
|
* to account for layout changes. In either case, view_output()
|
|
|
|
|
* gives the output closest to the current geometry (which may
|
|
|
|
|
* be different from the output originally maximized onto).
|
|
|
|
|
* view_output() can return NULL if all outputs are disabled.
|
|
|
|
|
*/
|
2021-12-31 17:30:55 -05:00
|
|
|
struct output *output = view_output(view);
|
2022-01-07 22:01:44 +00:00
|
|
|
if (!output) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-12-31 17:30:55 -05:00
|
|
|
struct wlr_box box = output_usable_area_in_layout_coords(output);
|
2022-04-04 20:53:36 +01:00
|
|
|
if (box.height == output->wlr_output->height
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
2021-12-31 17:30:55 -05:00
|
|
|
box.height /= output->wlr_output->scale;
|
|
|
|
|
}
|
2022-04-04 20:53:36 +01:00
|
|
|
if (box.width == output->wlr_output->width
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
2021-12-31 17:30:55 -05:00
|
|
|
box.width /= output->wlr_output->scale;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (view->ssd.enabled) {
|
|
|
|
|
struct border border = ssd_thickness(view);
|
|
|
|
|
box.x += border.left;
|
|
|
|
|
box.y += border.top;
|
|
|
|
|
box.width -= border.right + border.left;
|
|
|
|
|
box.height -= border.top + border.bottom;
|
|
|
|
|
}
|
|
|
|
|
view_move_resize(view, box);
|
|
|
|
|
}
|
2021-12-24 15:53:49 -05:00
|
|
|
|
2022-04-28 21:37:32 +01:00
|
|
|
#define LAB_FALLBACK_WIDTH (640)
|
|
|
|
|
#define LAB_FALLBACK_HEIGHT (480)
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
set_fallback_geometry(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
view->unmaximized_geometry.width = LAB_FALLBACK_WIDTH;
|
|
|
|
|
view->unmaximized_geometry.height = LAB_FALLBACK_HEIGHT;
|
|
|
|
|
view_compute_centered_position(view,
|
|
|
|
|
view->unmaximized_geometry.width,
|
|
|
|
|
view->unmaximized_geometry.height,
|
|
|
|
|
&view->unmaximized_geometry.x,
|
|
|
|
|
&view->unmaximized_geometry.y);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
static void
|
|
|
|
|
view_apply_unmaximized_geometry(struct view *view)
|
|
|
|
|
{
|
2022-04-28 21:37:32 +01:00
|
|
|
/*
|
|
|
|
|
* If an application was started maximized, its unmaximized_geometry
|
|
|
|
|
* width/height may still be zero in which case we set some fallback
|
|
|
|
|
* values. This is the case with foot and Qt applications.
|
|
|
|
|
*/
|
|
|
|
|
if (wlr_box_empty(&view->unmaximized_geometry)) {
|
|
|
|
|
set_fallback_geometry(view);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
struct wlr_output_layout *layout = view->server->output_layout;
|
2022-04-04 20:53:36 +01:00
|
|
|
if (wlr_output_layout_intersects(layout, NULL,
|
|
|
|
|
&view->unmaximized_geometry)) {
|
2021-12-31 17:30:55 -05:00
|
|
|
/* restore to original geometry */
|
|
|
|
|
view_move_resize(view, view->unmaximized_geometry);
|
|
|
|
|
} else {
|
|
|
|
|
/* reposition if original geometry is offscreen */
|
|
|
|
|
struct wlr_box box = view->unmaximized_geometry;
|
|
|
|
|
if (view_compute_centered_position(view, box.width, box.height,
|
|
|
|
|
&box.x, &box.y)) {
|
|
|
|
|
view_move_resize(view, box);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-07-09 21:39:20 +01:00
|
|
|
}
|
|
|
|
|
|
2021-02-27 17:10:53 -05:00
|
|
|
void
|
|
|
|
|
view_maximize(struct view *view, bool maximize)
|
|
|
|
|
{
|
2021-03-12 21:27:17 +00:00
|
|
|
if (view->maximized == maximize) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-01-02 10:19:48 -05:00
|
|
|
if (view->fullscreen) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-10-18 19:35:41 +01:00
|
|
|
if (view->impl->maximize) {
|
|
|
|
|
view->impl->maximize(view, maximize);
|
|
|
|
|
}
|
2021-08-05 12:52:42 +01:00
|
|
|
if (view->toplevel_handle) {
|
2021-11-26 18:49:44 +00:00
|
|
|
wlr_foreign_toplevel_handle_v1_set_maximized(
|
|
|
|
|
view->toplevel_handle, maximize);
|
2021-08-05 12:52:42 +01:00
|
|
|
}
|
2021-03-12 21:27:17 +00:00
|
|
|
if (maximize) {
|
2021-12-23 12:22:46 -05:00
|
|
|
interactive_end(view);
|
2021-02-28 18:12:10 +00:00
|
|
|
view->unmaximized_geometry.x = view->x;
|
|
|
|
|
view->unmaximized_geometry.y = view->y;
|
|
|
|
|
view->unmaximized_geometry.width = view->w;
|
|
|
|
|
view->unmaximized_geometry.height = view->h;
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
view_apply_maximized_geometry(view);
|
2021-02-27 17:10:53 -05:00
|
|
|
view->maximized = true;
|
2021-02-28 18:12:10 +00:00
|
|
|
} else {
|
|
|
|
|
/* unmaximize */
|
2022-07-01 20:40:18 +02:00
|
|
|
if (view->tiled) {
|
|
|
|
|
view_apply_tiled_geometry(view, NULL);
|
|
|
|
|
} else {
|
|
|
|
|
view_apply_unmaximized_geometry(view);
|
|
|
|
|
}
|
2021-02-27 17:10:53 -05:00
|
|
|
view->maximized = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-02 16:49:41 +01:00
|
|
|
void
|
|
|
|
|
view_toggle_maximize(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
view_maximize(view, !view->maximized);
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-19 22:16:56 +00:00
|
|
|
void
|
|
|
|
|
view_toggle_decorations(struct view *view)
|
|
|
|
|
{
|
2022-01-02 10:19:48 -05:00
|
|
|
if (!view->fullscreen) {
|
|
|
|
|
view->ssd.enabled = !view->ssd.enabled;
|
2022-02-21 03:18:38 +01:00
|
|
|
ssd_update_geometry(view);
|
2022-01-02 10:19:48 -05:00
|
|
|
if (view->maximized) {
|
|
|
|
|
view_apply_maximized_geometry(view);
|
2022-07-01 19:40:27 +02:00
|
|
|
} else if (view->tiled) {
|
|
|
|
|
view_apply_tiled_geometry(view, NULL);
|
2022-01-02 10:19:48 -05:00
|
|
|
}
|
2021-12-31 12:09:20 -05:00
|
|
|
}
|
2021-09-19 22:16:56 +00:00
|
|
|
}
|
|
|
|
|
|
2022-04-09 01:16:09 +02:00
|
|
|
static bool
|
|
|
|
|
is_always_on_top(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
return view->scene_tree->node.parent ==
|
2022-06-05 15:17:35 +02:00
|
|
|
view->server->view_tree_always_on_top;
|
2022-04-09 01:16:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
view_toggle_always_on_top(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
if (is_always_on_top(view)) {
|
2022-06-15 02:02:50 +02:00
|
|
|
view->workspace = view->server->workspace_current;
|
|
|
|
|
wlr_scene_node_reparent(&view->scene_tree->node, view->workspace->tree);
|
2022-04-09 01:16:09 +02:00
|
|
|
} else {
|
|
|
|
|
wlr_scene_node_reparent(&view->scene_tree->node,
|
2022-06-05 15:17:35 +02:00
|
|
|
view->server->view_tree_always_on_top);
|
2022-04-09 01:16:09 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-20 16:03:27 +00:00
|
|
|
void
|
|
|
|
|
view_set_decorations(struct view *view, bool decorations)
|
|
|
|
|
{
|
2022-01-02 10:19:48 -05:00
|
|
|
if (view->ssd.enabled != decorations && !view->fullscreen) {
|
2021-10-20 16:03:27 +00:00
|
|
|
view->ssd.enabled = decorations;
|
2022-02-21 03:18:38 +01:00
|
|
|
ssd_update_geometry(view);
|
2021-12-31 12:09:20 -05:00
|
|
|
if (view->maximized) {
|
2022-01-01 11:50:35 -05:00
|
|
|
view_apply_maximized_geometry(view);
|
2022-07-01 19:40:27 +02:00
|
|
|
} else if (view->tiled) {
|
|
|
|
|
view_apply_tiled_geometry(view, NULL);
|
2021-12-31 12:09:20 -05:00
|
|
|
}
|
2021-10-20 16:03:27 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-19 22:16:56 +00:00
|
|
|
void
|
|
|
|
|
view_toggle_fullscreen(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
view_set_fullscreen(view, !view->fullscreen, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-23 22:05:30 +01:00
|
|
|
void
|
|
|
|
|
view_set_fullscreen(struct view *view, bool fullscreen,
|
|
|
|
|
struct wlr_output *wlr_output)
|
|
|
|
|
{
|
2022-04-04 20:53:36 +01:00
|
|
|
if (fullscreen != !view->fullscreen) {
|
2021-08-23 22:05:30 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (view->impl->set_fullscreen) {
|
2021-08-24 21:53:20 +01:00
|
|
|
view->impl->set_fullscreen(view, fullscreen);
|
2021-08-23 22:05:30 +01:00
|
|
|
}
|
|
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_fullscreen(
|
|
|
|
|
view->toplevel_handle, fullscreen);
|
|
|
|
|
}
|
2022-03-07 21:18:23 +00:00
|
|
|
if (!wlr_output) {
|
|
|
|
|
wlr_output = view_wlr_output(view);
|
|
|
|
|
}
|
2021-08-23 22:05:30 +01:00
|
|
|
if (fullscreen) {
|
2021-12-28 00:55:34 +01:00
|
|
|
if (!view->maximized) {
|
|
|
|
|
view->unmaximized_geometry.x = view->x;
|
|
|
|
|
view->unmaximized_geometry.y = view->y;
|
|
|
|
|
view->unmaximized_geometry.width = view->w;
|
|
|
|
|
view->unmaximized_geometry.height = view->h;
|
|
|
|
|
}
|
2021-08-23 22:05:30 +01:00
|
|
|
view->fullscreen = wlr_output;
|
2021-12-31 17:30:55 -05:00
|
|
|
view_apply_fullscreen_geometry(view, view->fullscreen);
|
2021-08-23 22:05:30 +01:00
|
|
|
} else {
|
|
|
|
|
/* restore to normal */
|
2021-12-28 00:55:34 +01:00
|
|
|
if (view->maximized) {
|
2021-12-31 17:30:55 -05:00
|
|
|
view_apply_maximized_geometry(view);
|
2022-07-01 20:40:18 +02:00
|
|
|
} else if (view->tiled) {
|
|
|
|
|
view_apply_tiled_geometry(view, NULL);
|
2021-12-28 00:55:34 +01:00
|
|
|
} else {
|
2021-12-31 17:30:55 -05:00
|
|
|
view_apply_unmaximized_geometry(view);
|
2021-12-28 00:55:34 +01:00
|
|
|
}
|
2021-08-23 22:05:30 +01:00
|
|
|
view->fullscreen = false;
|
|
|
|
|
}
|
2022-03-07 21:18:23 +00:00
|
|
|
|
|
|
|
|
/* Show fullscreen views above top-layer */
|
|
|
|
|
struct output *output =
|
|
|
|
|
output_from_wlr_output(view->server, wlr_output);
|
|
|
|
|
uint32_t top = ZWLR_LAYER_SHELL_V1_LAYER_TOP;
|
|
|
|
|
wlr_scene_node_set_enabled(&output->layer_tree[top]->node, !fullscreen);
|
2021-08-23 22:05:30 +01:00
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
void
|
|
|
|
|
view_adjust_for_layout_change(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
struct wlr_output_layout *layout = view->server->output_layout;
|
|
|
|
|
if (view->fullscreen) {
|
|
|
|
|
if (wlr_output_layout_get(layout, view->fullscreen)) {
|
|
|
|
|
/* recompute fullscreen geometry */
|
|
|
|
|
view_apply_fullscreen_geometry(view, view->fullscreen);
|
|
|
|
|
} else {
|
|
|
|
|
/* output is gone, exit fullscreen */
|
|
|
|
|
view_set_fullscreen(view, false, NULL);
|
|
|
|
|
}
|
2022-01-08 11:46:46 -05:00
|
|
|
} else if (view->maximized) {
|
|
|
|
|
/* recompute maximized geometry */
|
|
|
|
|
view_apply_maximized_geometry(view);
|
2022-07-01 02:12:23 +02:00
|
|
|
} else if (view->tiled) {
|
|
|
|
|
/* recompute tiled geometry */
|
|
|
|
|
view_apply_tiled_geometry(view, NULL);
|
2022-01-08 11:46:46 -05:00
|
|
|
} else {
|
|
|
|
|
/* reposition view if it's offscreen */
|
|
|
|
|
struct wlr_box box = { view->x, view->y, view->w, view->h };
|
|
|
|
|
if (!wlr_output_layout_intersects(layout, NULL, &box)) {
|
|
|
|
|
view_center(view);
|
|
|
|
|
}
|
2021-12-31 17:30:55 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-15 14:59:03 -05:00
|
|
|
static void
|
2021-11-13 12:32:01 -05:00
|
|
|
view_output_enter(struct view *view, struct wlr_output *wlr_output)
|
|
|
|
|
{
|
|
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_output_enter(
|
|
|
|
|
view->toplevel_handle, wlr_output);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-15 14:59:03 -05:00
|
|
|
static void
|
2021-11-13 12:32:01 -05:00
|
|
|
view_output_leave(struct view *view, struct wlr_output *wlr_output)
|
|
|
|
|
{
|
|
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_output_leave(
|
|
|
|
|
view->toplevel_handle, wlr_output);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-15 14:59:03 -05:00
|
|
|
/*
|
|
|
|
|
* At present, a view can only 'enter' one output at a time, although the view
|
|
|
|
|
* may span multiple outputs. Ideally we would handle multiple outputs, but
|
|
|
|
|
* this method is the simplest form of what we want.
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
view_discover_output(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
struct output *old_output = view->output;
|
|
|
|
|
struct output *new_output = view_output(view);
|
|
|
|
|
if (old_output != new_output) {
|
|
|
|
|
view->output = new_output;
|
|
|
|
|
view_output_enter(view, new_output->wlr_output);
|
|
|
|
|
if (old_output) {
|
|
|
|
|
view_output_leave(view, old_output->wlr_output);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-20 19:54:57 +01:00
|
|
|
void
|
2021-07-20 20:24:39 +01:00
|
|
|
view_move_to_edge(struct view *view, const char *direction)
|
2021-07-20 19:54:57 +01:00
|
|
|
{
|
|
|
|
|
if (!view) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no view");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
struct output *output = view_output(view);
|
2021-10-17 01:14:01 +00:00
|
|
|
if (!output) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no output");
|
2021-10-18 19:35:41 +01:00
|
|
|
return;
|
|
|
|
|
}
|
2022-06-03 02:19:31 +02:00
|
|
|
if (!direction) {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid edge");
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-07-21 22:04:54 +01:00
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
2022-04-04 20:53:36 +01:00
|
|
|
if (usable.height == output->wlr_output->height
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
2021-11-24 17:15:37 -05:00
|
|
|
usable.height /= output->wlr_output->scale;
|
|
|
|
|
}
|
2022-04-04 20:53:36 +01:00
|
|
|
if (usable.width == output->wlr_output->width
|
|
|
|
|
&& output->wlr_output->scale != 1) {
|
2021-11-24 17:15:37 -05:00
|
|
|
usable.width /= output->wlr_output->scale;
|
|
|
|
|
}
|
2021-07-20 19:54:57 +01:00
|
|
|
|
2021-08-04 21:43:07 +01:00
|
|
|
int x = 0, y = 0;
|
2021-07-20 19:54:57 +01:00
|
|
|
if (!strcasecmp(direction, "left")) {
|
2022-04-20 17:00:43 +01:00
|
|
|
x = usable.x + view->margin.left + rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
y = view->y;
|
|
|
|
|
} else if (!strcasecmp(direction, "up")) {
|
|
|
|
|
x = view->x;
|
2022-04-20 17:00:43 +01:00
|
|
|
y = usable.y + view->margin.top + rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
} else if (!strcasecmp(direction, "right")) {
|
2022-04-20 17:00:43 +01:00
|
|
|
x = usable.x + usable.width - view->w - view->margin.right
|
|
|
|
|
- rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
y = view->y;
|
|
|
|
|
} else if (!strcasecmp(direction, "down")) {
|
|
|
|
|
x = view->x;
|
2022-04-20 17:00:43 +01:00
|
|
|
y = usable.y + usable.height - view->h - view->margin.bottom
|
|
|
|
|
- rc.gap;
|
2022-06-03 02:19:31 +02:00
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid edge");
|
|
|
|
|
return;
|
2021-07-20 19:54:57 +01:00
|
|
|
}
|
|
|
|
|
view_move(view, x, y);
|
|
|
|
|
}
|
2021-08-05 12:18:10 +01:00
|
|
|
|
2021-10-17 00:44:24 +00:00
|
|
|
static enum view_edge
|
|
|
|
|
view_edge_parse(const char *direction)
|
|
|
|
|
{
|
2022-06-03 02:19:31 +02:00
|
|
|
if (!direction) {
|
|
|
|
|
return VIEW_EDGE_INVALID;
|
|
|
|
|
}
|
2021-10-17 00:44:24 +00:00
|
|
|
if (!strcasecmp(direction, "left")) {
|
|
|
|
|
return VIEW_EDGE_LEFT;
|
|
|
|
|
} else if (!strcasecmp(direction, "up")) {
|
|
|
|
|
return VIEW_EDGE_UP;
|
|
|
|
|
} else if (!strcasecmp(direction, "right")) {
|
|
|
|
|
return VIEW_EDGE_RIGHT;
|
|
|
|
|
} else if (!strcasecmp(direction, "down")) {
|
|
|
|
|
return VIEW_EDGE_DOWN;
|
2021-11-26 18:49:44 +00:00
|
|
|
} else if (!strcasecmp(direction, "center")) {
|
2021-10-17 22:09:34 +00:00
|
|
|
return VIEW_EDGE_CENTER;
|
2021-10-17 00:44:24 +00:00
|
|
|
} else {
|
|
|
|
|
return VIEW_EDGE_INVALID;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
view_snap_to_edge(struct view *view, const char *direction)
|
|
|
|
|
{
|
|
|
|
|
if (!view) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no view");
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-07-01 20:42:09 +02:00
|
|
|
if (view->fullscreen) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-10-17 00:44:24 +00:00
|
|
|
struct output *output = view_output(view);
|
|
|
|
|
if (!output) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no output");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
enum view_edge edge = view_edge_parse(direction);
|
|
|
|
|
if (edge == VIEW_EDGE_INVALID) {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid edge");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-01 02:07:40 +02:00
|
|
|
if (view->tiled == edge) {
|
|
|
|
|
/* We are already tiled for this edge and thus should switch outputs */
|
|
|
|
|
struct wlr_output *new_output = NULL;
|
|
|
|
|
struct wlr_output *current_output = output->wlr_output;
|
|
|
|
|
struct wlr_output_layout *layout = view->server->output_layout;
|
2021-10-17 00:44:24 +00:00
|
|
|
switch (edge) {
|
2021-11-26 18:49:44 +00:00
|
|
|
case VIEW_EDGE_LEFT:
|
2022-07-01 02:07:40 +02:00
|
|
|
new_output = wlr_output_layout_adjacent_output(
|
|
|
|
|
layout, WLR_DIRECTION_LEFT, current_output, 1, 0);
|
2021-11-26 18:49:44 +00:00
|
|
|
break;
|
|
|
|
|
case VIEW_EDGE_RIGHT:
|
2022-07-01 02:07:40 +02:00
|
|
|
new_output = wlr_output_layout_adjacent_output(
|
|
|
|
|
layout, WLR_DIRECTION_RIGHT, current_output, 1, 0);
|
2021-11-26 18:49:44 +00:00
|
|
|
break;
|
|
|
|
|
case VIEW_EDGE_UP:
|
2022-07-01 02:07:40 +02:00
|
|
|
new_output = wlr_output_layout_adjacent_output(
|
|
|
|
|
layout, WLR_DIRECTION_UP, current_output, 0, 1);
|
2021-11-26 18:49:44 +00:00
|
|
|
break;
|
|
|
|
|
case VIEW_EDGE_DOWN:
|
2022-07-01 02:07:40 +02:00
|
|
|
new_output = wlr_output_layout_adjacent_output(
|
|
|
|
|
layout, WLR_DIRECTION_DOWN, current_output, 0, 1);
|
2021-11-26 18:49:44 +00:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2021-10-17 00:44:24 +00:00
|
|
|
}
|
2022-07-01 02:07:40 +02:00
|
|
|
if (new_output && new_output != current_output) {
|
|
|
|
|
/* Move to next output */
|
|
|
|
|
edge = view_edge_invert(edge);
|
|
|
|
|
output = output_from_wlr_output(view->server, new_output);
|
|
|
|
|
} else {
|
|
|
|
|
/* No more output to move to */
|
2021-10-17 00:44:24 +00:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-01 02:07:40 +02:00
|
|
|
/* TODO: store old geometry if !maximized && !fullscreen && !tiled */
|
|
|
|
|
view->tiled = edge;
|
|
|
|
|
view_apply_tiled_geometry(view, output);
|
2021-10-17 00:44:24 +00:00
|
|
|
}
|
|
|
|
|
|
2021-10-18 20:01:10 +01:00
|
|
|
const char *
|
|
|
|
|
view_get_string_prop(struct view *view, const char *prop)
|
|
|
|
|
{
|
|
|
|
|
if (view->impl->get_string_prop) {
|
2021-12-06 20:16:30 +00:00
|
|
|
return view->impl->get_string_prop(view, prop);
|
2021-10-18 20:01:10 +01:00
|
|
|
}
|
2021-10-18 20:06:47 +01:00
|
|
|
return "";
|
2021-10-18 20:01:10 +01:00
|
|
|
}
|
|
|
|
|
|
2021-08-05 12:18:10 +01:00
|
|
|
void
|
|
|
|
|
view_update_title(struct view *view)
|
|
|
|
|
{
|
2021-10-18 20:01:10 +01:00
|
|
|
const char *title = view_get_string_prop(view, "title");
|
2021-08-05 12:18:10 +01:00
|
|
|
if (!view->toplevel_handle || !title) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-08-07 09:35:53 +01:00
|
|
|
ssd_update_title(view);
|
2021-08-05 12:18:10 +01:00
|
|
|
wlr_foreign_toplevel_handle_v1_set_title(view->toplevel_handle, title);
|
|
|
|
|
}
|
2021-10-16 21:50:56 +01:00
|
|
|
|
|
|
|
|
void
|
|
|
|
|
view_update_app_id(struct view *view)
|
|
|
|
|
{
|
2021-10-18 20:01:10 +01:00
|
|
|
const char *app_id = view_get_string_prop(view, "app_id");
|
2021-10-16 21:50:56 +01:00
|
|
|
if (!view->toplevel_handle || !app_id) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_app_id(
|
|
|
|
|
view->toplevel_handle, app_id);
|
|
|
|
|
}
|
2022-04-21 23:33:44 +00:00
|
|
|
|
|
|
|
|
void
|
2022-04-23 03:44:41 +02:00
|
|
|
view_destroy(struct view *view)
|
2022-04-21 23:33:44 +00:00
|
|
|
{
|
|
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_destroy(view->toplevel_handle);
|
|
|
|
|
}
|
|
|
|
|
interactive_end(view);
|
|
|
|
|
|
2022-06-09 16:38:17 +02:00
|
|
|
struct server *server = view->server;
|
|
|
|
|
if (server->seat.pressed.view == view) {
|
|
|
|
|
server->seat.pressed.view = NULL;
|
|
|
|
|
server->seat.pressed.surface = NULL;
|
2022-04-21 23:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
2022-06-09 16:38:17 +02:00
|
|
|
if (server->cycle_view == view) {
|
2022-04-21 23:33:44 +00:00
|
|
|
/* If we are the current OSD selected view, cycle
|
|
|
|
|
* to the next because we are dying. */
|
2022-06-09 16:38:17 +02:00
|
|
|
server->cycle_view = desktop_cycle_view(server,
|
|
|
|
|
server->cycle_view, LAB_CYCLE_DIR_BACKWARD);
|
2022-04-21 23:33:44 +00:00
|
|
|
|
|
|
|
|
/* If we cycled back to ourselves, then we have no windows.
|
2022-04-23 03:44:41 +02:00
|
|
|
* just remove it and close the OSD for good. */
|
2022-06-09 16:38:17 +02:00
|
|
|
if (server->cycle_view == view || !server->cycle_view) {
|
|
|
|
|
server->cycle_view = NULL;
|
|
|
|
|
osd_finish(server);
|
2022-04-21 23:33:44 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-09 16:38:17 +02:00
|
|
|
if (server->cycle_view) {
|
2022-04-23 03:44:41 +02:00
|
|
|
/* Update the OSD to reflect the view has now gone. */
|
2022-06-09 16:38:17 +02:00
|
|
|
osd_update(server);
|
2022-04-21 23:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (view->scene_tree) {
|
|
|
|
|
ssd_destroy(view);
|
|
|
|
|
wlr_scene_node_destroy(&view->scene_tree->node);
|
|
|
|
|
view->scene_tree = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-07 22:28:25 +01:00
|
|
|
/*
|
|
|
|
|
* The layer-shell top-layer is disabled when an application is running
|
|
|
|
|
* in fullscreen mode, so if that's the case, we have to re-enable it
|
|
|
|
|
* here.
|
|
|
|
|
*/
|
|
|
|
|
if (view->fullscreen) {
|
|
|
|
|
struct output *output =
|
2022-06-09 16:38:17 +02:00
|
|
|
output_from_wlr_output(server, view->fullscreen);
|
2022-06-07 22:28:25 +01:00
|
|
|
uint32_t top = ZWLR_LAYER_SHELL_V1_LAYER_TOP;
|
|
|
|
|
wlr_scene_node_set_enabled(&output->layer_tree[top]->node, true);
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-09 17:10:36 +02:00
|
|
|
/* If we spawned a window menu, close it */
|
|
|
|
|
if (server->menu_current
|
|
|
|
|
&& server->menu_current->triggered_by_view == view) {
|
|
|
|
|
menu_close_root(server);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-23 03:44:41 +02:00
|
|
|
/* Remove view from server->views */
|
|
|
|
|
wl_list_remove(&view->link);
|
2022-04-21 23:33:44 +00:00
|
|
|
free(view);
|
|
|
|
|
}
|