mirror of
https://github.com/swaywm/sway.git
synced 2025-01-04 11:26:41 +01:00
Introduce common functions to create, map, unmap, destroy views
This commit is contained in:
parent
122b96abed
commit
b2c2ee693b
8 changed files with 212 additions and 150 deletions
|
@ -24,6 +24,7 @@ struct sway_output {
|
||||||
struct wl_listener destroy;
|
struct wl_listener destroy;
|
||||||
struct wl_listener mode;
|
struct wl_listener mode;
|
||||||
struct wl_listener transform;
|
struct wl_listener transform;
|
||||||
|
struct wl_listener scale;
|
||||||
|
|
||||||
struct wl_listener damage_destroy;
|
struct wl_listener damage_destroy;
|
||||||
struct wl_listener damage_frame;
|
struct wl_listener damage_frame;
|
||||||
|
|
|
@ -70,9 +70,8 @@ struct sway_container {
|
||||||
enum sway_container_layout prev_layout;
|
enum sway_container_layout prev_layout;
|
||||||
enum sway_container_layout workspace_layout;
|
enum sway_container_layout workspace_layout;
|
||||||
|
|
||||||
// TODO convert to layout coordinates
|
// in output-local coordinates
|
||||||
double x, y;
|
double x, y;
|
||||||
|
|
||||||
// does not include borders or gaps.
|
// does not include borders or gaps.
|
||||||
double width, height;
|
double width, height;
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,8 @@
|
||||||
#include <wlr/types/wlr_surface.h>
|
#include <wlr/types/wlr_surface.h>
|
||||||
#include <wlr/types/wlr_xdg_shell_v6.h>
|
#include <wlr/types/wlr_xdg_shell_v6.h>
|
||||||
#include <wlr/xwayland.h>
|
#include <wlr/xwayland.h>
|
||||||
|
#include "sway/input/input-manager.h"
|
||||||
|
#include "sway/input/seat.h"
|
||||||
|
|
||||||
struct sway_container;
|
struct sway_container;
|
||||||
struct sway_view;
|
struct sway_view;
|
||||||
|
@ -94,9 +96,13 @@ struct sway_view {
|
||||||
} iface;
|
} iface;
|
||||||
|
|
||||||
// only used for unmanaged views (shell specific)
|
// only used for unmanaged views (shell specific)
|
||||||
struct wl_list unmanaged_view_link; // sway_root::unmanaged views
|
struct wl_list unmanaged_view_link; // sway_root::unmanaged_views
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct sway_view *view_create(enum sway_view_type type);
|
||||||
|
|
||||||
|
void view_destroy(struct sway_view *view);
|
||||||
|
|
||||||
const char *view_get_title(struct sway_view *view);
|
const char *view_get_title(struct sway_view *view);
|
||||||
|
|
||||||
const char *view_get_app_id(struct sway_view *view);
|
const char *view_get_app_id(struct sway_view *view);
|
||||||
|
@ -113,7 +119,12 @@ void view_set_activated(struct sway_view *view, bool activated);
|
||||||
|
|
||||||
void view_close(struct sway_view *view);
|
void view_close(struct sway_view *view);
|
||||||
|
|
||||||
void view_update_outputs(struct sway_view *view, const struct wlr_box *before);
|
void view_map(struct sway_view *view, struct wlr_surface *wlr_surface);
|
||||||
|
|
||||||
|
void view_map_unmanaged(struct sway_view *view,
|
||||||
|
struct wlr_surface *wlr_surface);
|
||||||
|
|
||||||
|
void view_unmap(struct sway_view *view);
|
||||||
|
|
||||||
void view_damage_whole(struct sway_view *view);
|
void view_damage_whole(struct sway_view *view);
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,32 @@ static void rotate_child_position(double *sx, double *sy, double sw, double sh,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks whether a surface at (lx, ly) intersects an output. If `box` is not
|
||||||
|
* NULL, it populates it with the surface box in the output, in output-local
|
||||||
|
* coordinates.
|
||||||
|
*/
|
||||||
|
static bool surface_intersect_output(struct wlr_surface *surface,
|
||||||
|
struct wlr_output_layout *output_layout, struct wlr_output *wlr_output,
|
||||||
|
double lx, double ly, float rotation, struct wlr_box *box) {
|
||||||
|
double ox = lx, oy = ly;
|
||||||
|
wlr_output_layout_output_coords(output_layout, wlr_output, &ox, &oy);
|
||||||
|
|
||||||
|
if (box != NULL) {
|
||||||
|
box->x = ox * wlr_output->scale;
|
||||||
|
box->y = oy * wlr_output->scale;
|
||||||
|
box->width = surface->current->width * wlr_output->scale;
|
||||||
|
box->height = surface->current->height * wlr_output->scale;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wlr_box layout_box = {
|
||||||
|
.x = lx, .y = ly,
|
||||||
|
.width = surface->current->width, .height = surface->current->height,
|
||||||
|
};
|
||||||
|
wlr_box_rotated_bounds(&layout_box, rotation, &layout_box);
|
||||||
|
return wlr_output_layout_intersects(output_layout, wlr_output, &layout_box);
|
||||||
|
}
|
||||||
|
|
||||||
static void render_surface(struct wlr_surface *surface,
|
static void render_surface(struct wlr_surface *surface,
|
||||||
struct wlr_output *wlr_output, struct timespec *when,
|
struct wlr_output *wlr_output, struct timespec *when,
|
||||||
double lx, double ly, float rotation) {
|
double lx, double ly, float rotation) {
|
||||||
|
@ -48,29 +74,21 @@ static void render_surface(struct wlr_surface *surface,
|
||||||
if (!wlr_surface_has_buffer(surface)) {
|
if (!wlr_surface_has_buffer(surface)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
struct wlr_output_layout *layout = root_container.sway_root->output_layout;
|
|
||||||
int width = surface->current->width;
|
|
||||||
int height = surface->current->height;
|
|
||||||
int render_width = width * wlr_output->scale;
|
|
||||||
int render_height = height * wlr_output->scale;
|
|
||||||
int owidth, oheight;
|
|
||||||
wlr_output_effective_resolution(wlr_output, &owidth, &oheight);
|
|
||||||
|
|
||||||
// FIXME: view coords are inconsistently assumed to be in output or layout coords
|
struct wlr_output_layout *layout = root_container.sway_root->output_layout;
|
||||||
struct wlr_box layout_box = {
|
|
||||||
.x = lx + wlr_output->lx, .y = ly + wlr_output->ly,
|
struct wlr_box box;
|
||||||
.width = render_width, .height = render_height,
|
bool intersects = surface_intersect_output(surface, layout, wlr_output,
|
||||||
};
|
lx, ly, rotation, &box);
|
||||||
if (wlr_output_layout_intersects(layout, wlr_output, &layout_box)) {
|
if (intersects) {
|
||||||
struct wlr_box render_box = {
|
|
||||||
.x = lx, .y = ly,
|
|
||||||
.width = render_width, .height = render_height
|
|
||||||
};
|
|
||||||
float matrix[9];
|
float matrix[9];
|
||||||
wlr_matrix_project_box(matrix, &render_box, surface->current->transform,
|
enum wl_output_transform transform =
|
||||||
0, wlr_output->transform_matrix);
|
wlr_output_transform_invert(surface->current->transform);
|
||||||
wlr_render_texture_with_matrix(renderer, surface->texture, matrix,
|
wlr_matrix_project_box(matrix, &box, transform, rotation,
|
||||||
1.0f); // TODO: configurable alpha
|
wlr_output->transform_matrix);
|
||||||
|
|
||||||
|
// TODO: configurable alpha
|
||||||
|
wlr_render_texture_with_matrix(renderer, surface->texture, matrix, 1.0f);
|
||||||
|
|
||||||
wlr_surface_send_frame_done(surface, when);
|
wlr_surface_send_frame_done(surface, when);
|
||||||
}
|
}
|
||||||
|
@ -80,9 +98,8 @@ static void render_surface(struct wlr_surface *surface,
|
||||||
struct wlr_surface_state *state = subsurface->surface->current;
|
struct wlr_surface_state *state = subsurface->surface->current;
|
||||||
double sx = state->subsurface_position.x;
|
double sx = state->subsurface_position.x;
|
||||||
double sy = state->subsurface_position.y;
|
double sy = state->subsurface_position.y;
|
||||||
double sw = state->buffer_width / state->scale;
|
rotate_child_position(&sx, &sy, state->width, state->height,
|
||||||
double sh = state->buffer_height / state->scale;
|
surface->current->width, surface->current->height, rotation);
|
||||||
rotate_child_position(&sx, &sy, sw, sh, width, height, rotation);
|
|
||||||
|
|
||||||
render_surface(subsurface->surface, wlr_output, when,
|
render_surface(subsurface->surface, wlr_output, when,
|
||||||
lx + sx, ly + sy, rotation);
|
lx + sx, ly + sy, rotation);
|
||||||
|
@ -338,6 +355,12 @@ static void handle_transform(struct wl_listener *listener, void *data) {
|
||||||
arrange_windows(output->swayc, -1, -1);
|
arrange_windows(output->swayc, -1, -1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void handle_scale(struct wl_listener *listener, void *data) {
|
||||||
|
struct sway_output *output = wl_container_of(listener, output, scale);
|
||||||
|
arrange_layers(output);
|
||||||
|
arrange_windows(output->swayc, -1, -1);
|
||||||
|
}
|
||||||
|
|
||||||
void handle_new_output(struct wl_listener *listener, void *data) {
|
void handle_new_output(struct wl_listener *listener, void *data) {
|
||||||
struct sway_server *server = wl_container_of(listener, server, new_output);
|
struct sway_server *server = wl_container_of(listener, server, new_output);
|
||||||
struct wlr_output *wlr_output = data;
|
struct wlr_output *wlr_output = data;
|
||||||
|
@ -378,6 +401,8 @@ void handle_new_output(struct wl_listener *listener, void *data) {
|
||||||
output->mode.notify = handle_mode;
|
output->mode.notify = handle_mode;
|
||||||
wl_signal_add(&wlr_output->events.transform, &output->transform);
|
wl_signal_add(&wlr_output->events.transform, &output->transform);
|
||||||
output->transform.notify = handle_transform;
|
output->transform.notify = handle_transform;
|
||||||
|
wl_signal_add(&wlr_output->events.scale, &output->scale);
|
||||||
|
output->scale.notify = handle_scale;
|
||||||
|
|
||||||
wl_signal_add(&output->damage->events.frame, &output->damage_frame);
|
wl_signal_add(&output->damage->events.frame, &output->damage_frame);
|
||||||
output->damage_frame.notify = damage_handle_frame;
|
output->damage_frame.notify = damage_handle_frame;
|
||||||
|
|
|
@ -75,15 +75,13 @@ static void handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
wl_container_of(listener, sway_surface, destroy);
|
wl_container_of(listener, sway_surface, destroy);
|
||||||
wl_list_remove(&sway_surface->commit.link);
|
wl_list_remove(&sway_surface->commit.link);
|
||||||
wl_list_remove(&sway_surface->destroy.link);
|
wl_list_remove(&sway_surface->destroy.link);
|
||||||
struct sway_container *parent = container_view_destroy(sway_surface->view->swayc);
|
view_destroy(sway_surface->view);
|
||||||
free(sway_surface->view);
|
|
||||||
free(sway_surface);
|
free(sway_surface);
|
||||||
arrange_windows(parent, -1, -1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
|
void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
|
||||||
struct sway_server *server = wl_container_of(
|
struct sway_server *server = wl_container_of(listener, server,
|
||||||
listener, server, wl_shell_surface);
|
wl_shell_surface);
|
||||||
struct wlr_wl_shell_surface *shell_surface = data;
|
struct wlr_wl_shell_surface *shell_surface = data;
|
||||||
|
|
||||||
if (shell_surface->state == WLR_WL_SHELL_SURFACE_STATE_POPUP) {
|
if (shell_surface->state == WLR_WL_SHELL_SURFACE_STATE_POPUP) {
|
||||||
|
@ -103,20 +101,18 @@ void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct sway_view *sway_view = calloc(1, sizeof(struct sway_view));
|
struct sway_view *view = view_create(SWAY_WL_SHELL_VIEW);
|
||||||
if (!sway_assert(sway_view, "Failed to allocate view!")) {
|
if (!sway_assert(view, "Failed to allocate view")) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
sway_view->type = SWAY_WL_SHELL_VIEW;
|
view->iface.get_prop = get_prop;
|
||||||
sway_view->iface.get_prop = get_prop;
|
view->iface.set_size = set_size;
|
||||||
sway_view->iface.set_size = set_size;
|
view->iface.set_position = set_position;
|
||||||
sway_view->iface.set_position = set_position;
|
view->iface.set_activated = set_activated;
|
||||||
sway_view->iface.set_activated = set_activated;
|
view->iface.close = close;
|
||||||
sway_view->iface.close = close;
|
view->wlr_wl_shell_surface = shell_surface;
|
||||||
sway_view->wlr_wl_shell_surface = shell_surface;
|
view->sway_wl_shell_surface = sway_surface;
|
||||||
sway_view->sway_wl_shell_surface = sway_surface;
|
sway_surface->view = view;
|
||||||
sway_view->surface = shell_surface->surface;
|
|
||||||
sway_surface->view = sway_view;
|
|
||||||
|
|
||||||
// TODO:
|
// TODO:
|
||||||
// - Wire up listeners
|
// - Wire up listeners
|
||||||
|
@ -132,11 +128,5 @@ void handle_wl_shell_surface(struct wl_listener *listener, void *data) {
|
||||||
sway_surface->destroy.notify = handle_destroy;
|
sway_surface->destroy.notify = handle_destroy;
|
||||||
wl_signal_add(&shell_surface->events.destroy, &sway_surface->destroy);
|
wl_signal_add(&shell_surface->events.destroy, &sway_surface->destroy);
|
||||||
|
|
||||||
struct sway_seat *seat = input_manager_current_seat(input_manager);
|
view_map(view, shell_surface->surface);
|
||||||
struct sway_container *focus = sway_seat_get_focus_inactive(seat, &root_container);
|
|
||||||
struct sway_container *cont = container_view_create(focus, sway_view);
|
|
||||||
sway_view->swayc = cont;
|
|
||||||
|
|
||||||
arrange_windows(cont->parent, -1, -1);
|
|
||||||
sway_input_manager_set_focus(input_manager, cont);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -82,29 +82,14 @@ static void handle_commit(struct wl_listener *listener, void *data) {
|
||||||
static void handle_unmap(struct wl_listener *listener, void *data) {
|
static void handle_unmap(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xdg_surface_v6 *sway_surface =
|
struct sway_xdg_surface_v6 *sway_surface =
|
||||||
wl_container_of(listener, sway_surface, unmap);
|
wl_container_of(listener, sway_surface, unmap);
|
||||||
view_damage_whole(sway_surface->view);
|
view_unmap(sway_surface->view);
|
||||||
container_view_destroy(sway_surface->view->swayc);
|
|
||||||
sway_surface->view->swayc = NULL;
|
|
||||||
sway_surface->view->surface = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_map(struct wl_listener *listener, void *data) {
|
static void handle_map(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xdg_surface_v6 *sway_surface =
|
struct sway_xdg_surface_v6 *sway_surface =
|
||||||
wl_container_of(listener, sway_surface, map);
|
wl_container_of(listener, sway_surface, map);
|
||||||
struct sway_view *view = sway_surface->view;
|
struct sway_view *view = sway_surface->view;
|
||||||
|
view_map(view, view->wlr_xdg_surface_v6->surface);
|
||||||
sway_surface->view->surface = view->wlr_xdg_surface_v6->surface;
|
|
||||||
|
|
||||||
container_view_destroy(view->swayc);
|
|
||||||
|
|
||||||
struct sway_seat *seat = input_manager_current_seat(input_manager);
|
|
||||||
struct sway_container *focus = sway_seat_get_focus_inactive(seat, &root_container);
|
|
||||||
struct sway_container *cont = container_view_create(focus, view);
|
|
||||||
view->swayc = cont;
|
|
||||||
arrange_windows(cont->parent, -1, -1);
|
|
||||||
sway_input_manager_set_focus(input_manager, cont);
|
|
||||||
|
|
||||||
view_damage_whole(sway_surface->view);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_destroy(struct wl_listener *listener, void *data) {
|
static void handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
|
@ -112,8 +97,9 @@ static void handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
wl_container_of(listener, sway_xdg_surface, destroy);
|
wl_container_of(listener, sway_xdg_surface, destroy);
|
||||||
wl_list_remove(&sway_xdg_surface->commit.link);
|
wl_list_remove(&sway_xdg_surface->commit.link);
|
||||||
wl_list_remove(&sway_xdg_surface->destroy.link);
|
wl_list_remove(&sway_xdg_surface->destroy.link);
|
||||||
container_view_destroy(sway_xdg_surface->view->swayc);
|
wl_list_remove(&sway_xdg_surface->map.link);
|
||||||
free(sway_xdg_surface->view);
|
wl_list_remove(&sway_xdg_surface->unmap.link);
|
||||||
|
view_destroy(sway_xdg_surface->view);
|
||||||
free(sway_xdg_surface);
|
free(sway_xdg_surface);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -138,23 +124,21 @@ void handle_xdg_shell_v6_surface(struct wl_listener *listener, void *data) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct sway_view *sway_view = calloc(1, sizeof(struct sway_view));
|
struct sway_view *view = view_create(SWAY_XDG_SHELL_V6_VIEW);
|
||||||
if (!sway_assert(sway_view, "Failed to allocate view!")) {
|
if (!sway_assert(view, "Failed to allocate view")) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
sway_view->type = SWAY_XDG_SHELL_V6_VIEW;
|
view->iface.get_prop = get_prop;
|
||||||
sway_view->iface.get_prop = get_prop;
|
view->iface.set_size = set_size;
|
||||||
sway_view->iface.set_size = set_size;
|
view->iface.set_position = set_position;
|
||||||
sway_view->iface.set_position = set_position;
|
view->iface.set_activated = set_activated;
|
||||||
sway_view->iface.set_activated = set_activated;
|
view->iface.close = close;
|
||||||
sway_view->iface.close = close;
|
view->wlr_xdg_surface_v6 = xdg_surface;
|
||||||
sway_view->wlr_xdg_surface_v6 = xdg_surface;
|
view->sway_xdg_surface_v6 = sway_surface;
|
||||||
sway_view->sway_xdg_surface_v6 = sway_surface;
|
sway_surface->view = view;
|
||||||
sway_surface->view = sway_view;
|
|
||||||
|
|
||||||
// TODO:
|
// TODO:
|
||||||
// - Look up pid and open on appropriate workspace
|
// - Look up pid and open on appropriate workspace
|
||||||
// - Set new view to maximized so it behaves nicely
|
|
||||||
// - Criteria
|
// - Criteria
|
||||||
|
|
||||||
sway_surface->commit.notify = handle_commit;
|
sway_surface->commit.notify = handle_commit;
|
||||||
|
|
|
@ -102,56 +102,35 @@ static void handle_commit(struct wl_listener *listener, void *data) {
|
||||||
static void handle_destroy(struct wl_listener *listener, void *data) {
|
static void handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xwayland_surface *sway_surface =
|
struct sway_xwayland_surface *sway_surface =
|
||||||
wl_container_of(listener, sway_surface, destroy);
|
wl_container_of(listener, sway_surface, destroy);
|
||||||
|
|
||||||
wl_list_remove(&sway_surface->commit.link);
|
wl_list_remove(&sway_surface->commit.link);
|
||||||
wl_list_remove(&sway_surface->destroy.link);
|
wl_list_remove(&sway_surface->destroy.link);
|
||||||
wl_list_remove(&sway_surface->request_configure.link);
|
wl_list_remove(&sway_surface->request_configure.link);
|
||||||
wl_list_remove(&sway_surface->view->unmanaged_view_link);
|
wl_list_remove(&sway_surface->map.link);
|
||||||
container_view_destroy(sway_surface->view->swayc);
|
wl_list_remove(&sway_surface->unmap.link);
|
||||||
sway_surface->view->swayc = NULL;
|
view_destroy(sway_surface->view);
|
||||||
sway_surface->view->surface = NULL;
|
free(sway_surface);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_unmap(struct wl_listener *listener, void *data) {
|
static void handle_unmap(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xwayland_surface *sway_surface =
|
struct sway_xwayland_surface *sway_surface =
|
||||||
wl_container_of(listener, sway_surface, unmap);
|
wl_container_of(listener, sway_surface, unmap);
|
||||||
view_damage_whole(sway_surface->view);
|
view_unmap(sway_surface->view);
|
||||||
wl_list_remove(&sway_surface->view->unmanaged_view_link);
|
|
||||||
wl_list_init(&sway_surface->view->unmanaged_view_link);
|
|
||||||
container_view_destroy(sway_surface->view->swayc);
|
|
||||||
sway_surface->view->swayc = NULL;
|
|
||||||
sway_surface->view->surface = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_map(struct wl_listener *listener, void *data) {
|
static void handle_map(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xwayland_surface *sway_surface =
|
struct sway_xwayland_surface *sway_surface =
|
||||||
wl_container_of(listener, sway_surface, map);
|
wl_container_of(listener, sway_surface, map);
|
||||||
struct wlr_xwayland_surface *xsurface = data;
|
struct wlr_xwayland_surface *xsurface = data;
|
||||||
|
struct sway_view *view = sway_surface->view;
|
||||||
sway_surface->view->surface = xsurface->surface;
|
|
||||||
|
|
||||||
// put it back into the tree
|
// put it back into the tree
|
||||||
if (wlr_xwayland_surface_is_unmanaged(xsurface) ||
|
if (wlr_xwayland_surface_is_unmanaged(xsurface) ||
|
||||||
xsurface->override_redirect) {
|
xsurface->override_redirect) {
|
||||||
wl_list_remove(&sway_surface->view->unmanaged_view_link);
|
view_map_unmanaged(view, xsurface->surface);
|
||||||
wl_list_insert(&root_container.sway_root->unmanaged_views,
|
|
||||||
&sway_surface->view->unmanaged_view_link);
|
|
||||||
} else {
|
} else {
|
||||||
struct sway_view *view = sway_surface->view;
|
|
||||||
container_view_destroy(view->swayc);
|
|
||||||
|
|
||||||
wlr_xwayland_surface_set_maximized(xsurface, true);
|
wlr_xwayland_surface_set_maximized(xsurface, true);
|
||||||
|
view_map(view, xsurface->surface);
|
||||||
struct sway_seat *seat = input_manager_current_seat(input_manager);
|
|
||||||
struct sway_container *focus = sway_seat_get_focus_inactive(seat,
|
|
||||||
&root_container);
|
|
||||||
struct sway_container *cont = container_view_create(focus, view);
|
|
||||||
view->swayc = cont;
|
|
||||||
arrange_windows(cont->parent, -1, -1);
|
|
||||||
sway_input_manager_set_focus(input_manager, cont);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
view_damage_whole(sway_surface->view);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_request_configure(struct wl_listener *listener, void *data) {
|
static void handle_request_configure(struct wl_listener *listener, void *data) {
|
||||||
|
@ -180,25 +159,21 @@ void handle_xwayland_surface(struct wl_listener *listener, void *data) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct sway_view *sway_view = calloc(1, sizeof(struct sway_view));
|
struct sway_view *view = view_create(SWAY_XWAYLAND_VIEW);
|
||||||
if (!sway_assert(sway_view, "Failed to allocate view!")) {
|
if (!sway_assert(view, "Failed to allocate view")) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
sway_view->type = SWAY_XWAYLAND_VIEW;
|
view->iface.get_prop = get_prop;
|
||||||
sway_view->iface.get_prop = get_prop;
|
view->iface.set_size = set_size;
|
||||||
sway_view->iface.set_size = set_size;
|
view->iface.set_position = set_position;
|
||||||
sway_view->iface.set_position = set_position;
|
view->iface.set_activated = set_activated;
|
||||||
sway_view->iface.set_activated = set_activated;
|
view->iface.close = close_view;
|
||||||
sway_view->iface.close = close_view;
|
view->wlr_xwayland_surface = xsurface;
|
||||||
sway_view->wlr_xwayland_surface = xsurface;
|
view->sway_xwayland_surface = sway_surface;
|
||||||
sway_view->sway_xwayland_surface = sway_surface;
|
sway_surface->view = view;
|
||||||
sway_surface->view = sway_view;
|
|
||||||
|
|
||||||
wl_list_init(&sway_view->unmanaged_view_link);
|
|
||||||
|
|
||||||
// TODO:
|
// TODO:
|
||||||
// - Look up pid and open on appropriate workspace
|
// - Look up pid and open on appropriate workspace
|
||||||
// - Set new view to maximized so it behaves nicely
|
|
||||||
// - Criteria
|
// - Criteria
|
||||||
|
|
||||||
wl_signal_add(&xsurface->surface->events.commit, &sway_surface->commit);
|
wl_signal_add(&xsurface->surface->events.commit, &sway_surface->commit);
|
||||||
|
|
125
sway/tree/view.c
125
sway/tree/view.c
|
@ -1,3 +1,4 @@
|
||||||
|
#include <stdlib.h>
|
||||||
#include <wayland-server.h>
|
#include <wayland-server.h>
|
||||||
#include <wlr/types/wlr_output_layout.h>
|
#include <wlr/types/wlr_output_layout.h>
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
|
@ -6,6 +7,31 @@
|
||||||
#include "sway/tree/layout.h"
|
#include "sway/tree/layout.h"
|
||||||
#include "sway/tree/view.h"
|
#include "sway/tree/view.h"
|
||||||
|
|
||||||
|
struct sway_view *view_create(enum sway_view_type type) {
|
||||||
|
struct sway_view *view = calloc(1, sizeof(struct sway_view));
|
||||||
|
if (view == NULL) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
view->type = type;
|
||||||
|
wl_list_init(&view->unmanaged_view_link);
|
||||||
|
return view;
|
||||||
|
}
|
||||||
|
|
||||||
|
void view_destroy(struct sway_view *view) {
|
||||||
|
if (view == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (view->surface != NULL) {
|
||||||
|
view_unmap(view);
|
||||||
|
}
|
||||||
|
if (view->swayc != NULL) {
|
||||||
|
container_view_destroy(view->swayc);
|
||||||
|
}
|
||||||
|
|
||||||
|
free(view);
|
||||||
|
}
|
||||||
|
|
||||||
const char *view_get_title(struct sway_view *view) {
|
const char *view_get_title(struct sway_view *view) {
|
||||||
if (view->iface.get_prop) {
|
if (view->iface.get_prop) {
|
||||||
return view->iface.get_prop(view, VIEW_PROP_TITLE);
|
return view->iface.get_prop(view, VIEW_PROP_TITLE);
|
||||||
|
@ -34,6 +60,31 @@ const char *view_get_instance(struct sway_view *view) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void view_update_outputs(struct sway_view *view,
|
||||||
|
const struct wlr_box *before) {
|
||||||
|
struct wlr_output_layout *output_layout =
|
||||||
|
root_container.sway_root->output_layout;
|
||||||
|
struct wlr_box box = {
|
||||||
|
.x = view->swayc->x,
|
||||||
|
.y = view->swayc->y,
|
||||||
|
.width = view->width,
|
||||||
|
.height = view->height,
|
||||||
|
};
|
||||||
|
struct wlr_output_layout_output *layout_output;
|
||||||
|
wl_list_for_each(layout_output, &output_layout->outputs, link) {
|
||||||
|
bool intersected = before != NULL && wlr_output_layout_intersects(
|
||||||
|
output_layout, layout_output->output, before);
|
||||||
|
bool intersects = wlr_output_layout_intersects(output_layout,
|
||||||
|
layout_output->output, &box);
|
||||||
|
if (intersected && !intersects) {
|
||||||
|
wlr_surface_send_leave(view->surface, layout_output->output);
|
||||||
|
}
|
||||||
|
if (!intersected && intersects) {
|
||||||
|
wlr_surface_send_enter(view->surface, layout_output->output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void view_set_size(struct sway_view *view, int width, int height) {
|
void view_set_size(struct sway_view *view, int width, int height) {
|
||||||
if (view->iface.set_size) {
|
if (view->iface.set_size) {
|
||||||
struct wlr_box box = {
|
struct wlr_box box = {
|
||||||
|
@ -73,30 +124,6 @@ void view_close(struct sway_view *view) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void view_update_outputs(struct sway_view *view, const struct wlr_box *before) {
|
|
||||||
struct wlr_output_layout *output_layout =
|
|
||||||
root_container.sway_root->output_layout;
|
|
||||||
struct wlr_box box = {
|
|
||||||
.x = view->swayc->x,
|
|
||||||
.y = view->swayc->y,
|
|
||||||
.width = view->width,
|
|
||||||
.height = view->height,
|
|
||||||
};
|
|
||||||
struct wlr_output_layout_output *layout_output;
|
|
||||||
wl_list_for_each(layout_output, &output_layout->outputs, link) {
|
|
||||||
bool intersected = before != NULL && wlr_output_layout_intersects(
|
|
||||||
output_layout, layout_output->output, before);
|
|
||||||
bool intersects = wlr_output_layout_intersects(output_layout,
|
|
||||||
layout_output->output, &box);
|
|
||||||
if (intersected && !intersects) {
|
|
||||||
wlr_surface_send_leave(view->surface, layout_output->output);
|
|
||||||
}
|
|
||||||
if (!intersected && intersects) {
|
|
||||||
wlr_surface_send_enter(view->surface, layout_output->output);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct sway_container *container_view_destroy(struct sway_container *view) {
|
struct sway_container *container_view_destroy(struct sway_container *view) {
|
||||||
if (!view) {
|
if (!view) {
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -107,6 +134,56 @@ struct sway_container *container_view_destroy(struct sway_container *view) {
|
||||||
return parent;
|
return parent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
struct sway_container *focus = sway_seat_get_focus_inactive(seat,
|
||||||
|
&root_container);
|
||||||
|
struct sway_container *cont = container_view_create(focus, view);
|
||||||
|
|
||||||
|
view->surface = wlr_surface;
|
||||||
|
view->swayc = cont;
|
||||||
|
|
||||||
|
arrange_windows(cont->parent, -1, -1);
|
||||||
|
sway_input_manager_set_focus(input_manager, cont);
|
||||||
|
|
||||||
|
view_damage_whole(view);
|
||||||
|
}
|
||||||
|
|
||||||
|
void view_map_unmanaged(struct sway_view *view,
|
||||||
|
struct wlr_surface *wlr_surface) {
|
||||||
|
if (!sway_assert(view->surface == NULL, "cannot map mapped view")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
view->surface = wlr_surface;
|
||||||
|
view->swayc = NULL;
|
||||||
|
|
||||||
|
wl_list_insert(&root_container.sway_root->unmanaged_views,
|
||||||
|
&view->unmanaged_view_link);
|
||||||
|
|
||||||
|
view_damage_whole(view);
|
||||||
|
}
|
||||||
|
|
||||||
|
void view_unmap(struct sway_view *view) {
|
||||||
|
if (!sway_assert(view->surface != NULL, "cannot unmap unmapped view")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
view_damage_whole(view);
|
||||||
|
|
||||||
|
wl_list_remove(&view->unmanaged_view_link);
|
||||||
|
wl_list_init(&view->unmanaged_view_link);
|
||||||
|
|
||||||
|
container_view_destroy(view->swayc);
|
||||||
|
|
||||||
|
view->swayc = NULL;
|
||||||
|
view->surface = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
void view_damage_whole(struct sway_view *view) {
|
void view_damage_whole(struct sway_view *view) {
|
||||||
struct sway_container *cont = NULL;
|
struct sway_container *cont = NULL;
|
||||||
for (int i = 0; i < root_container.children->length; ++i) {
|
for (int i = 0; i < root_container.children->length; ++i) {
|
||||||
|
|
Loading…
Reference in a new issue