]> git.lizzy.rs Git - rust.git/blob - crates/ide_completion/src/completions/pattern.rs
feat: Add very simplistic ident completion for format_args! macro input
[rust.git] / crates / ide_completion / src / completions / pattern.rs
1 //! Completes constants and paths in unqualified patterns.
2
3 use hir::db::DefDatabase;
4
5 use crate::{
6     context::{PatternContext, PatternRefutability},
7     CompletionContext, Completions,
8 };
9
10 /// Completes constants and paths in unqualified patterns.
11 pub(crate) fn complete_pattern(acc: &mut Completions, ctx: &CompletionContext) {
12     let refutable = match ctx.pattern_ctx {
13         Some(PatternContext { refutability, .. }) if ctx.path_context.is_none() => {
14             refutability == PatternRefutability::Refutable
15         }
16         _ => return,
17     };
18     let single_variant_enum = |enum_: hir::Enum| ctx.db.enum_data(enum_.into()).variants.len() == 1;
19
20     if let Some(hir::Adt::Enum(e)) =
21         ctx.expected_type.as_ref().and_then(|ty| ty.strip_references().as_adt())
22     {
23         if refutable || single_variant_enum(e) {
24             super::enum_variants_with_paths(acc, ctx, e, |acc, ctx, variant, path| {
25                 acc.add_qualified_variant_pat(ctx, variant, path.clone());
26                 acc.add_qualified_enum_variant(ctx, variant, path);
27             });
28         }
29     }
30
31     // FIXME: ideally, we should look at the type we are matching against and
32     // suggest variants + auto-imports
33     ctx.process_all_names(&mut |name, res| {
34         let add_resolution = match res {
35             hir::ScopeDef::ModuleDef(def) => match def {
36                 hir::ModuleDef::Adt(hir::Adt::Struct(strukt)) => {
37                     acc.add_struct_pat(ctx, strukt, Some(name.clone()));
38                     true
39                 }
40                 hir::ModuleDef::Variant(variant)
41                     if refutable || single_variant_enum(variant.parent_enum(ctx.db)) =>
42                 {
43                     acc.add_variant_pat(ctx, variant, Some(name.clone()));
44                     true
45                 }
46                 hir::ModuleDef::Adt(hir::Adt::Enum(e)) => refutable || single_variant_enum(e),
47                 hir::ModuleDef::Const(..) | hir::ModuleDef::Module(..) => refutable,
48                 _ => false,
49             },
50             hir::ScopeDef::MacroDef(mac) => mac.is_fn_like(),
51             hir::ScopeDef::ImplSelfType(impl_) => match impl_.self_ty(ctx.db).as_adt() {
52                 Some(hir::Adt::Struct(strukt)) => {
53                     acc.add_struct_pat(ctx, strukt, Some(name.clone()));
54                     true
55                 }
56                 Some(hir::Adt::Enum(_)) => refutable,
57                 _ => true,
58             },
59             _ => false,
60         };
61         if add_resolution {
62             acc.add_resolution(ctx, name, res);
63         }
64     });
65 }