diff --git a/include/sway/config.h b/include/sway/config.h index d9f561571d..c659dc92ac 100644 --- a/include/sway/config.h +++ b/include/sway/config.h @@ -207,6 +207,12 @@ enum seat_config_hide_cursor_when_typing { HIDE_WHEN_TYPING_DISABLE, }; +enum seat_config_hide_cursor_but_keep_active { + HIDE_CURSOR_BUT_KEEP_ACTIVE_DEFAULT, // the default is currently disabled + HIDE_CURSOR_BUT_KEEP_ACTIVE_ENABLE, + HIDE_CURSOR_BUT_KEEP_ACTIVE_DISABLE, +}; + enum seat_config_allow_constrain { CONSTRAIN_DEFAULT, // the default is currently enabled CONSTRAIN_ENABLE, @@ -243,6 +249,7 @@ struct seat_config { list_t *attachments; // list of seat_attachment configs int hide_cursor_timeout; enum seat_config_hide_cursor_when_typing hide_cursor_when_typing; + enum seat_config_hide_cursor_but_keep_active hide_cursor_but_keep_active; enum seat_config_allow_constrain allow_constrain; enum seat_config_shortcuts_inhibit shortcuts_inhibit; enum seat_keyboard_grouping keyboard_grouping; diff --git a/include/sway/input/cursor.h b/include/sway/input/cursor.h index 527d03500c..d0da064159 100644 --- a/include/sway/input/cursor.h +++ b/include/sway/input/cursor.h @@ -74,10 +74,6 @@ struct sway_cursor { struct wl_event_source *hide_source; bool hidden; - // This field is just a cache of the field in seat_config in order to avoid - // costly seat_config lookups on every keypress. HIDE_WHEN_TYPING_DEFAULT - // indicates that there is no cached value. - enum seat_config_hide_cursor_when_typing hide_when_typing; size_t pressed_button_count; }; diff --git a/include/sway/input/seat.h b/include/sway/input/seat.h index 428f96796c..0adec3e1f0 100644 --- a/include/sway/input/seat.h +++ b/include/sway/input/seat.h @@ -83,6 +83,7 @@ struct sway_drag { struct sway_seat { struct wlr_seat *wlr_seat; struct sway_cursor *cursor; + struct seat_config *config; // Seat scene tree structure // - scene_tree @@ -243,10 +244,6 @@ void seat_for_each_node(struct sway_seat *seat, 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); - void seat_idle_notify_activity(struct sway_seat *seat, enum sway_input_idle_source source); diff --git a/sway/commands/seat/hide_cursor.c b/sway/commands/seat/hide_cursor.c index f5177a471f..b804a442e7 100644 --- a/sway/commands/seat/hide_cursor.c +++ b/sway/commands/seat/hide_cursor.c @@ -31,17 +31,15 @@ struct cmd_results *seat_cmd_hide_cursor(int argc, char **argv) { } seat_config->hide_cursor_timeout = timeout; } else { - if (strcmp(argv[0], "when-typing") != 0) { + if (strcmp(argv[0], "when-typing") == 0) { + seat_config->hide_cursor_when_typing = parse_boolean(argv[1], true) ? + HIDE_WHEN_TYPING_ENABLE : HIDE_WHEN_TYPING_DISABLE; + } else if (strcmp(argv[0], "but-keep-active") == 0) { + seat_config->hide_cursor_but_keep_active = parse_boolean(argv[1], true) ? + HIDE_CURSOR_BUT_KEEP_ACTIVE_ENABLE : HIDE_CURSOR_BUT_KEEP_ACTIVE_DISABLE; + } else { return cmd_results_new(CMD_INVALID, - "Expected 'hide_cursor |when-typing [enable|disable]'"); - } - seat_config->hide_cursor_when_typing = parse_boolean(argv[1], true) ? - HIDE_WHEN_TYPING_ENABLE : HIDE_WHEN_TYPING_DISABLE; - - // Invalidate all the caches for this config - struct sway_seat *seat = NULL; - wl_list_for_each(seat, &server.input->seats, link) { - seat->cursor->hide_when_typing = HIDE_WHEN_TYPING_DEFAULT; + "Expected 'hide_cursor |when-typing|but-keep-active [enable|disable]'"); } } diff --git a/sway/config/seat.c b/sway/config/seat.c index f2326189e0..1c1c40df29 100644 --- a/sway/config/seat.c +++ b/sway/config/seat.c @@ -29,6 +29,7 @@ struct seat_config *new_seat_config(const char* name) { } seat->hide_cursor_timeout = -1; seat->hide_cursor_when_typing = HIDE_WHEN_TYPING_DEFAULT; + seat->hide_cursor_but_keep_active = HIDE_CURSOR_BUT_KEEP_ACTIVE_DEFAULT; seat->allow_constrain = CONSTRAIN_DEFAULT; seat->shortcuts_inhibit = SHORTCUTS_INHIBIT_DEFAULT; seat->keyboard_grouping = KEYBOARD_GROUP_DEFAULT; @@ -154,6 +155,10 @@ void merge_seat_config(struct seat_config *dest, struct seat_config *source) { dest->hide_cursor_when_typing = source->hide_cursor_when_typing; } + if (source->hide_cursor_but_keep_active != HIDE_CURSOR_BUT_KEEP_ACTIVE_DEFAULT) { + dest->hide_cursor_but_keep_active = source->hide_cursor_but_keep_active; + } + if (source->allow_constrain != CONSTRAIN_DEFAULT) { dest->allow_constrain = source->allow_constrain; } diff --git a/sway/input/cursor.c b/sway/input/cursor.c index bbd16717ff..b256795504 100644 --- a/sway/input/cursor.c +++ b/sway/input/cursor.c @@ -183,7 +183,10 @@ void cursor_update_image(struct sway_cursor *cursor, static void cursor_hide(struct sway_cursor *cursor) { wlr_cursor_unset_image(cursor->cursor); cursor->hidden = true; - wlr_seat_pointer_notify_clear_focus(cursor->seat->wlr_seat); + if (cursor->seat->config->hide_cursor_but_keep_active + != HIDE_CURSOR_BUT_KEEP_ACTIVE_ENABLE) { + wlr_seat_pointer_notify_clear_focus(cursor->seat->wlr_seat); + } } static int hide_notify(void *data) { @@ -198,10 +201,8 @@ int cursor_get_timeout(struct sway_cursor *cursor) { return 0; } - struct seat_config *sc = seat_get_config(cursor->seat); - if (!sc) { - sc = seat_get_config_by_name("*"); - } + struct seat_config *sc = cursor->seat->config; + int timeout = sc ? sc->hide_cursor_timeout : 0; if (timeout < 0) { timeout = 0; @@ -214,23 +215,7 @@ void cursor_notify_key_press(struct sway_cursor *cursor) { return; } - if (cursor->hide_when_typing == HIDE_WHEN_TYPING_DEFAULT) { - // No cached value, need to lookup in the seat_config - const struct seat_config *seat_config = seat_get_config(cursor->seat); - if (!seat_config) { - seat_config = seat_get_config_by_name("*"); - if (!seat_config) { - return; - } - } - cursor->hide_when_typing = seat_config->hide_cursor_when_typing; - // The default is currently disabled - if (cursor->hide_when_typing == HIDE_WHEN_TYPING_DEFAULT) { - cursor->hide_when_typing = HIDE_WHEN_TYPING_DISABLE; - } - } - - if (cursor->hide_when_typing == HIDE_WHEN_TYPING_ENABLE) { + if (cursor->seat->config->hide_cursor_when_typing == HIDE_WHEN_TYPING_ENABLE) { cursor_hide(cursor); } } @@ -1339,10 +1324,7 @@ void handle_pointer_constraint(struct wl_listener *listener, void *data) { void sway_cursor_constrain(struct sway_cursor *cursor, struct wlr_pointer_constraint_v1 *constraint) { - struct seat_config *config = seat_get_config(cursor->seat); - if (!config) { - config = seat_get_config_by_name("*"); - } + struct seat_config *config = cursor->seat->config; if (!config || config->allow_constrain == CONSTRAIN_DISABLE) { return; diff --git a/sway/input/input-manager.c b/sway/input/input-manager.c index 248ca34eef..59bdb453fa 100644 --- a/sway/input/input-manager.c +++ b/sway/input/input-manager.c @@ -175,7 +175,15 @@ static struct sway_input_device *input_sway_device_from_wlr( static bool input_has_seat_fallback_configuration(void) { struct sway_seat *seat = NULL; wl_list_for_each(seat, &server.input->seats, link) { - struct seat_config *seat_config = seat_get_config(seat); + struct seat_config *seat_config = NULL; + struct seat_config *sc = NULL; + for (int i = 0; i < config->seat_configs->length; ++i ) { + sc = config->seat_configs->items[i]; + if (strcmp(seat->wlr_seat->name, sc->name) == 0) { + seat_config = sc; + break; + } + } if (seat_config && strcmp(seat_config->name, "*") != 0 && seat_config->fallback != -1) { return true; @@ -257,10 +265,9 @@ static void handle_new_input(struct wl_listener *listener, void *data) { bool added = false; struct sway_seat *seat = NULL; wl_list_for_each(seat, &input->seats, link) { - struct seat_config *seat_config = seat_get_config(seat); - bool has_attachment = seat_config && - (seat_config_get_attachment(seat_config, input_device->identifier) || - seat_config_get_attachment(seat_config, "*")); + bool has_attachment = seat->config && + (seat_config_get_attachment(seat->config, input_device->identifier) || + seat_config_get_attachment(seat->config, "*")); if (has_attachment) { seat_add_device(seat, input_device); @@ -270,8 +277,7 @@ static void handle_new_input(struct wl_listener *listener, void *data) { if (!added) { wl_list_for_each(seat, &input->seats, link) { - struct seat_config *seat_config = seat_get_config(seat); - if (seat_config && seat_config->fallback == 1) { + if (seat->config && seat->config->fallback == 1) { seat_add_device(seat, input_device); added = true; } @@ -336,13 +342,8 @@ static void handle_keyboard_shortcuts_inhibit_new_inhibitor( } if (inhibit == SHORTCUTS_INHIBIT_DEFAULT) { - struct seat_config *config = seat_get_config(seat); - if (!config) { - config = seat_get_config_by_name("*"); - } - - if (config) { - inhibit = config->shortcuts_inhibit; + if (seat->config) { + inhibit = seat->config->shortcuts_inhibit; } } @@ -633,7 +634,7 @@ void input_manager_apply_seat_config(struct seat_config *seat_config) { wl_list_for_each(seat, &server.input->seats, link) { // Only apply the wildcard config directly if there is no seat // specific config - struct seat_config *sc = seat_get_config(seat); + struct seat_config *sc = seat->config; if (!sc) { sc = seat_config; } @@ -655,7 +656,7 @@ void input_manager_apply_seat_config(struct seat_config *seat_config) { list_t *seat_list = create_list(); struct sway_seat *seat = NULL; wl_list_for_each(seat, &server.input->seats, link) { - struct seat_config *seat_config = seat_get_config(seat); + struct seat_config *seat_config = seat->config; if (!seat_config) { continue; } @@ -683,8 +684,7 @@ void input_manager_apply_seat_config(struct seat_config *seat_config) { } } else { wl_list_for_each(seat, &server.input->seats, link) { - struct seat_config *seat_config = seat_get_config(seat); - if (seat_config && seat_config->fallback == 1) { + if (seat->config && seat->config->fallback == 1) { seat_add_device(seat, input_device); } else { seat_remove_device(seat, input_device); diff --git a/sway/input/keyboard.c b/sway/input/keyboard.c index efb9ac394d..2d3f83d370 100644 --- a/sway/input/keyboard.c +++ b/sway/input/keyboard.c @@ -831,10 +831,7 @@ static void sway_keyboard_group_remove_invalid(struct sway_keyboard *keyboard) { } struct sway_seat *seat = keyboard->seat_device->sway_seat; - struct seat_config *sc = seat_get_config(seat); - if (!sc) { - sc = seat_get_config_by_name("*"); - } + struct seat_config *sc = seat->config; switch (sc ? sc->keyboard_grouping : KEYBOARD_GROUP_DEFAULT) { case KEYBOARD_GROUP_NONE: @@ -854,17 +851,13 @@ static void sway_keyboard_group_remove_invalid(struct sway_keyboard *keyboard) { static void sway_keyboard_group_add(struct sway_keyboard *keyboard) { struct sway_input_device *device = keyboard->seat_device->input_device; struct sway_seat *seat = keyboard->seat_device->sway_seat; - struct seat_config *sc = seat_get_config(seat); + struct seat_config *sc = seat->config; if (device->is_virtual) { // Virtual devices should not be grouped return; } - if (!sc) { - sc = seat_get_config_by_name("*"); - } - if (sc && sc->keyboard_grouping == KEYBOARD_GROUP_NONE) { // Keyboard grouping is disabled for the seat return; diff --git a/sway/input/seat.c b/sway/input/seat.c index 9a00a3e245..0fc899bc3b 100644 --- a/sway/input/seat.c +++ b/sway/input/seat.c @@ -984,13 +984,9 @@ void seat_configure_xcursor(struct sway_seat *seat) { unsigned cursor_size = 24; const char *cursor_theme = NULL; - const struct seat_config *seat_config = seat_get_config(seat); - if (!seat_config) { - seat_config = seat_get_config_by_name("*"); - } - if (seat_config) { - cursor_size = seat_config->xcursor_theme.size; - cursor_theme = seat_config->xcursor_theme.name; + if (seat->config) { + cursor_size = seat->config->xcursor_theme.size; + cursor_theme = seat->config->xcursor_theme.name; } if (seat == input_manager_get_default_seat()) { @@ -1493,6 +1489,7 @@ void seat_apply_config(struct sway_seat *seat, return; } + seat->config = seat_config; seat->idle_inhibit_sources = seat_config->idle_inhibit_sources; seat->idle_wake_sources = seat_config->idle_wake_sources; @@ -1503,30 +1500,6 @@ void seat_apply_config(struct sway_seat *seat, } } -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; -} - -struct seat_config *seat_get_config_by_name(const char *name) { - 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(name, seat_config->name) == 0) { - return seat_config; - } - } - - return NULL; -} - void seat_pointer_notify_button(struct sway_seat *seat, uint32_t time_msec, uint32_t button, enum wl_pointer_button_state state) { seat->last_button_serial = wlr_seat_pointer_notify_button(seat->wlr_seat, diff --git a/sway/sway-input.5.scd b/sway/sway-input.5.scd index fbef2a3213..5ae7e66b3b 100644 --- a/sway/sway-input.5.scd +++ b/sway/sway-input.5.scd @@ -258,7 +258,7 @@ correct seat. Set this seat as the fallback seat. A fallback seat will attach any device not explicitly attached to another seat (similar to a "default" seat). -*seat* hide_cursor |when-typing [enable|disable] +*seat* hide_cursor |when-typing|but-keep-active [enable|disable] Hides the cursor image after the specified event occurred. If _timeout_ is specified, then the cursor will be hidden after _timeout_ @@ -273,6 +273,9 @@ correct seat. certain types of software (Gimp, Blender etc) that rely on simultaneous input from mouse and keyboard. + If _but-keep-active_ is enabled, then the cursor will remain active when + hidden. This solves the issues with _when-typing_ as mentioned above. + *seat* idle_inhibit Sets the set of input event sources which can prevent the seat from becoming idle, as a space separated list of source names. Valid names are