2022-02-02 15:03:46 +01:00
|
|
|
//! Completion for use trees
|
|
|
|
|
|
|
|
use hir::ScopeDef;
|
|
|
|
use syntax::{ast, AstNode};
|
|
|
|
|
|
|
|
use crate::{
|
2022-02-02 16:01:46 +01:00
|
|
|
context::{CompletionContext, PathCompletionCtx, PathKind, PathQualifierCtx},
|
2022-02-02 15:03:46 +01:00
|
|
|
Completions,
|
|
|
|
};
|
|
|
|
|
|
|
|
pub(crate) fn complete_use_tree(acc: &mut Completions, ctx: &CompletionContext) {
|
2022-02-02 16:01:46 +01:00
|
|
|
let (is_absolute_path, qualifier) = match ctx.path_context {
|
|
|
|
Some(PathCompletionCtx {
|
2022-02-02 15:03:46 +01:00
|
|
|
kind: Some(PathKind::Use),
|
2022-02-02 16:01:46 +01:00
|
|
|
is_absolute_path,
|
2022-02-02 15:03:46 +01:00
|
|
|
ref qualifier,
|
|
|
|
..
|
2022-02-02 16:01:46 +01:00
|
|
|
}) => (is_absolute_path, qualifier),
|
2022-02-02 15:03:46 +01:00
|
|
|
_ => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
match qualifier {
|
2022-02-02 16:01:46 +01:00
|
|
|
Some(PathQualifierCtx { path, resolution, is_super_chain, use_tree_parent }) => {
|
|
|
|
if *is_super_chain {
|
2022-02-02 15:03:46 +01:00
|
|
|
acc.add_keyword(ctx, "super::");
|
|
|
|
}
|
|
|
|
// only show `self` in a new use-tree when the qualifier doesn't end in self
|
2022-02-02 16:01:46 +01:00
|
|
|
let not_preceded_by_self = *use_tree_parent
|
2022-02-02 15:03:46 +01:00
|
|
|
&& !matches!(
|
|
|
|
path.segment().and_then(|it| it.kind()),
|
|
|
|
Some(ast::PathSegmentKind::SelfKw)
|
|
|
|
);
|
|
|
|
if not_preceded_by_self {
|
|
|
|
acc.add_keyword(ctx, "self");
|
|
|
|
}
|
|
|
|
|
2022-02-02 16:01:46 +01:00
|
|
|
let resolution = match resolution {
|
2022-02-02 15:03:46 +01:00
|
|
|
Some(it) => it,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
2022-02-02 16:01:46 +01:00
|
|
|
match resolution {
|
2022-02-02 15:03:46 +01:00
|
|
|
hir::PathResolution::Def(hir::ModuleDef::Module(module)) => {
|
|
|
|
let module_scope = module.scope(ctx.db, ctx.module);
|
|
|
|
let unknown_is_current = |name: &hir::Name| {
|
|
|
|
matches!(
|
|
|
|
ctx.name_syntax.as_ref(),
|
|
|
|
Some(ast::NameLike::NameRef(name_ref))
|
|
|
|
if name_ref.syntax().text() == name.to_smol_str().as_str()
|
|
|
|
)
|
|
|
|
};
|
|
|
|
for (name, def) in module_scope {
|
|
|
|
let add_resolution = match def {
|
|
|
|
ScopeDef::Unknown if unknown_is_current(&name) => {
|
|
|
|
// for `use self::foo$0`, don't suggest `foo` as a completion
|
|
|
|
cov_mark::hit!(dont_complete_current_use);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
ScopeDef::ModuleDef(_) | ScopeDef::MacroDef(_) | ScopeDef::Unknown => {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
};
|
|
|
|
|
|
|
|
if add_resolution {
|
|
|
|
acc.add_resolution(ctx, name, def);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
hir::PathResolution::Def(hir::ModuleDef::Adt(hir::Adt::Enum(e))) => {
|
|
|
|
cov_mark::hit!(enum_plain_qualified_use_tree);
|
|
|
|
e.variants(ctx.db)
|
|
|
|
.into_iter()
|
|
|
|
.for_each(|variant| acc.add_enum_variant(ctx, variant, None));
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// fresh use tree with leading colon2, only show crate roots
|
2022-02-02 16:01:46 +01:00
|
|
|
None if is_absolute_path => {
|
2022-02-02 15:03:46 +01:00
|
|
|
cov_mark::hit!(use_tree_crate_roots_only);
|
|
|
|
ctx.process_all_names(&mut |name, res| match res {
|
|
|
|
ScopeDef::ModuleDef(hir::ModuleDef::Module(m)) if m.is_crate_root(ctx.db) => {
|
|
|
|
acc.add_resolution(ctx, name, res);
|
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// only show modules in a fresh UseTree
|
|
|
|
None => {
|
|
|
|
cov_mark::hit!(unqualified_path_only_modules_in_import);
|
|
|
|
ctx.process_all_names(&mut |name, res| {
|
|
|
|
if let ScopeDef::ModuleDef(hir::ModuleDef::Module(_)) = res {
|
|
|
|
acc.add_resolution(ctx, name, res);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
["self::", "super::", "crate::"].into_iter().for_each(|kw| acc.add_keyword(ctx, kw));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|