mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-19 08:24:00 +00:00
builtin bind: highlight output.
This highlights `bind` output, which is commands to reproduce the current bind state, for interactive sessions ala builtin complete.
This commit is contained in:
parent
8621852ec5
commit
864bd4a9cb
1 changed files with 42 additions and 28 deletions
|
@ -3,18 +3,22 @@
|
||||||
|
|
||||||
#include "bind.h"
|
#include "bind.h"
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
#include <cerrno>
|
#include <cerrno>
|
||||||
#include <cstddef>
|
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "../builtin.h"
|
#include "../builtin.h"
|
||||||
#include "../common.h"
|
#include "../common.h"
|
||||||
|
#include "../env.h"
|
||||||
#include "../fallback.h" // IWYU pragma: keep
|
#include "../fallback.h" // IWYU pragma: keep
|
||||||
|
#include "../highlight.h"
|
||||||
#include "../input.h"
|
#include "../input.h"
|
||||||
#include "../io.h"
|
#include "../io.h"
|
||||||
#include "../maybe.h"
|
#include "../maybe.h"
|
||||||
|
#include "../parser.h"
|
||||||
#include "../wgetopt.h"
|
#include "../wgetopt.h"
|
||||||
#include "../wutil.h" // IWYU pragma: keep
|
#include "../wutil.h" // IWYU pragma: keep
|
||||||
|
|
||||||
|
@ -50,7 +54,7 @@ class builtin_bind_t {
|
||||||
/// lock again.
|
/// lock again.
|
||||||
acquired_lock<input_mapping_set_t> input_mappings_;
|
acquired_lock<input_mapping_set_t> input_mappings_;
|
||||||
|
|
||||||
void list(const wchar_t *bind_mode, bool user, io_streams_t &streams);
|
void list(const wchar_t *bind_mode, bool user, parser_t &parser, io_streams_t &streams);
|
||||||
void key_names(bool all, io_streams_t &streams);
|
void key_names(bool all, io_streams_t &streams);
|
||||||
void function_names(io_streams_t &streams);
|
void function_names(io_streams_t &streams);
|
||||||
bool add(const wcstring &seq, const wchar_t *const *cmds, size_t cmds_len, const wchar_t *mode,
|
bool add(const wcstring &seq, const wchar_t *const *cmds, size_t cmds_len, const wchar_t *mode,
|
||||||
|
@ -58,56 +62,66 @@ class builtin_bind_t {
|
||||||
bool erase(const wchar_t *const *seq, bool all, const wchar_t *mode, bool use_terminfo,
|
bool erase(const wchar_t *const *seq, bool all, const wchar_t *mode, bool use_terminfo,
|
||||||
bool user, io_streams_t &streams);
|
bool user, io_streams_t &streams);
|
||||||
bool get_terminfo_sequence(const wcstring &seq, wcstring *out_seq, io_streams_t &streams) const;
|
bool get_terminfo_sequence(const wcstring &seq, wcstring *out_seq, io_streams_t &streams) const;
|
||||||
bool insert(int optind, int argc, const wchar_t **argv, io_streams_t &streams);
|
bool insert(int optind, int argc, const wchar_t **argv, parser_t &parser, io_streams_t &streams);
|
||||||
void list_modes(io_streams_t &streams);
|
void list_modes(io_streams_t &streams);
|
||||||
bool list_one(const wcstring &seq, const wcstring &bind_mode, bool user, io_streams_t &streams);
|
bool list_one(const wcstring &seq, const wcstring &bind_mode, bool user, parser_t &parser, io_streams_t &streams);
|
||||||
bool list_one(const wcstring &seq, const wcstring &bind_mode, bool user, bool preset,
|
bool list_one(const wcstring &seq, const wcstring &bind_mode, bool user, bool preset, parser_t &parser,
|
||||||
io_streams_t &streams);
|
io_streams_t &streams);
|
||||||
};
|
};
|
||||||
|
|
||||||
/// List a single key binding.
|
/// List a single key binding.
|
||||||
/// Returns false if no binding with that sequence and mode exists.
|
/// Returns false if no binding with that sequence and mode exists.
|
||||||
bool builtin_bind_t::list_one(const wcstring &seq, const wcstring &bind_mode, bool user,
|
bool builtin_bind_t::list_one(const wcstring &seq, const wcstring &bind_mode, bool user,
|
||||||
io_streams_t &streams) {
|
parser_t &parser, io_streams_t &streams) {
|
||||||
wcstring_list_t ecmds;
|
wcstring_list_t ecmds;
|
||||||
wcstring sets_mode;
|
wcstring sets_mode, out;
|
||||||
|
|
||||||
if (!input_mappings_->get(seq, bind_mode, &ecmds, user, &sets_mode)) {
|
if (!input_mappings_->get(seq, bind_mode, &ecmds, user, &sets_mode)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
streams.out.append(L"bind");
|
out.append(L"bind");
|
||||||
|
|
||||||
// Append the mode flags if applicable.
|
// Append the mode flags if applicable.
|
||||||
if (!user) {
|
if (!user) {
|
||||||
streams.out.append(L" --preset");
|
out.append(L" --preset");
|
||||||
}
|
}
|
||||||
if (bind_mode != DEFAULT_BIND_MODE) {
|
if (bind_mode != DEFAULT_BIND_MODE) {
|
||||||
streams.out.append(L" -M ");
|
out.append(L" -M ");
|
||||||
streams.out.append(escape_string(bind_mode));
|
out.append(escape_string(bind_mode));
|
||||||
}
|
}
|
||||||
if (!sets_mode.empty() && sets_mode != bind_mode) {
|
if (!sets_mode.empty() && sets_mode != bind_mode) {
|
||||||
streams.out.append(L" -m ");
|
out.append(L" -m ");
|
||||||
streams.out.append(escape_string(sets_mode));
|
out.append(escape_string(sets_mode));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append the name.
|
// Append the name.
|
||||||
wcstring tname;
|
wcstring tname;
|
||||||
if (input_terminfo_get_name(seq, &tname)) {
|
if (input_terminfo_get_name(seq, &tname)) {
|
||||||
// Note that we show -k here because we have an input key name.
|
// Note that we show -k here because we have an input key name.
|
||||||
streams.out.append_format(L" -k %ls", tname.c_str());
|
out.append(L" -k ");
|
||||||
|
out.append(tname);
|
||||||
} else {
|
} else {
|
||||||
// No key name, so no -k; we show the escape sequence directly.
|
// No key name, so no -k; we show the escape sequence directly.
|
||||||
const wcstring eseq = escape_string(seq);
|
const wcstring eseq = escape_string(seq);
|
||||||
streams.out.append_format(L" %ls", eseq.c_str());
|
out.append(L" ");
|
||||||
|
out.append(eseq);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now show the list of commands.
|
// Now show the list of commands.
|
||||||
for (const auto &ecmd : ecmds) {
|
for (const auto &ecmd : ecmds) {
|
||||||
streams.out.push_back(' ');
|
out.push_back(' ');
|
||||||
streams.out.append(escape_string(ecmd));
|
out.append(escape_string(ecmd));
|
||||||
|
}
|
||||||
|
out.push_back(L'\n');
|
||||||
|
|
||||||
|
if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) {
|
||||||
|
std::vector<highlight_spec_t> colors;
|
||||||
|
highlight_shell(out, colors, parser.context());
|
||||||
|
streams.out.append(str2wcstring(colorize(out, colors, parser.vars())));
|
||||||
|
} else {
|
||||||
|
streams.out.append(out);
|
||||||
}
|
}
|
||||||
streams.out.push_back(L'\n');
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -115,19 +129,19 @@ bool builtin_bind_t::list_one(const wcstring &seq, const wcstring &bind_mode, bo
|
||||||
// Overload with both kinds of bindings.
|
// Overload with both kinds of bindings.
|
||||||
// Returns false only if neither exists.
|
// Returns false only if neither exists.
|
||||||
bool builtin_bind_t::list_one(const wcstring &seq, const wcstring &bind_mode, bool user,
|
bool builtin_bind_t::list_one(const wcstring &seq, const wcstring &bind_mode, bool user,
|
||||||
bool preset, io_streams_t &streams) {
|
bool preset, parser_t &parser, io_streams_t &streams) {
|
||||||
bool retval = false;
|
bool retval = false;
|
||||||
if (preset) {
|
if (preset) {
|
||||||
retval |= list_one(seq, bind_mode, false, streams);
|
retval |= list_one(seq, bind_mode, false, parser, streams);
|
||||||
}
|
}
|
||||||
if (user) {
|
if (user) {
|
||||||
retval |= list_one(seq, bind_mode, true, streams);
|
retval |= list_one(seq, bind_mode, true, parser, streams);
|
||||||
}
|
}
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// List all current key bindings.
|
/// List all current key bindings.
|
||||||
void builtin_bind_t::list(const wchar_t *bind_mode, bool user, io_streams_t &streams) {
|
void builtin_bind_t::list(const wchar_t *bind_mode, bool user, parser_t &parser, io_streams_t &streams) {
|
||||||
const std::vector<input_mapping_name_t> lst = input_mappings_->get_names(user);
|
const std::vector<input_mapping_name_t> lst = input_mappings_->get_names(user);
|
||||||
|
|
||||||
for (const input_mapping_name_t &binding : lst) {
|
for (const input_mapping_name_t &binding : lst) {
|
||||||
|
@ -135,7 +149,7 @@ void builtin_bind_t::list(const wchar_t *bind_mode, bool user, io_streams_t &str
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_one(binding.seq, binding.mode, user, streams);
|
list_one(binding.seq, binding.mode, user, parser, streams);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -241,7 +255,7 @@ bool builtin_bind_t::erase(const wchar_t *const *seq, bool all, const wchar_t *m
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool builtin_bind_t::insert(int optind, int argc, const wchar_t **argv, io_streams_t &streams) {
|
bool builtin_bind_t::insert(int optind, int argc, const wchar_t **argv, parser_t &parser, io_streams_t &streams) {
|
||||||
const wchar_t *cmd = argv[0];
|
const wchar_t *cmd = argv[0];
|
||||||
int arg_count = argc - optind;
|
int arg_count = argc - optind;
|
||||||
|
|
||||||
|
@ -263,10 +277,10 @@ bool builtin_bind_t::insert(int optind, int argc, const wchar_t **argv, io_strea
|
||||||
// We don't overload this with user and def because we want them to be grouped.
|
// We don't overload this with user and def because we want them to be grouped.
|
||||||
// First the presets, then the users (because of scrolling).
|
// First the presets, then the users (because of scrolling).
|
||||||
if (opts->preset) {
|
if (opts->preset) {
|
||||||
list(opts->bind_mode_given ? opts->bind_mode : nullptr, false, streams);
|
list(opts->bind_mode_given ? opts->bind_mode : nullptr, false, parser, streams);
|
||||||
}
|
}
|
||||||
if (opts->user) {
|
if (opts->user) {
|
||||||
list(opts->bind_mode_given ? opts->bind_mode : nullptr, true, streams);
|
list(opts->bind_mode_given ? opts->bind_mode : nullptr, true, parser, streams);
|
||||||
}
|
}
|
||||||
} else if (arg_count == 1) {
|
} else if (arg_count == 1) {
|
||||||
wcstring seq;
|
wcstring seq;
|
||||||
|
@ -279,7 +293,7 @@ bool builtin_bind_t::insert(int optind, int argc, const wchar_t **argv, io_strea
|
||||||
seq = argv[optind];
|
seq = argv[optind];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!list_one(seq, opts->bind_mode, opts->user, opts->preset, streams)) {
|
if (!list_one(seq, opts->bind_mode, opts->user, opts->preset, parser, streams)) {
|
||||||
wcstring eseq = escape_string(argv[optind], ESCAPE_NO_PRINTABLES);
|
wcstring eseq = escape_string(argv[optind], ESCAPE_NO_PRINTABLES);
|
||||||
if (!opts->silent) {
|
if (!opts->silent) {
|
||||||
if (opts->use_terminfo) {
|
if (opts->use_terminfo) {
|
||||||
|
@ -467,7 +481,7 @@ maybe_t<int> builtin_bind_t::builtin_bind(parser_t &parser, io_streams_t &stream
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case BIND_INSERT: {
|
case BIND_INSERT: {
|
||||||
if (insert(optind, argc, argv, streams)) {
|
if (insert(optind, argc, argv, parser, streams)) {
|
||||||
return STATUS_CMD_ERROR;
|
return STATUS_CMD_ERROR;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
Loading…
Reference in a new issue