summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2021-03-18 21:37:55 +0100
committerBram Moolenaar <Bram@vim.org>2021-03-18 21:37:55 +0100
commit4b3e1964d85a25ac7b2202094d1abf27ab93cc23 (patch)
treee0a0c2fa156ce0ac2e884d02d873fb9c249aa2e2
parent5f91e74bf968c9033474086b2d9cb457281c8aa6 (diff)
downloadvim-git-4b3e1964d85a25ac7b2202094d1abf27ab93cc23.tar.gz
patch 8.2.2620: Vim9: Using #{ for a dictionary gives strange errorsv8.2.2620
Problem: Vim9: Using #{ for a dictionary gives strange errors. Solution: Give an error when using #{ for a comment after a command.
-rw-r--r--src/errors.h2
-rw-r--r--src/proto/vim9script.pro1
-rw-r--r--src/testdir/test_vim9_expr.vim6
-rw-r--r--src/testdir/test_vim9_script.vim2
-rw-r--r--src/version.c2
-rw-r--r--src/vim9compile.c19
-rw-r--r--src/vim9script.c21
7 files changed, 43 insertions, 10 deletions
diff --git a/src/errors.h b/src/errors.h
index c1dc547fb..ed55304bc 100644
--- a/src/errors.h
+++ b/src/errors.h
@@ -375,3 +375,5 @@ EXTERN char e_argument_already_declared_in_script_str[]
INIT(= N_("E1168: Argument already declared in the script: %s"));
EXTERN char e_import_as_name_not_supported_here[]
INIT(= N_("E1169: 'import * as {name}' not supported here"));
+EXTERN char e_cannot_use_hash_curly_to_start_comment[]
+ INIT(= N_("E1170: 'Cannot use #{ to start a comment"));
diff --git a/src/proto/vim9script.pro b/src/proto/vim9script.pro
index c80618f75..a7d821c0d 100644
--- a/src/proto/vim9script.pro
+++ b/src/proto/vim9script.pro
@@ -2,6 +2,7 @@
int in_vim9script(void);
void ex_vim9script(exarg_T *eap);
int not_in_vim9(exarg_T *eap);
+int vim9_bad_comment(char_u *p);
int vim9_comment_start(char_u *p);
void ex_export(exarg_T *eap);
void free_imports_and_script_vars(int sid);
diff --git a/src/testdir/test_vim9_expr.vim b/src/testdir/test_vim9_expr.vim
index 9829d8800..10468410f 100644
--- a/src/testdir/test_vim9_expr.vim
+++ b/src/testdir/test_vim9_expr.vim
@@ -2159,8 +2159,10 @@ def Test_expr7_dict()
CheckDefAndScriptSuccess(lines)
# legacy syntax doesn't work
- CheckDefFailure(["var x = #{key: 8}"], 'E1097:', 3)
- CheckDefFailure(["var x = 'a' .. #{a: 1}"], 'E1097:', 3)
+ CheckDefFailure(["var x = #{key: 8}"], 'E1170:', 1)
+ CheckDefFailure(["var x = 'a' #{a: 1}"], 'E1170:', 1)
+ CheckDefFailure(["var x = 'a' .. #{a: 1}"], 'E1170:', 1)
+ CheckDefFailure(["var x = true ? #{a: 1}"], 'E1170:', 1)
CheckDefFailure(["var x = {a:8}"], 'E1069:', 1)
CheckDefFailure(["var x = {a : 8}"], 'E1068:', 1)
diff --git a/src/testdir/test_vim9_script.vim b/src/testdir/test_vim9_script.vim
index 38d0b0abc..49ecbea65 100644
--- a/src/testdir/test_vim9_script.vim
+++ b/src/testdir/test_vim9_script.vim
@@ -2452,7 +2452,7 @@ def Test_while_loop()
assert_equal('1_3_', result)
var s = ''
- while s == 'x' #{comment}
+ while s == 'x' # {comment}
endwhile
enddef
diff --git a/src/version.c b/src/version.c
index 74eec468a..dfc0f033c 100644
--- a/src/version.c
+++ b/src/version.c
@@ -751,6 +751,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 2620,
+/**/
2619,
/**/
2618,
diff --git a/src/vim9compile.c b/src/vim9compile.c
index 458b4a1a3..238ce5d1f 100644
--- a/src/vim9compile.c
+++ b/src/vim9compile.c
@@ -1546,7 +1546,7 @@ generate_FUNCREF(cctx_T *cctx, ufunc_T *ufunc)
isn->isn_arg.funcref.fr_func = ufunc->uf_dfunc_idx;
cctx->ctx_has_closure = 1;
- // if the referenced function is a closure, it may use items further up in
+ // If the referenced function is a closure, it may use items further up in
// the nested context, including this one.
if (ufunc->uf_flags & FC_CLOSURE)
cctx->ctx_ufunc->uf_flags |= FC_CLOSURE;
@@ -2401,6 +2401,8 @@ peek_next_line_from_context(cctx_T *cctx)
if (line != NULL)
{
p = skipwhite(line);
+ if (vim9_bad_comment(p))
+ return NULL;
if (*p != NUL && !vim9_comment_start(p))
return p;
}
@@ -2465,6 +2467,8 @@ next_line_from_context(cctx_T *cctx, int skip_comment)
may_get_next_line(char_u *whitep, char_u **arg, cctx_T *cctx)
{
*arg = skipwhite(whitep);
+ if (vim9_bad_comment(*arg))
+ return FAIL;
if (**arg == NUL || (VIM_ISWHITE(*whitep) && vim9_comment_start(*arg)))
{
char_u *next = next_line_from_context(cctx, TRUE);
@@ -4277,10 +4281,13 @@ compile_expr7(
if (!eval_isnamec1(**arg))
{
- if (ends_excmd(*skipwhite(*arg)))
- semsg(_(e_empty_expression_str), *arg);
- else
- semsg(_(e_name_expected_str), *arg);
+ if (!vim9_bad_comment(*arg))
+ {
+ if (ends_excmd(*skipwhite(*arg)))
+ semsg(_(e_empty_expression_str), *arg);
+ else
+ semsg(_(e_name_expected_str), *arg);
+ }
return FAIL;
}
@@ -8297,6 +8304,8 @@ compile_def_function(
semsg(_(e_trailing_arg), line);
goto erret;
}
+ else if (line != NULL && vim9_bad_comment(skipwhite(line)))
+ goto erret;
else
{
line = next_line_from_context(&cctx, FALSE);
diff --git a/src/vim9script.c b/src/vim9script.c
index 05977b641..309d4bdaf 100644
--- a/src/vim9script.c
+++ b/src/vim9script.c
@@ -113,12 +113,29 @@ not_in_vim9(exarg_T *eap)
}
/*
- * Return TRUE if "p" points at a "#". Does not check for white space.
+ * Give an error message if "p" points at "#{" and return TRUE.
+ * This avoids that using a legacy style #{} dictionary leads to difficult to
+ * understand errors.
+ */
+ int
+vim9_bad_comment(char_u *p)
+{
+ if (p[0] == '#' && p[1] == '{')
+ {
+ emsg(_(e_cannot_use_hash_curly_to_start_comment));
+ return TRUE;
+ }
+ return FALSE;
+}
+
+/*
+ * Return TRUE if "p" points at a "#" not followed by '{'.
+ * Does not check for white space.
*/
int
vim9_comment_start(char_u *p)
{
- return p[0] == '#';
+ return p[0] == '#' && p[1] != '{';
}
#if defined(FEAT_EVAL) || defined(PROTO)