{
typval_T tv;
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_(e_cannot_mod));
*endp = cc;
listitem_T *ll_li = lp->ll_li;
int ll_n1 = lp->ll_n1;
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_("E996: Cannot lock a range"));
return;
/*
* Assign to a List or Dictionary item.
*/
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_("E996: Cannot lock a list or dict"));
return;
{
forinfo_T *fi = (forinfo_T *)fi_void;
int result;
- int flag = in_vim9script() ? ASSIGN_DECL : 0;
+ int flag = ASSIGN_FOR_LOOP | (in_vim9script()
+ ? (ASSIGN_FINAL | ASSIGN_DECL | ASSIGN_NO_MEMBER_TYPE)
+ : 0);
listitem_T *item;
if (fi->fi_blob != NULL)
// ":let $VAR = expr": Set environment variable.
if (*arg == '$')
{
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_("E996: Cannot lock an environment variable"));
return NULL;
// ":let &option = expr": Set option value.
// ":let &l:option = expr": Set local option value.
// ":let &g:option = expr": Set global option value.
+ // ":for &ts in range(8)": Set option value for for loop
else if (*arg == '&')
{
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_(e_const_option));
return NULL;
// ":let @r = expr": Set register contents.
else if (*arg == '@')
{
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_("E996: Cannot lock a register"));
return NULL;
type_T *type,
typval_T *tv_arg,
int copy, // make copy of value in "tv"
- int flags, // ASSIGN_CONST, ASSIGN_FINAL, etc.
+ int flags_arg, // ASSIGN_CONST, ASSIGN_FINAL, etc.
int var_idx) // index for ":let [a, b] = list"
{
typval_T *tv = tv_arg;
int is_script_local;
int vim9script = in_vim9script();
int var_in_vim9script;
+ int flags = flags_arg;
ht = find_var_ht(name, &varname);
if (ht == NULL || *varname == NUL)
vim9_declare_error(name);
goto failed;
}
+ if ((flags & ASSIGN_FOR_LOOP) && name[1] == ':'
+ && vim_strchr((char_u *)"gwbt", name[0]) != NULL)
+ // Do not make g:var, w:var, b:var or t:var final.
+ flags &= ~ASSIGN_FINAL;
+
var_in_vim9script = is_script_local && current_script_is_vim9();
if (var_in_vim9script && name[0] == '_' && name[1] == NUL)
{
// Item already exists. Allowed to replace when reloading.
if ((di->di_flags & DI_FLAGS_RELOAD) == 0)
{
- if (flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ if ((flags & (ASSIGN_CONST | ASSIGN_FINAL))
+ && (flags & ASSIGN_FOR_LOOP) == 0)
{
emsg(_(e_cannot_mod));
goto failed;
// A Vim9 script-local variable is also present in sn_all_vars and
// sn_var_vals. It may set "type" from "tv".
if (var_in_vim9script)
- update_vim9_script_var(FALSE, di, flags, tv, &type);
+ update_vim9_script_var(FALSE, di, flags, tv, &type,
+ (flags & ASSIGN_NO_MEMBER_TYPE) == 0);
}
// existing variable, need to clear the value
// A Vim9 script-local variable is also added to sn_all_vars and
// sn_var_vals. It may set "type" from "tv".
if (var_in_vim9script)
- update_vim9_script_var(TRUE, di, flags, tv, &type);
+ update_vim9_script_var(TRUE, di, flags, tv, &type,
+ (flags & ASSIGN_NO_MEMBER_TYPE) == 0);
}
if (copy || tv->v_type == VAR_NUMBER || tv->v_type == VAR_FLOAT)