2019-09-30 11:58:53 +03:00
|
|
|
//! FIXME: write short doc here
|
|
|
|
|
2020-06-26 20:09:11 -06:00
|
|
|
use ra_syntax::{ast, SyntaxKind};
|
2019-01-08 22:33:36 +03:00
|
|
|
|
2019-07-04 23:05:17 +03:00
|
|
|
use crate::completion::{
|
|
|
|
CompletionContext, CompletionItem, CompletionItemKind, CompletionKind, Completions,
|
|
|
|
};
|
2019-01-08 22:33:36 +03:00
|
|
|
|
|
|
|
pub(super) fn complete_use_tree_keyword(acc: &mut Completions, ctx: &CompletionContext) {
|
|
|
|
// complete keyword "crate" in use stmt
|
2019-01-20 12:02:00 +08:00
|
|
|
let source_range = ctx.source_range();
|
2019-01-08 22:33:36 +03:00
|
|
|
match (ctx.use_item_syntax.as_ref(), ctx.path_prefix.as_ref()) {
|
|
|
|
(Some(_), None) => {
|
2020-06-27 12:25:08 +02:00
|
|
|
CompletionItem::new(CompletionKind::Keyword, source_range, "crate::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.kind(CompletionItemKind::Keyword)
|
2019-01-19 22:02:50 +08:00
|
|
|
.insert_text("crate::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.add_to(acc);
|
2019-01-20 12:02:00 +08:00
|
|
|
CompletionItem::new(CompletionKind::Keyword, source_range, "self")
|
2019-01-08 22:33:36 +03:00
|
|
|
.kind(CompletionItemKind::Keyword)
|
|
|
|
.add_to(acc);
|
2020-06-27 12:25:08 +02:00
|
|
|
CompletionItem::new(CompletionKind::Keyword, source_range, "super::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.kind(CompletionItemKind::Keyword)
|
2019-01-19 22:02:50 +08:00
|
|
|
.insert_text("super::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.add_to(acc);
|
|
|
|
}
|
|
|
|
(Some(_), Some(_)) => {
|
2019-01-20 12:02:00 +08:00
|
|
|
CompletionItem::new(CompletionKind::Keyword, source_range, "self")
|
2019-01-08 22:33:36 +03:00
|
|
|
.kind(CompletionItemKind::Keyword)
|
|
|
|
.add_to(acc);
|
2020-06-27 12:25:08 +02:00
|
|
|
CompletionItem::new(CompletionKind::Keyword, source_range, "super::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.kind(CompletionItemKind::Keyword)
|
2019-01-19 22:02:50 +08:00
|
|
|
.insert_text("super::")
|
2019-01-08 22:33:36 +03:00
|
|
|
.add_to(acc);
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn complete_expr_keyword(acc: &mut Completions, ctx: &CompletionContext) {
|
2020-06-26 20:09:11 -06:00
|
|
|
if ctx.token.kind() == SyntaxKind::COMMENT {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-13 00:55:21 +02:00
|
|
|
let has_trait_or_impl_parent = ctx.has_impl_parent || ctx.has_trait_parent;
|
|
|
|
if ctx.trait_as_prev_sibling || ctx.impl_as_prev_sibling {
|
2020-06-13 13:47:30 +02:00
|
|
|
add_keyword(ctx, acc, "where", "where ");
|
2020-06-13 00:55:21 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if ctx.unsafe_is_prev {
|
2020-06-13 13:47:30 +02:00
|
|
|
if ctx.has_item_list_or_source_file_parent || ctx.block_expr_parent {
|
|
|
|
add_keyword(ctx, acc, "fn", "fn $0() {}")
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent)
|
|
|
|
|| ctx.block_expr_parent
|
|
|
|
{
|
|
|
|
add_keyword(ctx, acc, "trait", "trait $0 {}");
|
|
|
|
add_keyword(ctx, acc, "impl", "impl $0 {}");
|
|
|
|
}
|
|
|
|
|
2020-06-13 00:55:21 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-06-13 13:47:30 +02:00
|
|
|
if ctx.has_item_list_or_source_file_parent || ctx.block_expr_parent {
|
|
|
|
add_keyword(ctx, acc, "fn", "fn $0() {}");
|
|
|
|
}
|
|
|
|
if (ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent)
|
|
|
|
|| ctx.block_expr_parent
|
|
|
|
{
|
|
|
|
add_keyword(ctx, acc, "use", "use ");
|
|
|
|
add_keyword(ctx, acc, "impl", "impl $0 {}");
|
|
|
|
add_keyword(ctx, acc, "trait", "trait $0 {}");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent {
|
|
|
|
add_keyword(ctx, acc, "enum", "enum $0 {}");
|
|
|
|
add_keyword(ctx, acc, "struct", "struct $0 {}");
|
|
|
|
add_keyword(ctx, acc, "union", "union $0 {}");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.block_expr_parent || ctx.is_match_arm {
|
|
|
|
add_keyword(ctx, acc, "match", "match $0 {}");
|
|
|
|
add_keyword(ctx, acc, "loop", "loop {$0}");
|
|
|
|
}
|
|
|
|
if ctx.block_expr_parent {
|
|
|
|
add_keyword(ctx, acc, "while", "while $0 {}");
|
|
|
|
}
|
|
|
|
if ctx.if_is_prev || ctx.block_expr_parent {
|
|
|
|
add_keyword(ctx, acc, "let", "let ");
|
|
|
|
}
|
|
|
|
if ctx.if_is_prev || ctx.block_expr_parent || ctx.is_match_arm {
|
|
|
|
add_keyword(ctx, acc, "if", "if ");
|
|
|
|
add_keyword(ctx, acc, "if let", "if let ");
|
|
|
|
}
|
|
|
|
if ctx.after_if {
|
|
|
|
add_keyword(ctx, acc, "else", "else {$0}");
|
|
|
|
add_keyword(ctx, acc, "else if", "else if $0 {}");
|
|
|
|
}
|
|
|
|
if (ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent)
|
|
|
|
|| ctx.block_expr_parent
|
|
|
|
{
|
|
|
|
add_keyword(ctx, acc, "mod", "mod $0 {}");
|
|
|
|
}
|
|
|
|
if ctx.bind_pat_parent || ctx.ref_pat_parent {
|
|
|
|
add_keyword(ctx, acc, "mut", "mut ");
|
|
|
|
}
|
|
|
|
if ctx.has_item_list_or_source_file_parent || ctx.block_expr_parent {
|
|
|
|
add_keyword(ctx, acc, "const", "const ");
|
|
|
|
add_keyword(ctx, acc, "type", "type ");
|
|
|
|
}
|
|
|
|
if (ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent)
|
|
|
|
|| ctx.block_expr_parent
|
|
|
|
{
|
|
|
|
add_keyword(ctx, acc, "static", "static ");
|
|
|
|
};
|
|
|
|
if (ctx.has_item_list_or_source_file_parent && !has_trait_or_impl_parent)
|
|
|
|
|| ctx.block_expr_parent
|
|
|
|
{
|
|
|
|
add_keyword(ctx, acc, "extern", "extern ");
|
|
|
|
}
|
|
|
|
if ctx.has_item_list_or_source_file_parent || ctx.block_expr_parent || ctx.is_match_arm {
|
|
|
|
add_keyword(ctx, acc, "unsafe", "unsafe ");
|
|
|
|
}
|
|
|
|
if ctx.in_loop_body {
|
|
|
|
if ctx.can_be_stmt {
|
|
|
|
add_keyword(ctx, acc, "continue", "continue;");
|
|
|
|
add_keyword(ctx, acc, "break", "break;");
|
|
|
|
} else {
|
|
|
|
add_keyword(ctx, acc, "continue", "continue");
|
|
|
|
add_keyword(ctx, acc, "break", "break");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ctx.has_item_list_or_source_file_parent && !ctx.has_trait_parent {
|
|
|
|
add_keyword(ctx, acc, "pub", "pub ")
|
|
|
|
}
|
2019-01-08 22:33:36 +03:00
|
|
|
|
2020-06-13 01:21:48 +02:00
|
|
|
if !ctx.is_trivial_path {
|
|
|
|
return;
|
|
|
|
}
|
2019-07-19 12:56:47 +03:00
|
|
|
let fn_def = match &ctx.function_syntax {
|
2019-01-08 22:33:36 +03:00
|
|
|
Some(it) => it,
|
|
|
|
None => return,
|
|
|
|
};
|
2019-07-19 12:56:47 +03:00
|
|
|
acc.add_all(complete_return(ctx, &fn_def, ctx.can_be_stmt));
|
2019-01-08 22:33:36 +03:00
|
|
|
}
|
|
|
|
|
2020-06-13 13:47:30 +02:00
|
|
|
fn keyword(ctx: &CompletionContext, kw: &str, snippet: &str) -> CompletionItem {
|
|
|
|
let res = CompletionItem::new(CompletionKind::Keyword, ctx.source_range(), kw)
|
|
|
|
.kind(CompletionItemKind::Keyword);
|
|
|
|
|
|
|
|
match ctx.config.snippet_cap {
|
|
|
|
Some(cap) => res.insert_snippet(cap, snippet),
|
|
|
|
_ => res.insert_text(if snippet.contains('$') { kw } else { snippet }),
|
|
|
|
}
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_keyword(ctx: &CompletionContext, acc: &mut Completions, kw: &str, snippet: &str) {
|
|
|
|
acc.add(keyword(ctx, kw, snippet));
|
|
|
|
}
|
|
|
|
|
2019-01-19 22:02:50 +08:00
|
|
|
fn complete_return(
|
|
|
|
ctx: &CompletionContext,
|
|
|
|
fn_def: &ast::FnDef,
|
|
|
|
can_be_stmt: bool,
|
|
|
|
) -> Option<CompletionItem> {
|
2019-01-08 22:33:36 +03:00
|
|
|
let snip = match (can_be_stmt, fn_def.ret_type().is_some()) {
|
|
|
|
(true, true) => "return $0;",
|
|
|
|
(true, false) => "return;",
|
|
|
|
(false, true) => "return $0",
|
|
|
|
(false, false) => "return",
|
|
|
|
};
|
2019-01-19 22:02:50 +08:00
|
|
|
Some(keyword(ctx, "return", snip))
|
2019-01-08 22:33:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2020-07-03 12:51:18 +02:00
|
|
|
use expect::{expect, Expect};
|
|
|
|
|
2020-06-13 13:57:18 +02:00
|
|
|
use crate::completion::{test_utils::completion_list, CompletionKind};
|
2019-01-19 22:02:50 +08:00
|
|
|
|
2020-07-03 12:51:18 +02:00
|
|
|
fn check(ra_fixture: &str, expect: Expect) {
|
|
|
|
let actual = completion_list(ra_fixture, CompletionKind::Keyword);
|
|
|
|
expect.assert_eq(&actual)
|
2020-06-12 08:49:12 +02:00
|
|
|
}
|
|
|
|
|
2019-01-08 22:33:36 +03:00
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_in_use_stmt() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"use <|>",
|
|
|
|
expect![[r#"
|
|
|
|
kw crate::
|
|
|
|
kw self
|
|
|
|
kw super::
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"use a::<|>",
|
|
|
|
expect![[r#"
|
2020-06-13 13:57:18 +02:00
|
|
|
kw self
|
2020-06-27 12:25:08 +02:00
|
|
|
kw super::
|
2020-07-03 12:51:18 +02:00
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"use a::{b, <|>}",
|
|
|
|
expect![[r#"
|
2020-06-13 13:57:18 +02:00
|
|
|
kw self
|
2020-06-27 12:25:08 +02:00
|
|
|
kw super::
|
2020-07-03 12:51:18 +02:00
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-13 10:43:39 +02:00
|
|
|
#[test]
|
|
|
|
fn test_keywords_at_source_file_level() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"m<|>",
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw enum
|
|
|
|
kw extern
|
|
|
|
kw fn
|
|
|
|
kw impl
|
|
|
|
kw mod
|
|
|
|
kw pub
|
|
|
|
kw static
|
|
|
|
kw struct
|
|
|
|
kw trait
|
|
|
|
kw type
|
|
|
|
kw union
|
|
|
|
kw unsafe
|
|
|
|
kw use
|
|
|
|
"#]],
|
2020-06-13 10:43:39 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-01-08 22:33:36 +03:00
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_in_function() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn quux() { <|> }",
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw extern
|
|
|
|
kw fn
|
|
|
|
kw if
|
|
|
|
kw if let
|
|
|
|
kw impl
|
|
|
|
kw let
|
|
|
|
kw loop
|
|
|
|
kw match
|
|
|
|
kw mod
|
|
|
|
kw return
|
|
|
|
kw static
|
|
|
|
kw trait
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
kw use
|
|
|
|
kw while
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_inside_block() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn quux() { if true { <|> } }",
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw extern
|
|
|
|
kw fn
|
|
|
|
kw if
|
|
|
|
kw if let
|
|
|
|
kw impl
|
|
|
|
kw let
|
|
|
|
kw loop
|
|
|
|
kw match
|
|
|
|
kw mod
|
|
|
|
kw return
|
|
|
|
kw static
|
|
|
|
kw trait
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
kw use
|
|
|
|
kw while
|
|
|
|
"#]],
|
2020-06-13 00:55:21 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_keywords_after_if() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r#"fn quux() { if true { () } <|> }"#,
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw else
|
|
|
|
kw else if
|
|
|
|
kw extern
|
|
|
|
kw fn
|
|
|
|
kw if
|
|
|
|
kw if let
|
|
|
|
kw impl
|
|
|
|
kw let
|
|
|
|
kw loop
|
|
|
|
kw match
|
|
|
|
kw mod
|
|
|
|
kw return
|
|
|
|
kw static
|
|
|
|
kw trait
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
kw use
|
|
|
|
kw while
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_in_match_arm() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r#"
|
|
|
|
fn quux() -> i32 {
|
|
|
|
match () {
|
|
|
|
() => <|>
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
expect![[r#"
|
|
|
|
kw if
|
|
|
|
kw if let
|
|
|
|
kw loop
|
|
|
|
kw match
|
|
|
|
kw return
|
|
|
|
kw unsafe
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
2020-06-13 00:55:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_keywords_in_trait_def() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"trait My { <|> }",
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw fn
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_in_impl_def() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"impl My { <|> }",
|
|
|
|
expect![[r#"
|
|
|
|
kw const
|
|
|
|
kw fn
|
|
|
|
kw pub
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_in_loop() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn my() { loop { <|> } }",
|
|
|
|
expect![[r#"
|
|
|
|
kw break
|
|
|
|
kw const
|
|
|
|
kw continue
|
|
|
|
kw extern
|
|
|
|
kw fn
|
|
|
|
kw if
|
|
|
|
kw if let
|
|
|
|
kw impl
|
|
|
|
kw let
|
|
|
|
kw loop
|
|
|
|
kw match
|
|
|
|
kw mod
|
|
|
|
kw return
|
|
|
|
kw static
|
|
|
|
kw trait
|
|
|
|
kw type
|
|
|
|
kw unsafe
|
|
|
|
kw use
|
|
|
|
kw while
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
2020-06-13 00:55:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_keywords_after_unsafe_in_item_list() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"unsafe <|>",
|
|
|
|
expect![[r#"
|
|
|
|
kw fn
|
|
|
|
kw impl
|
|
|
|
kw trait
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_keywords_after_unsafe_in_block_expr() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn my_fn() { unsafe <|> }",
|
|
|
|
expect![[r#"
|
|
|
|
kw fn
|
|
|
|
kw impl
|
|
|
|
kw trait
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
2020-06-13 00:55:21 +02:00
|
|
|
}
|
2019-01-19 22:02:50 +08:00
|
|
|
|
2020-06-13 00:55:21 +02:00
|
|
|
#[test]
|
|
|
|
fn test_mut_in_ref_and_in_fn_parameters_list() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn my_fn(&<|>) {}",
|
|
|
|
expect![[r#"
|
|
|
|
kw mut
|
|
|
|
"#]],
|
2020-06-13 00:55:21 +02:00
|
|
|
);
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn my_fn(<|>) {}",
|
|
|
|
expect![[r#"
|
|
|
|
kw mut
|
|
|
|
"#]],
|
2020-06-13 00:55:21 +02:00
|
|
|
);
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"fn my_fn() { let &<|> }",
|
|
|
|
expect![[r#"
|
|
|
|
kw mut
|
|
|
|
"#]],
|
2019-01-08 22:33:36 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-06-13 00:55:21 +02:00
|
|
|
fn test_where_keyword() {
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"trait A <|>",
|
|
|
|
expect![[r#"
|
|
|
|
kw where
|
|
|
|
"#]],
|
2020-06-13 00:55:21 +02:00
|
|
|
);
|
2020-07-03 12:51:18 +02:00
|
|
|
check(
|
|
|
|
r"impl A <|>",
|
|
|
|
expect![[r#"
|
|
|
|
kw where
|
|
|
|
"#]],
|
2020-06-13 00:55:21 +02:00
|
|
|
);
|
2019-01-08 22:33:36 +03:00
|
|
|
}
|
|
|
|
}
|