labwc/src/view.c

547 lines
13 KiB
C
Raw Normal View History

2021-09-24 21:45:48 +01:00
// SPDX-License-Identifier: GPL-2.0-only
#include <stdio.h>
#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"
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);
}
}
void
2020-12-22 20:35:06 +00:00
view_move_resize(struct view *view, struct wlr_box geo)
{
if (view->impl->configure) {
view->impl->configure(view, geo);
}
ssd_update_title(view);
view_discover_output(view);
}
2020-12-23 18:52:46 +00:00
void
view_move(struct view *view, double x, double y)
{
if (view->impl->move) {
view->impl->move(view, x, y);
}
view_discover_output(view);
2020-12-23 18:52:46 +00:00
}
#define MIN_VIEW_WIDTH (100)
#define MIN_VIEW_HEIGHT (60)
void
view_min_size(struct view *view, int *w, int *h)
{
int min_width = MIN_VIEW_WIDTH;
int min_height = MIN_VIEW_HEIGHT;
#if HAVE_XWAYLAND
2021-11-26 18:49:44 +00:00
if (view->type != LAB_XWAYLAND_VIEW) {
goto out;
}
if (!view->xwayland_surface->size_hints) {
goto out;
}
2021-11-26 18:49:44 +00:00
if (view->xwayland_surface->size_hints->min_width > 0
|| view->xwayland_surface->size_hints->min_height > 0) {
min_width = view->xwayland_surface->size_hints->min_width;
min_height = view->xwayland_surface->size_hints->min_height;
}
out:
#endif
if (w) {
*w = min_width;
}
if (h) {
*h = min_height;
}
}
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;
}
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);
desktop_move_view_to_end_of_cycle(view);
2021-08-05 13:00:34 +01:00
} else {
view->impl->map(view);
}
2020-09-08 20:51:33 +01:00
}
2020-09-29 20:48:50 +01: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
}
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)
{
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 =
wlr_output_layout_get(layout, wlr_output);
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);
}
void
view_maximize(struct view *view, bool maximize)
{
2021-03-12 21:27:17 +00:00
if (view->maximized == maximize) {
return;
}
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-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;
struct output *output = view_output(view);
struct wlr_box box = output_usable_area_in_layout_coords(output);
if (box.height == output->wlr_output->height && output->wlr_output->scale != 1) {
box.height /= output->wlr_output->scale;
}
if (box.width == output->wlr_output->width && output->wlr_output->scale != 1) {
box.width /= output->wlr_output->scale;
}
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;
}
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);
view->maximized = false;
}
}
2021-08-02 16:49:41 +01:00
void
view_toggle_maximize(struct view *view)
{
view_maximize(view, !view->maximized);
}
void
view_toggle_decorations(struct view *view)
{
view->ssd.enabled = !view->ssd.enabled;
ssd_update_geometry(view, true);
}
void
view_set_decorations(struct view *view, bool decorations)
{
if (view->ssd.enabled != decorations) {
view->ssd.enabled = decorations;
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) {
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)
{
if (view->impl->for_each_surface) {
view->impl->for_each_surface(view, iterator, user_data);
}
2020-09-29 20:48:50 +01:00
}
2021-01-09 22:51:20 +00:00
void
view_for_each_popup_surface(struct view *view,
wlr_surface_iterator_func_t iterator, void *data)
2021-01-09 22:51:20 +00:00
{
if (view->impl->for_each_popup_surface) {
view->impl->for_each_popup_surface(view, iterator, data);
2021-01-09 22:51:20 +00:00
}
}
2021-10-23 22:31:39 -04:00
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;
}
static void
surface_enter_for_each_surface(struct wlr_surface *surface, int sx, int sy,
void *user_data)
{
struct wlr_output *wlr_output = user_data;
wlr_surface_send_enter(surface, wlr_output);
}
static void
surface_leave_for_each_surface(struct wlr_surface *surface, int sx, int sy,
void *user_data)
{
struct wlr_output *wlr_output = user_data;
wlr_surface_send_leave(surface, wlr_output);
}
static void
view_output_enter(struct view *view, struct wlr_output *wlr_output)
{
view_for_each_surface(view, surface_enter_for_each_surface,
wlr_output);
if (view->toplevel_handle) {
wlr_foreign_toplevel_handle_v1_output_enter(
view->toplevel_handle, wlr_output);
}
}
static void
view_output_leave(struct view *view, struct wlr_output *wlr_output)
{
view_for_each_surface(view, surface_leave_for_each_surface,
wlr_output);
if (view->toplevel_handle) {
wlr_foreign_toplevel_handle_v1_output_leave(
view->toplevel_handle, wlr_output);
}
}
/*
* 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);
}
}
}
void
2021-07-20 20:24:39 +01:00
view_move_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;
}
struct border border = view_border(view);
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 = 0, y = 0;
if (!strcasecmp(direction, "left")) {
x = usable.x + border.left + rc.gap;
y = view->y;
} else if (!strcasecmp(direction, "up")) {
x = view->x;
y = usable.y + border.top + rc.gap;
} else if (!strcasecmp(direction, "right")) {
x = usable.x + usable.width - view->w - border.right - rc.gap;
y = view->y;
} else if (!strcasecmp(direction, "down")) {
x = view->x;
y = usable.y + usable.height - view->h - border.bottom - rc.gap;
}
view_move(view, x, y);
}
enum view_edge {
VIEW_EDGE_INVALID,
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) {
2021-11-26 18:49:44 +00:00
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 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;
2021-11-26 18:49:44 +00:00
} else if (!strcasecmp(direction, "center")) {
return VIEW_EDGE_CENTER;
} else {
return VIEW_EDGE_INVALID;
}
}
static struct wlr_box
2021-11-26 18:49:44 +00:00
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);
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 + border.left,
.y = y_offset + usable.y + border.top,
.width = base_width - border.left - border.right,
.height = base_height - border.top - border.bottom,
};
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);
2021-11-26 18:49:44 +00:00
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. */
2021-11-26 18:49:44 +00:00
struct wlr_box usable =
output_usable_area_in_layout_coords(output);
switch (edge) {
2021-11-26 18:49:44 +00:00
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;
}
2021-11-26 18:49:44 +00:00
struct wlr_output *new_wlr_output = wlr_output_layout_output_at(
view->server->output_layout, dst.x, dst.y);
struct output *new_output =
2021-11-26 18:49:44 +00:00
output_from_wlr_output(view->server, new_wlr_output);
2021-11-26 18:49:44 +00:00
if (new_output == output || !new_output
|| edge == VIEW_EDGE_CENTER) {
return;
}
2021-11-26 18:49:44 +00:00
dst = view_get_edge_snap_box(view, new_output,
view_edge_invert(edge));
}
view_move_resize(view, dst);
}
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
}
void
view_update_title(struct view *view)
{
2021-10-18 20:01:10 +01:00
const char *title = view_get_string_prop(view, "title");
if (!view->toplevel_handle || !title) {
return;
}
ssd_update_title(view);
wlr_foreign_toplevel_handle_v1_set_title(view->toplevel_handle, title);
damage_all_outputs(view->server);
}
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");
if (!view->toplevel_handle || !app_id) {
return;
}
wlr_foreign_toplevel_handle_v1_set_app_id(
view->toplevel_handle, app_id);
}