2018-05-05 12:36:50 +10:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
2018-04-01 07:49:40 +10:00
|
|
|
#include <stdlib.h>
|
2018-02-26 09:23:36 +11:00
|
|
|
#include <wayland-server.h>
|
2018-04-30 21:24:13 +10:00
|
|
|
#include <wlr/render/wlr_renderer.h>
|
2018-02-26 09:23:36 +11:00
|
|
|
#include <wlr/types/wlr_output_layout.h>
|
2018-03-31 04:18:50 +11:00
|
|
|
#include "log.h"
|
2018-04-24 14:59:49 +10:00
|
|
|
#include "sway/criteria.h"
|
|
|
|
#include "sway/commands.h"
|
2018-04-16 20:36:40 +10:00
|
|
|
#include "sway/ipc-server.h"
|
2018-03-31 04:18:50 +11:00
|
|
|
#include "sway/output.h"
|
2018-04-24 14:59:49 +10:00
|
|
|
#include "sway/input/seat.h"
|
2018-04-28 11:26:14 +10:00
|
|
|
#include "sway/tree/arrange.h"
|
2018-03-30 14:41:33 +11:00
|
|
|
#include "sway/tree/container.h"
|
|
|
|
#include "sway/tree/layout.h"
|
|
|
|
#include "sway/tree/view.h"
|
2018-04-17 09:31:34 +10:00
|
|
|
#include "sway/tree/workspace.h"
|
2018-05-08 02:30:45 +10:00
|
|
|
#include "sway/config.h"
|
|
|
|
#include "pango.h"
|
2018-01-22 01:09:53 +11:00
|
|
|
|
2018-04-06 01:38:14 +10:00
|
|
|
void view_init(struct sway_view *view, enum sway_view_type type,
|
2018-04-01 08:07:44 +10:00
|
|
|
const struct sway_view_impl *impl) {
|
2018-04-01 07:49:40 +10:00
|
|
|
view->type = type;
|
2018-04-01 08:07:44 +10:00
|
|
|
view->impl = impl;
|
2018-04-06 04:46:02 +10:00
|
|
|
wl_signal_init(&view->events.unmap);
|
2018-04-01 07:49:40 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void view_destroy(struct sway_view *view) {
|
|
|
|
if (view == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (view->surface != NULL) {
|
|
|
|
view_unmap(view);
|
|
|
|
}
|
|
|
|
|
2018-04-04 02:25:19 +10:00
|
|
|
container_destroy(view->swayc);
|
2018-04-06 01:38:14 +10:00
|
|
|
|
|
|
|
if (view->impl->destroy) {
|
|
|
|
view->impl->destroy(view);
|
|
|
|
} else {
|
|
|
|
free(view);
|
|
|
|
}
|
2018-04-01 07:49:40 +10:00
|
|
|
}
|
|
|
|
|
2018-01-22 01:09:53 +11:00
|
|
|
const char *view_get_title(struct sway_view *view) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->get_prop) {
|
|
|
|
return view->impl->get_prop(view, VIEW_PROP_TITLE);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *view_get_app_id(struct sway_view *view) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->get_prop) {
|
|
|
|
return view->impl->get_prop(view, VIEW_PROP_APP_ID);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *view_get_class(struct sway_view *view) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->get_prop) {
|
|
|
|
return view->impl->get_prop(view, VIEW_PROP_CLASS);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *view_get_instance(struct sway_view *view) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->get_prop) {
|
|
|
|
return view->impl->get_prop(view, VIEW_PROP_INSTANCE);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-05-05 12:36:50 +10:00
|
|
|
const char *view_get_type(struct sway_view *view) {
|
|
|
|
switch(view->type) {
|
|
|
|
case SWAY_VIEW_WL_SHELL:
|
|
|
|
return "wl_shell";
|
|
|
|
case SWAY_VIEW_XDG_SHELL_V6:
|
|
|
|
return "xdg_shell_v6";
|
|
|
|
case SWAY_VIEW_XWAYLAND:
|
|
|
|
return "xwayland";
|
|
|
|
}
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
|
2018-04-03 00:57:45 +10:00
|
|
|
void view_configure(struct sway_view *view, double ox, double oy, int width,
|
|
|
|
int height) {
|
|
|
|
if (view->impl->configure) {
|
|
|
|
view->impl->configure(view, ox, oy, width, height);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-30 21:24:13 +10:00
|
|
|
void view_autoconfigure(struct sway_view *view) {
|
|
|
|
if (!sway_assert(view->swayc,
|
|
|
|
"Called view_autoconfigure() on a view without a swayc")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (view->is_fullscreen) {
|
|
|
|
struct sway_container *output = container_parent(view->swayc, C_OUTPUT);
|
|
|
|
view_configure(view, 0, 0, output->width, output->height);
|
|
|
|
view->x = view->y = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-13 02:56:26 +10:00
|
|
|
int other_views = 1;
|
|
|
|
if (config->hide_edge_borders == E_SMART) {
|
|
|
|
struct sway_container *ws = container_parent(view->swayc, C_WORKSPACE);
|
|
|
|
other_views = container_count_descendants_of_type(ws, C_VIEW) - 1;
|
|
|
|
}
|
2018-05-12 08:44:56 +10:00
|
|
|
|
2018-04-30 21:24:13 +10:00
|
|
|
double x, y, width, height;
|
2018-05-02 19:40:45 +10:00
|
|
|
x = y = width = height = 0;
|
2018-04-30 21:24:13 +10:00
|
|
|
switch (view->border) {
|
2018-05-01 17:36:12 +10:00
|
|
|
case B_NONE:
|
|
|
|
x = view->swayc->x;
|
|
|
|
y = view->swayc->y;
|
|
|
|
width = view->swayc->width;
|
|
|
|
height = view->swayc->height;
|
|
|
|
break;
|
|
|
|
case B_PIXEL:
|
2018-05-12 08:44:56 +10:00
|
|
|
if (view->swayc->layout > L_VERT
|
|
|
|
|| config->hide_edge_borders == E_NONE
|
|
|
|
|| config->hide_edge_borders == E_HORIZONTAL
|
|
|
|
|| (config->hide_edge_borders == E_SMART && other_views)) {
|
|
|
|
x = view->swayc->x + view->border_thickness;
|
|
|
|
width = view->swayc->width - view->border_thickness * 2;
|
|
|
|
} else {
|
|
|
|
x = view->swayc->x;
|
|
|
|
width = view->swayc->width;
|
|
|
|
}
|
|
|
|
if (view->swayc->layout > L_VERT
|
|
|
|
|| config->hide_edge_borders == E_NONE
|
|
|
|
|| config->hide_edge_borders == E_VERTICAL
|
|
|
|
|| (config->hide_edge_borders == E_SMART && other_views)) {
|
|
|
|
y = view->swayc->y + view->border_thickness;
|
|
|
|
height = view->swayc->height - view->border_thickness * 2;
|
|
|
|
} else {
|
|
|
|
y = view->swayc->y;
|
|
|
|
height = view->swayc->height;
|
|
|
|
}
|
2018-05-01 17:36:12 +10:00
|
|
|
break;
|
|
|
|
case B_NORMAL:
|
2018-05-12 08:44:56 +10:00
|
|
|
if (view->swayc->layout > L_VERT
|
|
|
|
|| config->hide_edge_borders == E_NONE
|
|
|
|
|| config->hide_edge_borders == E_HORIZONTAL
|
|
|
|
|| (config->hide_edge_borders == E_SMART && other_views)) {
|
|
|
|
x = view->swayc->x + view->border_thickness;
|
|
|
|
width = view->swayc->width - view->border_thickness * 2;
|
|
|
|
} else {
|
|
|
|
x = view->swayc->x;
|
|
|
|
width = view->swayc->width;
|
|
|
|
}
|
|
|
|
if (view->swayc->layout > L_VERT
|
|
|
|
|| config->hide_edge_borders == E_NONE
|
|
|
|
|| config->hide_edge_borders == E_VERTICAL
|
|
|
|
|| (config->hide_edge_borders == E_SMART && other_views)) {
|
|
|
|
// Height is: border + title height + border + view height + border
|
|
|
|
y = view->swayc->y + config->font_height
|
|
|
|
+ view->border_thickness * 2;
|
|
|
|
height = view->swayc->height - config->font_height
|
|
|
|
- view->border_thickness * 3;
|
|
|
|
} else {
|
|
|
|
y = view->swayc->y;
|
|
|
|
height = view->swayc->height;
|
|
|
|
}
|
2018-05-01 17:36:12 +10:00
|
|
|
break;
|
2018-04-30 21:24:13 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
view->x = x;
|
|
|
|
view->y = y;
|
|
|
|
view_configure(view, x, y, width, height);
|
|
|
|
}
|
|
|
|
|
2018-01-22 01:09:53 +11:00
|
|
|
void view_set_activated(struct sway_view *view, bool activated) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->set_activated) {
|
|
|
|
view->impl->set_activated(view, activated);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-28 11:26:14 +10:00
|
|
|
// Set fullscreen, but without IPC events or arranging windows.
|
|
|
|
void view_set_fullscreen_raw(struct sway_view *view, bool fullscreen) {
|
2018-04-16 20:36:40 +10:00
|
|
|
if (view->is_fullscreen == fullscreen) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-28 11:26:14 +10:00
|
|
|
struct sway_container *workspace =
|
|
|
|
container_parent(view->swayc, C_WORKSPACE);
|
2018-04-16 20:36:40 +10:00
|
|
|
|
|
|
|
if (view->impl->set_fullscreen) {
|
|
|
|
view->impl->set_fullscreen(view, fullscreen);
|
|
|
|
}
|
|
|
|
|
2018-04-17 08:11:50 +10:00
|
|
|
view->is_fullscreen = fullscreen;
|
|
|
|
|
2018-04-16 20:36:40 +10:00
|
|
|
if (fullscreen) {
|
2018-04-18 00:10:32 +10:00
|
|
|
if (workspace->sway_workspace->fullscreen) {
|
|
|
|
view_set_fullscreen(workspace->sway_workspace->fullscreen, false);
|
|
|
|
}
|
2018-04-17 09:31:34 +10:00
|
|
|
workspace->sway_workspace->fullscreen = view;
|
2018-04-28 11:26:14 +10:00
|
|
|
view->swayc->saved_width = view->swayc->width;
|
|
|
|
view->swayc->saved_height = view->swayc->height;
|
2018-04-18 00:10:32 +10:00
|
|
|
|
|
|
|
struct sway_seat *seat;
|
|
|
|
struct sway_container *focus, *focus_ws;
|
|
|
|
wl_list_for_each(seat, &input_manager->seats, link) {
|
|
|
|
focus = seat_get_focus(seat);
|
2018-05-11 11:20:20 +10:00
|
|
|
if (focus) {
|
|
|
|
focus_ws = focus;
|
|
|
|
if (focus && focus_ws->type != C_WORKSPACE) {
|
|
|
|
focus_ws = container_parent(focus_ws, C_WORKSPACE);
|
|
|
|
}
|
|
|
|
seat_set_focus(seat, view->swayc);
|
|
|
|
if (focus_ws != workspace) {
|
|
|
|
seat_set_focus(seat, focus);
|
|
|
|
}
|
2018-04-18 00:10:32 +10:00
|
|
|
}
|
|
|
|
}
|
2018-04-16 20:36:40 +10:00
|
|
|
} else {
|
2018-04-17 09:31:34 +10:00
|
|
|
workspace->sway_workspace->fullscreen = NULL;
|
2018-04-28 11:26:14 +10:00
|
|
|
view->swayc->width = view->swayc->saved_width;
|
|
|
|
view->swayc->height = view->swayc->saved_height;
|
2018-04-16 20:36:40 +10:00
|
|
|
}
|
2018-04-28 11:26:14 +10:00
|
|
|
}
|
2018-04-16 20:36:40 +10:00
|
|
|
|
2018-04-28 11:26:14 +10:00
|
|
|
void view_set_fullscreen(struct sway_view *view, bool fullscreen) {
|
|
|
|
if (view->is_fullscreen == fullscreen) {
|
|
|
|
return;
|
|
|
|
}
|
2018-04-16 20:36:40 +10:00
|
|
|
|
2018-04-28 11:26:14 +10:00
|
|
|
view_set_fullscreen_raw(view, fullscreen);
|
|
|
|
|
|
|
|
struct sway_container *workspace =
|
|
|
|
container_parent(view->swayc, C_WORKSPACE);
|
|
|
|
arrange_workspace(workspace);
|
|
|
|
output_damage_whole(workspace->parent->sway_output);
|
2018-04-16 20:36:40 +10:00
|
|
|
ipc_event_window(view->swayc, "fullscreen_mode");
|
|
|
|
}
|
|
|
|
|
2018-01-22 01:09:53 +11:00
|
|
|
void view_close(struct sway_view *view) {
|
2018-04-01 08:07:44 +10:00
|
|
|
if (view->impl->close) {
|
|
|
|
view->impl->close(view);
|
2018-01-22 01:09:53 +11:00
|
|
|
}
|
|
|
|
}
|
2018-02-26 09:23:36 +11:00
|
|
|
|
2018-05-06 04:43:12 +10:00
|
|
|
void view_damage_from(struct sway_view *view) {
|
2018-04-03 00:57:45 +10:00
|
|
|
for (int i = 0; i < root_container.children->length; ++i) {
|
|
|
|
struct sway_container *cont = root_container.children->items[i];
|
|
|
|
if (cont->type == C_OUTPUT) {
|
2018-05-06 04:43:12 +10:00
|
|
|
output_damage_from_view(cont->sway_output, view);
|
2018-04-03 00:57:45 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void view_get_layout_box(struct sway_view *view, struct wlr_box *box) {
|
2018-04-03 04:35:43 +10:00
|
|
|
struct sway_container *output = container_parent(view->swayc, C_OUTPUT);
|
2018-04-03 00:57:45 +10:00
|
|
|
|
2018-04-03 04:35:43 +10:00
|
|
|
box->x = output->x + view->swayc->x;
|
|
|
|
box->y = output->y + view->swayc->y;
|
2018-04-03 00:57:45 +10:00
|
|
|
box->width = view->width;
|
|
|
|
box->height = view->height;
|
|
|
|
}
|
|
|
|
|
2018-04-07 00:26:32 +10:00
|
|
|
void view_for_each_surface(struct sway_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);
|
|
|
|
} else {
|
|
|
|
wlr_surface_for_each_surface(view->surface, iterator, user_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:46:02 +10:00
|
|
|
static void view_subsurface_create(struct sway_view *view,
|
|
|
|
struct wlr_subsurface *subsurface);
|
|
|
|
|
|
|
|
static void view_init_subsurfaces(struct sway_view *view,
|
|
|
|
struct wlr_surface *surface);
|
|
|
|
|
|
|
|
static void view_handle_surface_new_subsurface(struct wl_listener *listener,
|
|
|
|
void *data) {
|
|
|
|
struct sway_view *view =
|
|
|
|
wl_container_of(listener, view, surface_new_subsurface);
|
|
|
|
struct wlr_subsurface *subsurface = data;
|
|
|
|
view_subsurface_create(view, subsurface);
|
|
|
|
}
|
|
|
|
|
2018-04-07 00:26:32 +10:00
|
|
|
static void surface_send_enter_iterator(struct wlr_surface *surface,
|
|
|
|
int x, int y, void *data) {
|
|
|
|
struct wlr_output *wlr_output = data;
|
|
|
|
wlr_surface_send_enter(surface, wlr_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void surface_send_leave_iterator(struct wlr_surface *surface,
|
|
|
|
int x, int y, void *data) {
|
|
|
|
struct wlr_output *wlr_output = data;
|
|
|
|
wlr_surface_send_leave(surface, wlr_output);
|
|
|
|
}
|
|
|
|
|
2018-04-06 13:50:21 +10:00
|
|
|
static void view_handle_container_reparent(struct wl_listener *listener,
|
|
|
|
void *data) {
|
|
|
|
struct sway_view *view =
|
|
|
|
wl_container_of(listener, view, container_reparent);
|
|
|
|
struct sway_container *old_parent = data;
|
|
|
|
|
|
|
|
struct sway_container *old_output = old_parent;
|
|
|
|
if (old_output != NULL && old_output->type != C_OUTPUT) {
|
|
|
|
old_output = container_parent(old_output, C_OUTPUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct sway_container *new_output = view->swayc->parent;
|
|
|
|
if (new_output != NULL && new_output->type != C_OUTPUT) {
|
|
|
|
new_output = container_parent(new_output, C_OUTPUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_output == new_output) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_output != NULL) {
|
2018-04-07 00:26:32 +10:00
|
|
|
view_for_each_surface(view, surface_send_leave_iterator,
|
2018-04-06 13:50:21 +10:00
|
|
|
old_output->sway_output->wlr_output);
|
|
|
|
}
|
|
|
|
if (new_output != NULL) {
|
2018-04-07 00:26:32 +10:00
|
|
|
view_for_each_surface(view, surface_send_enter_iterator,
|
2018-04-06 13:50:21 +10:00
|
|
|
new_output->sway_output->wlr_output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 14:59:49 +10:00
|
|
|
static void view_execute_criteria(struct sway_view *view) {
|
2018-04-24 20:14:22 +10:00
|
|
|
if (!sway_assert(view->swayc, "cannot run criteria for unmapped view")) {
|
2018-04-24 20:06:35 +10:00
|
|
|
return;
|
|
|
|
}
|
2018-04-24 14:59:49 +10:00
|
|
|
struct sway_seat *seat = input_manager_current_seat(input_manager);
|
|
|
|
struct sway_container *prior_workspace =
|
|
|
|
container_parent(view->swayc, C_WORKSPACE);
|
|
|
|
list_t *criteria = criteria_for(view->swayc);
|
|
|
|
for (int i = 0; i < criteria->length; i++) {
|
|
|
|
struct criteria *crit = criteria->items[i];
|
|
|
|
wlr_log(L_DEBUG, "for_window '%s' matches new view %p, cmd: '%s'",
|
|
|
|
crit->crit_raw, view, crit->cmdlist);
|
|
|
|
struct cmd_results *res = execute_command(crit->cmdlist, NULL);
|
|
|
|
if (res->status != CMD_SUCCESS) {
|
|
|
|
wlr_log(L_ERROR, "Command '%s' failed: %s", res->input, res->error);
|
|
|
|
}
|
|
|
|
free_cmd_results(res);
|
|
|
|
// view must be focused for commands to affect it,
|
|
|
|
// so always refocus in-between command lists
|
|
|
|
seat_set_focus(seat, view->swayc);
|
|
|
|
}
|
|
|
|
list_free(criteria);
|
|
|
|
seat_set_focus(seat, seat_get_focus_inactive(seat, prior_workspace));
|
|
|
|
}
|
|
|
|
|
2018-04-01 07:49:40 +10:00
|
|
|
void view_map(struct sway_view *view, struct wlr_surface *wlr_surface) {
|
|
|
|
if (!sway_assert(view->surface == NULL, "cannot map mapped view")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct sway_seat *seat = input_manager_current_seat(input_manager);
|
2018-04-03 05:30:58 +10:00
|
|
|
struct sway_container *focus = seat_get_focus_inactive(seat,
|
2018-04-01 07:49:40 +10:00
|
|
|
&root_container);
|
|
|
|
struct sway_container *cont = container_view_create(focus, view);
|
|
|
|
|
|
|
|
view->surface = wlr_surface;
|
|
|
|
view->swayc = cont;
|
2018-04-30 21:24:13 +10:00
|
|
|
view->border = config->border;
|
|
|
|
view->border_thickness = config->border_thickness;
|
2018-04-01 07:49:40 +10:00
|
|
|
|
2018-04-06 04:46:02 +10:00
|
|
|
view_init_subsurfaces(view, wlr_surface);
|
|
|
|
wl_signal_add(&wlr_surface->events.new_subsurface,
|
|
|
|
&view->surface_new_subsurface);
|
|
|
|
view->surface_new_subsurface.notify = view_handle_surface_new_subsurface;
|
|
|
|
|
2018-04-06 13:50:21 +10:00
|
|
|
wl_signal_add(&view->swayc->events.reparent, &view->container_reparent);
|
|
|
|
view->container_reparent.notify = view_handle_container_reparent;
|
|
|
|
|
2018-04-28 11:26:14 +10:00
|
|
|
arrange_children_of(cont->parent);
|
2018-04-03 05:30:58 +10:00
|
|
|
input_manager_set_focus(input_manager, cont);
|
2018-04-01 07:49:40 +10:00
|
|
|
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(cont);
|
2018-04-06 13:50:21 +10:00
|
|
|
view_handle_container_reparent(&view->container_reparent, NULL);
|
2018-04-24 14:59:49 +10:00
|
|
|
|
|
|
|
view_execute_criteria(view);
|
2018-04-01 07:49:40 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void view_unmap(struct sway_view *view) {
|
|
|
|
if (!sway_assert(view->surface != NULL, "cannot unmap unmapped view")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:46:02 +10:00
|
|
|
wl_signal_emit(&view->events.unmap, view);
|
|
|
|
|
2018-04-17 11:06:03 +10:00
|
|
|
if (view->is_fullscreen) {
|
|
|
|
struct sway_container *ws = container_parent(view->swayc, C_WORKSPACE);
|
|
|
|
ws->sway_workspace->fullscreen = NULL;
|
|
|
|
}
|
|
|
|
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(view->swayc);
|
2018-04-01 07:49:40 +10:00
|
|
|
|
2018-04-06 06:48:11 +10:00
|
|
|
wl_list_remove(&view->surface_new_subsurface.link);
|
2018-04-06 13:50:21 +10:00
|
|
|
wl_list_remove(&view->container_reparent.link);
|
|
|
|
|
|
|
|
struct sway_container *parent = container_destroy(view->swayc);
|
2018-04-06 06:48:11 +10:00
|
|
|
|
2018-04-01 07:49:40 +10:00
|
|
|
view->swayc = NULL;
|
|
|
|
view->surface = NULL;
|
2018-04-03 12:37:21 +10:00
|
|
|
|
2018-05-05 12:36:50 +10:00
|
|
|
if (view->title_format) {
|
|
|
|
free(view->title_format);
|
|
|
|
view->title_format = NULL;
|
|
|
|
}
|
|
|
|
|
2018-05-01 17:18:20 +10:00
|
|
|
if (parent->type == C_OUTPUT) {
|
|
|
|
arrange_output(parent);
|
|
|
|
} else {
|
|
|
|
arrange_children_of(parent);
|
|
|
|
}
|
2018-04-01 07:49:40 +10:00
|
|
|
}
|
|
|
|
|
2018-05-01 19:59:36 +10:00
|
|
|
void view_update_position(struct sway_view *view, double ox, double oy) {
|
|
|
|
if (view->swayc->x == ox && view->swayc->y == oy) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Only allow this if the view is floating (this function will only be
|
|
|
|
// called in response to wayland clients wanting to reposition themselves).
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(view->swayc);
|
2018-05-01 19:59:36 +10:00
|
|
|
view->swayc->x = ox;
|
|
|
|
view->swayc->y = oy;
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(view->swayc);
|
2018-05-01 19:59:36 +10:00
|
|
|
}
|
|
|
|
|
2018-04-03 00:57:45 +10:00
|
|
|
void view_update_size(struct sway_view *view, int width, int height) {
|
|
|
|
if (view->width == width && view->height == height) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(view->swayc);
|
2018-04-30 21:24:13 +10:00
|
|
|
// Should we update the swayc width/height here too?
|
2018-04-03 00:57:45 +10:00
|
|
|
view->width = width;
|
|
|
|
view->height = height;
|
2018-05-06 04:43:12 +10:00
|
|
|
container_damage_whole(view->swayc);
|
2018-03-31 04:18:50 +11:00
|
|
|
}
|
2018-04-06 04:46:02 +10:00
|
|
|
|
|
|
|
|
|
|
|
static void view_subsurface_create(struct sway_view *view,
|
|
|
|
struct wlr_subsurface *subsurface) {
|
|
|
|
struct sway_view_child *child = calloc(1, sizeof(struct sway_view_child));
|
|
|
|
if (child == NULL) {
|
|
|
|
wlr_log(L_ERROR, "Allocation failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
view_child_init(child, NULL, view, subsurface->surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void view_child_handle_surface_commit(struct wl_listener *listener,
|
|
|
|
void *data) {
|
|
|
|
struct sway_view_child *child =
|
|
|
|
wl_container_of(listener, child, surface_commit);
|
|
|
|
// TODO: only accumulate damage from the child
|
2018-05-06 04:43:12 +10:00
|
|
|
view_damage_from(child->view);
|
2018-04-06 04:46:02 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
static void view_child_handle_surface_new_subsurface(
|
|
|
|
struct wl_listener *listener, void *data) {
|
|
|
|
struct sway_view_child *child =
|
|
|
|
wl_container_of(listener, child, surface_new_subsurface);
|
|
|
|
struct wlr_subsurface *subsurface = data;
|
|
|
|
view_subsurface_create(child->view, subsurface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void view_child_handle_surface_destroy(struct wl_listener *listener,
|
|
|
|
void *data) {
|
|
|
|
struct sway_view_child *child =
|
|
|
|
wl_container_of(listener, child, surface_destroy);
|
|
|
|
view_child_destroy(child);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void view_child_handle_view_unmap(struct wl_listener *listener,
|
|
|
|
void *data) {
|
|
|
|
struct sway_view_child *child =
|
|
|
|
wl_container_of(listener, child, view_unmap);
|
|
|
|
view_child_destroy(child);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void view_init_subsurfaces(struct sway_view *view,
|
|
|
|
struct wlr_surface *surface) {
|
|
|
|
struct wlr_subsurface *subsurface;
|
2018-04-22 04:12:49 +10:00
|
|
|
wl_list_for_each(subsurface, &surface->subsurfaces, parent_link) {
|
2018-04-06 04:46:02 +10:00
|
|
|
view_subsurface_create(view, subsurface);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void view_child_init(struct sway_view_child *child,
|
|
|
|
const struct sway_view_child_impl *impl, struct sway_view *view,
|
|
|
|
struct wlr_surface *surface) {
|
|
|
|
child->impl = impl;
|
|
|
|
child->view = view;
|
|
|
|
child->surface = surface;
|
|
|
|
|
|
|
|
wl_signal_add(&surface->events.commit, &child->surface_commit);
|
|
|
|
child->surface_commit.notify = view_child_handle_surface_commit;
|
|
|
|
wl_signal_add(&surface->events.new_subsurface,
|
|
|
|
&child->surface_new_subsurface);
|
|
|
|
child->surface_new_subsurface.notify =
|
|
|
|
view_child_handle_surface_new_subsurface;
|
|
|
|
wl_signal_add(&surface->events.destroy, &child->surface_destroy);
|
|
|
|
child->surface_destroy.notify = view_child_handle_surface_destroy;
|
|
|
|
wl_signal_add(&view->events.unmap, &child->view_unmap);
|
|
|
|
child->view_unmap.notify = view_child_handle_view_unmap;
|
|
|
|
|
2018-04-07 00:26:32 +10:00
|
|
|
struct sway_container *output = child->view->swayc->parent;
|
|
|
|
if (output != NULL) {
|
|
|
|
if (output->type != C_OUTPUT) {
|
|
|
|
output = container_parent(output, C_OUTPUT);
|
|
|
|
}
|
|
|
|
wlr_surface_send_enter(child->surface, output->sway_output->wlr_output);
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:46:02 +10:00
|
|
|
view_init_subsurfaces(child->view, surface);
|
|
|
|
|
|
|
|
// TODO: only damage the whole child
|
2018-05-06 04:43:12 +10:00
|
|
|
if (child->view->swayc) {
|
|
|
|
container_damage_whole(child->view->swayc);
|
|
|
|
}
|
2018-04-06 04:46:02 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void view_child_destroy(struct sway_view_child *child) {
|
|
|
|
// TODO: only damage the whole child
|
2018-05-06 04:43:12 +10:00
|
|
|
if (child->view->swayc) {
|
|
|
|
container_damage_whole(child->view->swayc);
|
|
|
|
}
|
2018-04-06 04:46:02 +10:00
|
|
|
|
|
|
|
wl_list_remove(&child->surface_commit.link);
|
|
|
|
wl_list_remove(&child->surface_destroy.link);
|
|
|
|
wl_list_remove(&child->view_unmap.link);
|
|
|
|
|
|
|
|
if (child->impl && child->impl->destroy) {
|
|
|
|
child->impl->destroy(child);
|
|
|
|
} else {
|
|
|
|
free(child);
|
|
|
|
}
|
|
|
|
}
|
2018-05-05 12:36:50 +10:00
|
|
|
|
2018-05-05 18:18:54 +10:00
|
|
|
/**
|
|
|
|
* Calculate and return the length of the formatted title.
|
|
|
|
* If buffer is not NULL, also populate the buffer with the formatted title.
|
|
|
|
*/
|
|
|
|
static size_t parse_title_format(struct sway_view *view, char *buffer) {
|
2018-05-05 12:36:50 +10:00
|
|
|
if (!view->title_format || strcmp(view->title_format, "%title") == 0) {
|
2018-05-05 18:18:54 +10:00
|
|
|
const char *title = view_get_title(view);
|
2018-05-05 18:28:35 +10:00
|
|
|
if (buffer && title) {
|
2018-05-05 18:18:54 +10:00
|
|
|
strcpy(buffer, title);
|
|
|
|
}
|
2018-05-05 18:28:35 +10:00
|
|
|
return title ? strlen(title) : 0;
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|
|
|
|
const char *title = view_get_title(view);
|
|
|
|
const char *class = view_get_class(view);
|
|
|
|
const char *instance = view_get_instance(view);
|
|
|
|
const char *shell = view_get_type(view);
|
|
|
|
size_t title_len = title ? strlen(title) : 0;
|
|
|
|
size_t class_len = class ? strlen(class) : 0;
|
|
|
|
size_t instance_len = instance ? strlen(instance) : 0;
|
|
|
|
size_t shell_len = shell ? strlen(shell) : 0;
|
|
|
|
|
|
|
|
size_t len = 0;
|
|
|
|
char *format = view->title_format;
|
|
|
|
char *next = strchr(format, '%');
|
|
|
|
while (next) {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer) {
|
|
|
|
// Copy everything up to the %
|
|
|
|
strncat(buffer, format, next - format);
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
len += next - format;
|
2018-05-05 12:36:50 +10:00
|
|
|
format = next;
|
|
|
|
|
|
|
|
if (strncmp(next, "%title", 6) == 0) {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer && title) {
|
2018-05-05 12:36:50 +10:00
|
|
|
strcat(buffer, title);
|
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
len += title_len;
|
2018-05-05 12:36:50 +10:00
|
|
|
format += 6;
|
|
|
|
} else if (strncmp(next, "%class", 6) == 0) {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer && class) {
|
2018-05-05 12:36:50 +10:00
|
|
|
strcat(buffer, class);
|
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
len += class_len;
|
2018-05-05 12:36:50 +10:00
|
|
|
format += 6;
|
|
|
|
} else if (strncmp(next, "%instance", 9) == 0) {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer && instance) {
|
2018-05-05 12:36:50 +10:00
|
|
|
strcat(buffer, instance);
|
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
len += instance_len;
|
2018-05-05 12:36:50 +10:00
|
|
|
format += 9;
|
|
|
|
} else if (strncmp(next, "%shell", 6) == 0) {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer) {
|
|
|
|
strcat(buffer, shell);
|
|
|
|
}
|
|
|
|
len += shell_len;
|
2018-05-05 12:36:50 +10:00
|
|
|
format += 6;
|
|
|
|
} else {
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer) {
|
|
|
|
strcat(buffer, "%");
|
|
|
|
}
|
2018-05-05 12:36:50 +10:00
|
|
|
++format;
|
2018-05-05 18:18:54 +10:00
|
|
|
++len;
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|
|
|
|
next = strchr(format, '%');
|
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
if (buffer) {
|
|
|
|
strcat(buffer, format);
|
|
|
|
}
|
|
|
|
len += strlen(format);
|
2018-05-05 12:36:50 +10:00
|
|
|
|
2018-05-05 18:18:54 +10:00
|
|
|
return len;
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|
|
|
|
|
2018-05-08 02:30:45 +10:00
|
|
|
static char *escape_title(char *buffer) {
|
|
|
|
int length = escape_markup_text(buffer, NULL, 0);
|
|
|
|
char *escaped_title = calloc(length + 1, sizeof(char));
|
|
|
|
int result = escape_markup_text(buffer, escaped_title, length);
|
|
|
|
if (result != length) {
|
|
|
|
wlr_log(L_ERROR, "Could not escape title: %s", buffer);
|
|
|
|
free(escaped_title);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
free(buffer);
|
|
|
|
return escaped_title;
|
|
|
|
}
|
|
|
|
|
2018-05-05 12:36:50 +10:00
|
|
|
void view_update_title(struct sway_view *view, bool force) {
|
|
|
|
if (!view->swayc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const char *title = view_get_title(view);
|
|
|
|
|
|
|
|
if (!force) {
|
|
|
|
if (title && view->swayc->name && strcmp(title, view->swayc->name) == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!title && !view->swayc->name) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-05 18:18:54 +10:00
|
|
|
free(view->swayc->name);
|
|
|
|
free(view->swayc->formatted_title);
|
2018-05-05 12:36:50 +10:00
|
|
|
if (title) {
|
2018-05-05 18:18:54 +10:00
|
|
|
size_t len = parse_title_format(view, NULL);
|
2018-05-08 02:30:45 +10:00
|
|
|
char *buffer = calloc(len + 1, sizeof(char));
|
2018-05-05 18:18:54 +10:00
|
|
|
if (!sway_assert(buffer, "Unable to allocate title string")) {
|
|
|
|
return;
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
parse_title_format(view, buffer);
|
2018-05-08 02:30:45 +10:00
|
|
|
// now we have the title, but needs to be escaped when using pango markup
|
|
|
|
if (config->pango_markup) {
|
|
|
|
buffer = escape_title(buffer);
|
|
|
|
}
|
2018-05-05 18:18:54 +10:00
|
|
|
|
2018-05-05 12:36:50 +10:00
|
|
|
view->swayc->name = strdup(title);
|
2018-05-05 18:18:54 +10:00
|
|
|
view->swayc->formatted_title = buffer;
|
2018-05-05 12:36:50 +10:00
|
|
|
} else {
|
|
|
|
view->swayc->name = NULL;
|
|
|
|
view->swayc->formatted_title = NULL;
|
|
|
|
}
|
|
|
|
container_calculate_title_height(view->swayc);
|
|
|
|
container_update_title_textures(view->swayc);
|
|
|
|
container_notify_child_title_changed(view->swayc->parent);
|
2018-05-05 18:25:31 +10:00
|
|
|
config_update_font_height(false);
|
2018-05-05 12:36:50 +10:00
|
|
|
}
|