2021-09-24 21:45:48 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2021-02-27 17:10:53 -05:00
|
|
|
#include <stdio.h>
|
2021-07-20 19:54:57 +01:00
|
|
|
#include <strings.h>
|
2021-03-20 14:36:40 +00:00
|
|
|
#include "labwc.h"
|
2021-03-21 20:54:55 +00:00
|
|
|
#include "ssd.h"
|
2021-03-02 20:37:23 +00:00
|
|
|
|
2021-10-16 19:24:26 +01:00
|
|
|
void
|
|
|
|
|
view_set_activated(struct view *view, bool activated)
|
|
|
|
|
{
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
void
|
2020-12-22 20:35:06 +00:00
|
|
|
view_move_resize(struct view *view, struct wlr_box geo)
|
2020-05-26 12:56:33 +01:00
|
|
|
{
|
2020-12-22 20:35:06 +00:00
|
|
|
view->impl->configure(view, geo);
|
2021-08-22 14:35:34 +01:00
|
|
|
ssd_update_title(view);
|
2020-05-26 12:56:33 +01:00
|
|
|
}
|
|
|
|
|
|
2020-12-23 18:52:46 +00:00
|
|
|
void
|
|
|
|
|
view_move(struct view *view, double x, double y)
|
|
|
|
|
{
|
|
|
|
|
view->impl->move(view, x, y);
|
|
|
|
|
}
|
|
|
|
|
|
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) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_minimized(view->toplevel_handle,
|
|
|
|
|
minimized);
|
|
|
|
|
}
|
|
|
|
|
view->minimized = minimized;
|
|
|
|
|
if (minimized) {
|
|
|
|
|
view->impl->unmap(view);
|
|
|
|
|
} 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-07-21 22:04:54 +01:00
|
|
|
/* view_wlr_output - return the output that a view is mostly on */
|
2021-08-05 12:18:10 +01:00
|
|
|
struct wlr_output *
|
2021-07-20 19:40:37 +01:00
|
|
|
view_wlr_output(struct view *view)
|
2021-03-20 14:36:40 +00:00
|
|
|
{
|
2021-07-21 22:04:54 +01:00
|
|
|
return wlr_output_layout_output_at(view->server->output_layout,
|
|
|
|
|
view->x + view->w / 2, view->y + view->h / 2);
|
2021-03-20 14:36:40 +00: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-07-09 21:39:20 +01:00
|
|
|
void
|
|
|
|
|
view_center(struct view *view)
|
|
|
|
|
{
|
2021-07-20 19:40:37 +01:00
|
|
|
struct wlr_output *wlr_output = view_wlr_output(view);
|
|
|
|
|
if (!wlr_output) {
|
2021-07-09 21:39:20 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct wlr_output_layout *layout = view->server->output_layout;
|
2021-08-25 20:45:39 +01:00
|
|
|
struct wlr_output_layout_output *ol_output =
|
2021-07-20 19:40:37 +01:00
|
|
|
wlr_output_layout_get(layout, wlr_output);
|
2021-08-17 07:24:27 +01:00
|
|
|
int center_x = ol_output->x + wlr_output->width / wlr_output->scale / 2;
|
|
|
|
|
int center_y = ol_output->y + wlr_output->height / wlr_output->scale / 2;
|
2021-07-09 21:39:20 +01:00
|
|
|
view_move(view, center_x - view->w / 2, center_y - view->h / 2);
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
view->impl->maximize(view, maximize);
|
2021-08-05 12:52:42 +01:00
|
|
|
if (view->toplevel_handle) {
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_maximized(view->toplevel_handle,
|
|
|
|
|
maximize);
|
|
|
|
|
}
|
2021-03-12 21:27:17 +00:00
|
|
|
if (maximize) {
|
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-07-20 19:40:37 +01:00
|
|
|
struct output *output = view_output(view);
|
2021-07-21 22:04:54 +01:00
|
|
|
struct wlr_box box = output_usable_area_in_layout_coords(output);
|
2021-07-12 21:39:09 +01:00
|
|
|
|
2021-03-21 21:46:16 +00:00
|
|
|
if (view->ssd.enabled) {
|
2021-03-20 14:41:39 +00:00
|
|
|
struct border border = ssd_thickness(view);
|
2021-03-01 18:15:02 +00:00
|
|
|
box.x += border.left;
|
|
|
|
|
box.y += border.top;
|
|
|
|
|
box.width -= border.right + border.left;
|
|
|
|
|
box.height -= border.top + border.bottom;
|
|
|
|
|
}
|
2021-02-27 17:10:53 -05:00
|
|
|
view_move_resize(view, box);
|
|
|
|
|
view->maximized = true;
|
2021-02-28 18:12:10 +00:00
|
|
|
} else {
|
|
|
|
|
/* unmaximize */
|
|
|
|
|
view_move_resize(view, view->unmaximized_geometry);
|
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)
|
|
|
|
|
{
|
|
|
|
|
view->ssd.enabled = !view->ssd.enabled;
|
|
|
|
|
ssd_update_geometry(view, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
if (fullscreen == (view->fullscreen != NULL)) {
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
if (fullscreen) {
|
|
|
|
|
view->unmaximized_geometry.x = view->x;
|
|
|
|
|
view->unmaximized_geometry.y = view->y;
|
|
|
|
|
view->unmaximized_geometry.width = view->w;
|
|
|
|
|
view->unmaximized_geometry.height = view->h;
|
|
|
|
|
|
|
|
|
|
if (!wlr_output) {
|
|
|
|
|
wlr_output = view_wlr_output(view);
|
|
|
|
|
}
|
|
|
|
|
view->fullscreen = 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);
|
|
|
|
|
} else {
|
|
|
|
|
/* restore to normal */
|
|
|
|
|
view_move_resize(view, view->unmaximized_geometry);
|
|
|
|
|
view->fullscreen = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-29 20:48:50 +01:00
|
|
|
void
|
|
|
|
|
view_for_each_surface(struct view *view, wlr_surface_iterator_func_t iterator,
|
|
|
|
|
void *user_data)
|
|
|
|
|
{
|
|
|
|
|
view->impl->for_each_surface(view, iterator, user_data);
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-09 22:51:20 +00:00
|
|
|
void
|
2021-03-13 21:07:11 +00:00
|
|
|
view_for_each_popup_surface(struct view *view, wlr_surface_iterator_func_t iterator,
|
2021-01-09 22:51:20 +00:00
|
|
|
void *data)
|
|
|
|
|
{
|
2021-03-13 21:07:11 +00:00
|
|
|
if (!view->impl->for_each_popup_surface) {
|
2021-01-09 22:51:20 +00:00
|
|
|
return;
|
|
|
|
|
}
|
2021-03-13 21:07:11 +00:00
|
|
|
view->impl->for_each_popup_surface(view, iterator, data);
|
2021-01-09 22:51:20 +00:00
|
|
|
}
|
|
|
|
|
|
2021-07-20 19:54:57 +01:00
|
|
|
static struct border
|
|
|
|
|
view_border(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
struct border border = {
|
|
|
|
|
.left = view->margin.left - view->padding.left,
|
|
|
|
|
.top = view->margin.top - view->padding.top,
|
|
|
|
|
.right = view->margin.right + view->padding.right,
|
|
|
|
|
.bottom = view->margin.bottom + view->padding.bottom,
|
|
|
|
|
};
|
|
|
|
|
return border;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
struct border border = view_border(view);
|
2021-07-21 22:04:54 +01:00
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
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")) {
|
2021-08-22 14:32:19 +01:00
|
|
|
x = usable.x + border.left + rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
y = view->y;
|
|
|
|
|
} else if (!strcasecmp(direction, "up")) {
|
|
|
|
|
x = view->x;
|
2021-08-22 14:32:19 +01:00
|
|
|
y = usable.y + border.top + rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
} else if (!strcasecmp(direction, "right")) {
|
2021-08-22 14:32:19 +01:00
|
|
|
x = usable.x + usable.width - view->w - border.right - rc.gap;
|
2021-07-20 19:54:57 +01:00
|
|
|
y = view->y;
|
|
|
|
|
} else if (!strcasecmp(direction, "down")) {
|
|
|
|
|
x = view->x;
|
2021-08-22 14:32:19 +01:00
|
|
|
y = usable.y + usable.height - view->h - border.bottom - rc.gap;
|
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
|
|
|
enum view_edge {
|
|
|
|
|
VIEW_EDGE_INVALID,
|
|
|
|
|
|
|
|
|
|
VIEW_EDGE_LEFT,
|
|
|
|
|
VIEW_EDGE_RIGHT,
|
|
|
|
|
VIEW_EDGE_UP,
|
|
|
|
|
VIEW_EDGE_DOWN,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
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_INVALID:
|
|
|
|
|
default:
|
|
|
|
|
return VIEW_EDGE_INVALID;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static enum view_edge
|
|
|
|
|
view_edge_parse(const char *direction)
|
|
|
|
|
{
|
|
|
|
|
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;
|
|
|
|
|
} else {
|
|
|
|
|
return VIEW_EDGE_INVALID;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct wlr_box
|
|
|
|
|
view_get_edge_snap_box(struct view *view, struct output *output, enum view_edge edge)
|
|
|
|
|
{
|
|
|
|
|
struct border border = view_border(view);
|
|
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
|
|
|
|
|
|
|
|
|
int x_offset = edge == VIEW_EDGE_RIGHT ? usable.width / 2 : 0;
|
|
|
|
|
int y_offset = edge == VIEW_EDGE_DOWN ? usable.height / 2 : 0;
|
|
|
|
|
int base_width = (edge == VIEW_EDGE_LEFT || edge == VIEW_EDGE_RIGHT) ? usable.width / 2 : usable.width;
|
|
|
|
|
int base_height = (edge == VIEW_EDGE_UP || edge == VIEW_EDGE_DOWN) ? usable.height / 2 : usable.height;
|
|
|
|
|
struct wlr_box dst = {
|
|
|
|
|
.x = x_offset + usable.x + border.left + rc.gap,
|
|
|
|
|
.y = y_offset + usable.y + border.top + rc.gap,
|
|
|
|
|
.width = base_width - border.left - border.right - 2 * rc.gap,
|
|
|
|
|
.height = base_height - border.top - border.bottom - 2 * rc.gap,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return dst;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
view_snap_to_edge(struct view *view, const char *direction)
|
|
|
|
|
{
|
|
|
|
|
if (!view) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no view");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct wlr_box dst = view_get_edge_snap_box(view, output, edge);
|
|
|
|
|
|
|
|
|
|
if (view->x == dst.x && view->y == dst.y && view->w == dst.width && view->h == dst.height) {
|
|
|
|
|
/* Move over to the next screen if this is already snapped. */
|
|
|
|
|
struct wlr_box usable = output_usable_area_in_layout_coords(output);
|
|
|
|
|
switch (edge) {
|
|
|
|
|
case VIEW_EDGE_LEFT: dst.x -= (usable.width / 2) + 1; break;
|
|
|
|
|
case VIEW_EDGE_RIGHT: dst.x += (usable.width / 2) + 1; break;
|
|
|
|
|
case VIEW_EDGE_UP: dst.y -= (usable.height / 2) + 1; break;
|
|
|
|
|
case VIEW_EDGE_DOWN: dst.y += (usable.height / 2) + 1; break;
|
|
|
|
|
default: break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct output *new_output =
|
|
|
|
|
output_from_wlr_output(view->server,
|
|
|
|
|
wlr_output_layout_output_at(view->server->output_layout, dst.x, dst.y));
|
|
|
|
|
|
|
|
|
|
if (new_output == output || !new_output) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dst = view_get_edge_snap_box(view, new_output, view_edge_invert(edge));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
view_move_resize(view, dst);
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-05 12:18:10 +01:00
|
|
|
void
|
|
|
|
|
view_update_title(struct view *view)
|
|
|
|
|
{
|
|
|
|
|
const char *title = view->impl->get_string_prop(view, "title");
|
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
const char *app_id = view->impl->get_string_prop(view, "app_id");
|
|
|
|
|
if (!view->toplevel_handle || !app_id) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
wlr_foreign_toplevel_handle_v1_set_app_id(
|
|
|
|
|
view->toplevel_handle, app_id);
|
|
|
|
|
}
|