aboutsummaryrefslogtreecommitdiff
path: root/src/nvim/digraph.c
diff options
context:
space:
mode:
authorzeertzjq <zeertzjq@outlook.com>2022-02-17 15:23:17 +0800
committerzeertzjq <zeertzjq@outlook.com>2022-04-12 21:26:30 +0800
commitcbc54cf484d1082e8c09b955e86aff4579ad8f8a (patch)
tree5d210dd3ff92eb80d2fd6c127c96451fb925c89b /src/nvim/digraph.c
parent7e1e906738ae8cf2d38ded4be3bef50d72b7c2a7 (diff)
downloadrneovim-cbc54cf484d1082e8c09b955e86aff4579ad8f8a.tar.gz
rneovim-cbc54cf484d1082e8c09b955e86aff4579ad8f8a.tar.bz2
rneovim-cbc54cf484d1082e8c09b955e86aff4579ad8f8a.zip
vim-patch:8.2.3184: cannot add a digraph with a leading space
Problem: Cannot add a digraph with a leading space. It is not easy to list existing digraphs. Solution: Add setdigraph(), setdigraphlist(), getdigraph() and getdigraphlist(). (closes vim/vim#8580) https://github.com/vim/vim/commit/6106504e9edc8500131f7a36e59bc146f90180fa Use GA_APPEND_VIA_PTR in registerdigraph(). Use tv_list_append_*() in getdigraphlist_appendpair(). Put the error messages in digraph.c. E196 is N/A. Remove mentions about 'encoding' being non-Unicode. Nvim doesn't support setting encoding=japan, so skip a test.
Diffstat (limited to 'src/nvim/digraph.c')
-rw-r--r--src/nvim/digraph.c257
1 files changed, 232 insertions, 25 deletions
diff --git a/src/nvim/digraph.c b/src/nvim/digraph.c
index 6e5389c979..4046869871 100644
--- a/src/nvim/digraph.c
+++ b/src/nvim/digraph.c
@@ -12,6 +12,7 @@
#include "nvim/ascii.h"
#include "nvim/charset.h"
#include "nvim/digraph.h"
+#include "nvim/eval/typval.h"
#include "nvim/ex_cmds2.h"
#include "nvim/ex_docmd.h"
#include "nvim/ex_getln.h"
@@ -34,6 +35,12 @@ typedef struct digraph {
result_T result;
} digr_T;
+static char e_digraph_must_be_just_two_characters_str[]
+ = N_("E1214: Digraph must be just two characters: %s");
+static char e_digraph_argument_must_be_one_character_str[]
+ = N_("E1215: Digraph must be one character: %s");
+static char e_setdigraphlist_argument_must_be_list_of_lists_with_two_items[]
+ = N_("E1216: setdigraphlist() argument must be a list of lists with two items");
#ifdef INCLUDE_GENERATED_DECLARATIONS
# include "digraph.c.generated.h"
@@ -1607,6 +1614,44 @@ int getdigraph(int char1, int char2, bool meta_char)
return retval;
}
+/// Add a digraph to the digraph table.
+static void registerdigraph(int char1, int char2, int n)
+{
+ // If the digraph already exists, replace "result".
+ digr_T *dp = (digr_T *)user_digraphs.ga_data;
+ for (int i = 0; i < user_digraphs.ga_len; i++) {
+ if ((int)dp->char1 == char1 && (int)dp->char2 == char2) {
+ dp->result = n;
+ return;
+ }
+ dp++;
+ }
+
+ // Add a new digraph to the table.
+ dp = GA_APPEND_VIA_PTR(digr_T, &user_digraphs);
+ dp->char1 = (char_u)char1;
+ dp->char2 = (char_u)char2;
+ dp->result = n;
+}
+
+/// Check the characters are valid for a digraph.
+/// If they are valid, returns true; otherwise, give an error message and
+/// returns false.
+bool check_digraph_chars_valid(int char1, int char2)
+{
+ if (char2 == 0) {
+ char_u msg[MB_MAXBYTES + 1];
+ msg[utf_char2bytes(char1, msg)] = NUL;
+ semsg(_(e_digraph_must_be_just_two_characters_str), msg);
+ return false;
+ }
+ if (char1 == ESC || char2 == ESC) {
+ emsg(_("E104: Escape not allowed in digraph"));
+ return false;
+ }
+ return true;
+}
+
/// Add the digraphs in the argument to the digraph table.
/// format: {c1}{c2} char {c1}{c2} char ...
///
@@ -1622,15 +1667,10 @@ void putdigraph(char_u *str)
char_u char1 = *str++;
char_u char2 = *str++;
- if (char2 == 0) {
- emsg(_(e_invarg));
+ if (!check_digraph_chars_valid(char1, char2)) {
return;
}
- if ((char1 == ESC) || (char2 == ESC)) {
- emsg(_("E104: Escape not allowed in digraph"));
- return;
- }
str = skipwhite(str);
if (!ascii_isdigit(*str)) {
@@ -1639,25 +1679,7 @@ void putdigraph(char_u *str)
}
int n = getdigits_int(&str, true, 0);
- // If the digraph already exists, replace the result.
- digr_T *dp = (digr_T *)user_digraphs.ga_data;
-
- int i;
- for (i = 0; i < user_digraphs.ga_len; i++) {
- if (((int)dp->char1 == char1) && ((int)dp->char2 == char2)) {
- dp->result = n;
- break;
- }
- dp++;
- }
-
- // Add a new digraph to the table.
- if (i == user_digraphs.ga_len) {
- dp = GA_APPEND_VIA_PTR(digr_T, &user_digraphs);
- dp->char1 = char1;
- dp->char2 = char2;
- dp->result = n;
- }
+ registerdigraph(char1, char2, n);
}
}
@@ -1707,6 +1729,50 @@ void listdigraphs(bool use_headers)
}
}
+static void getdigraphlist_appendpair(digr_T *dp, list_T *l)
+{
+ list_T *l2 = tv_list_alloc(2);
+ tv_list_append_list(l, l2);
+
+ char_u buf[30];
+ buf[0] = dp->char1;
+ buf[1] = dp->char2;
+ buf[2] = NUL;
+ tv_list_append_string(l2, (char *)buf, -1);
+
+ char_u *p = buf;
+ p += utf_char2bytes(dp->result, p);
+ *p = NUL;
+ tv_list_append_string(l2, (char *)buf, -1);
+}
+
+void getdigraphlist_common(bool list_all, typval_T *rettv)
+{
+ tv_list_alloc_ret(rettv, (int)sizeof(digraphdefault) + user_digraphs.ga_len);
+
+ digr_T *dp;
+
+ if (list_all) {
+ dp = digraphdefault;
+ for (int i = 0; dp->char1 != NUL && !got_int; i++) {
+ digr_T tmp;
+ tmp.char1 = dp->char1;
+ tmp.char2 = dp->char2;
+ tmp.result = getexactdigraph(tmp.char1, tmp.char2, false);
+ if (tmp.result != 0 && tmp.result != tmp.char2) {
+ getdigraphlist_appendpair(&tmp, rettv->vval.v_list);
+ }
+ dp++;
+ }
+ }
+
+ dp = (digr_T *)user_digraphs.ga_data;
+ for (int i = 0; i < user_digraphs.ga_len && !got_int; i++) {
+ getdigraphlist_appendpair(dp, rettv->vval.v_list);
+ dp++;
+ }
+}
+
struct dg_header_entry {
int dg_start;
const char *dg_header;
@@ -1797,6 +1863,147 @@ static void printdigraph(const digr_T *dp, result_T *previous)
}
}
+/// Get the two digraph characters from a typval.
+/// @return OK or FAIL.
+static int get_digraph_chars(typval_T *arg, int *char1, int *char2)
+{
+ char buf_chars[NUMBUFLEN];
+ const char *chars = tv_get_string_buf_chk(arg, buf_chars);
+ const char_u *p = (const char_u *)chars;
+
+ if (p != NULL) {
+ if (*p != NUL) {
+ *char1 = mb_cptr2char_adv(&p);
+ if (*p != NUL) {
+ *char2 = mb_cptr2char_adv(&p);
+ if (*p == NUL) {
+ if (check_digraph_chars_valid(*char1, *char2)) {
+ return OK;
+ }
+ return FAIL;
+ }
+ }
+ }
+ }
+ semsg(_(e_digraph_must_be_just_two_characters_str), chars);
+ return FAIL;
+}
+
+static bool setdigraph_common(typval_T *argchars, typval_T *argdigraph)
+{
+ int char1, char2;
+ if (get_digraph_chars(argchars, &char1, &char2) == FAIL) {
+ return false;
+ }
+
+ char buf_digraph[NUMBUFLEN];
+ const char *digraph = tv_get_string_buf_chk(argdigraph, buf_digraph);
+ if (digraph == NULL) {
+ return false;
+ }
+ const char_u *p = (const char_u *)digraph;
+ int n = mb_cptr2char_adv(&p);
+ if (*p != NUL) {
+ semsg(_(e_digraph_argument_must_be_one_character_str), digraph);
+ return false;
+ }
+
+ registerdigraph(char1, char2, n);
+ return true;
+}
+
+/// "getdigraph()" function
+void f_getdigraph(typval_T *argvars, typval_T *rettv, FunPtr fptr)
+{
+ rettv->v_type = VAR_STRING;
+ rettv->vval.v_string = NULL; // Return empty string for failure
+ const char *digraphs = tv_get_string_chk(&argvars[0]);
+
+ if (digraphs == NULL) {
+ return;
+ }
+ if (STRLEN(digraphs) != 2) {
+ semsg(_(e_digraph_must_be_just_two_characters_str), digraphs);
+ return;
+ }
+ int code = getdigraph(digraphs[0], digraphs[1], false);
+
+ char_u buf[NUMBUFLEN];
+ buf[utf_char2bytes(code, buf)] = NUL;
+ rettv->vval.v_string = vim_strsave(buf);
+}
+
+/// "getdigraphlist()" function
+void f_getdigraphlist(typval_T *argvars, typval_T *rettv, FunPtr fptr)
+{
+ bool flag_list_all;
+
+ if (argvars[0].v_type == VAR_UNKNOWN) {
+ flag_list_all = false;
+ } else {
+ bool error = false;
+ varnumber_T flag = tv_get_number_chk(&argvars[0], &error);
+ if (error) {
+ return;
+ }
+ flag_list_all = flag != 0;
+ }
+
+ getdigraphlist_common(flag_list_all, rettv);
+}
+
+/// "setdigraph()" function
+void f_setdigraph(typval_T *argvars, typval_T *rettv, FunPtr fptr)
+{
+ rettv->v_type = VAR_BOOL;
+ rettv->vval.v_bool = kBoolVarFalse;
+
+ if (!setdigraph_common(&argvars[0], &argvars[1])) {
+ return;
+ }
+
+ rettv->vval.v_bool = kBoolVarTrue;
+}
+
+/// "setdigraphlist()" function
+void f_setdigraphlist(typval_T *argvars, typval_T *rettv, FunPtr fptr)
+{
+ rettv->v_type = VAR_BOOL;
+ rettv->vval.v_bool = kBoolVarFalse;
+
+ if (argvars[0].v_type != VAR_LIST) {
+ emsg(_(e_setdigraphlist_argument_must_be_list_of_lists_with_two_items));
+ return;
+ }
+
+ list_T *pl = argvars[0].vval.v_list;
+ if (pl == NULL) {
+ // Empty list always results in success.
+ rettv->vval.v_bool = kBoolVarTrue;
+ return;
+ }
+
+ TV_LIST_ITER_CONST(pl, pli, {
+ if (TV_LIST_ITEM_TV(pli)->v_type != VAR_LIST) {
+ emsg(_(e_setdigraphlist_argument_must_be_list_of_lists_with_two_items));
+ return;
+ }
+
+ list_T *l = TV_LIST_ITEM_TV(pli)->vval.v_list;
+ if (l == NULL || tv_list_len(l) != 2) {
+ emsg(_(e_setdigraphlist_argument_must_be_list_of_lists_with_two_items));
+ return;
+ }
+
+ if (!setdigraph_common(TV_LIST_ITEM_TV(tv_list_first(l)),
+ TV_LIST_ITEM_TV(TV_LIST_ITEM_NEXT(l, tv_list_first(l))))) {
+ return;
+ }
+ });
+
+ rettv->vval.v_bool = kBoolVarTrue;
+}
+
/// structure used for b_kmap_ga.ga_data
typedef struct {
char_u *from;