diff options
author | zeertzjq <zeertzjq@outlook.com> | 2022-06-23 21:17:11 +0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-06-23 21:17:11 +0800 |
commit | 7718b758461265d8966468c104ce5454538471e2 (patch) | |
tree | c58cec1a66defb6c84c113cb76c03e759db47feb /src/nvim/keycodes.c | |
parent | 05ca14a8810555495c309b8add3002773c77123d (diff) | |
download | rneovim-7718b758461265d8966468c104ce5454538471e2.tar.gz rneovim-7718b758461265d8966468c104ce5454538471e2.tar.bz2 rneovim-7718b758461265d8966468c104ce5454538471e2.zip |
refactor: move some mapping-related code to a separate file (#19061)
This marks the following Vim patches as ported:
vim-patch:8.1.1785: map functionality mixed with character input
Problem: Map functionality mixed with character input.
Solution: Move the map functionality to a separate file. (Yegappan
Lakshmanan, closes vim/vim#4740) Graduate the +localmap feature.
https://github.com/vim/vim/commit/b66bab381c8ba71fd6e92327d1d34c6f8a65f2a7
vim-patch:8.2.3643: header for source file is outdated
Problem: Header for source file is outdated.
Solution: Make the header more accurate. (closes vim/vim#9186)
https://github.com/vim/vim/commit/a3f83feb63eae5464a620ae793c002eb45f7a838
Also cherry-pick a change for <unique> mappings from patch 8.2.0807.
Rename map_clear_mode() to do_mapclear().
Diffstat (limited to 'src/nvim/keycodes.c')
-rw-r--r-- | src/nvim/keycodes.c | 72 |
1 files changed, 72 insertions, 0 deletions
diff --git a/src/nvim/keycodes.c b/src/nvim/keycodes.c index 676ddcf8d4..e0ada32ec2 100644 --- a/src/nvim/keycodes.c +++ b/src/nvim/keycodes.c @@ -1005,6 +1005,78 @@ char *replace_termcodes(const char *const from, const size_t from_len, char **co return *bufp; } +/// Add character "c" to buffer "s" +/// +/// Escapes the special meaning of K_SPECIAL, handles multi-byte +/// characters. +/// +/// @param[in] c Character to add. +/// @param[out] s Buffer to add to. Must have at least MB_MAXBYTES + 1 bytes. +/// +/// @return Pointer to after the added bytes. +char_u *add_char2buf(int c, char_u *s) + FUNC_ATTR_NONNULL_ALL FUNC_ATTR_WARN_UNUSED_RESULT +{ + char_u temp[MB_MAXBYTES + 1]; + const int len = utf_char2bytes(c, (char *)temp); + for (int i = 0; i < len; i++) { + c = (uint8_t)temp[i]; + // Need to escape K_SPECIAL like in the typeahead buffer. + if (c == K_SPECIAL) { + *s++ = K_SPECIAL; + *s++ = KS_SPECIAL; + *s++ = KE_FILLER; + } else { + *s++ = (char_u)c; + } + } + return s; +} + +/// Copy "p" to allocated memory, escaping K_SPECIAL so that the result +/// can be put in the typeahead buffer. +char *vim_strsave_escape_ks(char *p) +{ + // Need a buffer to hold up to three times as much. Four in case of an + // illegal utf-8 byte: + // 0xc0 -> 0xc3 - 0x80 -> 0xc3 K_SPECIAL KS_SPECIAL KE_FILLER + char_u *res = xmalloc(STRLEN(p) * 4 + 1); + char_u *d = res; + for (char_u *s = (char_u *)p; *s != NUL;) { + if (s[0] == K_SPECIAL && s[1] != NUL && s[2] != NUL) { + // Copy special key unmodified. + *d++ = *s++; + *d++ = *s++; + *d++ = *s++; + } else { + // Add character, possibly multi-byte to destination, escaping + // K_SPECIAL. Be careful, it can be an illegal byte! + d = add_char2buf(utf_ptr2char((char *)s), d); + s += utf_ptr2len((char *)s); + } + } + *d = NUL; + + return (char *)res; +} + +/// Remove escaping from K_SPECIAL characters. Reverse of +/// vim_strsave_escape_ks(). Works in-place. +void vim_unescape_ks(char_u *p) +{ + char_u *s = p, *d = p; + + while (*s != NUL) { + if (s[0] == K_SPECIAL && s[1] == KS_SPECIAL && s[2] == KE_FILLER) { + *d++ = K_SPECIAL; + s += 3; + } else { + *d++ = *s++; + } + } + *d = NUL; +} + /// Logs a single key as a human-readable keycode. void log_key(int log_level, int key) { |