add ; to last return in block

This commit is contained in:
Aleksey Kladov 2018-12-30 17:09:27 +03:00
parent c182aab546
commit 0f75ac1ae0
2 changed files with 46 additions and 15 deletions

View File

@ -35,7 +35,7 @@ pub(super) fn complete_expr_keyword(acc: &mut Completions, ctx: &CompletionConte
acc.add(keyword("continue", "continue"));
acc.add(keyword("break", "break"));
}
acc.add_all(complete_return(fn_def, ctx.is_stmt));
acc.add_all(complete_return(fn_def, ctx.can_be_stmt));
}
fn is_in_loop_body(leaf: SyntaxNodeRef) -> bool {
@ -57,8 +57,8 @@ fn is_in_loop_body(leaf: SyntaxNodeRef) -> bool {
false
}
fn complete_return(fn_def: ast::FnDef, is_stmt: bool) -> Option<CompletionItem> {
let snip = match (is_stmt, fn_def.ret_type().is_some()) {
fn complete_return(fn_def: ast::FnDef, can_be_stmt: bool) -> Option<CompletionItem> {
let snip = match (can_be_stmt, fn_def.ret_type().is_some()) {
(true, true) => "return $0;",
(true, false) => "return;",
(false, true) => "return $0",
@ -75,7 +75,7 @@ mod tests {
}
#[test]
fn test_completion_kewords() {
fn completes_various_keywords_in_function() {
check_keyword_completion(
r"
fn quux() {
@ -87,13 +87,13 @@ mod tests {
match "match $0 {}"
while "while $0 {}"
loop "loop {$0}"
return "return"
return "return;"
"#,
);
}
#[test]
fn test_completion_else() {
fn completes_else_after_if() {
check_keyword_completion(
r"
fn quux() {
@ -109,7 +109,7 @@ mod tests {
loop "loop {$0}"
else "else {$0}"
else if "else if $0 {}"
return "return"
return "return;"
"#,
);
}
@ -149,7 +149,7 @@ mod tests {
}
#[test]
fn test_completion_return_no_stmt() {
fn dont_add_semi_after_return_if_not_a_statement() {
check_keyword_completion(
r"
fn quux() -> i32 {
@ -169,7 +169,27 @@ mod tests {
}
#[test]
fn test_continue_break_completion() {
fn last_return_in_block_has_semi() {
check_keyword_completion(
r"
fn quux() -> i32 {
if condition {
<|>
}
}
",
r#"
if "if $0 {}"
match "match $0 {}"
while "while $0 {}"
loop "loop {$0}"
return "return $0;"
"#,
);
}
#[test]
fn completes_break_and_continue_in_loops() {
check_keyword_completion(
r"
fn quux() -> i32 {
@ -183,9 +203,10 @@ mod tests {
loop "loop {$0}"
continue "continue"
break "break"
return "return $0"
return "return $0;"
"#,
);
// No completion: lambda isolates control flow
check_keyword_completion(
r"
fn quux() -> i32 {
@ -197,7 +218,7 @@ mod tests {
match "match $0 {}"
while "while $0 {}"
loop "loop {$0}"
return "return $0"
return "return $0;"
"#,
);
}

View File

@ -31,7 +31,8 @@ pub(super) struct CompletionContext<'a> {
/// If not a trivial, path, the prefix (qualifier).
pub(super) path_prefix: Option<hir::Path>,
pub(super) after_if: bool,
pub(super) is_stmt: bool,
/// `true` if we are a statement or a last expr in the block.
pub(super) can_be_stmt: bool,
/// Something is typed at the "top" level, in module or impl/trait.
pub(super) is_new_item: bool,
/// The receiver if this is a field or method access, i.e. writing something.<|>
@ -61,7 +62,7 @@ impl<'a> CompletionContext<'a> {
is_trivial_path: false,
path_prefix: None,
after_if: false,
is_stmt: false,
can_be_stmt: false,
is_new_item: false,
dot_receiver: None,
is_method_call: false,
@ -147,13 +148,22 @@ impl<'a> CompletionContext<'a> {
if path.qualifier().is_none() {
self.is_trivial_path = true;
self.is_stmt = match name_ref
self.can_be_stmt = match name_ref
.syntax()
.ancestors()
.filter_map(ast::ExprStmt::cast)
.next()
{
None => false,
None => {
name_ref
.syntax()
.ancestors()
.filter_map(ast::Block::cast)
.next()
.and_then(|block| block.expr())
.map(|e| e.syntax().range())
== Some(name_ref.syntax().range())
}
Some(expr_stmt) => expr_stmt.syntax().range() == name_ref.syntax().range(),
};