mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-16 06:54:03 +00:00
Use Unicode symbols for rendering control characters in pager
The history pager will show multiline commands in single-line cells. We escape newline characters as \\n but that looks awkward if the next line starts with a letter. Let's render control characters using their corresponding symbol from the Control Pictures Unicode block. This means there is also no need to escape backslashes, which further improves the history pager - now the rendering has exactly as many backslashes as the eventual command. This means that (multiline) commands in the history pager will be rendered with the same amount of characters as are in the actual command (unless they contain funny nonprintables). This makes it easy for the next commit to highlight multiline commands correctly in the history pager. The font size for these symbols (for example ␉) is quite small, but that's okay since for the proposed uses it's not so important that they readable. The important thing is that the stand out from surrounding text.
This commit is contained in:
parent
02fcc50b9a
commit
b64cec1d7e
3 changed files with 37 additions and 18 deletions
|
@ -883,6 +883,8 @@ static void escape_string_script(const wchar_t *orig_in, size_t in_len, wcstring
|
||||||
const bool no_quoted = static_cast<bool>(flags & ESCAPE_NO_QUOTED);
|
const bool no_quoted = static_cast<bool>(flags & ESCAPE_NO_QUOTED);
|
||||||
const bool no_tilde = static_cast<bool>(flags & ESCAPE_NO_TILDE);
|
const bool no_tilde = static_cast<bool>(flags & ESCAPE_NO_TILDE);
|
||||||
const bool no_qmark = feature_test(features_t::qmark_noglob);
|
const bool no_qmark = feature_test(features_t::qmark_noglob);
|
||||||
|
const bool symbolic = static_cast<bool>(flags & ESCAPE_SYMBOLIC) && (MB_CUR_MAX > 1);
|
||||||
|
assert((!symbolic || !escape_printables) && "symbolic implies escape-no-printables");
|
||||||
|
|
||||||
bool need_escape = false;
|
bool need_escape = false;
|
||||||
bool need_complex_escape = false;
|
bool need_complex_escape = false;
|
||||||
|
@ -911,47 +913,57 @@ static void escape_string_script(const wchar_t *orig_in, size_t in_len, wcstring
|
||||||
wchar_t c = *in;
|
wchar_t c = *in;
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case L'\t': {
|
case L'\t': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L't';
|
out += L'␉';
|
||||||
|
else
|
||||||
|
out += L"\\t";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\n': {
|
case L'\n': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L'n';
|
out += L'';
|
||||||
|
else
|
||||||
|
out += L"\\n";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\b': {
|
case L'\b': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L'b';
|
out += L'␈';
|
||||||
|
else
|
||||||
|
out += L"\\b";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\r': {
|
case L'\r': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L'r';
|
out += L'␍';
|
||||||
|
else
|
||||||
|
out += L"\\r";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\x1B': {
|
case L'\x1B': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L'e';
|
out += L'␛';
|
||||||
|
else
|
||||||
|
out += L"\\e";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\x7F': {
|
case L'\x7F': {
|
||||||
out += L'\\';
|
if (symbolic)
|
||||||
out += L'x';
|
out += L'␡';
|
||||||
out += L'7';
|
else
|
||||||
out += L'f';
|
out += L"\\x7f";
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case L'\\':
|
case L'\\':
|
||||||
case L'\'': {
|
case L'\'': {
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
if (escape_printables || c == L'\\') out += L'\\';
|
if (escape_printables || (c == L'\\' && !symbolic)) out += L'\\';
|
||||||
out += *in;
|
out += *in;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1001,6 +1013,11 @@ static void escape_string_script(const wchar_t *orig_in, size_t in_len, wcstring
|
||||||
if (*in >= 0 && *in < 32) {
|
if (*in >= 0 && *in < 32) {
|
||||||
need_escape = need_complex_escape = true;
|
need_escape = need_complex_escape = true;
|
||||||
|
|
||||||
|
if (symbolic) {
|
||||||
|
out += L'\u2400' + *in;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if (*in < 27 && *in != 0) {
|
if (*in < 27 && *in != 0) {
|
||||||
out += L'\\';
|
out += L'\\';
|
||||||
out += L'c';
|
out += L'c';
|
||||||
|
|
|
@ -148,7 +148,9 @@ enum {
|
||||||
/// string.
|
/// string.
|
||||||
ESCAPE_NO_QUOTED = 1 << 1,
|
ESCAPE_NO_QUOTED = 1 << 1,
|
||||||
/// Do not escape tildes.
|
/// Do not escape tildes.
|
||||||
ESCAPE_NO_TILDE = 1 << 2
|
ESCAPE_NO_TILDE = 1 << 2,
|
||||||
|
/// Replace nonprintable control characters with Unicode symbols.
|
||||||
|
ESCAPE_SYMBOLIC = 1 << 3
|
||||||
};
|
};
|
||||||
typedef unsigned int escape_flags_t;
|
typedef unsigned int escape_flags_t;
|
||||||
|
|
||||||
|
|
|
@ -313,8 +313,8 @@ static comp_info_list_t process_completions_into_infos(const completion_list_t &
|
||||||
comp_t *comp_info = &result.at(i);
|
comp_t *comp_info = &result.at(i);
|
||||||
|
|
||||||
// Append the single completion string. We may later merge these into multiple.
|
// Append the single completion string. We may later merge these into multiple.
|
||||||
comp_info->comp.push_back(
|
comp_info->comp.push_back(escape_string(
|
||||||
escape_string(comp.completion, ESCAPE_NO_PRINTABLES | ESCAPE_NO_QUOTED));
|
comp.completion, ESCAPE_NO_PRINTABLES | ESCAPE_NO_QUOTED | ESCAPE_SYMBOLIC));
|
||||||
|
|
||||||
// Append the mangled description.
|
// Append the mangled description.
|
||||||
comp_info->desc = comp.description;
|
comp_info->desc = comp.description;
|
||||||
|
|
Loading…
Reference in a new issue