summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2022-08-30 14:34:52 +0100
committerBram Moolenaar <Bram@vim.org>2022-08-30 14:34:52 +0100
commit976f859763b215050a03248dbc2bb62fa5d0d059 (patch)
tree36bc5a92c8eb8cf57c0116e21c9a36679993fba4
parent0e412be00f8290e0575c7f72ec080725631eff38 (diff)
downloadvim-git-976f859763b215050a03248dbc2bb62fa5d0d059.tar.gz
patch 9.0.0327: items() does not work on a listv9.0.0327
Problem: items() does not work on a list. (Sergey Vlasov) Solution: Make items() work on a list. (closes #11013)
-rw-r--r--src/dict.c39
-rw-r--r--src/evalfunc.c2
-rw-r--r--src/list.c32
-rw-r--r--src/proto/list.pro1
-rw-r--r--src/testdir/test_listdict.vim11
-rw-r--r--src/testdir/test_vim9_builtin.vim6
-rw-r--r--src/version.c2
7 files changed, 73 insertions, 20 deletions
diff --git a/src/dict.c b/src/dict.c
index 29608bd88..6bac4d541 100644
--- a/src/dict.c
+++ b/src/dict.c
@@ -1440,14 +1440,17 @@ dict_remove(typval_T *argvars, typval_T *rettv, char_u *arg_errmsg)
dictitem_remove(d, di);
}
+typedef enum {
+ DICT2LIST_KEYS,
+ DICT2LIST_VALUES,
+ DICT2LIST_ITEMS,
+} dict2list_T;
+
/*
- * Turn a dict into a list:
- * "what" == 0: list of keys
- * "what" == 1: list of values
- * "what" == 2: list of items
+ * Turn a dict into a list.
*/
static void
-dict_list(typval_T *argvars, typval_T *rettv, int what)
+dict2list(typval_T *argvars, typval_T *rettv, dict2list_T what)
{
list_T *l2;
dictitem_T *di;
@@ -1460,16 +1463,13 @@ dict_list(typval_T *argvars, typval_T *rettv, int what)
if (rettv_list_alloc(rettv) == FAIL)
return;
- if (in_vim9script() && check_for_dict_arg(argvars, 0) == FAIL)
- return;
-
- if (argvars[0].v_type != VAR_DICT)
- {
- emsg(_(e_dictionary_required));
+ if ((what == DICT2LIST_ITEMS
+ ? check_for_list_or_dict_arg(argvars, 0)
+ : check_for_dict_arg(argvars, 0)) == FAIL)
return;
- }
- if ((d = argvars[0].vval.v_dict) == NULL)
+ d = argvars[0].vval.v_dict;
+ if (d == NULL)
// empty dict behaves like an empty dict
return;
@@ -1486,14 +1486,14 @@ dict_list(typval_T *argvars, typval_T *rettv, int what)
break;
list_append(rettv->vval.v_list, li);
- if (what == 0)
+ if (what == DICT2LIST_KEYS)
{
// keys()
li->li_tv.v_type = VAR_STRING;
li->li_tv.v_lock = 0;
li->li_tv.vval.v_string = vim_strsave(di->di_key);
}
- else if (what == 1)
+ else if (what == DICT2LIST_VALUES)
{
// values()
copy_tv(&di->di_tv, &li->li_tv);
@@ -1533,7 +1533,10 @@ dict_list(typval_T *argvars, typval_T *rettv, int what)
void
f_items(typval_T *argvars, typval_T *rettv)
{
- dict_list(argvars, rettv, 2);
+ if (argvars[0].v_type == VAR_LIST)
+ list2items(argvars, rettv);
+ else
+ dict2list(argvars, rettv, DICT2LIST_ITEMS);
}
/*
@@ -1542,7 +1545,7 @@ f_items(typval_T *argvars, typval_T *rettv)
void
f_keys(typval_T *argvars, typval_T *rettv)
{
- dict_list(argvars, rettv, 0);
+ dict2list(argvars, rettv, DICT2LIST_KEYS);
}
/*
@@ -1551,7 +1554,7 @@ f_keys(typval_T *argvars, typval_T *rettv)
void
f_values(typval_T *argvars, typval_T *rettv)
{
- dict_list(argvars, rettv, 1);
+ dict2list(argvars, rettv, DICT2LIST_VALUES);
}
/*
diff --git a/src/evalfunc.c b/src/evalfunc.c
index 1b629ff84..c43db65b9 100644
--- a/src/evalfunc.c
+++ b/src/evalfunc.c
@@ -2029,7 +2029,7 @@ static funcentry_T global_functions[] =
ret_number_bool, f_islocked},
{"isnan", 1, 1, FEARG_1, arg1_float_or_nr,
ret_number_bool, MATH_FUNC(f_isnan)},
- {"items", 1, 1, FEARG_1, arg1_dict_any,
+ {"items", 1, 1, FEARG_1, arg1_list_or_dict,
ret_list_items, f_items},
{"job_getchannel", 1, 1, FEARG_1, arg1_job,
ret_channel, JOB_FUNC(f_job_getchannel)},
diff --git a/src/list.c b/src/list.c
index 10a7aacd3..056210d24 100644
--- a/src/list.c
+++ b/src/list.c
@@ -1053,6 +1053,38 @@ f_flattennew(typval_T *argvars, typval_T *rettv)
}
/*
+ * "items(list)" function
+ * Caller must have already checked that argvars[0] is a List.
+ */
+ void
+list2items(typval_T *argvars, typval_T *rettv)
+{
+ list_T *l = argvars[0].vval.v_list;
+ listitem_T *li;
+ varnumber_T idx;
+
+ if (rettv_list_alloc(rettv) == FAIL)
+ return;
+
+ if (l == NULL)
+ return; // empty list behaves like an empty list
+
+ // TODO: would be more efficient to not materialize the argument
+ CHECK_LIST_MATERIALIZE(l);
+ for (idx = 0, li = l->lv_first; li != NULL; li = li->li_next, ++idx)
+ {
+ list_T *l2 = list_alloc();
+
+ if (l2 == NULL)
+ break;
+ if (list_append_list(rettv->vval.v_list, l2) == FAIL
+ || list_append_number(l2, idx) == FAIL
+ || list_append_tv(l2, &li->li_tv) == FAIL)
+ break;
+ }
+}
+
+/*
* Extend "l1" with "l2". "l1" must not be NULL.
* If "bef" is NULL append at the end, otherwise insert before this item.
* Returns FAIL when out of memory.
diff --git a/src/proto/list.pro b/src/proto/list.pro
index 3f77f04d1..611aeac40 100644
--- a/src/proto/list.pro
+++ b/src/proto/list.pro
@@ -35,6 +35,7 @@ int check_range_index_two(list_T *l, long *n1, listitem_T *li1, long *n2, int qu
int list_assign_range(list_T *dest, list_T *src, long idx1_arg, long idx2, int empty_idx2, char_u *op, char_u *varname);
void f_flatten(typval_T *argvars, typval_T *rettv);
void f_flattennew(typval_T *argvars, typval_T *rettv);
+void list2items(typval_T *argvars, typval_T *rettv);
int list_extend(list_T *l1, list_T *l2, listitem_T *bef);
int list_concat(list_T *l1, list_T *l2, typval_T *tv);
list_T *list_slice(list_T *ol, long n1, long n2);
diff --git a/src/testdir/test_listdict.vim b/src/testdir/test_listdict.vim
index bba4e3e1e..c87d701a3 100644
--- a/src/testdir/test_listdict.vim
+++ b/src/testdir/test_listdict.vim
@@ -198,6 +198,17 @@ func Test_list_range_assign()
call v9.CheckDefAndScriptFailure(lines, 'E1012:', 2)
endfunc
+func Test_list_items()
+ let r = []
+ let l = ['a', 'b', 'c']
+ for [idx, val] in items(l)
+ call extend(r, [[idx, val]])
+ endfor
+ call assert_equal([[0, 'a'], [1, 'b'], [2, 'c']], r)
+
+ call assert_fails('call items(3)', 'E1227:')
+endfunc
+
" Test removing items in list
func Test_list_func_remove()
let lines =<< trim END
diff --git a/src/testdir/test_vim9_builtin.vim b/src/testdir/test_vim9_builtin.vim
index d68ddce7a..5eb64adef 100644
--- a/src/testdir/test_vim9_builtin.vim
+++ b/src/testdir/test_vim9_builtin.vim
@@ -2244,10 +2244,14 @@ def Test_islocked()
enddef
def Test_items()
- v9.CheckDefFailure(['[]->items()'], 'E1013: Argument 1: type mismatch, expected dict<any> but got list<unknown>')
+ v9.CheckDefFailure(['"x"->items()'], 'E1013: Argument 1: type mismatch, expected list<any> but got string')
assert_equal([['a', 10], ['b', 20]], {'a': 10, 'b': 20}->items())
assert_equal([], {}->items())
assert_equal(['x', 'x'], {'a': 10, 'b': 20}->items()->map((_, _) => 'x'))
+
+ assert_equal([[0, 'a'], [1, 'b']], ['a', 'b']->items())
+ assert_equal([], []->items())
+ assert_equal([], test_null_list()->items())
enddef
def Test_job_getchannel()
diff --git a/src/version.c b/src/version.c
index 505fe4fbe..92a9f3ad9 100644
--- a/src/version.c
+++ b/src/version.c
@@ -708,6 +708,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 327,
+/**/
326,
/**/
325,