2020-04-01 00:50:45 +03:00
|
|
|
//! Complete fields in record literals and patterns.
|
2021-12-22 02:25:38 +01:00
|
|
|
use ide_db::SymbolKind;
|
2021-08-10 17:03:38 +02:00
|
|
|
use syntax::{ast::Expr, T};
|
2020-11-13 17:17:16 +01:00
|
|
|
|
2021-05-30 21:23:42 +02:00
|
|
|
use crate::{
|
2022-02-02 12:05:21 +01:00
|
|
|
patterns::ImmediateLocation, CompletionContext, CompletionItem, CompletionItemKind,
|
2022-04-01 20:50:27 +03:00
|
|
|
CompletionRelevance, CompletionRelevancePostfixMatch, Completions,
|
2021-05-30 21:23:42 +02:00
|
|
|
};
|
2020-04-07 15:07:18 +02:00
|
|
|
|
2020-10-25 10:59:15 +03:00
|
|
|
pub(crate) fn complete_record(acc: &mut Completions, ctx: &CompletionContext) -> Option<()> {
|
2021-05-30 21:23:42 +02:00
|
|
|
let missing_fields = match &ctx.completion_location {
|
2021-08-10 17:03:38 +02:00
|
|
|
Some(
|
|
|
|
ImmediateLocation::RecordExpr(record_expr)
|
|
|
|
| ImmediateLocation::RecordExprUpdate(record_expr),
|
|
|
|
) => {
|
2021-05-30 21:23:42 +02:00
|
|
|
let ty = ctx.sema.type_of_expr(&Expr::RecordExpr(record_expr.clone()));
|
2022-03-12 06:58:43 -08:00
|
|
|
|
2022-03-12 07:35:13 -08:00
|
|
|
if let Some(hir::Adt::Union(un)) = ty.as_ref().and_then(|t| t.original.as_adt()) {
|
|
|
|
// ctx.sema.record_literal_missing_fields will always return
|
|
|
|
// an empty Vec on a union literal. This is normally
|
|
|
|
// reasonable, but here we'd like to present the full list
|
|
|
|
// of fields if the literal is empty.
|
|
|
|
let were_fields_specified = record_expr
|
|
|
|
.record_expr_field_list()
|
|
|
|
.and_then(|fl| fl.fields().next())
|
|
|
|
.is_some();
|
2022-03-12 06:58:43 -08:00
|
|
|
|
2022-03-12 07:35:13 -08:00
|
|
|
match were_fields_specified {
|
|
|
|
false => un.fields(ctx.db).into_iter().map(|f| (f, f.ty(ctx.db))).collect(),
|
|
|
|
true => vec![],
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let missing_fields = ctx.sema.record_literal_missing_fields(record_expr);
|
|
|
|
|
|
|
|
let default_trait = ctx.famous_defs().core_default_Default();
|
|
|
|
let impl_default_trait =
|
|
|
|
default_trait.zip(ty.as_ref()).map_or(false, |(default_trait, ty)| {
|
|
|
|
ty.original.impls_trait(ctx.db, default_trait, &[])
|
|
|
|
});
|
2020-11-13 17:17:16 +01:00
|
|
|
|
2022-03-12 07:35:13 -08:00
|
|
|
if impl_default_trait && !missing_fields.is_empty() && ctx.path_qual().is_none() {
|
|
|
|
let completion_text = "..Default::default()";
|
|
|
|
let mut item =
|
|
|
|
CompletionItem::new(SymbolKind::Field, ctx.source_range(), completion_text);
|
|
|
|
let completion_text =
|
|
|
|
completion_text.strip_prefix(ctx.token.text()).unwrap_or(completion_text);
|
|
|
|
item.insert_text(completion_text).set_relevance(CompletionRelevance {
|
2022-04-01 20:50:27 +03:00
|
|
|
postfix_match: Some(CompletionRelevancePostfixMatch::Exact),
|
2022-03-12 07:35:13 -08:00
|
|
|
..Default::default()
|
|
|
|
});
|
|
|
|
item.add_to(acc);
|
2022-03-12 06:58:43 -08:00
|
|
|
}
|
2022-03-12 07:35:13 -08:00
|
|
|
if ctx.previous_token_is(T![.]) {
|
|
|
|
let mut item =
|
|
|
|
CompletionItem::new(CompletionItemKind::Snippet, ctx.source_range(), "..");
|
|
|
|
item.insert_text(".");
|
|
|
|
item.add_to(acc);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
missing_fields
|
2021-08-10 17:03:38 +02:00
|
|
|
}
|
2020-11-13 17:17:16 +01:00
|
|
|
}
|
2021-05-30 21:23:42 +02:00
|
|
|
Some(ImmediateLocation::RecordPat(record_pat)) => {
|
|
|
|
ctx.sema.record_pattern_missing_fields(record_pat)
|
|
|
|
}
|
|
|
|
_ => return None,
|
2020-04-07 17:09:02 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
for (field, ty) in missing_fields {
|
2021-05-28 14:02:53 +02:00
|
|
|
acc.add_field(ctx, None, field, &ty);
|
2020-04-01 00:50:45 +03:00
|
|
|
}
|
|
|
|
|
2020-04-07 17:09:02 +02:00
|
|
|
Some(())
|
2020-04-01 00:50:45 +03:00
|
|
|
}
|
2021-07-22 19:59:01 +02:00
|
|
|
|
2021-08-04 00:46:50 +02:00
|
|
|
pub(crate) fn complete_record_literal(
|
|
|
|
acc: &mut Completions,
|
|
|
|
ctx: &CompletionContext,
|
|
|
|
) -> Option<()> {
|
|
|
|
if !ctx.expects_expression() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2022-03-12 06:58:43 -08:00
|
|
|
match ctx.expected_type.as_ref()?.as_adt()? {
|
2022-03-12 07:35:13 -08:00
|
|
|
hir::Adt::Struct(strukt) if ctx.path_qual().is_none() => {
|
2022-03-31 11:12:08 +02:00
|
|
|
let path = ctx
|
|
|
|
.module
|
2022-03-16 16:27:55 +01:00
|
|
|
.find_use_path(ctx.db, hir::ModuleDef::from(strukt))
|
|
|
|
.filter(|it| it.len() > 1);
|
2021-11-23 10:57:29 -06:00
|
|
|
|
2022-03-12 07:35:13 -08:00
|
|
|
acc.add_struct_literal(ctx, strukt, path, None);
|
2022-02-21 13:50:16 +01:00
|
|
|
}
|
2022-03-12 07:35:13 -08:00
|
|
|
hir::Adt::Union(un) if ctx.path_qual().is_none() => {
|
2022-03-31 11:12:08 +02:00
|
|
|
let path = ctx
|
|
|
|
.module
|
|
|
|
.find_use_path(ctx.db, hir::ModuleDef::from(un))
|
|
|
|
.filter(|it| it.len() > 1);
|
2022-03-12 06:58:43 -08:00
|
|
|
|
2022-03-12 07:35:13 -08:00
|
|
|
acc.add_union_literal(ctx, un, path, None);
|
2022-03-12 06:58:43 -08:00
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
};
|
2021-08-04 00:46:50 +02:00
|
|
|
|
|
|
|
Some(())
|
|
|
|
}
|
|
|
|
|
2021-07-22 19:59:01 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use crate::tests::check_edit;
|
|
|
|
|
|
|
|
#[test]
|
2021-08-04 00:46:50 +02:00
|
|
|
fn literal_struct_completion_edit() {
|
|
|
|
check_edit(
|
|
|
|
"FooDesc {…}",
|
|
|
|
r#"
|
|
|
|
struct FooDesc { pub bar: bool }
|
|
|
|
|
|
|
|
fn create_foo(foo_desc: &FooDesc) -> () { () }
|
|
|
|
|
|
|
|
fn baz() {
|
|
|
|
let foo = create_foo(&$0);
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct FooDesc { pub bar: bool }
|
|
|
|
|
|
|
|
fn create_foo(foo_desc: &FooDesc) -> () { () }
|
|
|
|
|
|
|
|
fn baz() {
|
|
|
|
let foo = create_foo(&FooDesc { bar: ${1:()} }$0);
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-11-23 10:57:29 -06:00
|
|
|
#[test]
|
|
|
|
fn literal_struct_completion_from_sub_modules() {
|
|
|
|
check_edit(
|
2022-03-16 16:27:55 +01:00
|
|
|
"submod::Struct {…}",
|
2021-11-23 10:57:29 -06:00
|
|
|
r#"
|
|
|
|
mod submod {
|
|
|
|
pub struct Struct {
|
|
|
|
pub a: u64,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn f() -> submod::Struct {
|
|
|
|
Stru$0
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
mod submod {
|
|
|
|
pub struct Struct {
|
|
|
|
pub a: u64,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn f() -> submod::Struct {
|
|
|
|
submod::Struct { a: ${1:()} }$0
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-04 00:46:50 +02:00
|
|
|
#[test]
|
|
|
|
fn literal_struct_complexion_module() {
|
|
|
|
check_edit(
|
|
|
|
"FooDesc {…}",
|
|
|
|
r#"
|
|
|
|
mod _69latrick {
|
|
|
|
pub struct FooDesc { pub six: bool, pub neuf: Vec<String>, pub bar: bool }
|
|
|
|
pub fn create_foo(foo_desc: &FooDesc) -> () { () }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn baz() {
|
|
|
|
use _69latrick::*;
|
|
|
|
|
|
|
|
let foo = create_foo(&$0);
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
mod _69latrick {
|
|
|
|
pub struct FooDesc { pub six: bool, pub neuf: Vec<String>, pub bar: bool }
|
|
|
|
pub fn create_foo(foo_desc: &FooDesc) -> () { () }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn baz() {
|
|
|
|
use _69latrick::*;
|
|
|
|
|
|
|
|
let foo = create_foo(&FooDesc { six: ${1:()}, neuf: ${2:()}, bar: ${3:()} }$0);
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-07-22 19:59:01 +02:00
|
|
|
fn default_completion_edit() {
|
|
|
|
check_edit(
|
|
|
|
"..Default::default()",
|
|
|
|
r#"
|
|
|
|
//- minicore: default
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
.$0
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
check_edit(
|
|
|
|
"..Default::default()",
|
|
|
|
r#"
|
|
|
|
//- minicore: default
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
$0
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
2021-08-10 14:58:14 +02:00
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
check_edit(
|
|
|
|
"..Default::default()",
|
|
|
|
r#"
|
|
|
|
//- minicore: default
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
..$0
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
r#"
|
|
|
|
struct Struct { foo: u32, bar: usize }
|
|
|
|
|
|
|
|
impl Default for Struct {
|
|
|
|
fn default() -> Self {}
|
|
|
|
}
|
|
|
|
|
2021-07-22 19:59:01 +02:00
|
|
|
fn foo() {
|
|
|
|
let other = Struct {
|
|
|
|
foo: 5,
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|