mirror of
https://github.com/neovim/neovim.git
synced 2025-09-09 21:08:17 +00:00
451 lines
11 KiB
C
451 lines
11 KiB
C
// This is an open source non-commercial project. Dear PVS-Studio, please check
|
|
// it. PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
|
|
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
#include <stdbool.h>
|
|
|
|
#include <uv.h>
|
|
|
|
#include "nvim/api/private/defs.h"
|
|
#include "nvim/os/input.h"
|
|
#include "nvim/event/loop.h"
|
|
#include "nvim/event/rstream.h"
|
|
#include "nvim/ascii.h"
|
|
#include "nvim/vim.h"
|
|
#include "nvim/ui.h"
|
|
#include "nvim/memory.h"
|
|
#include "nvim/keymap.h"
|
|
#include "nvim/mbyte.h"
|
|
#include "nvim/fileio.h"
|
|
#include "nvim/ex_cmds2.h"
|
|
#include "nvim/getchar.h"
|
|
#include "nvim/main.h"
|
|
#include "nvim/misc1.h"
|
|
#include "nvim/state.h"
|
|
#include "nvim/msgpack_rpc/channel.h"
|
|
|
|
#define READ_BUFFER_SIZE 0xfff
|
|
#define INPUT_BUFFER_SIZE (READ_BUFFER_SIZE * 4)
|
|
|
|
typedef enum {
|
|
kInputNone,
|
|
kInputAvail,
|
|
kInputEof
|
|
} InbufPollResult;
|
|
|
|
static Stream read_stream = {.closed = true};
|
|
static RBuffer *input_buffer = NULL;
|
|
static bool input_eof = false;
|
|
static int global_fd = -1;
|
|
static int events_enabled = 0;
|
|
static bool blocking = false;
|
|
|
|
#ifdef INCLUDE_GENERATED_DECLARATIONS
|
|
# include "os/input.c.generated.h"
|
|
#endif
|
|
|
|
void input_init(void)
|
|
{
|
|
input_buffer = rbuffer_new(INPUT_BUFFER_SIZE + MAX_KEY_CODE_LEN);
|
|
}
|
|
|
|
/// Gets the file from which input was gathered at startup.
|
|
int input_global_fd(void)
|
|
{
|
|
return global_fd;
|
|
}
|
|
|
|
void input_start(int fd)
|
|
{
|
|
if (!read_stream.closed) {
|
|
return;
|
|
}
|
|
|
|
global_fd = fd;
|
|
rstream_init_fd(&main_loop, &read_stream, fd, READ_BUFFER_SIZE);
|
|
rstream_start(&read_stream, read_cb, NULL);
|
|
}
|
|
|
|
void input_stop(void)
|
|
{
|
|
if (read_stream.closed) {
|
|
return;
|
|
}
|
|
|
|
rstream_stop(&read_stream);
|
|
stream_close(&read_stream, NULL, NULL);
|
|
}
|
|
|
|
static void cursorhold_event(void **argv)
|
|
{
|
|
event_T event = State & INSERT ? EVENT_CURSORHOLDI : EVENT_CURSORHOLD;
|
|
apply_autocmds(event, NULL, NULL, false, curbuf);
|
|
did_cursorhold = true;
|
|
}
|
|
|
|
static void create_cursorhold_event(void)
|
|
{
|
|
// If events are enabled and the queue has any items, this function should not
|
|
// have been called(inbuf_poll would return kInputAvail)
|
|
// TODO(tarruda): Cursorhold should be implemented as a timer set during the
|
|
// `state_check` callback for the states where it can be triggered.
|
|
assert(!events_enabled || multiqueue_empty(main_loop.events));
|
|
multiqueue_put(main_loop.events, cursorhold_event, 0);
|
|
}
|
|
|
|
// Low level input function
|
|
int os_inchar(uint8_t *buf, int maxlen, int ms, int tb_change_cnt)
|
|
{
|
|
if (maxlen && rbuffer_size(input_buffer)) {
|
|
return (int)rbuffer_read(input_buffer, (char *)buf, (size_t)maxlen);
|
|
}
|
|
|
|
InbufPollResult result;
|
|
if (ms >= 0) {
|
|
if ((result = inbuf_poll(ms)) == kInputNone) {
|
|
return 0;
|
|
}
|
|
} else {
|
|
if ((result = inbuf_poll((int)p_ut)) == kInputNone) {
|
|
if (trigger_cursorhold() && !typebuf_changed(tb_change_cnt)) {
|
|
create_cursorhold_event();
|
|
} else {
|
|
before_blocking();
|
|
result = inbuf_poll(-1);
|
|
}
|
|
}
|
|
}
|
|
|
|
// If input was put directly in typeahead buffer bail out here.
|
|
if (typebuf_changed(tb_change_cnt)) {
|
|
return 0;
|
|
}
|
|
|
|
if (maxlen && rbuffer_size(input_buffer)) {
|
|
// Safe to convert rbuffer_read to int, it will never overflow since we use
|
|
// relatively small buffers.
|
|
return (int)rbuffer_read(input_buffer, (char *)buf, (size_t)maxlen);
|
|
}
|
|
|
|
// If there are events, return the keys directly
|
|
if (maxlen && pending_events()) {
|
|
return push_event_key(buf, maxlen);
|
|
}
|
|
|
|
if (result == kInputEof) {
|
|
read_error_exit();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// Check if a character is available for reading
|
|
bool os_char_avail(void)
|
|
{
|
|
return inbuf_poll(0) == kInputAvail;
|
|
}
|
|
|
|
// Check for CTRL-C typed by reading all available characters.
|
|
void os_breakcheck(void)
|
|
{
|
|
int save_us = updating_screen;
|
|
// We do not want screen_resize() to redraw here.
|
|
updating_screen++;
|
|
|
|
if (!got_int) {
|
|
loop_poll_events(&main_loop, 0);
|
|
}
|
|
|
|
updating_screen = save_us;
|
|
}
|
|
|
|
void input_enable_events(void)
|
|
{
|
|
events_enabled++;
|
|
}
|
|
|
|
void input_disable_events(void)
|
|
{
|
|
events_enabled--;
|
|
}
|
|
|
|
/// Test whether a file descriptor refers to a terminal.
|
|
///
|
|
/// @param fd File descriptor.
|
|
/// @return `true` if file descriptor refers to a terminal.
|
|
bool os_isatty(int fd)
|
|
{
|
|
return uv_guess_handle(fd) == UV_TTY;
|
|
}
|
|
|
|
size_t input_enqueue(String keys)
|
|
{
|
|
char *ptr = keys.data;
|
|
char *end = ptr + keys.size;
|
|
|
|
while (rbuffer_space(input_buffer) >= 6 && ptr < end) {
|
|
uint8_t buf[6] = { 0 };
|
|
unsigned int new_size
|
|
= trans_special((const uint8_t **)&ptr, (size_t)(end - ptr), buf, true,
|
|
true);
|
|
|
|
if (new_size) {
|
|
new_size = handle_mouse_event(&ptr, buf, new_size);
|
|
rbuffer_write(input_buffer, (char *)buf, new_size);
|
|
continue;
|
|
}
|
|
|
|
if (*ptr == '<') {
|
|
char *old_ptr = ptr;
|
|
// Invalid or incomplete key sequence, skip until the next '>' or *end.
|
|
do {
|
|
ptr++;
|
|
} while (ptr < end && *ptr != '>');
|
|
if (*ptr != '>') {
|
|
// Incomplete key sequence, return without consuming.
|
|
ptr = old_ptr;
|
|
break;
|
|
}
|
|
ptr++;
|
|
continue;
|
|
}
|
|
|
|
// copy the character, escaping CSI and K_SPECIAL
|
|
if ((uint8_t)*ptr == CSI) {
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){K_SPECIAL}, 1);
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){KS_EXTRA}, 1);
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){KE_CSI}, 1);
|
|
} else if ((uint8_t)*ptr == K_SPECIAL) {
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){K_SPECIAL}, 1);
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){KS_SPECIAL}, 1);
|
|
rbuffer_write(input_buffer, (char *)&(uint8_t){KE_FILLER}, 1);
|
|
} else {
|
|
rbuffer_write(input_buffer, ptr, 1);
|
|
}
|
|
ptr++;
|
|
}
|
|
|
|
size_t rv = (size_t)(ptr - keys.data);
|
|
process_interrupts();
|
|
return rv;
|
|
}
|
|
|
|
// Mouse event handling code(Extract row/col if available and detect multiple
|
|
// clicks)
|
|
static unsigned int handle_mouse_event(char **ptr, uint8_t *buf,
|
|
unsigned int bufsize)
|
|
{
|
|
int mouse_code = 0;
|
|
int type = 0;
|
|
|
|
if (bufsize == 3) {
|
|
mouse_code = buf[2];
|
|
type = buf[1];
|
|
} else if (bufsize == 6) {
|
|
// prefixed with K_SPECIAL KS_MODIFIER mod
|
|
mouse_code = buf[5];
|
|
type = buf[4];
|
|
}
|
|
|
|
if (type != KS_EXTRA
|
|
|| !((mouse_code >= KE_LEFTMOUSE && mouse_code <= KE_RIGHTRELEASE)
|
|
|| (mouse_code >= KE_MOUSEDOWN && mouse_code <= KE_MOUSERIGHT))) {
|
|
return bufsize;
|
|
}
|
|
|
|
// a <[COL],[ROW]> sequence can follow and will set the mouse_row/mouse_col
|
|
// global variables. This is ugly but its how the rest of the code expects to
|
|
// find mouse coordinates, and it would be too expensive to refactor this
|
|
// now.
|
|
int col, row, advance;
|
|
if (sscanf(*ptr, "<%d,%d>%n", &col, &row, &advance) != EOF && advance) {
|
|
if (col >= 0 && row >= 0) {
|
|
// Make sure the mouse position is valid. Some terminals may
|
|
// return weird values.
|
|
if (col >= Columns) {
|
|
col = (int)Columns - 1;
|
|
}
|
|
if (row >= Rows) {
|
|
row = (int)Rows - 1;
|
|
}
|
|
mouse_row = row;
|
|
mouse_col = col;
|
|
}
|
|
*ptr += advance;
|
|
}
|
|
|
|
static int orig_num_clicks = 0;
|
|
if (mouse_code != KE_LEFTRELEASE && mouse_code != KE_RIGHTRELEASE
|
|
&& mouse_code != KE_MIDDLERELEASE) {
|
|
static int orig_mouse_code = 0;
|
|
static int orig_mouse_col = 0;
|
|
static int orig_mouse_row = 0;
|
|
static uint64_t orig_mouse_time = 0; // time of previous mouse click
|
|
uint64_t mouse_time = os_hrtime(); // time of current mouse click (ns)
|
|
|
|
// compute the time elapsed since the previous mouse click and
|
|
// convert p_mouse from ms to ns
|
|
uint64_t timediff = mouse_time - orig_mouse_time;
|
|
uint64_t mouset = (uint64_t)p_mouset * 1000000;
|
|
if (mouse_code == orig_mouse_code
|
|
&& timediff < mouset
|
|
&& orig_num_clicks != 4
|
|
&& orig_mouse_col == mouse_col
|
|
&& orig_mouse_row == mouse_row) {
|
|
orig_num_clicks++;
|
|
} else {
|
|
orig_num_clicks = 1;
|
|
}
|
|
orig_mouse_code = mouse_code;
|
|
orig_mouse_col = mouse_col;
|
|
orig_mouse_row = mouse_row;
|
|
orig_mouse_time = mouse_time;
|
|
}
|
|
|
|
uint8_t modifiers = 0;
|
|
if (orig_num_clicks == 2) {
|
|
modifiers |= MOD_MASK_2CLICK;
|
|
} else if (orig_num_clicks == 3) {
|
|
modifiers |= MOD_MASK_3CLICK;
|
|
} else if (orig_num_clicks == 4) {
|
|
modifiers |= MOD_MASK_4CLICK;
|
|
}
|
|
|
|
if (modifiers) {
|
|
if (buf[1] != KS_MODIFIER) {
|
|
// no modifiers in the buffer yet, shift the bytes 3 positions
|
|
memcpy(buf + 3, buf, 3);
|
|
// add the modifier sequence
|
|
buf[0] = K_SPECIAL;
|
|
buf[1] = KS_MODIFIER;
|
|
buf[2] = modifiers;
|
|
bufsize += 3;
|
|
} else {
|
|
buf[2] |= modifiers;
|
|
}
|
|
}
|
|
|
|
return bufsize;
|
|
}
|
|
|
|
/// @return true if the main loop is blocked and waiting for input.
|
|
bool input_blocking(void)
|
|
{
|
|
return blocking;
|
|
}
|
|
|
|
static bool input_poll(int ms)
|
|
{
|
|
if (do_profiling == PROF_YES && ms) {
|
|
prof_inchar_enter();
|
|
}
|
|
|
|
if ((ms == - 1 || ms > 0) && !events_enabled && !input_eof) {
|
|
// The pending input provoked a blocking wait. Do special events now. #6247
|
|
blocking = true;
|
|
multiqueue_process_events(ch_before_blocking_events);
|
|
}
|
|
LOOP_PROCESS_EVENTS_UNTIL(&main_loop, NULL, ms, input_ready() || input_eof);
|
|
blocking = false;
|
|
|
|
if (do_profiling == PROF_YES && ms) {
|
|
prof_inchar_exit();
|
|
}
|
|
|
|
return input_ready();
|
|
}
|
|
|
|
void input_done(void)
|
|
{
|
|
input_eof = true;
|
|
}
|
|
|
|
bool input_available(void)
|
|
{
|
|
return rbuffer_size(input_buffer) != 0;
|
|
}
|
|
|
|
// This is a replacement for the old `WaitForChar` function in os_unix.c
|
|
static InbufPollResult inbuf_poll(int ms)
|
|
{
|
|
if (input_ready() || input_poll(ms)) {
|
|
return kInputAvail;
|
|
}
|
|
|
|
return input_eof ? kInputEof : kInputNone;
|
|
}
|
|
|
|
static void read_cb(Stream *stream, RBuffer *buf, size_t c, void *data,
|
|
bool at_eof)
|
|
{
|
|
if (at_eof) {
|
|
input_eof = true;
|
|
}
|
|
|
|
assert(rbuffer_space(input_buffer) >= rbuffer_size(buf));
|
|
RBUFFER_UNTIL_EMPTY(buf, ptr, len) {
|
|
(void)rbuffer_write(input_buffer, ptr, len);
|
|
rbuffer_consumed(buf, len);
|
|
}
|
|
}
|
|
|
|
static void process_interrupts(void)
|
|
{
|
|
if ((mapped_ctrl_c | curbuf->b_mapped_ctrl_c) & get_real_state()) {
|
|
return;
|
|
}
|
|
|
|
size_t consume_count = 0;
|
|
RBUFFER_EACH_REVERSE(input_buffer, c, i) {
|
|
if ((uint8_t)c == 3) {
|
|
got_int = true;
|
|
consume_count = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (got_int && consume_count) {
|
|
// Remove everything typed before the CTRL-C
|
|
rbuffer_consumed(input_buffer, consume_count);
|
|
}
|
|
}
|
|
|
|
// Helper function used to push bytes from the 'event' key sequence partially
|
|
// between calls to os_inchar when maxlen < 3
|
|
static int push_event_key(uint8_t *buf, int maxlen)
|
|
{
|
|
static const uint8_t key[3] = { K_SPECIAL, KS_EXTRA, KE_EVENT };
|
|
static int key_idx = 0;
|
|
int buf_idx = 0;
|
|
|
|
do {
|
|
buf[buf_idx++] = key[key_idx++];
|
|
key_idx %= 3;
|
|
} while (key_idx > 0 && buf_idx < maxlen);
|
|
|
|
return buf_idx;
|
|
}
|
|
|
|
// Check if there's pending input
|
|
static bool input_ready(void)
|
|
{
|
|
return (typebuf_was_filled // API call filled typeahead
|
|
|| rbuffer_size(input_buffer) // Input buffer filled
|
|
|| pending_events()); // Events must be processed
|
|
}
|
|
|
|
// Exit because of an input read error.
|
|
static void read_error_exit(void)
|
|
{
|
|
if (silent_mode) /* Normal way to exit for "ex -s" */
|
|
getout(0);
|
|
STRCPY(IObuff, _("Vim: Error reading input, exiting...\n"));
|
|
preserve_exit();
|
|
}
|
|
|
|
static bool pending_events(void)
|
|
{
|
|
return events_enabled && !multiqueue_empty(main_loop.events);
|
|
}
|