dce549c537
This moves the arrange_windows call into the arrange_layers function, where we know the output actually needs to be arranged. Additionally, we shouldn't set focus to the parent of an unknown container type, because the parent may be an output and this causes a crash because outputs can't have direct focus. Fixes #2543
1035 lines
30 KiB
C
1035 lines
30 KiB
C
#define _XOPEN_SOURCE 700
|
|
#define _POSIX_C_SOURCE 199309L
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#ifdef __linux__
|
|
#include <linux/input-event-codes.h>
|
|
#elif __FreeBSD__
|
|
#include <dev/evdev/input-event-codes.h>
|
|
#endif
|
|
#include <strings.h>
|
|
#include <time.h>
|
|
#include <wlr/types/wlr_cursor.h>
|
|
#include <wlr/types/wlr_output_layout.h>
|
|
#include <wlr/types/wlr_xcursor_manager.h>
|
|
#include "log.h"
|
|
#include "config.h"
|
|
#include "sway/debug.h"
|
|
#include "sway/desktop.h"
|
|
#include "sway/input/cursor.h"
|
|
#include "sway/input/input-manager.h"
|
|
#include "sway/input/keyboard.h"
|
|
#include "sway/input/seat.h"
|
|
#include "sway/ipc-server.h"
|
|
#include "sway/layers.h"
|
|
#include "sway/output.h"
|
|
#include "sway/tree/arrange.h"
|
|
#include "sway/tree/container.h"
|
|
#include "sway/tree/root.h"
|
|
#include "sway/tree/view.h"
|
|
#include "sway/tree/workspace.h"
|
|
|
|
static void seat_device_destroy(struct sway_seat_device *seat_device) {
|
|
if (!seat_device) {
|
|
return;
|
|
}
|
|
|
|
sway_keyboard_destroy(seat_device->keyboard);
|
|
wlr_cursor_detach_input_device(seat_device->sway_seat->cursor->cursor,
|
|
seat_device->input_device->wlr_device);
|
|
wl_list_remove(&seat_device->link);
|
|
free(seat_device);
|
|
}
|
|
|
|
void seat_destroy(struct sway_seat *seat) {
|
|
struct sway_seat_device *seat_device, *next;
|
|
wl_list_for_each_safe(seat_device, next, &seat->devices, link) {
|
|
seat_device_destroy(seat_device);
|
|
}
|
|
sway_cursor_destroy(seat->cursor);
|
|
wl_list_remove(&seat->new_container.link);
|
|
wl_list_remove(&seat->new_drag_icon.link);
|
|
wl_list_remove(&seat->link);
|
|
wlr_seat_destroy(seat->wlr_seat);
|
|
}
|
|
|
|
static struct sway_seat_container *seat_container_from_container(
|
|
struct sway_seat *seat, struct sway_container *con);
|
|
|
|
static void seat_container_destroy(struct sway_seat_container *seat_con) {
|
|
struct sway_container *con = seat_con->container;
|
|
struct sway_container *child = NULL;
|
|
|
|
if (con->children != NULL) {
|
|
for (int i = 0; i < con->children->length; ++i) {
|
|
child = con->children->items[i];
|
|
struct sway_seat_container *seat_child =
|
|
seat_container_from_container(seat_con->seat, child);
|
|
seat_container_destroy(seat_child);
|
|
}
|
|
}
|
|
|
|
wl_list_remove(&seat_con->destroy.link);
|
|
wl_list_remove(&seat_con->link);
|
|
free(seat_con);
|
|
}
|
|
|
|
/**
|
|
* Activate all views within this container recursively.
|
|
*/
|
|
static void seat_send_activate(struct sway_container *con,
|
|
struct sway_seat *seat) {
|
|
if (con->type == C_VIEW) {
|
|
if (!seat_is_input_allowed(seat, con->sway_view->surface)) {
|
|
wlr_log(WLR_DEBUG, "Refusing to set focus, input is inhibited");
|
|
return;
|
|
}
|
|
view_set_activated(con->sway_view, true);
|
|
} else {
|
|
for (int i = 0; i < con->children->length; ++i) {
|
|
struct sway_container *child = con->children->items[i];
|
|
seat_send_activate(child, seat);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* If con is a view, set it as active and enable keyboard input.
|
|
* If con is a container, set all child views as active and don't enable
|
|
* keyboard input on any.
|
|
*/
|
|
static void seat_send_focus(struct sway_container *con,
|
|
struct sway_seat *seat) {
|
|
seat_send_activate(con, seat);
|
|
|
|
if (con->type == C_VIEW
|
|
&& seat_is_input_allowed(seat, con->sway_view->surface)) {
|
|
#ifdef HAVE_XWAYLAND
|
|
if (con->sway_view->type == SWAY_VIEW_XWAYLAND) {
|
|
struct wlr_xwayland *xwayland =
|
|
seat->input->server->xwayland.wlr_xwayland;
|
|
wlr_xwayland_set_seat(xwayland, seat->wlr_seat);
|
|
}
|
|
#endif
|
|
struct wlr_keyboard *keyboard = wlr_seat_get_keyboard(seat->wlr_seat);
|
|
if (keyboard) {
|
|
wlr_seat_keyboard_notify_enter(seat->wlr_seat,
|
|
con->sway_view->surface, keyboard->keycodes,
|
|
keyboard->num_keycodes, &keyboard->modifiers);
|
|
} else {
|
|
wlr_seat_keyboard_notify_enter(
|
|
seat->wlr_seat, con->sway_view->surface, NULL, 0, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
void seat_focus_inactive_children_for_each(struct sway_seat *seat,
|
|
struct sway_container *container,
|
|
void (*f)(struct sway_container *container, void *data), void *data) {
|
|
struct sway_seat_container *current = NULL;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
if (current->container->parent == NULL) {
|
|
continue;
|
|
}
|
|
if (current->container->parent == container) {
|
|
f(current->container, data);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct sway_container *seat_get_focus_inactive_view(struct sway_seat *seat,
|
|
struct sway_container *ancestor) {
|
|
if (ancestor->type == C_VIEW) {
|
|
return ancestor;
|
|
}
|
|
struct sway_seat_container *current;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
struct sway_container *con = current->container;
|
|
if (con->type == C_VIEW && container_has_ancestor(con, ancestor)) {
|
|
return con;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void handle_seat_container_destroy(struct wl_listener *listener,
|
|
void *data) {
|
|
struct sway_seat_container *seat_con =
|
|
wl_container_of(listener, seat_con, destroy);
|
|
struct sway_seat *seat = seat_con->seat;
|
|
struct sway_container *con = seat_con->container;
|
|
struct sway_container *parent = con->parent;
|
|
struct sway_container *focus = seat_get_focus(seat);
|
|
|
|
bool set_focus =
|
|
focus != NULL &&
|
|
(focus == con || container_has_ancestor(focus, con)) &&
|
|
con->type != C_WORKSPACE;
|
|
|
|
seat_container_destroy(seat_con);
|
|
|
|
if (set_focus) {
|
|
struct sway_container *next_focus = NULL;
|
|
while (next_focus == NULL) {
|
|
next_focus = seat_get_focus_inactive_view(seat, parent);
|
|
|
|
if (next_focus == NULL && parent->type == C_WORKSPACE) {
|
|
next_focus = parent;
|
|
break;
|
|
}
|
|
|
|
parent = parent->parent;
|
|
}
|
|
|
|
// the structure change might have caused it to move up to the top of
|
|
// the focus stack without sending focus notifications to the view
|
|
if (seat_get_focus(seat) == next_focus) {
|
|
seat_send_focus(next_focus, seat);
|
|
} else {
|
|
seat_set_focus(seat, next_focus);
|
|
}
|
|
}
|
|
}
|
|
|
|
static struct sway_seat_container *seat_container_from_container(
|
|
struct sway_seat *seat, struct sway_container *con) {
|
|
if (con->type == C_ROOT || con->type == C_OUTPUT) {
|
|
// these don't get seat containers ever
|
|
return NULL;
|
|
}
|
|
|
|
struct sway_seat_container *seat_con = NULL;
|
|
wl_list_for_each(seat_con, &seat->focus_stack, link) {
|
|
if (seat_con->container == con) {
|
|
return seat_con;
|
|
}
|
|
}
|
|
|
|
seat_con = calloc(1, sizeof(struct sway_seat_container));
|
|
if (seat_con == NULL) {
|
|
wlr_log(WLR_ERROR, "could not allocate seat container");
|
|
return NULL;
|
|
}
|
|
|
|
seat_con->container = con;
|
|
seat_con->seat = seat;
|
|
wl_list_insert(seat->focus_stack.prev, &seat_con->link);
|
|
wl_signal_add(&con->events.destroy, &seat_con->destroy);
|
|
seat_con->destroy.notify = handle_seat_container_destroy;
|
|
|
|
return seat_con;
|
|
}
|
|
|
|
static void handle_new_container(struct wl_listener *listener, void *data) {
|
|
struct sway_seat *seat = wl_container_of(listener, seat, new_container);
|
|
struct sway_container *con = data;
|
|
seat_container_from_container(seat, con);
|
|
}
|
|
|
|
static void drag_icon_damage_whole(struct sway_drag_icon *icon) {
|
|
if (!icon->wlr_drag_icon->mapped) {
|
|
return;
|
|
}
|
|
desktop_damage_surface(icon->wlr_drag_icon->surface, icon->x, icon->y, true);
|
|
}
|
|
|
|
void drag_icon_update_position(struct sway_drag_icon *icon) {
|
|
drag_icon_damage_whole(icon);
|
|
|
|
struct wlr_drag_icon *wlr_icon = icon->wlr_drag_icon;
|
|
struct sway_seat *seat = icon->seat;
|
|
struct wlr_cursor *cursor = seat->cursor->cursor;
|
|
if (wlr_icon->is_pointer) {
|
|
icon->x = cursor->x + wlr_icon->sx;
|
|
icon->y = cursor->y + wlr_icon->sy;
|
|
} else {
|
|
struct wlr_touch_point *point =
|
|
wlr_seat_touch_get_point(seat->wlr_seat, wlr_icon->touch_id);
|
|
if (point == NULL) {
|
|
return;
|
|
}
|
|
icon->x = seat->touch_x + wlr_icon->sx;
|
|
icon->y = seat->touch_y + wlr_icon->sy;
|
|
}
|
|
|
|
drag_icon_damage_whole(icon);
|
|
}
|
|
|
|
static void drag_icon_handle_surface_commit(struct wl_listener *listener,
|
|
void *data) {
|
|
struct sway_drag_icon *icon =
|
|
wl_container_of(listener, icon, surface_commit);
|
|
drag_icon_update_position(icon);
|
|
}
|
|
|
|
static void drag_icon_handle_map(struct wl_listener *listener, void *data) {
|
|
struct sway_drag_icon *icon = wl_container_of(listener, icon, map);
|
|
drag_icon_damage_whole(icon);
|
|
}
|
|
|
|
static void drag_icon_handle_unmap(struct wl_listener *listener, void *data) {
|
|
struct sway_drag_icon *icon = wl_container_of(listener, icon, unmap);
|
|
drag_icon_damage_whole(icon);
|
|
}
|
|
|
|
static void drag_icon_handle_destroy(struct wl_listener *listener,
|
|
void *data) {
|
|
struct sway_drag_icon *icon = wl_container_of(listener, icon, destroy);
|
|
icon->wlr_drag_icon->data = NULL;
|
|
wl_list_remove(&icon->link);
|
|
wl_list_remove(&icon->surface_commit.link);
|
|
wl_list_remove(&icon->unmap.link);
|
|
wl_list_remove(&icon->destroy.link);
|
|
free(icon);
|
|
}
|
|
|
|
static void handle_new_drag_icon(struct wl_listener *listener, void *data) {
|
|
struct sway_seat *seat = wl_container_of(listener, seat, new_drag_icon);
|
|
struct wlr_drag_icon *wlr_drag_icon = data;
|
|
|
|
struct sway_drag_icon *icon = calloc(1, sizeof(struct sway_drag_icon));
|
|
if (icon == NULL) {
|
|
wlr_log(WLR_ERROR, "Allocation failed");
|
|
return;
|
|
}
|
|
icon->seat = seat;
|
|
icon->wlr_drag_icon = wlr_drag_icon;
|
|
wlr_drag_icon->data = icon;
|
|
|
|
icon->surface_commit.notify = drag_icon_handle_surface_commit;
|
|
wl_signal_add(&wlr_drag_icon->surface->events.commit, &icon->surface_commit);
|
|
icon->unmap.notify = drag_icon_handle_unmap;
|
|
wl_signal_add(&wlr_drag_icon->events.unmap, &icon->unmap);
|
|
icon->map.notify = drag_icon_handle_map;
|
|
wl_signal_add(&wlr_drag_icon->events.map, &icon->map);
|
|
icon->destroy.notify = drag_icon_handle_destroy;
|
|
wl_signal_add(&wlr_drag_icon->events.destroy, &icon->destroy);
|
|
|
|
wl_list_insert(&root_container.sway_root->drag_icons, &icon->link);
|
|
|
|
drag_icon_update_position(icon);
|
|
}
|
|
|
|
static void collect_focus_iter(struct sway_container *con, void *data) {
|
|
struct sway_seat *seat = data;
|
|
struct sway_seat_container *seat_con =
|
|
seat_container_from_container(seat, con);
|
|
if (!seat_con) {
|
|
return;
|
|
}
|
|
wl_list_remove(&seat_con->link);
|
|
wl_list_insert(&seat->focus_stack, &seat_con->link);
|
|
}
|
|
|
|
struct sway_seat *seat_create(struct sway_input_manager *input,
|
|
const char *seat_name) {
|
|
struct sway_seat *seat = calloc(1, sizeof(struct sway_seat));
|
|
if (!seat) {
|
|
return NULL;
|
|
}
|
|
|
|
seat->wlr_seat = wlr_seat_create(input->server->wl_display, seat_name);
|
|
if (!sway_assert(seat->wlr_seat, "could not allocate seat")) {
|
|
free(seat);
|
|
return NULL;
|
|
}
|
|
seat->wlr_seat->data = seat;
|
|
|
|
seat->cursor = sway_cursor_create(seat);
|
|
if (!seat->cursor) {
|
|
wlr_seat_destroy(seat->wlr_seat);
|
|
free(seat);
|
|
return NULL;
|
|
}
|
|
|
|
// init the focus stack
|
|
wl_list_init(&seat->focus_stack);
|
|
|
|
root_for_each_workspace(collect_focus_iter, seat);
|
|
root_for_each_container(collect_focus_iter, seat);
|
|
|
|
wl_signal_add(&root_container.sway_root->events.new_container,
|
|
&seat->new_container);
|
|
seat->new_container.notify = handle_new_container;
|
|
|
|
wl_signal_add(&seat->wlr_seat->events.new_drag_icon, &seat->new_drag_icon);
|
|
seat->new_drag_icon.notify = handle_new_drag_icon;
|
|
|
|
seat->input = input;
|
|
wl_list_init(&seat->devices);
|
|
|
|
wlr_seat_set_capabilities(seat->wlr_seat,
|
|
WL_SEAT_CAPABILITY_KEYBOARD |
|
|
WL_SEAT_CAPABILITY_POINTER |
|
|
WL_SEAT_CAPABILITY_TOUCH);
|
|
|
|
|
|
wl_list_insert(&input->seats, &seat->link);
|
|
|
|
return seat;
|
|
}
|
|
|
|
static void seat_apply_input_config(struct sway_seat *seat,
|
|
struct sway_seat_device *sway_device) {
|
|
const char *mapped_to_output = NULL;
|
|
|
|
struct input_config *ic = input_device_get_config(
|
|
sway_device->input_device);
|
|
if (ic != NULL) {
|
|
wlr_log(WLR_DEBUG, "Applying input config to %s",
|
|
sway_device->input_device->identifier);
|
|
|
|
mapped_to_output = ic->mapped_to_output;
|
|
}
|
|
|
|
if (mapped_to_output == NULL) {
|
|
mapped_to_output = sway_device->input_device->wlr_device->output_name;
|
|
}
|
|
if (mapped_to_output != NULL) {
|
|
wlr_log(WLR_DEBUG, "Mapping input device %s to output %s",
|
|
sway_device->input_device->identifier, mapped_to_output);
|
|
struct sway_container *output = NULL;
|
|
for (int i = 0; i < root_container.children->length; ++i) {
|
|
struct sway_container *_output = root_container.children->items[i];
|
|
if (strcasecmp(_output->name, mapped_to_output) == 0) {
|
|
output = _output;
|
|
break;
|
|
}
|
|
}
|
|
if (output) {
|
|
wlr_cursor_map_input_to_output(seat->cursor->cursor,
|
|
sway_device->input_device->wlr_device,
|
|
output->sway_output->wlr_output);
|
|
wlr_log(WLR_DEBUG, "Mapped to output %s", output->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void seat_configure_pointer(struct sway_seat *seat,
|
|
struct sway_seat_device *sway_device) {
|
|
seat_configure_xcursor(seat);
|
|
wlr_cursor_attach_input_device(seat->cursor->cursor,
|
|
sway_device->input_device->wlr_device);
|
|
seat_apply_input_config(seat, sway_device);
|
|
}
|
|
|
|
static void seat_configure_keyboard(struct sway_seat *seat,
|
|
struct sway_seat_device *seat_device) {
|
|
if (!seat_device->keyboard) {
|
|
sway_keyboard_create(seat, seat_device);
|
|
}
|
|
struct wlr_keyboard *wlr_keyboard =
|
|
seat_device->input_device->wlr_device->keyboard;
|
|
sway_keyboard_configure(seat_device->keyboard);
|
|
wlr_seat_set_keyboard(seat->wlr_seat,
|
|
seat_device->input_device->wlr_device);
|
|
struct sway_container *focus = seat_get_focus(seat);
|
|
if (focus && focus->type == C_VIEW) {
|
|
// force notify reenter to pick up the new configuration
|
|
wlr_seat_keyboard_clear_focus(seat->wlr_seat);
|
|
wlr_seat_keyboard_notify_enter(seat->wlr_seat,
|
|
focus->sway_view->surface, wlr_keyboard->keycodes,
|
|
wlr_keyboard->num_keycodes, &wlr_keyboard->modifiers);
|
|
}
|
|
}
|
|
|
|
static void seat_configure_touch(struct sway_seat *seat,
|
|
struct sway_seat_device *sway_device) {
|
|
wlr_cursor_attach_input_device(seat->cursor->cursor,
|
|
sway_device->input_device->wlr_device);
|
|
seat_apply_input_config(seat, sway_device);
|
|
}
|
|
|
|
static void seat_configure_tablet_tool(struct sway_seat *seat,
|
|
struct sway_seat_device *sway_device) {
|
|
wlr_cursor_attach_input_device(seat->cursor->cursor,
|
|
sway_device->input_device->wlr_device);
|
|
seat_apply_input_config(seat, sway_device);
|
|
}
|
|
|
|
static struct sway_seat_device *seat_get_device(struct sway_seat *seat,
|
|
struct sway_input_device *input_device) {
|
|
struct sway_seat_device *seat_device = NULL;
|
|
wl_list_for_each(seat_device, &seat->devices, link) {
|
|
if (seat_device->input_device == input_device) {
|
|
return seat_device;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void seat_configure_device(struct sway_seat *seat,
|
|
struct sway_input_device *input_device) {
|
|
struct sway_seat_device *seat_device =
|
|
seat_get_device(seat, input_device);
|
|
if (!seat_device) {
|
|
return;
|
|
}
|
|
|
|
switch (input_device->wlr_device->type) {
|
|
case WLR_INPUT_DEVICE_POINTER:
|
|
seat_configure_pointer(seat, seat_device);
|
|
break;
|
|
case WLR_INPUT_DEVICE_KEYBOARD:
|
|
seat_configure_keyboard(seat, seat_device);
|
|
break;
|
|
case WLR_INPUT_DEVICE_TOUCH:
|
|
seat_configure_touch(seat, seat_device);
|
|
break;
|
|
case WLR_INPUT_DEVICE_TABLET_TOOL:
|
|
seat_configure_tablet_tool(seat, seat_device);
|
|
break;
|
|
case WLR_INPUT_DEVICE_TABLET_PAD:
|
|
wlr_log(WLR_DEBUG, "TODO: configure tablet pad");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void seat_add_device(struct sway_seat *seat,
|
|
struct sway_input_device *input_device) {
|
|
if (seat_get_device(seat, input_device)) {
|
|
seat_configure_device(seat, input_device);
|
|
return;
|
|
}
|
|
|
|
struct sway_seat_device *seat_device =
|
|
calloc(1, sizeof(struct sway_seat_device));
|
|
if (!seat_device) {
|
|
wlr_log(WLR_DEBUG, "could not allocate seat device");
|
|
return;
|
|
}
|
|
|
|
wlr_log(WLR_DEBUG, "adding device %s to seat %s",
|
|
input_device->identifier, seat->wlr_seat->name);
|
|
|
|
seat_device->sway_seat = seat;
|
|
seat_device->input_device = input_device;
|
|
wl_list_insert(&seat->devices, &seat_device->link);
|
|
|
|
seat_configure_device(seat, input_device);
|
|
}
|
|
|
|
void seat_remove_device(struct sway_seat *seat,
|
|
struct sway_input_device *input_device) {
|
|
struct sway_seat_device *seat_device =
|
|
seat_get_device(seat, input_device);
|
|
|
|
if (!seat_device) {
|
|
return;
|
|
}
|
|
|
|
wlr_log(WLR_DEBUG, "removing device %s from seat %s",
|
|
input_device->identifier, seat->wlr_seat->name);
|
|
|
|
seat_device_destroy(seat_device);
|
|
}
|
|
|
|
void seat_configure_xcursor(struct sway_seat *seat) {
|
|
// TODO configure theme and size
|
|
const char *cursor_theme = NULL;
|
|
|
|
if (!seat->cursor->xcursor_manager) {
|
|
seat->cursor->xcursor_manager =
|
|
wlr_xcursor_manager_create(cursor_theme, 24);
|
|
if (sway_assert(seat->cursor->xcursor_manager,
|
|
"Cannot create XCursor manager for theme %s",
|
|
cursor_theme)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < root_container.children->length; ++i) {
|
|
struct sway_container *output_container =
|
|
root_container.children->items[i];
|
|
struct wlr_output *output =
|
|
output_container->sway_output->wlr_output;
|
|
bool result =
|
|
wlr_xcursor_manager_load(seat->cursor->xcursor_manager,
|
|
output->scale);
|
|
|
|
sway_assert(!result,
|
|
"Cannot load xcursor theme for output '%s' with scale %f",
|
|
// TODO: Fractional scaling
|
|
output->name, (double)output->scale);
|
|
}
|
|
|
|
wlr_xcursor_manager_set_cursor_image(seat->cursor->xcursor_manager,
|
|
"left_ptr", seat->cursor->cursor);
|
|
wlr_cursor_warp(seat->cursor->cursor, NULL, seat->cursor->cursor->x,
|
|
seat->cursor->cursor->y);
|
|
}
|
|
|
|
bool seat_is_input_allowed(struct sway_seat *seat,
|
|
struct wlr_surface *surface) {
|
|
struct wl_client *client = wl_resource_get_client(surface->resource);
|
|
return !seat->exclusive_client || seat->exclusive_client == client;
|
|
}
|
|
|
|
// Unfocus the container and any children (eg. when leaving `focus parent`)
|
|
static void seat_send_unfocus(struct sway_container *container,
|
|
struct sway_seat *seat) {
|
|
if (container->type == C_VIEW) {
|
|
wlr_seat_keyboard_clear_focus(seat->wlr_seat);
|
|
view_set_activated(container->sway_view, false);
|
|
} else {
|
|
for (int i = 0; i < container->children->length; ++i) {
|
|
struct sway_container *child = container->children->items[i];
|
|
seat_send_unfocus(child, seat);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int handle_urgent_timeout(void *data) {
|
|
struct sway_view *view = data;
|
|
view_set_urgent(view, false);
|
|
return 0;
|
|
}
|
|
|
|
void seat_set_focus_warp(struct sway_seat *seat,
|
|
struct sway_container *container, bool warp, bool notify) {
|
|
if (seat->focused_layer) {
|
|
return;
|
|
}
|
|
|
|
struct sway_container *last_focus = seat_get_focus(seat);
|
|
if (last_focus == container) {
|
|
return;
|
|
}
|
|
|
|
struct sway_container *last_workspace = last_focus;
|
|
if (last_workspace && last_workspace->type != C_WORKSPACE) {
|
|
last_workspace = container_parent(last_workspace, C_WORKSPACE);
|
|
}
|
|
|
|
if (container == NULL) {
|
|
// Close any popups on the old focus
|
|
if (last_focus->type == C_VIEW) {
|
|
view_close_popups(last_focus->sway_view);
|
|
}
|
|
seat_send_unfocus(last_focus, seat);
|
|
seat->has_focus = false;
|
|
update_debug_tree();
|
|
return;
|
|
}
|
|
|
|
struct sway_container *new_workspace = container;
|
|
if (new_workspace->type != C_WORKSPACE) {
|
|
new_workspace = container_parent(new_workspace, C_WORKSPACE);
|
|
}
|
|
|
|
if (last_workspace == new_workspace
|
|
&& last_workspace->sway_workspace->fullscreen
|
|
&& !container_is_fullscreen_or_child(container)) {
|
|
return;
|
|
}
|
|
|
|
struct sway_container *last_output = last_focus;
|
|
if (last_output && last_output->type != C_OUTPUT) {
|
|
last_output = container_parent(last_output, C_OUTPUT);
|
|
}
|
|
struct sway_container *new_output = container;
|
|
if (new_output->type != C_OUTPUT) {
|
|
new_output = container_parent(new_output, C_OUTPUT);
|
|
}
|
|
|
|
// find new output's old workspace, which might have to be removed if empty
|
|
struct sway_container *new_output_last_ws = NULL;
|
|
if (last_output != new_output) {
|
|
new_output_last_ws = seat_get_active_child(seat, new_output);
|
|
}
|
|
|
|
if (container->parent) {
|
|
struct sway_seat_container *seat_con =
|
|
seat_container_from_container(seat, container);
|
|
if (seat_con == NULL) {
|
|
return;
|
|
}
|
|
|
|
// put all the ancestors of this container on top of the focus stack
|
|
struct sway_seat_container *parent =
|
|
seat_container_from_container(seat, container->parent);
|
|
while (parent) {
|
|
wl_list_remove(&parent->link);
|
|
wl_list_insert(&seat->focus_stack, &parent->link);
|
|
container_set_dirty(parent->container);
|
|
|
|
parent = seat_container_from_container(seat,
|
|
parent->container->parent);
|
|
}
|
|
|
|
wl_list_remove(&seat_con->link);
|
|
wl_list_insert(&seat->focus_stack, &seat_con->link);
|
|
|
|
if (last_focus) {
|
|
seat_send_unfocus(last_focus, seat);
|
|
container_set_dirty(last_focus);
|
|
}
|
|
seat_send_focus(container, seat);
|
|
|
|
container_set_dirty(container);
|
|
container_set_dirty(container->parent); // for focused_inactive_child
|
|
}
|
|
|
|
// emit ipc events
|
|
if (notify && new_workspace && last_workspace != new_workspace) {
|
|
ipc_event_workspace(last_workspace, new_workspace, "focus");
|
|
}
|
|
if (container->type == C_VIEW) {
|
|
ipc_event_window(container, "focus");
|
|
}
|
|
|
|
if (new_output_last_ws) {
|
|
workspace_consider_destroy(new_output_last_ws);
|
|
}
|
|
|
|
// Close any popups on the old focus
|
|
if (last_focus && last_focus->type == C_VIEW) {
|
|
view_close_popups(last_focus->sway_view);
|
|
}
|
|
|
|
// If urgent, either unset the urgency or start a timer to unset it
|
|
if (container->type == C_VIEW && view_is_urgent(container->sway_view) &&
|
|
!container->sway_view->urgent_timer) {
|
|
struct sway_view *view = container->sway_view;
|
|
if (last_workspace && last_workspace != new_workspace &&
|
|
config->urgent_timeout > 0) {
|
|
view->urgent_timer = wl_event_loop_add_timer(server.wl_event_loop,
|
|
handle_urgent_timeout, view);
|
|
if (view->urgent_timer) {
|
|
wl_event_source_timer_update(view->urgent_timer,
|
|
config->urgent_timeout);
|
|
} else {
|
|
wlr_log(WLR_ERROR, "Unable to create urgency timer (%s)",
|
|
strerror(errno));
|
|
handle_urgent_timeout(view);
|
|
}
|
|
} else {
|
|
view_set_urgent(view, false);
|
|
}
|
|
}
|
|
|
|
// If we've focused a floating container, bring it to the front.
|
|
// We do this by putting it at the end of the floating list.
|
|
struct sway_container *floater = container;
|
|
while (floater->parent && floater->parent->type != C_WORKSPACE) {
|
|
floater = floater->parent;
|
|
}
|
|
if (container_is_floating(floater)) {
|
|
list_move_to_end(floater->parent->sway_workspace->floating, floater);
|
|
}
|
|
|
|
if (last_focus) {
|
|
if (last_workspace) {
|
|
workspace_consider_destroy(last_workspace);
|
|
}
|
|
|
|
if (config->mouse_warping && warp && new_output != last_output) {
|
|
double x = container->x + container->width / 2.0;
|
|
double y = container->y + container->height / 2.0;
|
|
struct wlr_output *wlr_output =
|
|
new_output->sway_output->wlr_output;
|
|
if (!wlr_output_layout_contains_point(
|
|
root_container.sway_root->output_layout,
|
|
wlr_output, seat->cursor->cursor->x,
|
|
seat->cursor->cursor->y)) {
|
|
wlr_cursor_warp(seat->cursor->cursor, NULL, x, y);
|
|
cursor_send_pointer_motion(seat->cursor, 0, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
seat->has_focus = true;
|
|
|
|
update_debug_tree();
|
|
}
|
|
|
|
void seat_set_focus(struct sway_seat *seat,
|
|
struct sway_container *container) {
|
|
seat_set_focus_warp(seat, container, true, true);
|
|
}
|
|
|
|
void seat_set_focus_surface(struct sway_seat *seat,
|
|
struct wlr_surface *surface, bool unfocus) {
|
|
if (seat->focused_layer != NULL) {
|
|
return;
|
|
}
|
|
if (seat->has_focus && unfocus) {
|
|
struct sway_container *focus = seat_get_focus(seat);
|
|
seat_send_unfocus(focus, seat);
|
|
seat->has_focus = false;
|
|
}
|
|
struct wlr_keyboard *keyboard =
|
|
wlr_seat_get_keyboard(seat->wlr_seat);
|
|
if (keyboard) {
|
|
wlr_seat_keyboard_notify_enter(seat->wlr_seat, surface,
|
|
keyboard->keycodes, keyboard->num_keycodes, &keyboard->modifiers);
|
|
} else {
|
|
wlr_seat_keyboard_notify_enter(seat->wlr_seat, surface, NULL, 0, NULL);
|
|
}
|
|
}
|
|
|
|
void seat_set_focus_layer(struct sway_seat *seat,
|
|
struct wlr_layer_surface *layer) {
|
|
if (!layer && seat->focused_layer) {
|
|
seat->focused_layer = NULL;
|
|
struct sway_container *previous =
|
|
seat_get_focus_inactive(seat, &root_container);
|
|
if (previous) {
|
|
wlr_log(WLR_DEBUG, "Returning focus to %p %s '%s'", previous,
|
|
container_type_to_str(previous->type), previous->name);
|
|
// Hack to get seat to re-focus the return value of get_focus
|
|
seat_set_focus(seat, NULL);
|
|
seat_set_focus(seat, previous);
|
|
}
|
|
return;
|
|
} else if (!layer || seat->focused_layer == layer) {
|
|
return;
|
|
}
|
|
seat_set_focus_surface(seat, layer->surface, true);
|
|
if (layer->layer >= ZWLR_LAYER_SHELL_V1_LAYER_TOP) {
|
|
seat->focused_layer = layer;
|
|
}
|
|
}
|
|
|
|
void seat_set_exclusive_client(struct sway_seat *seat,
|
|
struct wl_client *client) {
|
|
if (!client) {
|
|
seat->exclusive_client = client;
|
|
// Triggers a refocus of the topmost surface layer if necessary
|
|
// TODO: Make layer surface focus per-output based on cursor position
|
|
for (int i = 0; i < root_container.children->length; ++i) {
|
|
struct sway_container *output = root_container.children->items[i];
|
|
if (!sway_assert(output->type == C_OUTPUT,
|
|
"root container has non-output child")) {
|
|
continue;
|
|
}
|
|
arrange_layers(output->sway_output);
|
|
}
|
|
return;
|
|
}
|
|
if (seat->focused_layer) {
|
|
if (wl_resource_get_client(seat->focused_layer->resource) != client) {
|
|
seat_set_focus_layer(seat, NULL);
|
|
}
|
|
}
|
|
if (seat->has_focus) {
|
|
struct sway_container *focus = seat_get_focus(seat);
|
|
if (focus->type == C_VIEW && wl_resource_get_client(
|
|
focus->sway_view->surface->resource) != client) {
|
|
seat_set_focus(seat, NULL);
|
|
}
|
|
}
|
|
if (seat->wlr_seat->pointer_state.focused_client) {
|
|
if (seat->wlr_seat->pointer_state.focused_client->client != client) {
|
|
wlr_seat_pointer_clear_focus(seat->wlr_seat);
|
|
}
|
|
}
|
|
struct timespec now;
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
struct wlr_touch_point *point;
|
|
wl_list_for_each(point, &seat->wlr_seat->touch_state.touch_points, link) {
|
|
if (point->client->client != client) {
|
|
wlr_seat_touch_point_clear_focus(seat->wlr_seat,
|
|
now.tv_nsec / 1000, point->touch_id);
|
|
}
|
|
}
|
|
seat->exclusive_client = client;
|
|
}
|
|
|
|
struct sway_container *seat_get_focus_inactive(struct sway_seat *seat,
|
|
struct sway_container *con) {
|
|
if (con->type == C_WORKSPACE && !con->children->length &&
|
|
!con->sway_workspace->floating->length) {
|
|
return con;
|
|
}
|
|
if (con->type == C_VIEW) {
|
|
return con;
|
|
}
|
|
struct sway_seat_container *current;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
if (container_has_ancestor(current->container, con)) {
|
|
return current->container;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct sway_container *seat_get_focus_inactive_tiling(struct sway_seat *seat,
|
|
struct sway_container *ancestor) {
|
|
if (ancestor->type == C_WORKSPACE && !ancestor->children->length) {
|
|
return ancestor;
|
|
}
|
|
struct sway_seat_container *current;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
struct sway_container *con = current->container;
|
|
if (!container_is_floating_or_child(con) &&
|
|
container_has_ancestor(current->container, ancestor)) {
|
|
return con;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct sway_container *seat_get_focus_inactive_floating(struct sway_seat *seat,
|
|
struct sway_container *ancestor) {
|
|
if (ancestor->type == C_WORKSPACE &&
|
|
!ancestor->sway_workspace->floating->length) {
|
|
return NULL;
|
|
}
|
|
struct sway_seat_container *current;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
struct sway_container *con = current->container;
|
|
if (container_is_floating_or_child(con) &&
|
|
container_has_ancestor(current->container, ancestor)) {
|
|
return con;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct sway_container *seat_get_active_child(struct sway_seat *seat,
|
|
struct sway_container *parent) {
|
|
if (parent->type == C_VIEW) {
|
|
return parent;
|
|
}
|
|
struct sway_seat_container *current;
|
|
wl_list_for_each(current, &seat->focus_stack, link) {
|
|
struct sway_container *con = current->container;
|
|
if (con->parent == parent) {
|
|
return con;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct sway_container *seat_get_focus(struct sway_seat *seat) {
|
|
if (!seat->has_focus) {
|
|
return NULL;
|
|
}
|
|
struct sway_seat_container *current =
|
|
wl_container_of(seat->focus_stack.next, current, link);
|
|
return current->container;
|
|
}
|
|
|
|
void seat_apply_config(struct sway_seat *seat,
|
|
struct seat_config *seat_config) {
|
|
struct sway_seat_device *seat_device = NULL;
|
|
|
|
if (!seat_config) {
|
|
return;
|
|
}
|
|
|
|
wl_list_for_each(seat_device, &seat->devices, link) {
|
|
seat_configure_device(seat, seat_device->input_device);
|
|
}
|
|
}
|
|
|
|
struct seat_config *seat_get_config(struct sway_seat *seat) {
|
|
struct seat_config *seat_config = NULL;
|
|
for (int i = 0; i < config->seat_configs->length; ++i ) {
|
|
seat_config = config->seat_configs->items[i];
|
|
if (strcmp(seat->wlr_seat->name, seat_config->name) == 0) {
|
|
return seat_config;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void seat_begin_down(struct sway_seat *seat, struct sway_container *con,
|
|
uint32_t button, double sx, double sy) {
|
|
seat->operation = OP_DOWN;
|
|
seat->op_container = con;
|
|
seat->op_button = button;
|
|
seat->op_ref_lx = seat->cursor->cursor->x;
|
|
seat->op_ref_ly = seat->cursor->cursor->y;
|
|
seat->op_ref_con_lx = sx;
|
|
seat->op_ref_con_ly = sy;
|
|
seat->op_moved = false;
|
|
}
|
|
|
|
void seat_begin_move(struct sway_seat *seat, struct sway_container *con,
|
|
uint32_t button) {
|
|
if (!seat->cursor) {
|
|
wlr_log(WLR_DEBUG, "Ignoring move request due to no cursor device");
|
|
return;
|
|
}
|
|
seat->operation = OP_MOVE;
|
|
seat->op_container = con;
|
|
seat->op_button = button;
|
|
cursor_set_image(seat->cursor, "grab", NULL);
|
|
}
|
|
|
|
void seat_begin_resize_floating(struct sway_seat *seat,
|
|
struct sway_container *con, uint32_t button, enum wlr_edges edge) {
|
|
if (!seat->cursor) {
|
|
wlr_log(WLR_DEBUG, "Ignoring resize request due to no cursor device");
|
|
return;
|
|
}
|
|
struct wlr_keyboard *keyboard = wlr_seat_get_keyboard(seat->wlr_seat);
|
|
seat->operation = OP_RESIZE_FLOATING;
|
|
seat->op_container = con;
|
|
seat->op_resize_preserve_ratio = keyboard &&
|
|
(wlr_keyboard_get_modifiers(keyboard) & WLR_MODIFIER_SHIFT);
|
|
seat->op_resize_edge = edge == WLR_EDGE_NONE ?
|
|
WLR_EDGE_BOTTOM | WLR_EDGE_RIGHT : edge;
|
|
seat->op_button = button;
|
|
seat->op_ref_lx = seat->cursor->cursor->x;
|
|
seat->op_ref_ly = seat->cursor->cursor->y;
|
|
seat->op_ref_con_lx = con->x;
|
|
seat->op_ref_con_ly = con->y;
|
|
seat->op_ref_width = con->width;
|
|
seat->op_ref_height = con->height;
|
|
|
|
const char *image = edge == WLR_EDGE_NONE ?
|
|
"se-resize" : wlr_xcursor_get_resize_name(edge);
|
|
cursor_set_image(seat->cursor, image, NULL);
|
|
}
|
|
|
|
void seat_begin_resize_tiling(struct sway_seat *seat,
|
|
struct sway_container *con, uint32_t button, enum wlr_edges edge) {
|
|
seat->operation = OP_RESIZE_TILING;
|
|
seat->op_container = con;
|
|
seat->op_resize_edge = edge;
|
|
seat->op_button = button;
|
|
seat->op_ref_lx = seat->cursor->cursor->x;
|
|
seat->op_ref_ly = seat->cursor->cursor->y;
|
|
seat->op_ref_con_lx = con->x;
|
|
seat->op_ref_con_ly = con->y;
|
|
seat->op_ref_width = con->width;
|
|
seat->op_ref_height = con->height;
|
|
}
|
|
|
|
void seat_end_mouse_operation(struct sway_seat *seat) {
|
|
enum sway_seat_operation operation = seat->operation;
|
|
if (seat->operation == OP_MOVE) {
|
|
// We "move" the container to its own location so it discovers its
|
|
// output again.
|
|
struct sway_container *con = seat->op_container;
|
|
container_floating_move_to(con, con->x, con->y);
|
|
}
|
|
seat->operation = OP_NONE;
|
|
seat->op_container = NULL;
|
|
if (operation == OP_DOWN) {
|
|
// Set the cursor's previous coords to the x/y at the start of the
|
|
// operation, so the container change will be detected if using
|
|
// focus_follows_mouse and the cursor moved off the original container
|
|
// during the operation.
|
|
seat->cursor->previous.x = seat->op_ref_lx;
|
|
seat->cursor->previous.y = seat->op_ref_ly;
|
|
if (seat->op_moved) {
|
|
cursor_send_pointer_motion(seat->cursor, 0, true);
|
|
}
|
|
} else {
|
|
cursor_set_image(seat->cursor, "left_ptr", NULL);
|
|
}
|
|
}
|
|
|
|
void seat_pointer_notify_button(struct sway_seat *seat, uint32_t time_msec,
|
|
uint32_t button, enum wlr_button_state state) {
|
|
seat->last_button = button;
|
|
seat->last_button_serial = wlr_seat_pointer_notify_button(seat->wlr_seat,
|
|
time_msec, button, state);
|
|
}
|