// SPDX-License-Identifier: GPL-2.0-only #include #include #include "labwc.h" #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 view_move_resize(struct view *view, struct wlr_box geo) { view->impl->configure(view, geo); ssd_update_title(view); } void view_move(struct view *view, double x, double y) { view->impl->move(view, x, y); } void view_minimize(struct view *view, bool minimized) { if (view->minimized == minimized) { return; } 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); } } /* view_wlr_output - return the output that a view is mostly on */ struct wlr_output * view_wlr_output(struct view *view) { return wlr_output_layout_output_at(view->server->output_layout, view->x + view->w / 2, view->y + view->h / 2); } 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); } void view_center(struct view *view) { struct wlr_output *wlr_output = view_wlr_output(view); if (!wlr_output) { return; } struct wlr_output_layout *layout = view->server->output_layout; 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; view_move(view, center_x - view->w / 2, center_y - view->h / 2); } void view_maximize(struct view *view, bool maximize) { if (view->maximized == maximize) { return; } view->impl->maximize(view, maximize); if (view->toplevel_handle) { wlr_foreign_toplevel_handle_v1_set_maximized(view->toplevel_handle, maximize); } if (maximize) { 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 (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); view->maximized = true; } else { /* unmaximize */ view_move_resize(view, view->unmaximized_geometry); view->maximized = false; } } 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_toggle_fullscreen(struct view *view) { view_set_fullscreen(view, !view->fullscreen, NULL); } 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); } 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; } } 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); } void view_for_each_popup_surface(struct view *view, wlr_surface_iterator_func_t iterator, void *data) { if (!view->impl->for_each_popup_surface) { return; } view->impl->for_each_popup_surface(view, iterator, data); } 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 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); struct border border = view_border(view); struct wlr_box usable = output_usable_area_in_layout_coords(output); 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, }; 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); } void view_update_title(struct view *view) { const char *title = view->impl->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); } 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); }