mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-28 04:35:09 +00:00
Introduces "smartcase" completions
"smartcase" performs case-insensitive matching if the input string is all lowercase, and case-sensitive matching otherwise. When completing e.g. files, we will now show both case sensitive and insensitive completions if the input string does not contain uppercase characters. This is a delicate fix in an interactive component with low test coverage. It's likely something will regress here. Fixes #3978
This commit is contained in:
parent
b38a23a46d
commit
2b8d2deb0c
5 changed files with 49 additions and 18 deletions
|
@ -140,8 +140,7 @@ Interactive improvements
|
|||
- ``help`` works properly on MSYS2 (:issue:`7113`).
|
||||
- Resuming a piped job by its number, like ``fg %1`` has been fixed (:issue:`7406`).
|
||||
- Commands run from key bindings now use the same tty modes as normal commands (:issue:`7483`).
|
||||
- Autosuggestions from history are now case-sensitive (:issue:`3978`).
|
||||
|
||||
- Autosuggestions from history are now case-sensitive, and tab completions are "smartcase": they offer case-insensitive matches if the input string is lowercase (:issue:`3978`).
|
||||
|
||||
New or improved bindings
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
|
|
@ -290,9 +290,12 @@ void completions_sort_and_prioritize(completion_list_t *comps, completion_reques
|
|||
|
||||
// Lastly, if this is for an autosuggestion, prefer to avoid completions that duplicate
|
||||
// arguments, and penalize files that end in tilde - they're frequently autosave files from e.g.
|
||||
// emacs.
|
||||
// emacs. Also prefer samecase to smartcase.
|
||||
if (flags & completion_request_t::autosuggestion) {
|
||||
stable_sort(comps->begin(), comps->end(), [](const completion_t &a, const completion_t &b) {
|
||||
if (a.match.case_fold != b.match.case_fold) {
|
||||
return a.match.case_fold < b.match.case_fold;
|
||||
}
|
||||
return compare_completions_by_duplicate_arguments(a, b) ||
|
||||
compare_completions_by_tilde(a, b);
|
||||
});
|
||||
|
|
|
@ -2132,9 +2132,13 @@ static void test_fuzzy_match() {
|
|||
do_test(test_fuzzy(L"", L"", type_t::exact, case_fold_t::samecase));
|
||||
do_test(test_fuzzy(L"alpha", L"alpha", type_t::exact, case_fold_t::samecase));
|
||||
do_test(test_fuzzy(L"alp", L"alpha", type_t::prefix, case_fold_t::samecase));
|
||||
do_test(test_fuzzy(L"alpha", L"AlPhA", type_t::exact, case_fold_t::smartcase));
|
||||
do_test(test_fuzzy(L"alpha", L"AlPhA!", type_t::prefix, case_fold_t::smartcase));
|
||||
do_test(test_fuzzy(L"ALPHA", L"alpha!", type_t::prefix, case_fold_t::icase));
|
||||
do_test(test_fuzzy(L"ALPHA!", L"alPhA!", type_t::exact, case_fold_t::icase));
|
||||
do_test(test_fuzzy(L"alPh", L"ALPHA!", type_t::prefix, case_fold_t::icase));
|
||||
do_test(test_fuzzy(L"LPH", L"ALPHA!", type_t::substr, case_fold_t::samecase));
|
||||
do_test(test_fuzzy(L"lph", L"AlPhA!", type_t::substr, case_fold_t::smartcase));
|
||||
do_test(test_fuzzy(L"lPh", L"ALPHA!", type_t::substr, case_fold_t::icase));
|
||||
do_test(test_fuzzy(L"AA", L"ALPHA!", type_t::subseq, case_fold_t::samecase));
|
||||
do_test(!string_fuzzy_match_string(L"lh", L"ALPHA!").has_value()); // no subseq icase
|
||||
|
@ -2897,7 +2901,8 @@ static void test_complete() {
|
|||
struct test_complete_vars_t : environment_t {
|
||||
wcstring_list_t get_names(int flags) const override {
|
||||
UNUSED(flags);
|
||||
return {L"Foo1", L"Foo2", L"Foo3", L"Bar1", L"Bar2", L"Bar3"};
|
||||
return {L"Foo1", L"Foo2", L"Foo3", L"Bar1", L"Bar2",
|
||||
L"Bar3", L"alpha", L"ALPHA!", L"gamma1", L"GAMMA2"};
|
||||
}
|
||||
|
||||
maybe_t<env_var_t> get(const wcstring &key,
|
||||
|
@ -2921,13 +2926,24 @@ static void test_complete() {
|
|||
|
||||
completions = do_complete(L"$", {});
|
||||
completions_sort_and_prioritize(&completions);
|
||||
do_test(completions.size() == 6);
|
||||
do_test(completions.at(0).completion == L"Bar1");
|
||||
do_test(completions.at(1).completion == L"Bar2");
|
||||
do_test(completions.at(2).completion == L"Bar3");
|
||||
do_test(completions.at(3).completion == L"Foo1");
|
||||
do_test(completions.at(4).completion == L"Foo2");
|
||||
do_test(completions.at(5).completion == L"Foo3");
|
||||
do_test(completions.size() == 10);
|
||||
do_test(completions.at(0).completion == L"alpha");
|
||||
do_test(completions.at(1).completion == L"ALPHA!");
|
||||
do_test(completions.at(2).completion == L"Bar1");
|
||||
do_test(completions.at(3).completion == L"Bar2");
|
||||
do_test(completions.at(4).completion == L"Bar3");
|
||||
do_test(completions.at(5).completion == L"Foo1");
|
||||
do_test(completions.at(6).completion == L"Foo2");
|
||||
do_test(completions.at(7).completion == L"Foo3");
|
||||
do_test(completions.at(8).completion == L"gamma1");
|
||||
do_test(completions.at(9).completion == L"GAMMA2");
|
||||
|
||||
// Smartcase test. Lowercase inputs match both lowercase and uppercase.
|
||||
completions = do_complete(L"$a", {});
|
||||
completions_sort_and_prioritize(&completions);
|
||||
do_test(completions.size() == 2);
|
||||
do_test(completions.at(0).completion == L"$ALPHA!");
|
||||
do_test(completions.at(1).completion == L"lpha");
|
||||
|
||||
completions = do_complete(L"$F", {});
|
||||
completions_sort_and_prioritize(&completions);
|
||||
|
@ -2942,9 +2958,10 @@ static void test_complete() {
|
|||
|
||||
completions = do_complete(L"$1", completion_request_t::fuzzy_match);
|
||||
completions_sort_and_prioritize(&completions);
|
||||
do_test(completions.size() == 2);
|
||||
do_test(completions.size() == 3);
|
||||
do_test(completions.at(0).completion == L"$Bar1");
|
||||
do_test(completions.at(1).completion == L"$Foo1");
|
||||
do_test(completions.at(2).completion == L"$gamma1");
|
||||
|
||||
if (system("mkdir -p 'test/complete_test'")) err(L"mkdir failed");
|
||||
if (system("touch 'test/complete_test/has space'")) err(L"touch failed");
|
||||
|
|
|
@ -158,6 +158,15 @@ static bool subsequence_in_string(const wcstring &needle, const wcstring &haysta
|
|||
maybe_t<string_fuzzy_match_t> string_fuzzy_match_t::try_create(const wcstring &string,
|
||||
const wcstring &match_against,
|
||||
bool anchor_start) {
|
||||
// Helper to lazily compute if case insensitive matches should use icase or smartcase.
|
||||
// Use icase if the input contains any uppercase characters, smartcase otherwise.
|
||||
auto get_case_fold = [&] {
|
||||
for (wchar_t c : string) {
|
||||
if (towlower(c) != c) return case_fold_t::icase;
|
||||
}
|
||||
return case_fold_t::smartcase;
|
||||
};
|
||||
|
||||
// A string cannot fuzzy match against a shorter string.
|
||||
if (string.size() > match_against.size()) {
|
||||
return none();
|
||||
|
@ -175,12 +184,12 @@ maybe_t<string_fuzzy_match_t> string_fuzzy_match_t::try_create(const wcstring &s
|
|||
|
||||
// exact icase
|
||||
if (wcscasecmp(string.c_str(), match_against.c_str()) == 0) {
|
||||
return string_fuzzy_match_t{contain_type_t::exact, case_fold_t::icase};
|
||||
return string_fuzzy_match_t{contain_type_t::exact, get_case_fold()};
|
||||
}
|
||||
|
||||
// prefix icase
|
||||
if (string_prefixes_string_case_insensitive(string, match_against)) {
|
||||
return string_fuzzy_match_t{contain_type_t::prefix, case_fold_t::icase};
|
||||
return string_fuzzy_match_t{contain_type_t::prefix, get_case_fold()};
|
||||
}
|
||||
|
||||
// If anchor_start is set, this is as far as we go.
|
||||
|
@ -196,7 +205,7 @@ maybe_t<string_fuzzy_match_t> string_fuzzy_match_t::try_create(const wcstring &s
|
|||
|
||||
// substr icase
|
||||
if ((location = ifind(match_against, string, true /* fuzzy */)) != wcstring::npos) {
|
||||
return string_fuzzy_match_t{contain_type_t::substr, case_fold_t::icase};
|
||||
return string_fuzzy_match_t{contain_type_t::substr, get_case_fold()};
|
||||
}
|
||||
|
||||
// subseq samecase
|
||||
|
@ -212,10 +221,12 @@ uint32_t string_fuzzy_match_t::rank() const {
|
|||
// Combine our type and our case fold into a single number, such that better matches are
|
||||
// smaller. Treat 'exact' types the same as 'prefix' types; this is because we do not
|
||||
// prefer exact matches to prefix matches when presenting completions to the user.
|
||||
// Treat smartcase the same as samecase; see #3978.
|
||||
auto effective_type = (type == contain_type_t::exact ? contain_type_t::prefix : type);
|
||||
auto effective_case = (case_fold == case_fold_t::smartcase ? case_fold_t::samecase : case_fold);
|
||||
|
||||
// Type dominates fold.
|
||||
return static_cast<uint32_t>(effective_type) * 8 + static_cast<uint32_t>(case_fold);
|
||||
return static_cast<uint32_t>(effective_type) * 8 + static_cast<uint32_t>(effective_case);
|
||||
}
|
||||
|
||||
template <bool Fuzzy, typename T>
|
||||
|
|
|
@ -49,7 +49,8 @@ struct string_fuzzy_match_t {
|
|||
// The case-folding required for the match.
|
||||
enum class case_fold_t : uint8_t {
|
||||
samecase, // exact match: foobar matches foobar
|
||||
icase, // case insensitive: FoBaR matches foobar
|
||||
smartcase, // case insensitive match with lowercase input. foobar matches FoBar.
|
||||
icase, // case insensitive: FoBaR matches foobAr
|
||||
};
|
||||
case_fold_t case_fold;
|
||||
|
||||
|
|
Loading…
Reference in a new issue