mirror of
https://github.com/neovim/neovim.git
synced 2025-09-24 20:18:32 +00:00

Problem: ModeChanged autocmd not executed when Visual mode is ended with
CTRL-C.
Solution: Do not trigger the autocmd when got_int is set. (closes vim/vim#11394)
61c4b04799
Cherry-pick removal of cmdwin feature check from patch 9.0.0663.
260 lines
7.4 KiB
C
260 lines
7.4 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 "klib/kvec.h"
|
|
#include "nvim/ascii.h"
|
|
#include "nvim/autocmd.h"
|
|
#include "nvim/drawscreen.h"
|
|
#include "nvim/eval.h"
|
|
#include "nvim/ex_docmd.h"
|
|
#include "nvim/getchar.h"
|
|
#include "nvim/insexpand.h"
|
|
#include "nvim/log.h"
|
|
#include "nvim/main.h"
|
|
#include "nvim/option.h"
|
|
#include "nvim/option_defs.h"
|
|
#include "nvim/os/input.h"
|
|
#include "nvim/state.h"
|
|
#include "nvim/ui.h"
|
|
#include "nvim/vim.h"
|
|
|
|
#ifdef INCLUDE_GENERATED_DECLARATIONS
|
|
# include "state.c.generated.h"
|
|
#endif
|
|
|
|
void state_enter(VimState *s)
|
|
{
|
|
for (;;) {
|
|
int check_result = s->check ? s->check(s) : 1;
|
|
|
|
if (!check_result) {
|
|
break; // Terminate this state.
|
|
} else if (check_result == -1) {
|
|
continue; // check() again.
|
|
}
|
|
// Execute this state.
|
|
|
|
int key;
|
|
|
|
getkey:
|
|
// Apply mappings first by calling vpeekc() directly.
|
|
// - If vpeekc() returns non-NUL, there is a character already available for processing, so
|
|
// don't block for events. vgetc() may still block, in case of an incomplete UTF-8 sequence.
|
|
// - If vpeekc() returns NUL, vgetc() will block, and there are three cases:
|
|
// - There is no input available.
|
|
// - All of available input maps to an empty string.
|
|
// - There is an incomplete mapping.
|
|
// A blocking wait for a character should only be done in the third case, which is the only
|
|
// case of the three where typebuf.tb_len > 0 after vpeekc() returns NUL.
|
|
if (vpeekc() != NUL || typebuf.tb_len > 0) {
|
|
key = safe_vgetc();
|
|
} else if (!multiqueue_empty(main_loop.events)) {
|
|
// Event was made available after the last multiqueue_process_events call
|
|
key = K_EVENT;
|
|
} else {
|
|
// Duplicate display updating logic in vgetorpeek()
|
|
if (((State & MODE_INSERT) != 0 || p_lz) && (State & MODE_CMDLINE) == 0
|
|
&& must_redraw != 0 && !need_wait_return) {
|
|
update_screen();
|
|
setcursor(); // put cursor back where it belongs
|
|
}
|
|
// Flush screen updates before blocking
|
|
ui_flush();
|
|
// Call `os_inchar` directly to block for events or user input without
|
|
// consuming anything from `input_buffer`(os/input.c) or calling the
|
|
// mapping engine.
|
|
(void)os_inchar(NULL, 0, -1, typebuf.tb_change_cnt, main_loop.events);
|
|
// If an event was put into the queue, we send K_EVENT directly.
|
|
if (!multiqueue_empty(main_loop.events)) {
|
|
key = K_EVENT;
|
|
} else {
|
|
goto getkey;
|
|
}
|
|
}
|
|
|
|
if (key == K_EVENT) {
|
|
// An event handler may use the value of reg_executing.
|
|
// Clear it if it should be cleared when getting the next character.
|
|
check_end_reg_executing(true);
|
|
may_sync_undo();
|
|
}
|
|
|
|
#if MIN_LOG_LEVEL <= LOGLVL_DBG
|
|
log_key(LOGLVL_DBG, key);
|
|
#endif
|
|
|
|
int execute_result = s->execute(s, key);
|
|
if (!execute_result) {
|
|
break;
|
|
} else if (execute_result == -1) {
|
|
goto getkey;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// process events on main_loop, but interrupt if input is available
|
|
///
|
|
/// This should be used to handle K_EVENT in states accepting input
|
|
/// otherwise bursts of events can block break checking indefinitely.
|
|
void state_handle_k_event(void)
|
|
{
|
|
while (true) {
|
|
Event event = multiqueue_get(main_loop.events);
|
|
if (event.handler) {
|
|
event.handler(event.argv);
|
|
}
|
|
|
|
if (multiqueue_empty(main_loop.events)) {
|
|
// don't breakcheck before return, caller should return to main-loop
|
|
// and handle input already.
|
|
return;
|
|
}
|
|
|
|
// TODO(bfredl): as an further micro-optimization, we could check whether
|
|
// event.handler already checked input.
|
|
os_breakcheck();
|
|
if (input_available() || got_int) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Return true if in the current mode we need to use virtual.
|
|
bool virtual_active(void)
|
|
{
|
|
unsigned int cur_ve_flags = get_ve_flags();
|
|
|
|
// While an operator is being executed we return "virtual_op", because
|
|
// VIsual_active has already been reset, thus we can't check for "block"
|
|
// being used.
|
|
if (virtual_op != kNone) {
|
|
return virtual_op;
|
|
}
|
|
return cur_ve_flags == VE_ALL
|
|
|| ((cur_ve_flags & VE_BLOCK) && VIsual_active && VIsual_mode == Ctrl_V)
|
|
|| ((cur_ve_flags & VE_INSERT) && (State & MODE_INSERT));
|
|
}
|
|
|
|
/// MODE_VISUAL, MODE_SELECT and MODE_OP_PENDING State are never set, they are
|
|
/// equal to MODE_NORMAL State with a condition. This function returns the real
|
|
/// State.
|
|
int get_real_state(void)
|
|
{
|
|
if (State & MODE_NORMAL) {
|
|
if (VIsual_active) {
|
|
if (VIsual_select) {
|
|
return MODE_SELECT;
|
|
}
|
|
return MODE_VISUAL;
|
|
} else if (finish_op) {
|
|
return MODE_OP_PENDING;
|
|
}
|
|
}
|
|
return State;
|
|
}
|
|
|
|
/// Returns the current mode as a string in "buf[MODE_MAX_LENGTH]", NUL
|
|
/// terminated.
|
|
/// The first character represents the major mode, the following ones the minor
|
|
/// ones.
|
|
void get_mode(char *buf)
|
|
{
|
|
int i = 0;
|
|
|
|
if (VIsual_active) {
|
|
if (VIsual_select) {
|
|
buf[i++] = (char)(VIsual_mode + 's' - 'v');
|
|
} else {
|
|
buf[i++] = (char)VIsual_mode;
|
|
if (restart_VIsual_select) {
|
|
buf[i++] = 's';
|
|
}
|
|
}
|
|
} else if (State == MODE_HITRETURN || State == MODE_ASKMORE || State == MODE_SETWSIZE
|
|
|| State == MODE_CONFIRM) {
|
|
buf[i++] = 'r';
|
|
if (State == MODE_ASKMORE) {
|
|
buf[i++] = 'm';
|
|
} else if (State == MODE_CONFIRM) {
|
|
buf[i++] = '?';
|
|
}
|
|
} else if (State == MODE_EXTERNCMD) {
|
|
buf[i++] = '!';
|
|
} else if (State & MODE_INSERT) {
|
|
if (State & VREPLACE_FLAG) {
|
|
buf[i++] = 'R';
|
|
buf[i++] = 'v';
|
|
} else {
|
|
if (State & REPLACE_FLAG) {
|
|
buf[i++] = 'R';
|
|
} else {
|
|
buf[i++] = 'i';
|
|
}
|
|
}
|
|
if (ins_compl_active()) {
|
|
buf[i++] = 'c';
|
|
} else if (ctrl_x_mode_not_defined_yet()) {
|
|
buf[i++] = 'x';
|
|
}
|
|
} else if ((State & MODE_CMDLINE) || exmode_active) {
|
|
buf[i++] = 'c';
|
|
if (exmode_active) {
|
|
buf[i++] = 'v';
|
|
}
|
|
} else if (State & MODE_TERMINAL) {
|
|
buf[i++] = 't';
|
|
} else {
|
|
buf[i++] = 'n';
|
|
if (finish_op) {
|
|
buf[i++] = 'o';
|
|
// to be able to detect force-linewise/blockwise/charwise operations
|
|
buf[i++] = (char)motion_force;
|
|
} else if (curbuf->terminal) {
|
|
buf[i++] = 't';
|
|
if (restart_edit == 'I') {
|
|
buf[i++] = 'T';
|
|
}
|
|
} else if (restart_edit == 'I' || restart_edit == 'R'
|
|
|| restart_edit == 'V') {
|
|
buf[i++] = 'i';
|
|
buf[i++] = (char)restart_edit;
|
|
}
|
|
}
|
|
|
|
buf[i] = NUL;
|
|
}
|
|
|
|
/// Fires a ModeChanged autocmd if appropriate.
|
|
void may_trigger_modechanged(void)
|
|
{
|
|
// Skip this when got_int is set, the autocommand will not be executed.
|
|
// Better trigger it next time.
|
|
if (!has_event(EVENT_MODECHANGED) || got_int) {
|
|
return;
|
|
}
|
|
|
|
char curr_mode[MODE_MAX_LENGTH];
|
|
char pattern_buf[2 * MODE_MAX_LENGTH];
|
|
|
|
get_mode(curr_mode);
|
|
if (strcmp(curr_mode, last_mode) == 0) {
|
|
return;
|
|
}
|
|
|
|
save_v_event_T save_v_event;
|
|
dict_T *v_event = get_v_event(&save_v_event);
|
|
tv_dict_add_str(v_event, S_LEN("new_mode"), curr_mode);
|
|
tv_dict_add_str(v_event, S_LEN("old_mode"), last_mode);
|
|
tv_dict_set_keys_readonly(v_event);
|
|
|
|
// concatenate modes in format "old_mode:new_mode"
|
|
vim_snprintf(pattern_buf, sizeof(pattern_buf), "%s:%s", last_mode, curr_mode);
|
|
|
|
apply_autocmds(EVENT_MODECHANGED, pattern_buf, NULL, false, curbuf);
|
|
STRCPY(last_mode, curr_mode);
|
|
|
|
restore_v_event(v_event, &save_v_event);
|
|
}
|