]> granicus.if.org Git - vim/commitdiff
patch 8.2.3682: Vim9: assigning to a script variable drops the type v8.2.3682
authorBram Moolenaar <Bram@vim.org>
Fri, 26 Nov 2021 17:36:51 +0000 (17:36 +0000)
committerBram Moolenaar <Bram@vim.org>
Fri, 26 Nov 2021 17:36:51 +0000 (17:36 +0000)
Problem:    Vim9: assigning to a script variable drops the required type.
Solution:   Lookup the type of the variable and use it. (closes #9219)

src/evalvars.c
src/proto/vim9script.pro
src/testdir/test_vim9_assign.vim
src/version.c
src/vim9script.c

index b1d7b78c8130b2bcb13b7f4da513166cbfbf43bf..434fd96119b029841d907104768e58ff2d3e3398 100644 (file)
@@ -3206,13 +3206,14 @@ set_var(
     void
 set_var_const(
     char_u     *name,
-    type_T     *type,
+    type_T     *type_arg,
     typval_T   *tv_arg,
     int                copy,       // make copy of value in "tv"
     int                flags_arg,  // ASSIGN_CONST, ASSIGN_FINAL, etc.
     int                var_idx)    // index for ":let [a, b] = list"
 {
     typval_T   *tv = tv_arg;
+    type_T     *type = type_arg;
     typval_T   bool_tv;
     dictitem_T *di;
     typval_T   *dest_tv = NULL;
@@ -3334,13 +3335,18 @@ set_var_const(
                if (var_in_vim9script && (flags & ASSIGN_FOR_LOOP) == 0)
                {
                    where_T where = WHERE_INIT;
+                   svar_T  *sv = find_typval_in_script(&di->di_tv);
 
-                   // check the type and adjust to bool if needed
-                   where.wt_index = var_idx;
-                   where.wt_variable = TRUE;
-                   if (check_script_var_type(&di->di_tv, tv, name, where)
-                                                                      == FAIL)
-                       goto failed;
+                   if (sv != NULL)
+                   {
+                       // check the type and adjust to bool if needed
+                       where.wt_index = var_idx;
+                       where.wt_variable = TRUE;
+                       if (check_script_var_type(sv, tv, name, where) == FAIL)
+                           goto failed;
+                       if (type == NULL)
+                           type = sv->sv_type;
+                   }
                }
 
                if ((flags & ASSIGN_FOR_LOOP) == 0
index 97c1c92ad4d214e1e11590c276ab003b74142113..1b21819235eb0bf63f2270731a9202f0e56b92e6 100644 (file)
@@ -16,6 +16,6 @@ char_u *vim9_declare_scriptvar(exarg_T *eap, char_u *arg);
 void update_vim9_script_var(int create, dictitem_T *di, int flags, typval_T *tv, type_T **type, int do_member);
 void hide_script_var(scriptitem_T *si, int idx, int func_defined);
 svar_T *find_typval_in_script(typval_T *dest);
-int check_script_var_type(typval_T *dest, typval_T *value, char_u *name, where_T where);
+int check_script_var_type(svar_T *sv, typval_T *value, char_u *name, where_T where);
 int check_reserved_name(char_u *name);
 /* vim: set ft=c : */
index 0d07e07eb71ef6f04ca411ea12b0fcbebb8d806f..df8f486f9a0f33ed47dd8163fba7f583a1483c0f 100644 (file)
@@ -322,6 +322,16 @@ def Test_skipped_assignment()
   CheckDefAndScriptSuccess(lines)
 enddef
 
+def Test_assign_keep_type()
+  var lines =<< trim END
+      vim9script
+      var l: list<number> = [123]
+      l = [123]
+      l->add('string')
+  END
+  CheckScriptFailure(lines, 'E1012:', 4)
+enddef
+
 def Test_assign_unpack()
   var lines =<< trim END
     var v1: number
index 532b59e825463635152500c6b2f09bd4d752681d..f328f833c8022c99c607486bd92826f8859b4c41 100644 (file)
@@ -757,6 +757,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    3682,
 /**/
     3681,
 /**/
index 9696dfe8125a562ae059628b7e0b44977e770ecf..62341b0cacaa455a210fcf62f1b77ed523f24761 100644 (file)
@@ -999,35 +999,29 @@ find_typval_in_script(typval_T *dest)
  */
     int
 check_script_var_type(
-       typval_T    *dest,
+       svar_T      *sv,
        typval_T    *value,
        char_u      *name,
        where_T     where)
 {
-    svar_T  *sv = find_typval_in_script(dest);
     int            ret;
 
-    if (sv != NULL)
+    if (sv->sv_const != 0)
     {
-       if (sv->sv_const != 0)
-       {
-           semsg(_(e_cannot_change_readonly_variable_str), name);
-           return FAIL;
-       }
-       ret = check_typval_type(sv->sv_type, value, where);
-       if (ret == OK && need_convert_to_bool(sv->sv_type, value))
-       {
-           int val = tv2bool(value);
-
-           clear_tv(value);
-           value->v_type = VAR_BOOL;
-           value->v_lock = 0;
-           value->vval.v_number = val ? VVAL_TRUE : VVAL_FALSE;
-       }
-       return ret;
+       semsg(_(e_cannot_change_readonly_variable_str), name);
+       return FAIL;
     }
+    ret = check_typval_type(sv->sv_type, value, where);
+    if (ret == OK && need_convert_to_bool(sv->sv_type, value))
+    {
+       int     val = tv2bool(value);
 
-    return OK; // not really
+       clear_tv(value);
+       value->v_type = VAR_BOOL;
+       value->v_lock = 0;
+       value->vval.v_number = val ? VVAL_TRUE : VVAL_FALSE;
+    }
+    return ret;
 }
 
 // words that cannot be used as a variable