5d882cb5fc
A wlr_keyboard_group allows for multiple keyboard devices to be combined into one logical keyboard. This is useful for keyboards that are split into multiple input devices despite appearing as one physical keyboard in the user's mind. This adds support for wlr_keyboard_groups to sway. There are two keyboard groupings currently supported, which can be set on a per-seat basis. The first keyboard grouping is none, which disables all grouping and provides no functional change. The second is keymap, which groups the keyboard devices in the seat by their keymap. With this grouping, the effective layout and repeat info is also synced across keyboard devices in the seat. Device specific bindings will still be executed as normal, but everything else related to key and modifier events will be handled by the keyboard group's keyboard.
267 lines
8.2 KiB
C
267 lines
8.2 KiB
C
#ifndef _SWAY_INPUT_SEAT_H
|
|
#define _SWAY_INPUT_SEAT_H
|
|
|
|
#include <wlr/types/wlr_layer_shell_v1.h>
|
|
#include <wlr/types/wlr_seat.h>
|
|
#include <wlr/util/edges.h>
|
|
#include "sway/input/input-manager.h"
|
|
|
|
struct sway_seat;
|
|
|
|
struct sway_seatop_impl {
|
|
void (*button)(struct sway_seat *seat, uint32_t time_msec,
|
|
struct wlr_input_device *device, uint32_t button,
|
|
enum wlr_button_state state);
|
|
void (*motion)(struct sway_seat *seat, uint32_t time_msec,
|
|
double dx, double dy);
|
|
void (*axis)(struct sway_seat *seat, struct wlr_event_pointer_axis *event);
|
|
void (*rebase)(struct sway_seat *seat, uint32_t time_msec);
|
|
void (*end)(struct sway_seat *seat);
|
|
void (*unref)(struct sway_seat *seat, struct sway_container *con);
|
|
void (*render)(struct sway_seat *seat, struct sway_output *output,
|
|
pixman_region32_t *damage);
|
|
bool allow_set_cursor;
|
|
};
|
|
|
|
struct sway_seat_device {
|
|
struct sway_seat *sway_seat;
|
|
struct sway_input_device *input_device;
|
|
struct sway_keyboard *keyboard;
|
|
struct sway_switch *switch_device;
|
|
struct sway_tablet *tablet;
|
|
struct sway_tablet_pad *tablet_pad;
|
|
struct wl_list link; // sway_seat::devices
|
|
};
|
|
|
|
struct sway_seat_node {
|
|
struct sway_seat *seat;
|
|
struct sway_node *node;
|
|
|
|
struct wl_list link; // sway_seat::focus_stack
|
|
|
|
struct wl_listener destroy;
|
|
};
|
|
|
|
struct sway_drag_icon {
|
|
struct sway_seat *seat;
|
|
struct wlr_drag_icon *wlr_drag_icon;
|
|
struct wl_list link; // sway_root::drag_icons
|
|
|
|
double x, y; // in layout-local coordinates
|
|
|
|
struct wl_listener surface_commit;
|
|
struct wl_listener map;
|
|
struct wl_listener unmap;
|
|
struct wl_listener destroy;
|
|
};
|
|
|
|
struct sway_seat {
|
|
struct wlr_seat *wlr_seat;
|
|
struct sway_cursor *cursor;
|
|
|
|
bool has_focus;
|
|
struct wl_list focus_stack; // list of containers in focus order
|
|
struct sway_workspace *workspace;
|
|
char *prev_workspace_name; // for workspace back_and_forth
|
|
|
|
// If the focused layer is set, views cannot receive keyboard focus
|
|
struct wlr_layer_surface_v1 *focused_layer;
|
|
|
|
// If exclusive_client is set, no other clients will receive input events
|
|
struct wl_client *exclusive_client;
|
|
|
|
// Last touch point
|
|
int32_t touch_id;
|
|
double touch_x, touch_y;
|
|
|
|
// Seat operations (drag and resize)
|
|
const struct sway_seatop_impl *seatop_impl;
|
|
void *seatop_data;
|
|
|
|
uint32_t last_button_serial;
|
|
|
|
list_t *deferred_bindings; // struct sway_binding
|
|
|
|
struct wl_listener focus_destroy;
|
|
struct wl_listener new_node;
|
|
struct wl_listener request_start_drag;
|
|
struct wl_listener start_drag;
|
|
struct wl_listener request_set_selection;
|
|
struct wl_listener request_set_primary_selection;
|
|
|
|
struct wl_list devices; // sway_seat_device::link
|
|
struct wl_list keyboard_groups; // sway_keyboard_group::link
|
|
|
|
struct wl_list link; // input_manager::seats
|
|
};
|
|
|
|
struct sway_pointer_constraint {
|
|
struct wlr_pointer_constraint_v1 *constraint;
|
|
|
|
struct wl_listener destroy;
|
|
};
|
|
|
|
struct sway_seat *seat_create(const char *seat_name);
|
|
|
|
void seat_destroy(struct sway_seat *seat);
|
|
|
|
void seat_add_device(struct sway_seat *seat,
|
|
struct sway_input_device *device);
|
|
|
|
void seat_configure_device(struct sway_seat *seat,
|
|
struct sway_input_device *device);
|
|
|
|
void seat_reset_device(struct sway_seat *seat,
|
|
struct sway_input_device *input_device);
|
|
|
|
void seat_remove_device(struct sway_seat *seat,
|
|
struct sway_input_device *device);
|
|
|
|
void seat_configure_xcursor(struct sway_seat *seat);
|
|
|
|
void seat_set_focus(struct sway_seat *seat, struct sway_node *node);
|
|
|
|
void seat_set_focus_container(struct sway_seat *seat,
|
|
struct sway_container *con);
|
|
|
|
void seat_set_focus_workspace(struct sway_seat *seat,
|
|
struct sway_workspace *ws);
|
|
|
|
/**
|
|
* Manipulate the focus stack without triggering any other behaviour.
|
|
*
|
|
* This can be used to set focus_inactive by calling the function a second time
|
|
* with the real focus.
|
|
*/
|
|
void seat_set_raw_focus(struct sway_seat *seat, struct sway_node *node);
|
|
|
|
void seat_set_focus_surface(struct sway_seat *seat,
|
|
struct wlr_surface *surface, bool unfocus);
|
|
|
|
void seat_set_focus_layer(struct sway_seat *seat,
|
|
struct wlr_layer_surface_v1 *layer);
|
|
|
|
void seat_set_exclusive_client(struct sway_seat *seat,
|
|
struct wl_client *client);
|
|
|
|
struct sway_node *seat_get_focus(struct sway_seat *seat);
|
|
|
|
struct sway_workspace *seat_get_focused_workspace(struct sway_seat *seat);
|
|
|
|
// If a scratchpad container is fullscreen global, this can be used to try to
|
|
// determine the last focused workspace. Otherwise, this should yield the same
|
|
// results as seat_get_focused_workspace.
|
|
struct sway_workspace *seat_get_last_known_workspace(struct sway_seat *seat);
|
|
|
|
struct sway_container *seat_get_focused_container(struct sway_seat *seat);
|
|
|
|
/**
|
|
* Return the last container to be focused for the seat (or the most recently
|
|
* opened if no container has received focused) that is a child of the given
|
|
* container. The focus-inactive container of the root window is the focused
|
|
* container for the seat (if the seat does have focus). This function can be
|
|
* used to determine what container gets focused next if the focused container
|
|
* is destroyed, or focus moves to a container with children and we need to
|
|
* descend into the next leaf in focus order.
|
|
*/
|
|
struct sway_node *seat_get_focus_inactive(struct sway_seat *seat,
|
|
struct sway_node *node);
|
|
|
|
struct sway_container *seat_get_focus_inactive_tiling(struct sway_seat *seat,
|
|
struct sway_workspace *workspace);
|
|
|
|
/**
|
|
* Descend into the focus stack to find the focus-inactive view. Useful for
|
|
* container placement when they change position in the tree.
|
|
*/
|
|
struct sway_container *seat_get_focus_inactive_view(struct sway_seat *seat,
|
|
struct sway_node *ancestor);
|
|
|
|
/**
|
|
* Return the immediate child of container which was most recently focused.
|
|
*/
|
|
struct sway_node *seat_get_active_tiling_child(struct sway_seat *seat,
|
|
struct sway_node *parent);
|
|
|
|
/**
|
|
* Iterate over the focus-inactive children of the container calling the
|
|
* function on each.
|
|
*/
|
|
void seat_for_each_node(struct sway_seat *seat,
|
|
void (*f)(struct sway_node *node, void *data), void *data);
|
|
|
|
void seat_apply_config(struct sway_seat *seat, struct seat_config *seat_config);
|
|
|
|
struct seat_config *seat_get_config(struct sway_seat *seat);
|
|
|
|
struct seat_config *seat_get_config_by_name(const char *name);
|
|
|
|
bool seat_is_input_allowed(struct sway_seat *seat, struct wlr_surface *surface);
|
|
|
|
void drag_icon_update_position(struct sway_drag_icon *icon);
|
|
|
|
void seatop_begin_default(struct sway_seat *seat);
|
|
|
|
void seatop_begin_down(struct sway_seat *seat, struct sway_container *con,
|
|
uint32_t time_msec, int sx, int sy);
|
|
|
|
void seatop_begin_move_floating(struct sway_seat *seat,
|
|
struct sway_container *con);
|
|
|
|
void seatop_begin_move_tiling_threshold(struct sway_seat *seat,
|
|
struct sway_container *con);
|
|
|
|
void seatop_begin_move_tiling(struct sway_seat *seat,
|
|
struct sway_container *con);
|
|
|
|
void seatop_begin_resize_floating(struct sway_seat *seat,
|
|
struct sway_container *con, enum wlr_edges edge);
|
|
|
|
void seatop_begin_resize_tiling(struct sway_seat *seat,
|
|
struct sway_container *con, enum wlr_edges edge);
|
|
|
|
struct sway_container *seat_get_focus_inactive_floating(struct sway_seat *seat,
|
|
struct sway_workspace *workspace);
|
|
|
|
void seat_pointer_notify_button(struct sway_seat *seat, uint32_t time_msec,
|
|
uint32_t button, enum wlr_button_state state);
|
|
|
|
void seat_consider_warp_to_focus(struct sway_seat *seat);
|
|
|
|
void seatop_button(struct sway_seat *seat, uint32_t time_msec,
|
|
struct wlr_input_device *device, uint32_t button,
|
|
enum wlr_button_state state);
|
|
|
|
/**
|
|
* dx and dy are distances relative to previous position.
|
|
*/
|
|
void seatop_motion(struct sway_seat *seat, uint32_t time_msec,
|
|
double dx, double dy);
|
|
|
|
void seatop_axis(struct sway_seat *seat, struct wlr_event_pointer_axis *event);
|
|
|
|
void seatop_rebase(struct sway_seat *seat, uint32_t time_msec);
|
|
|
|
/**
|
|
* End a seatop (ie. free any seatop specific resources).
|
|
*/
|
|
void seatop_end(struct sway_seat *seat);
|
|
|
|
/**
|
|
* Instructs the seatop implementation to drop any references to the given
|
|
* container (eg. because the container is destroying).
|
|
* The seatop may choose to abort itself in response to this.
|
|
*/
|
|
void seatop_unref(struct sway_seat *seat, struct sway_container *con);
|
|
|
|
/**
|
|
* Instructs a seatop to render anything that it needs to render
|
|
* (eg. dropzone for move-tiling)
|
|
*/
|
|
void seatop_render(struct sway_seat *seat, struct sway_output *output,
|
|
pixman_region32_t *damage);
|
|
|
|
bool seatop_allows_set_cursor(struct sway_seat *seat);
|
|
|
|
#endif
|