Get rid of ncurses based input parsing in favor of custom code
This commit is contained in:
parent
6d085f995e
commit
f2f99580f8
|
@ -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);
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user