Rework mode handling of bind

Binds with the same sequence in multiple modes was not working right.
Fix up the implementation to propagate modes everywhere as necessary.
This means that `bind` will properly list distinct binds with the same
sequence, and `bind -e` will take mode into account properly as well.
Note that `bind -e seq` now assumes the bind is in the default bind
mode, whereas before it would erase the first binding with that sequence
regardless of mode.

`bind -e -a` still erases all binds in all modes, though `bind -M mode
-e -a` still only erases all binds in the selected mode.
This commit is contained in:
Kevin Ballard 2014-09-22 21:04:06 -07:00
parent 5eee7d17f6
commit 0a3f220572
4 changed files with 102 additions and 85 deletions

View file

@ -399,71 +399,80 @@ static void builtin_missing_argument(parser_t &parser, const wchar_t *cmd, const
/* builtin_test lives in builtin_test.cpp */ /* builtin_test lives in builtin_test.cpp */
int builtin_test(parser_t &parser, wchar_t **argv); int builtin_test(parser_t &parser, wchar_t **argv);
/**
List a single key binding.
Returns false if no binding with that sequence and mode exists.
*/
static bool builtin_bind_list_one(const wcstring& seq, const wcstring& bind_mode)
{
std::vector<wcstring> ecmds;
wcstring sets_mode;
if (!input_mapping_get(seq, bind_mode, &ecmds, &sets_mode))
{
return false;
}
stdout_buffer.append(L"bind");
// Append the mode flags if applicable
if (bind_mode != DEFAULT_BIND_MODE)
{
const wcstring emode = escape_string(bind_mode, ESCAPE_ALL);
stdout_buffer.append(L" -M ");
stdout_buffer.append(emode);
}
if (sets_mode != bind_mode)
{
const wcstring esets_mode = escape_string(sets_mode, ESCAPE_ALL);
stdout_buffer.append(L" -m ");
stdout_buffer.append(esets_mode);
}
// Append the name
wcstring tname;
if (input_terminfo_get_name(seq, &tname))
{
// Note that we show -k here because we have an input key name
append_format(stdout_buffer, L" -k %ls", tname.c_str());
}
else
{
// No key name, so no -k; we show the escape sequence directly
const wcstring eseq = escape_string(seq, ESCAPE_ALL);
append_format(stdout_buffer, L" %ls", eseq.c_str());
}
// Now show the list of commands
for (size_t i = 0; i < ecmds.size(); i++)
{
const wcstring &ecmd = ecmds.at(i);
const wcstring escaped_ecmd = escape_string(ecmd, ESCAPE_ALL);
stdout_buffer.push_back(' ');
stdout_buffer.append(escaped_ecmd);
}
stdout_buffer.push_back(L'\n');
return true;
}
/** /**
List all current key bindings List all current key bindings
*/ */
static void builtin_bind_list(const wchar_t *bind_mode) static void builtin_bind_list(const wchar_t *bind_mode)
{ {
size_t i; const std::vector<input_mapping_name_t> lst = input_mapping_get_names();
const wcstring_list_t lst = input_mapping_get_names();
for (i=0; i<lst.size(); i++) for (std::vector<input_mapping_name_t>::const_iterator it = lst.begin(), end = lst.end();
it != end;
++it)
{ {
wcstring seq = lst.at(i); if (bind_mode != NULL && bind_mode != it->mode)
std::vector<wcstring> ecmds;
wcstring mode;
wcstring sets_mode;
if (! input_mapping_get(seq, &ecmds, &mode, &sets_mode))
{ {
continue; continue;
} }
if (bind_mode != NULL && bind_mode != mode) builtin_bind_list_one(it->seq, it->mode);
{
continue;
}
stdout_buffer.append(L"bind");
// Append the mode flags if applicable
if (mode != DEFAULT_BIND_MODE)
{
const wcstring emode = escape_string(mode, ESCAPE_ALL);
stdout_buffer.append(L" -M ");
stdout_buffer.append(emode);
}
if (sets_mode != mode)
{
const wcstring esets_mode = escape_string(sets_mode, ESCAPE_ALL);
stdout_buffer.append(L" -m ");
stdout_buffer.append(esets_mode);
}
// Append the name
wcstring tname;
if (input_terminfo_get_name(seq, &tname))
{
// Note that we show -k here because we have an input key name
append_format(stdout_buffer, L" -k %ls", tname.c_str());
}
else
{
// No key name, so no -k; we show the escape sequence directly
const wcstring eseq = escape_string(seq, ESCAPE_ALL);
append_format(stdout_buffer, L" %ls", eseq.c_str());
}
// Now show the list of commands
for (size_t i = 0; i < ecmds.size(); i++)
{
const wcstring &ecmd = ecmds.at(i);
const wcstring escaped_ecmd = escape_string(ecmd, ESCAPE_ALL);
stdout_buffer.push_back(' ');
stdout_buffer.append(escaped_ecmd);
}
stdout_buffer.push_back(L'\n');
} }
} }
@ -564,15 +573,21 @@ static int builtin_bind_add(const wchar_t *seq, const wchar_t **cmds, size_t cmd
\param seq an array of all key bindings to erase \param seq an array of all key bindings to erase
\param all if specified, _all_ key bindings will be erased \param all if specified, _all_ key bindings will be erased
\param mode if specified, only bindings from that mode will be erased. If not given and \c all is \c false, \c DEFAULT_BIND_MODE will be used.
*/ */
static int builtin_bind_erase(wchar_t **seq, int all, const wchar_t *mode, int use_terminfo) static int builtin_bind_erase(wchar_t **seq, int all, const wchar_t *mode, int use_terminfo)
{ {
if (all) if (all)
{ {
const wcstring_list_t lst = input_mapping_get_names(); const std::vector<input_mapping_name_t> lst = input_mapping_get_names();
for (size_t i=0; i<lst.size(); i++) for (std::vector<input_mapping_name_t>::const_iterator it = lst.begin(), end = lst.end();
it != end;
++it)
{ {
input_mapping_erase(lst.at(i).c_str(), mode); if (mode == NULL || mode == it->mode)
{
input_mapping_erase(it->seq, it->mode);
}
} }
return 0; return 0;
@ -581,6 +596,8 @@ static int builtin_bind_erase(wchar_t **seq, int all, const wchar_t *mode, int u
{ {
int res = 0; int res = 0;
if (mode == NULL) mode = DEFAULT_BIND_MODE;
while (*seq) while (*seq)
{ {
if (use_terminfo) if (use_terminfo)

View file

@ -43,7 +43,7 @@ The following parameters are available:
- `-m NEW_MODE` or `--sets-mode NEW_MODE` Change the current mode to `NEW_MODE` after this binding is executed - `-m NEW_MODE` or `--sets-mode NEW_MODE` Change the current mode to `NEW_MODE` after this binding is executed
- `-e` or `--erase` Erase the binding with the given sequence and mode instead of defining a new one. Multiple sequences can be specified with this flag. Specifying `-a` or `--all` erases all binds in this mode regardless of sequence. - `-e` or `--erase` Erase the binding with the given sequence and mode instead of defining a new one. Multiple sequences can be specified with this flag. Specifying `-a` or `--all` with `-M` or `--mode` erases all binds in the given mode regardless of sequence. Specifying `-a` or `--all` without `-M` or `--mode` erases all binds in all modes regardless of sequence.
- `-a` or `--all` See `--erase` and `--key-names` - `-a` or `--all` See `--erase` and `--key-names`

View file

@ -772,59 +772,53 @@ wint_t input_readch(bool allow_commands)
} }
} }
wcstring_list_t input_mapping_get_names() std::vector<input_mapping_name_t> input_mapping_get_names()
{ {
// Sort the mappings by the user specification order, so we can return them in the same order that the user specified them in // Sort the mappings by the user specification order, so we can return them in the same order that the user specified them in
std::vector<input_mapping_t> local_list = mapping_list; std::vector<input_mapping_t> local_list = mapping_list;
std::sort(local_list.begin(), local_list.end(), specification_order_is_less_than); std::sort(local_list.begin(), local_list.end(), specification_order_is_less_than);
wcstring_list_t result; std::vector<input_mapping_name_t> result;
result.reserve(local_list.size()); result.reserve(local_list.size());
for (size_t i=0; i<local_list.size(); i++) for (size_t i=0; i<local_list.size(); i++)
{ {
const input_mapping_t &m = local_list.at(i); const input_mapping_t &m = local_list.at(i);
result.push_back(m.seq); result.push_back((input_mapping_name_t){m.seq, m.mode});
} }
return result; return result;
} }
bool input_mapping_erase(const wchar_t *sequence, const wchar_t *mode) bool input_mapping_erase(const wcstring &sequence, const wcstring &mode)
{ {
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
bool result = false; bool result = false;
size_t i, sz = mapping_list.size();
for (i=0; i<sz; i++) for (std::vector<input_mapping_t>::const_iterator it = mapping_list.begin(), end = mapping_list.end();
it != end;
++it)
{ {
const input_mapping_t &m = mapping_list.at(i); if (sequence == it->seq && mode == it->mode)
if (sequence == m.seq && (mode == NULL || mode == m.mode))
{ {
if (i != (sz-1)) mapping_list.erase(it);
{
mapping_list[i] = mapping_list[sz-1];
}
mapping_list.pop_back();
result = true; result = true;
break; break;
} }
} }
return result; return result;
} }
bool input_mapping_get(const wcstring &sequence, wcstring_list_t *out_cmds, wcstring *out_mode, wcstring *out_sets_mode) bool input_mapping_get(const wcstring &sequence, const wcstring &mode, wcstring_list_t *out_cmds, wcstring *out_sets_mode)
{ {
bool result = false; bool result = false;
size_t sz = mapping_list.size(); for (std::vector<input_mapping_t>::const_iterator it = mapping_list.begin(), end = mapping_list.end();
for (size_t i=0; i<sz; i++) it != end;
++it)
{ {
const input_mapping_t &m = mapping_list.at(i); if (sequence == it->seq && mode == it->mode)
if (sequence == m.seq)
{ {
*out_cmds = m.commands; *out_cmds = it->commands;
*out_mode = m.mode; *out_sets_mode = it->sets_mode;
*out_sets_mode = m.sets_mode;
result = true; result = true;
break; break;
} }

16
input.h
View file

@ -9,6 +9,7 @@ inputrc information for key bindings.
#define FISH_INPUT_H #define FISH_INPUT_H
#include <wchar.h> #include <wchar.h>
#include <utility>
#include "input_common.h" #include "input_common.h"
@ -131,20 +132,25 @@ void input_mapping_add(const wchar_t *sequence, const wchar_t *command,
void input_mapping_add(const wchar_t *sequence, const wchar_t **commands, size_t commands_len, void input_mapping_add(const wchar_t *sequence, const wchar_t **commands, size_t commands_len,
const wchar_t *mode = DEFAULT_BIND_MODE, const wchar_t *new_mode = DEFAULT_BIND_MODE); const wchar_t *mode = DEFAULT_BIND_MODE, const wchar_t *new_mode = DEFAULT_BIND_MODE);
struct input_mapping_name_t {
wcstring seq;
wcstring mode;
};
/** /**
Returns all mapping names Returns all mapping names and modes
*/ */
wcstring_list_t input_mapping_get_names(); std::vector<input_mapping_name_t> input_mapping_get_names();
/** /**
Erase binding for specified key sequence Erase binding for specified key sequence
*/ */
bool input_mapping_erase(const wchar_t *sequence, const wchar_t *mode = DEFAULT_BIND_MODE); bool input_mapping_erase(const wcstring &sequence, const wcstring &mode = DEFAULT_BIND_MODE);
/** /**
Gets the command bound to the specified key sequence. Returns true if it exists, false if not. Gets the command bound to the specified key sequence in the specified mode. Returns true if it exists, false if not.
*/ */
bool input_mapping_get(const wcstring &sequence, wcstring_list_t *out_cmds, wcstring *out_mode, wcstring *out_new_mode); bool input_mapping_get(const wcstring &sequence, const wcstring &mode, wcstring_list_t *out_cmds, wcstring *out_new_mode);
/** /**
Return the current bind mode Return the current bind mode