mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-07 10:38:47 +00:00
Merge branch 'self-insert-notfirst'
This merges support for the self-insert-notfirst binding, which is used for efficient space-stripping on paste. This will also merge into 3.1.1.
This commit is contained in:
commit
e3e82f5873
7 changed files with 73 additions and 44 deletions
|
@ -120,6 +120,8 @@ The following special input functions are available:
|
||||||
|
|
||||||
- ``execute`` run the current commandline
|
- ``execute`` run the current commandline
|
||||||
|
|
||||||
|
- ``force-repaint`` reexecute the prompt functions without coalescing
|
||||||
|
|
||||||
- ``forward-bigword``, move one whitespace-delimited word to the right
|
- ``forward-bigword``, move one whitespace-delimited word to the right
|
||||||
|
|
||||||
- ``forward-char``, move one character to the right
|
- ``forward-char``, move one character to the right
|
||||||
|
@ -160,7 +162,9 @@ The following special input functions are available:
|
||||||
|
|
||||||
- ``repaint-mode`` reexecutes the fish_mode_prompt function and redraws the prompt. This is useful for vi-mode. If no fish_mode_prompt exists, it acts like a normal repaint.
|
- ``repaint-mode`` reexecutes the fish_mode_prompt function and redraws the prompt. This is useful for vi-mode. If no fish_mode_prompt exists, it acts like a normal repaint.
|
||||||
|
|
||||||
- ``force-repaint`` reexecute the prompt functions without coalescing.
|
- ``self-insert``, inserts the matching sequence into the command line
|
||||||
|
|
||||||
|
- ``self-insert-notfirst``, inserts the matching sequence into the command line, unless the cursor is at the beginning
|
||||||
|
|
||||||
- ``suppress-autosuggestion``, remove the current autosuggestion
|
- ``suppress-autosuggestion``, remove the current autosuggestion
|
||||||
|
|
||||||
|
|
|
@ -161,7 +161,7 @@ function __fish_shared_key_bindings -d "Bindings shared between emacs and vi mod
|
||||||
bind --preset -M paste \\ "__fish_commandline_insert_escaped \\\ \$__fish_paste_quoted"
|
bind --preset -M paste \\ "__fish_commandline_insert_escaped \\\ \$__fish_paste_quoted"
|
||||||
# Only insert spaces if we're either quoted or not at the beginning of the commandline
|
# Only insert spaces if we're either quoted or not at the beginning of the commandline
|
||||||
# - this strips leading spaces if they would trigger histignore.
|
# - this strips leading spaces if they would trigger histignore.
|
||||||
bind --preset -M paste \ 'if set -q __fish_paste_quoted[1]; or string length -q -- (commandline -c); commandline -i " "; end'
|
bind --preset -M paste " " self-insert-notfirst
|
||||||
end
|
end
|
||||||
|
|
||||||
function __fish_commandline_insert_escaped --description 'Insert the first arg escaped if a second arg is given'
|
function __fish_commandline_insert_escaped --description 'Insert the first arg escaped if a second arg is given'
|
||||||
|
|
|
@ -117,6 +117,7 @@ static const input_function_metadata_t input_function_metadata[] = {
|
||||||
{readline_cmd_t::history_token_search_backward, L"history-token-search-backward"},
|
{readline_cmd_t::history_token_search_backward, L"history-token-search-backward"},
|
||||||
{readline_cmd_t::history_token_search_forward, L"history-token-search-forward"},
|
{readline_cmd_t::history_token_search_forward, L"history-token-search-forward"},
|
||||||
{readline_cmd_t::self_insert, L"self-insert"},
|
{readline_cmd_t::self_insert, L"self-insert"},
|
||||||
|
{readline_cmd_t::self_insert_notfirst, L"self-insert-notfirst"},
|
||||||
{readline_cmd_t::transpose_chars, L"transpose-chars"},
|
{readline_cmd_t::transpose_chars, L"transpose-chars"},
|
||||||
{readline_cmd_t::transpose_words, L"transpose-words"},
|
{readline_cmd_t::transpose_words, L"transpose-words"},
|
||||||
{readline_cmd_t::upcase_word, L"upcase-word"},
|
{readline_cmd_t::upcase_word, L"upcase-word"},
|
||||||
|
@ -497,7 +498,8 @@ char_event_t inputter_t::readch(bool allow_commands) {
|
||||||
|
|
||||||
if (evt.is_readline()) {
|
if (evt.is_readline()) {
|
||||||
switch (evt.get_readline()) {
|
switch (evt.get_readline()) {
|
||||||
case readline_cmd_t::self_insert: {
|
case readline_cmd_t::self_insert:
|
||||||
|
case readline_cmd_t::self_insert_notfirst: {
|
||||||
// Typically self-insert is generated by the generic (empty) binding.
|
// Typically self-insert is generated by the generic (empty) binding.
|
||||||
// However if it is generated by a real sequence, then insert that sequence.
|
// However if it is generated by a real sequence, then insert that sequence.
|
||||||
for (auto iter = evt.seq.crbegin(); iter != evt.seq.crend(); ++iter) {
|
for (auto iter = evt.seq.crbegin(); iter != evt.seq.crend(); ++iter) {
|
||||||
|
@ -505,7 +507,13 @@ char_event_t inputter_t::readch(bool allow_commands) {
|
||||||
}
|
}
|
||||||
// Issue #1595: ensure we only insert characters, not readline functions. The
|
// Issue #1595: ensure we only insert characters, not readline functions. The
|
||||||
// common case is that this will be empty.
|
// common case is that this will be empty.
|
||||||
return read_characters_no_readline();
|
char_event_t res = read_characters_no_readline();
|
||||||
|
|
||||||
|
// Hackish: mark the input style.
|
||||||
|
res.input_style = evt.get_readline() == readline_cmd_t::self_insert_notfirst
|
||||||
|
? char_event_t::style_notfirst
|
||||||
|
: char_event_t::style_normal;
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
case readline_cmd_t::func_and: {
|
case readline_cmd_t::func_and: {
|
||||||
if (function_status_) {
|
if (function_status_) {
|
||||||
|
|
|
@ -42,6 +42,7 @@ enum class readline_cmd_t {
|
||||||
history_token_search_backward,
|
history_token_search_backward,
|
||||||
history_token_search_forward,
|
history_token_search_forward,
|
||||||
self_insert,
|
self_insert,
|
||||||
|
self_insert_notfirst,
|
||||||
transpose_chars,
|
transpose_chars,
|
||||||
transpose_words,
|
transpose_words,
|
||||||
upcase_word,
|
upcase_word,
|
||||||
|
@ -111,6 +112,17 @@ class char_event_t {
|
||||||
/// The type of event.
|
/// The type of event.
|
||||||
char_event_type_t type;
|
char_event_type_t type;
|
||||||
|
|
||||||
|
/// Hackish: the input style, which describes how char events (only) are applied to the command
|
||||||
|
/// line. Note this is set only after applying bindings; it is not set from readb().
|
||||||
|
enum input_style_t : uint8_t {
|
||||||
|
// Insert characters normally.
|
||||||
|
style_normal,
|
||||||
|
|
||||||
|
// Insert characters only if the cursor is not at the beginning. Otherwise, discard them.
|
||||||
|
style_notfirst,
|
||||||
|
};
|
||||||
|
input_style_t input_style{style_normal};
|
||||||
|
|
||||||
/// The sequence of characters in the input mapping which generated this event.
|
/// The sequence of characters in the input mapping which generated this event.
|
||||||
/// Note that the generic self-insert case does not have any characters, so this would be empty.
|
/// Note that the generic self-insert case does not have any characters, so this would be empty.
|
||||||
wcstring seq{};
|
wcstring seq{};
|
||||||
|
|
|
@ -1180,6 +1180,7 @@ static bool command_ends_paging(readline_cmd_t c, bool focused_on_search_field)
|
||||||
case rl::backward_kill_path_component:
|
case rl::backward_kill_path_component:
|
||||||
case rl::backward_kill_bigword:
|
case rl::backward_kill_bigword:
|
||||||
case rl::self_insert:
|
case rl::self_insert:
|
||||||
|
case rl::self_insert_notfirst:
|
||||||
case rl::transpose_chars:
|
case rl::transpose_chars:
|
||||||
case rl::transpose_words:
|
case rl::transpose_words:
|
||||||
case rl::upcase_word:
|
case rl::upcase_word:
|
||||||
|
@ -2558,39 +2559,27 @@ struct readline_loop_state_t {
|
||||||
/// Read normal characters, inserting them into the command line.
|
/// Read normal characters, inserting them into the command line.
|
||||||
/// \return the next unhandled event.
|
/// \return the next unhandled event.
|
||||||
maybe_t<char_event_t> reader_data_t::read_normal_chars(readline_loop_state_t &rls) {
|
maybe_t<char_event_t> reader_data_t::read_normal_chars(readline_loop_state_t &rls) {
|
||||||
maybe_t<char_event_t> event_needing_handling = inputter.readch();
|
maybe_t<char_event_t> event_needing_handling{};
|
||||||
|
wcstring accumulated_chars;
|
||||||
if (!event_is_normal_char(*event_needing_handling) || !can_read(STDIN_FILENO))
|
|
||||||
return event_needing_handling;
|
|
||||||
|
|
||||||
// This is a normal character input.
|
|
||||||
// We are going to handle it directly, accumulating more.
|
|
||||||
char_event_t evt = event_needing_handling.acquire();
|
|
||||||
size_t limit = std::min(rls.nchars - command_line.size(), READAHEAD_MAX);
|
size_t limit = std::min(rls.nchars - command_line.size(), READAHEAD_MAX);
|
||||||
|
while (accumulated_chars.size() < limit) {
|
||||||
wchar_t arr[READAHEAD_MAX + 1] = {};
|
bool allow_commands = (accumulated_chars.empty());
|
||||||
arr[0] = evt.get_char();
|
auto evt = inputter.readch(allow_commands);
|
||||||
|
if (!event_is_normal_char(evt) || !can_read(STDIN_FILENO)) {
|
||||||
for (size_t i = 1; i < limit; ++i) {
|
event_needing_handling = std::move(evt);
|
||||||
if (!can_read(0)) {
|
|
||||||
break;
|
break;
|
||||||
}
|
} else if (evt.input_style == char_event_t::style_notfirst && accumulated_chars.empty() &&
|
||||||
// Only allow commands on the first key; otherwise, we might have data we
|
active_edit_line()->position() == 0) {
|
||||||
// need to insert on the commandline that the command might need to be able
|
// The cursor is at the beginning and nothing is accumulated, so skip this character.
|
||||||
// to see.
|
continue;
|
||||||
auto next_event = inputter.readch(false);
|
|
||||||
if (event_is_normal_char(next_event)) {
|
|
||||||
arr[i] = next_event.get_char();
|
|
||||||
} else {
|
} else {
|
||||||
// We need to process this in the outer loop.
|
accumulated_chars.push_back(evt.get_char());
|
||||||
assert(!event_needing_handling && "Should not have an unhandled event");
|
|
||||||
event_needing_handling = next_event;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!accumulated_chars.empty()) {
|
||||||
editable_line_t *el = active_edit_line();
|
editable_line_t *el = active_edit_line();
|
||||||
insert_string(el, arr);
|
insert_string(el, accumulated_chars);
|
||||||
|
|
||||||
// End paging upon inserting into the normal command line.
|
// End paging upon inserting into the normal command line.
|
||||||
if (el == &command_line) {
|
if (el == &command_line) {
|
||||||
|
@ -2599,6 +2588,7 @@ maybe_t<char_event_t> reader_data_t::read_normal_chars(readline_loop_state_t &rl
|
||||||
|
|
||||||
// Since we handled a normal character, we don't have a last command.
|
// Since we handled a normal character, we don't have a last command.
|
||||||
rls.last_cmd.reset();
|
rls.last_cmd.reset();
|
||||||
|
}
|
||||||
return event_needing_handling;
|
return event_needing_handling;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3373,12 +3363,11 @@ void reader_data_t::handle_readline_command(readline_cmd_t c, readline_loop_stat
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// Some commands should have been handled internally by input_readch().
|
// Some commands should have been handled internally by inputter_t::readch().
|
||||||
case rl::self_insert: {
|
case rl::self_insert:
|
||||||
DIE("self-insert should have been handled by inputter_t::readch");
|
case rl::self_insert_notfirst:
|
||||||
}
|
|
||||||
case rl::func_and: {
|
case rl::func_and: {
|
||||||
DIE("self-insert should have been handled by inputter_t::readch");
|
DIE("should have been handled by inputter_t::readch");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3492,7 +3481,10 @@ maybe_t<wcstring> reader_data_t::readline(int nchars_or_0) {
|
||||||
} else {
|
} else {
|
||||||
// Ordinary char.
|
// Ordinary char.
|
||||||
wchar_t c = event_needing_handling->get_char();
|
wchar_t c = event_needing_handling->get_char();
|
||||||
if (!fish_reserved_codepoint(c) && (c >= L' ' || c == L'\n' || c == L'\r') &&
|
if (event_needing_handling->input_style == char_event_t::style_notfirst &&
|
||||||
|
active_edit_line()->position() == 0) {
|
||||||
|
// This character is skipped.
|
||||||
|
} else if (!fish_reserved_codepoint(c) && (c >= L' ' || c == L'\n' || c == L'\r') &&
|
||||||
c != 0x7F) {
|
c != 0x7F) {
|
||||||
// Regular character.
|
// Regular character.
|
||||||
editable_line_t *el = active_edit_line();
|
editable_line_t *el = active_edit_line();
|
||||||
|
|
|
@ -299,3 +299,15 @@ expect_prompt -re {nul seen\r\nnul seen\r\nnul seen} {
|
||||||
} unmatched {
|
} unmatched {
|
||||||
puts stderr "nul not seen"
|
puts stderr "nul not seen"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Test self-insert-notfirst. (#6603)
|
||||||
|
# Here the leading 'q's should be stripped, but the trailing ones not.
|
||||||
|
send "bind q self-insert-notfirst\r"
|
||||||
|
expect_prompt
|
||||||
|
send "qqqecho qqq"
|
||||||
|
send "\r"
|
||||||
|
expect_prompt -re {qqq} {
|
||||||
|
puts "Leading q properly stripped"
|
||||||
|
} unmatched {
|
||||||
|
puts stderr "Leading qs not stripped"
|
||||||
|
}
|
||||||
|
|
|
@ -23,3 +23,4 @@ ctrl-o seen
|
||||||
ctrl-w stops at :
|
ctrl-w stops at :
|
||||||
ctrl-w stops at @
|
ctrl-w stops at @
|
||||||
nul seen
|
nul seen
|
||||||
|
Leading q properly stripped
|
||||||
|
|
Loading…
Reference in a new issue