Get rid of ncurses based input parsing in favor of custom code

This commit is contained in:
Maxime Coste 2019-09-08 17:19:41 +10:00
parent 6d085f995e
commit f2f99580f8
3 changed files with 108 additions and 204 deletions

View File

@ -457,8 +457,7 @@ void register_options()
" ncurses_wheel_up_button int\n" " ncurses_wheel_up_button int\n"
" ncurses_wheel_down_button int\n" " ncurses_wheel_down_button int\n"
" ncurses_wheel_scroll_amount int\n" " ncurses_wheel_scroll_amount int\n"
" ncurses_shift_function_key int\n" " ncurses_shift_function_key int\n",
" ncurses_builtin_key_parser bool\n",
UserInterface::Options{}); UserInterface::Options{});
reg.declare_option("modelinefmt", "format string used to generate the modeline", reg.declare_option("modelinefmt", "format string used to generate the modeline",
"%val{bufname} %val{cursor_line}:%val{cursor_char_column} {{context_info}} {{mode_info}} - %val{client}@[%val{session}]"_str); "%val{bufname} %val{cursor_line}:%val{cursor_char_column} {{context_info}} {{mode_info}} - %val{client}@[%val{session}]"_str);

View File

@ -144,16 +144,6 @@ void NCursesUI::Window::set_background_color(Palette& palette, Face face)
wbkgdset(win, COLOR_PAIR(palette.get_color_pair(face))); wbkgdset(win, COLOR_PAIR(palette.get_color_pair(face)));
} }
int NCursesUI::Window::get_char()
{
return wgetch(win);
}
void NCursesUI::Window::set_blocking(bool blocking)
{
wtimeout(win, blocking ? -1 : 0);
}
constexpr int NCursesUI::default_shift_function_key; constexpr int NCursesUI::default_shift_function_key;
static constexpr StringView assistant_cat[] = static constexpr StringView assistant_cat[] =
@ -385,6 +375,8 @@ NCursesUI::NCursesUI()
enable_mouse(true); enable_mouse(true);
fcntl(0, F_SETFL, fcntl(0, F_GETFL, 0) | O_NONBLOCK);
set_signal_handler(SIGWINCH, &signal_handler<&resize_pending>); set_signal_handler(SIGWINCH, &signal_handler<&resize_pending>);
set_signal_handler(SIGHUP, &signal_handler<&sighup_raised>); set_signal_handler(SIGHUP, &signal_handler<&sighup_raised>);
@ -552,7 +544,6 @@ void NCursesUI::check_resize(bool force)
m_window.create({0, 0}, {ws.ws_row, ws.ws_col}); m_window.create({0, 0}, {ws.ws_row, ws.ws_col});
kak_assert(m_window); kak_assert(m_window);
keypad(m_window.win, not m_builtin_key_parser);
m_dimensions = DisplayCoord{ws.ws_row-1, ws.ws_col}; m_dimensions = DisplayCoord{ws.ws_row-1, ws.ws_col};
@ -591,73 +582,18 @@ Optional<Key> NCursesUI::get_next_key()
return resize(dimensions()); return resize(dimensions());
} }
m_window.set_blocking(false); static auto get_char = []() -> Optional<unsigned char> {
const int c = m_window.get_char(); unsigned char c = 0;
m_window.set_blocking(true); if (read(0, &c, 1) == 1)
return c;
return {};
};
if (c == ERR) const auto c = get_char();
if (not c)
return {}; return {};
if (c == KEY_MOUSE) auto parse_key = [this](unsigned char c) -> Key {
{
MEVENT ev;
if (getmouse(&ev) == OK)
{
const auto mask = ev.bstate;
auto coord = encode_coord({ ev.y - content_line_offset(), ev.x });
Key::Modifiers mod{};
if (mask & BUTTON_CTRL)
mod |= Key::Modifiers::Control;
if (mask & BUTTON_ALT)
mod |= Key::Modifiers::Alt;
if (BUTTON_PRESS(mask, 1))
return Key{mod | Key::Modifiers::MousePressLeft, coord};
if (BUTTON_PRESS(mask, 3))
return Key{mod | Key::Modifiers::MousePressRight, coord};
if (BUTTON_RELEASE(mask, 1))
return Key{mod | Key::Modifiers::MouseReleaseLeft, coord};
if (BUTTON_RELEASE(mask, 3))
return Key{mod | Key::Modifiers::MouseReleaseRight, coord};
if (BUTTON_PRESS(mask, m_wheel_down_button))
return Key{mod | Key::Modifiers::Scroll, static_cast<Codepoint>(m_wheel_scroll_amount)};
if (BUTTON_PRESS(mask, m_wheel_up_button))
return Key{mod | Key::Modifiers::Scroll, static_cast<Codepoint>(-m_wheel_scroll_amount)};
return Key{mod | Key::Modifiers::MousePos, coord};
}
}
auto parse_key = [this](int c) -> Optional<Key> {
switch (c)
{
case KEY_BACKSPACE: case 127: return {Key::Backspace};
case KEY_DC: return {Key::Delete};
case KEY_SDC: return shift(Key::Delete);
case KEY_UP: return {Key::Up};
case KEY_SR: return shift(Key::Up);
case KEY_DOWN: return {Key::Down};
case KEY_SF: return shift(Key::Down);
case KEY_LEFT: return {Key::Left};
case KEY_SLEFT: return shift(Key::Left);
case KEY_RIGHT: return {Key::Right};
case KEY_SRIGHT: return shift(Key::Right);
case KEY_PPAGE: return {Key::PageUp};
case KEY_SPREVIOUS: return shift(Key::PageUp);
case KEY_NPAGE: return {Key::PageDown};
case KEY_SNEXT: return shift(Key::PageDown);
case KEY_HOME: return {Key::Home};
case KEY_SHOME: return shift(Key::Home);
case KEY_END: return {Key::End};
case KEY_SEND: return shift(Key::End);
case KEY_IC: return {Key::Insert};
case KEY_SIC: return shift(Key::Insert);
case KEY_BTAB: return shift(Key::Tab);
case KEY_RESIZE: return resize(dimensions());
}
if (c > 0 and c < 27)
{
if (c == control('m') or c == control('j')) if (c == control('m') or c == control('j'))
return {Key::Return}; return {Key::Return};
if (c == control('i')) if (c == control('i'))
@ -675,49 +611,32 @@ Optional<Key> NCursesUI::get_next_key()
enable_mouse(mouse_enabled); enable_mouse(mouse_enabled);
return {}; return {};
} }
if (c < 27)
return ctrl(Codepoint(c) - 1 + 'a'); return ctrl(Codepoint(c) - 1 + 'a');
} if (c == 127)
return {Key::Backspace};
for (int i = 0; i < 12; ++i)
{
if (c == KEY_F(i + 1))
return {Key::F1 + i};
if (c == KEY_F(m_shift_function_key + i + 1))
return shift(Key::F1 + i);
}
if (c >= 0 and c < 256)
{
ungetch(c);
struct getch_iterator struct getch_iterator
{ {
getch_iterator(Window& win) : window(win) {} unsigned char operator*() { return *c; }
int operator*() { return window.get_char(); } getch_iterator& operator++() { c = get_char(); return *this; }
getch_iterator& operator++() { return *this; } bool operator==(const getch_iterator&) const { return not c; }
getch_iterator& operator++(int) { return *this; } Optional<unsigned char> c;
bool operator== (const getch_iterator&) const { return false; }
Window& window;
}; };
return Key{utf8::codepoint(getch_iterator{m_window}, return Key{utf8::codepoint(getch_iterator{c}, getch_iterator{})};
getch_iterator{m_window})};
}
return {};
}; };
constexpr auto direction = make_array({Key::Up, Key::Down, Key::Right, Key::Left, Key::Home, Key::End}); auto parse_csi = [this]() -> Optional<Key> {
constexpr auto special = make_array({Key::Insert, Key::Delete, Key::NamedKey{}, Key::PageUp, Key::PageDown, Key::Home, Key::End, Key::NamedKey{}, Key::NamedKey{}, auto next_char = [] { return get_char().value_or((unsigned char)0xff); };
Key::F1, Key::F2, Key::F3, Key::F4, Key::NamedKey{}, Key::F5, Key::F6, Key::F7, Key::F8, Key::F9, Key::F10, Key::NamedKey{}, Key::F11, Key::F12});
auto parse_csi = [this, &direction, &special]() -> Optional<Key> {
int params[16] = {}; int params[16] = {};
int c = m_window.get_char(); auto c = next_char();
char private_mode = 0; char private_mode = 0;
if (c == '?' or c == '<' or c == '=' or c == '>') if (c == '?' or c == '<' or c == '=' or c == '>')
{ {
private_mode = c; private_mode = c;
c = m_window.get_char(); c = next_char();
} }
for (int count = 0; count < 16 and c >= 0x30 && c <= 0x3f; c = m_window.get_char()) for (int count = 0; count < 16 and c >= 0x30 && c <= 0x3f; c = next_char())
{ {
if ('0' <= 'c' and c <= '9') if ('0' <= 'c' and c <= '9')
params[count] = params[count] * 10 + c - '0'; params[count] = params[count] * 10 + c - '0';
@ -730,7 +649,6 @@ Optional<Key> NCursesUI::get_next_key()
return {}; return {};
auto parse_mask = [](int mask) { auto parse_mask = [](int mask) {
mask = std::max(0, mask - 1);
Key::Modifiers mod = Key::Modifiers::None; Key::Modifiers mod = Key::Modifiers::None;
if (mask & 1) if (mask & 1)
mod |= Key::Modifiers::Shift; mod |= Key::Modifiers::Shift;
@ -756,83 +674,88 @@ Optional<Key> NCursesUI::get_next_key()
return Key{mod, coord}; return Key{mod, coord};
}; };
if (c >= 'A' and c <= 'F') auto mouse_scroll = [this](Key::Modifiers mod, bool down) -> Key {
return Key{parse_mask(params[1]), direction[c - 'A']}; return {mod | Key::Modifiers::Scroll,
if (c == '~' and 2 <= params[0] and params[0] <= 24) (Codepoint)((down ? 1 : -1) * m_wheel_scroll_amount)};
return Key{parse_mask(params[1]), special[params[0] - 2]}; };
if (c == 'Z')
return shift(Key::Tab); auto masked_key = [&](Codepoint key) { return Key{parse_mask(std::max(params[1] - 1, 0)), key}; };
if (c == 'I')
return {Key::FocusIn}; switch (c)
if (c == 'O')
return {Key::FocusOut};
if ((c == 'M' or c == 'm') and private_mode == '<')
{ {
auto coord = encode_coord({params[2] - content_line_offset() - 1, params[1] - 1}); case 'A': return masked_key(Key::Up);
Key::Modifiers mod = parse_mask(1 + ((params[0] >> 2) & 0x7)); case 'B': return masked_key(Key::Down);
switch (params[0] & 0x43) case 'C': return masked_key(Key::Right);
case 'D': return masked_key(Key::Left);
case 'F': return masked_key(Key::End);
case 'H': return masked_key(Key::Home);
case 'P': return masked_key(Key::F1);
case 'Q': return masked_key(Key::F2);
case 'R': return masked_key(Key::F3);
case 'S': return masked_key(Key::F4);
case '~':
switch (params[0])
{ {
case 0: case 2: return masked_key(Key::Insert);
return mouse_button(mod, coord, true, c == 'm'); case 3: return masked_key(Key::Delete);
case 2: case 5: return masked_key(Key::PageUp);
return mouse_button(mod, coord, false, c == 'm'); case 6: return masked_key(Key::PageDown);
case 64: case 7: return masked_key(Key::Home);
return Key{mod | Key::Modifiers::Scroll, case 8: return masked_key(Key::End);
static_cast<Codepoint>(-m_wheel_scroll_amount)}; case 11: case 12: case 13: case 14: case 15:
case 65: return masked_key(Key::F1 + params[0] - 11);
return Key{mod | Key::Modifiers::Scroll, case 17: case 18: case 19: case 20: case 21:
static_cast<Codepoint>(m_wheel_scroll_amount)}; return masked_key(Key::F6 + params[0] - 17);
case 23: case 24:
return masked_key(Key::F11 + params[0] - 23);
} }
} return {};
if (c == 'M') case 'Z': return shift(Key::Tab);
{ case 'I': return {Key::FocusIn};
const Codepoint b = m_window.get_char() - 32; case 'O': return {Key::FocusOut};
const int x = m_window.get_char() - 32 - 1; case 'M': case 'm':
const int y = m_window.get_char() - 32 - 1; const bool sgr = private_mode == '<';
if (not sgr and c != 'M')
return {};
const Codepoint b = sgr ? params[0] : next_char() - 32;
const int x = (sgr ? params[1] : next_char() - 32) - 1;
const int y = (sgr ? params[2] : next_char() - 32) - 1;
auto coord = encode_coord({y - content_line_offset(), x}); auto coord = encode_coord({y - content_line_offset(), x});
Key::Modifiers mod = parse_mask(1 + ((b >> 2) & 0x7)); Key::Modifiers mod = parse_mask((b >> 2) & 0x7);
switch (b & 0x43) switch (b & 0x43)
{ {
case 0: case 0: return mouse_button(mod, coord, true, c == 'm');
return mouse_button(mod, coord, true, false); case 2: return mouse_button(mod, coord, false, c == 'm');
case 2:
return mouse_button(mod, coord, false, false);
case 3: case 3:
if (sgr)
return {};
if (m_mouse_state & 0x1) if (m_mouse_state & 0x1)
return mouse_button(mod, coord, true, true); return mouse_button(mod, coord, true, true);
else if (m_mouse_state & 0x2) else if (m_mouse_state & 0x2)
return mouse_button(mod, coord, false, true); return mouse_button(mod, coord, false, true);
break; break;
case 64: case 64: return mouse_scroll(mod, false);
return Key{mod | Key::Modifiers::Scroll, case 65: return mouse_scroll(mod, true);
static_cast<Codepoint>(-m_wheel_scroll_amount)};
case 65:
return Key{mod | Key::Modifiers::Scroll,
static_cast<Codepoint>(m_wheel_scroll_amount)};
} }
return Key{Key::Modifiers::MousePos, coord}; return Key{Key::Modifiers::MousePos, coord};
} }
return {}; return {};
}; };
if (c == 27) if (*c != 27)
return parse_key(*c);
if (auto next = get_char())
{ {
m_window.set_blocking(false); if (*next == '[') // potential CSI
const int new_c = m_window.get_char();
if (new_c == '[') // potential CSI
{ {
if (auto key = parse_csi()) if (auto key = parse_csi())
return key; return key;
} }
m_window.set_blocking(true); return alt(parse_key(*next));
if (auto key = parse_key(new_c))
return alt(*key);
else
return {Key::Escape};
} }
return Key{Key::Escape};
return parse_key(c);
} }
template<typename T> template<typename T>
@ -1300,10 +1223,7 @@ void NCursesUI::enable_mouse(bool enabled)
m_mouse_enabled = enabled; m_mouse_enabled = enabled;
if (enabled) if (enabled)
{ {
mousemask(ALL_MOUSE_EVENTS | REPORT_MOUSE_POSITION, nullptr);
mouseinterval(0);
// force SGR mode // force SGR mode
if (m_builtin_key_parser)
fputs("\033[?1006h", stdout); fputs("\033[?1006h", stdout);
// force enable report focus events // force enable report focus events
fputs("\033[?1004h", stdout); fputs("\033[?1004h", stdout);
@ -1314,7 +1234,6 @@ void NCursesUI::enable_mouse(bool enabled)
} }
else else
{ {
mousemask(0, nullptr);
fputs("\033[?1002l", stdout); fputs("\033[?1002l", stdout);
fputs("\033[?1000l", stdout); fputs("\033[?1000l", stdout);
fputs("\033[?1004l", stdout); fputs("\033[?1004l", stdout);
@ -1385,17 +1304,6 @@ void NCursesUI::set_ui_options(const Options& options)
m_wheel_scroll_amount = wheel_scroll_amount_it != options.end() ? m_wheel_scroll_amount = wheel_scroll_amount_it != options.end() ?
str_to_int_ifp(wheel_scroll_amount_it->value).value_or(3) : 3; str_to_int_ifp(wheel_scroll_amount_it->value).value_or(3) : 3;
} }
{
auto builtin_key_parser_it = options.find("ncurses_builtin_key_parser"_sv);
bool mouse_enabled = m_mouse_enabled;
enable_mouse(false);
m_builtin_key_parser = builtin_key_parser_it != options.end() and
(builtin_key_parser_it->value == "yes" or
builtin_key_parser_it->value == "true");
keypad(m_window.win, not m_builtin_key_parser);
enable_mouse(mouse_enabled);
}
} }
} }

View File

@ -105,8 +105,6 @@ private:
void clear_to_end_of_line(); void clear_to_end_of_line();
void set_face(Palette& palette, Face face, const Face& default_face); void set_face(Palette& palette, Face face, const Face& default_face);
void set_background_color(Palette& palette, Face face); void set_background_color(Palette& palette, Face face);
int get_char();
void set_blocking(bool blocking);
explicit operator bool() const { return win; } explicit operator bool() const { return win; }
@ -169,7 +167,6 @@ private:
int m_shift_function_key = default_shift_function_key; int m_shift_function_key = default_shift_function_key;
bool m_set_title = true; bool m_set_title = true;
bool m_builtin_key_parser = false;
bool m_dirty = false; bool m_dirty = false;