Unnamed repository; edit this file 'description' to name the repository.
Diffstat (limited to 'crates/ide-completion/src/completions/record.rs')
-rw-r--r--crates/ide-completion/src/completions/record.rs110
1 files changed, 54 insertions, 56 deletions
diff --git a/crates/ide-completion/src/completions/record.rs b/crates/ide-completion/src/completions/record.rs
index 05fbe8513e..6717ca0a0e 100644
--- a/crates/ide-completion/src/completions/record.rs
+++ b/crates/ide-completion/src/completions/record.rs
@@ -3,67 +3,65 @@ use ide_db::SymbolKind;
use syntax::{ast::Expr, T};
use crate::{
- patterns::ImmediateLocation, CompletionContext, CompletionItem, CompletionItemKind,
- CompletionRelevance, CompletionRelevancePostfixMatch, Completions,
+ context::{NameRefContext, PatternContext},
+ CompletionContext, CompletionItem, CompletionItemKind, CompletionRelevance,
+ CompletionRelevancePostfixMatch, Completions,
};
pub(crate) fn complete_record(acc: &mut Completions, ctx: &CompletionContext) -> Option<()> {
- let missing_fields = match &ctx.completion_location {
- Some(
- ImmediateLocation::RecordExpr(record_expr)
- | ImmediateLocation::RecordExprUpdate(record_expr),
- ) => {
- let ty = ctx.sema.type_of_expr(&Expr::RecordExpr(record_expr.clone()));
-
- 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();
-
- 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, &[])
- });
-
- 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 {
- postfix_match: Some(CompletionRelevancePostfixMatch::Exact),
- ..Default::default()
- });
- item.add_to(acc);
- }
- 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
+ let missing_fields = if let Some(PatternContext { record_pat: Some(record_pat), .. }) =
+ &ctx.pattern_ctx
+ {
+ ctx.sema.record_pattern_missing_fields(record_pat)
+ } else if let Some(NameRefContext { record_expr: Some((record_expr, _)), .. }) =
+ ctx.nameref_ctx()
+ {
+ let ty = ctx.sema.type_of_expr(&Expr::RecordExpr(record_expr.clone()));
+
+ 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();
+
+ 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, &[])
+ });
+
+ 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 {
+ postfix_match: Some(CompletionRelevancePostfixMatch::Exact),
+ ..Default::default()
+ });
+ item.add_to(acc);
+ }
+ 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
}
- Some(ImmediateLocation::RecordPat(record_pat)) => {
- ctx.sema.record_pattern_missing_fields(record_pat)
- }
- _ => return None,
+ } else {
+ return None;
};
for (field, ty) in missing_fields {