//! `render` module provides utilities for rendering completion suggestions //! into code pieces that will be presented to user. pub(crate) mod const_; pub(crate) mod function; pub(crate) mod literal; pub(crate) mod macro_; pub(crate) mod pattern; pub(crate) mod type_alias; pub(crate) mod union_literal; pub(crate) mod variant; use hir::{sym, AsAssocItem, HasAttrs, HirDisplay, ImportPathConfig, ModuleDef, ScopeDef, Type}; use ide_db::{ documentation::{Documentation, HasDocs}, helpers::item_name, imports::import_assets::LocatedImport, RootDatabase, SnippetCap, SymbolKind, }; use syntax::{ast, format_smolstr, AstNode, SmolStr, SyntaxKind, TextRange, ToSmolStr}; use text_edit::TextEdit; use crate::{ context::{DotAccess, DotAccessKind, PathCompletionCtx, PathKind, PatternContext}, item::{Builder, CompletionRelevanceTypeMatch}, render::{ function::render_fn, literal::render_variant_lit, macro_::{render_macro, render_macro_pat}, }, CompletionContext, CompletionItem, CompletionItemKind, CompletionRelevance, }; /// Interface for data and methods required for items rendering. #[derive(Debug, Clone)] pub(crate) struct RenderContext<'a> { completion: &'a CompletionContext<'a>, is_private_editable: bool, import_to_add: Option, doc_aliases: Vec, } impl<'a> RenderContext<'a> { pub(crate) fn new(completion: &'a CompletionContext<'a>) -> RenderContext<'a> { RenderContext { completion, is_private_editable: false, import_to_add: None, doc_aliases: vec![], } } pub(crate) fn private_editable(mut self, private_editable: bool) -> Self { self.is_private_editable = private_editable; self } pub(crate) fn import_to_add(mut self, import_to_add: Option) -> Self { self.import_to_add = import_to_add; self } pub(crate) fn doc_aliases(mut self, doc_aliases: Vec) -> Self { self.doc_aliases = doc_aliases; self } fn snippet_cap(&self) -> Option { self.completion.config.snippet_cap } fn db(&self) -> &'a RootDatabase { self.completion.db } fn source_range(&self) -> TextRange { self.completion.source_range() } fn completion_relevance(&self) -> CompletionRelevance { CompletionRelevance { is_private_editable: self.is_private_editable, requires_import: self.import_to_add.is_some(), ..Default::default() } } fn is_immediately_after_macro_bang(&self) -> bool { self.completion.token.kind() == SyntaxKind::BANG && self .completion .token .parent() .map_or(false, |it| it.kind() == SyntaxKind::MACRO_CALL) } fn is_deprecated(&self, def: impl HasAttrs) -> bool { let attrs = def.attrs(self.db()); attrs.by_key(&sym::deprecated).exists() } fn is_deprecated_assoc_item(&self, as_assoc_item: impl AsAssocItem) -> bool { let db = self.db(); let assoc = match as_assoc_item.as_assoc_item(db) { Some(assoc) => assoc, None => return false, }; let is_assoc_deprecated = match assoc { hir::AssocItem::Function(it) => self.is_deprecated(it), hir::AssocItem::Const(it) => self.is_deprecated(it), hir::AssocItem::TypeAlias(it) => self.is_deprecated(it), }; is_assoc_deprecated || assoc .container_or_implemented_trait(db) .map(|trait_| self.is_deprecated(trait_)) .unwrap_or(false) } // FIXME: remove this fn docs(&self, def: impl HasDocs) -> Option { def.docs(self.db()) } } pub(crate) fn render_field( ctx: RenderContext<'_>, dot_access: &DotAccess, receiver: Option, field: hir::Field, ty: &hir::Type, ) -> CompletionItem { let db = ctx.db(); let is_deprecated = ctx.is_deprecated(field); let name = field.name(db); let (name, escaped_name) = (name.unescaped().display(db).to_smolstr(), name.display_no_db().to_smolstr()); let mut item = CompletionItem::new( SymbolKind::Field, ctx.source_range(), field_with_receiver(db, receiver.as_ref(), &name), ); item.set_relevance(CompletionRelevance { type_match: compute_type_match(ctx.completion, ty), exact_name_match: compute_exact_name_match(ctx.completion, name.as_str()), ..CompletionRelevance::default() }); item.detail(ty.display(db).to_string()) .set_documentation(field.docs(db)) .set_deprecated(is_deprecated) .lookup_by(name); let is_field_access = matches!(dot_access.kind, DotAccessKind::Field { .. }); if !is_field_access || ty.is_fn() || ty.is_closure() { let mut builder = TextEdit::builder(); // Using TextEdit, insert '(' before the struct name and ')' before the // dot access, then comes the field name and optionally insert function // call parens. builder.replace( ctx.source_range(), field_with_receiver(db, receiver.as_ref(), &escaped_name).into(), ); let expected_fn_type = ctx.completion.expected_type.as_ref().is_some_and(|ty| ty.is_fn() || ty.is_closure()); if !expected_fn_type { if let Some(receiver) = &dot_access.receiver { if let Some(receiver) = ctx.completion.sema.original_ast_node(receiver.clone()) { builder.insert(receiver.syntax().text_range().start(), "(".to_owned()); builder.insert(ctx.source_range().end(), ")".to_owned()); let is_parens_needed = !matches!(dot_access.kind, DotAccessKind::Method { has_parens: true }); if is_parens_needed { builder.insert(ctx.source_range().end(), "()".to_owned()); } } } } item.text_edit(builder.finish()); } else { item.insert_text(field_with_receiver(db, receiver.as_ref(), &escaped_name)); } if let Some(receiver) = &dot_access.receiver { if let Some(original) = ctx.completion.sema.original_ast_node(receiver.clone()) { if let Some(ref_match) = compute_ref_match(ctx.completion, ty) { item.ref_match(ref_match, original.syntax().text_range().start()); } } } item.doc_aliases(ctx.doc_aliases); item.build(db) } fn field_with_receiver( db: &RootDatabase, receiver: Option<&hir::Name>, field_name: &str, ) -> SmolStr { receiver.map_or_else( || field_name.into(), |receiver| format_smolstr!("{}.{field_name}", receiver.display(db)), ) } pub(crate) fn render_tuple_field( ctx: RenderContext<'_>, receiver: Option, field: usize, ty: &hir::Type, ) -> CompletionItem { let mut item = CompletionItem::new( SymbolKind::Field, ctx.source_range(), field_with_receiver(ctx.db(), receiver.as_ref(), &field.to_string()), ); item.detail(ty.display(ctx.db()).to_string()).lookup_by(field.to_string()); item.build(ctx.db()) } pub(crate) fn render_type_inference( ty_string: String, ctx: &CompletionContext<'_>, ) -> CompletionItem { let mut builder = CompletionItem::new(CompletionItemKind::InferredType, ctx.source_range(), ty_string); builder.set_relevance(CompletionRelevance { is_definite: true, ..Default::default() }); builder.build(ctx.db) } pub(crate) fn render_path_resolution( ctx: RenderContext<'_>, path_ctx: &PathCompletionCtx, local_name: hir::Name, resolution: ScopeDef, ) -> Builder { render_resolution_path(ctx, path_ctx, local_name, None, resolution) } pub(crate) fn render_pattern_resolution( ctx: RenderContext<'_>, pattern_ctx: &PatternContext, local_name: hir::Name, resolution: ScopeDef, ) -> Builder { render_resolution_pat(ctx, pattern_ctx, local_name, None, resolution) } pub(crate) fn render_resolution_with_import( ctx: RenderContext<'_>, path_ctx: &PathCompletionCtx, import_edit: LocatedImport, ) -> Option { let resolution = ScopeDef::from(import_edit.original_item); let local_name = scope_def_to_name(resolution, &ctx, &import_edit)?; //this now just renders the alias text, but we need to find the aliases earlier and call this with the alias instead let doc_aliases = ctx.completion.doc_aliases_in_scope(resolution); let ctx = ctx.doc_aliases(doc_aliases); Some(render_resolution_path(ctx, path_ctx, local_name, Some(import_edit), resolution)) } pub(crate) fn render_resolution_with_import_pat( ctx: RenderContext<'_>, pattern_ctx: &PatternContext, import_edit: LocatedImport, ) -> Option { let resolution = ScopeDef::from(import_edit.original_item); let local_name = scope_def_to_name(resolution, &ctx, &import_edit)?; Some(render_resolution_pat(ctx, pattern_ctx, local_name, Some(import_edit), resolution)) } pub(crate) fn render_expr( ctx: &CompletionContext<'_>, expr: &hir::term_search::Expr, ) -> Option { let mut i = 1; let mut snippet_formatter = |ty: &hir::Type| { let arg_name = ty .as_adt() .map(|adt| stdx::to_lower_snake_case(adt.name(ctx.db).as_str())) .unwrap_or_else(|| String::from("_")); let res = format!("${{{i}:{arg_name}}}"); i += 1; res }; let mut label_formatter = |ty: &hir::Type| { ty.as_adt() .map(|adt| stdx::to_lower_snake_case(adt.name(ctx.db).as_str())) .unwrap_or_else(|| String::from("...")) }; let cfg = ImportPathConfig { prefer_no_std: ctx.config.prefer_no_std, prefer_prelude: ctx.config.prefer_prelude, prefer_absolute: ctx.config.prefer_absolute, }; let label = expr.gen_source_code(&ctx.scope, &mut label_formatter, cfg).ok()?; let source_range = match ctx.original_token.parent() { Some(node) => match node.ancestors().find_map(ast::Path::cast) { Some(path) => path.syntax().text_range(), None => node.text_range(), }, None => ctx.source_range(), }; let mut item = CompletionItem::new(CompletionItemKind::Expression, source_range, label); let snippet = format!("{}$0", expr.gen_source_code(&ctx.scope, &mut snippet_formatter, cfg).ok()?); let edit = TextEdit::replace(source_range, snippet); item.snippet_edit(ctx.config.snippet_cap?, edit); item.documentation(Documentation::new(String::from("Autogenerated expression by term search"))); item.set_relevance(crate::CompletionRelevance { type_match: compute_type_match(ctx, &expr.ty(ctx.db)), ..Default::default() }); for trait_ in expr.traits_used(ctx.db) { let trait_item = hir::ItemInNs::from(hir::ModuleDef::from(trait_)); let Some(path) = ctx.module.find_path(ctx.db, trait_item, cfg) else { continue; }; item.add_import(LocatedImport::new(path, trait_item, trait_item)); } Some(item) } fn scope_def_to_name( resolution: ScopeDef, ctx: &RenderContext<'_>, import_edit: &LocatedImport, ) -> Option { Some(match resolution { ScopeDef::ModuleDef(hir::ModuleDef::Function(f)) => f.name(ctx.completion.db), ScopeDef::ModuleDef(hir::ModuleDef::Const(c)) => c.name(ctx.completion.db)?, ScopeDef::ModuleDef(hir::ModuleDef::TypeAlias(t)) => t.name(ctx.completion.db), _ => item_name(ctx.db(), import_edit.original_item)?, }) } fn render_resolution_pat( ctx: RenderContext<'_>, pattern_ctx: &PatternContext, local_name: hir::Name, import_to_add: Option, resolution: ScopeDef, ) -> Builder { let _p = tracing::info_span!("render_resolution_pat").entered(); use hir::ModuleDef::*; if let ScopeDef::ModuleDef(Macro(mac)) = resolution { let ctx = ctx.import_to_add(import_to_add); render_macro_pat(ctx, pattern_ctx, local_name, mac) } else { render_resolution_simple_(ctx, &local_name, import_to_add, resolution) } } fn render_resolution_path( ctx: RenderContext<'_>, path_ctx: &PathCompletionCtx, local_name: hir::Name, import_to_add: Option, resolution: ScopeDef, ) -> Builder { let _p = tracing::info_span!("render_resolution_path").entered(); use hir::ModuleDef::*; match resolution { ScopeDef::ModuleDef(Macro(mac)) => { let ctx = ctx.import_to_add(import_to_add); return render_macro(ctx, path_ctx, local_name, mac); } ScopeDef::ModuleDef(Function(func)) => { let ctx = ctx.import_to_add(import_to_add); return render_fn(ctx, path_ctx, Some(local_name), func); } ScopeDef::ModuleDef(Variant(var)) => { let ctx = ctx.clone().import_to_add(import_to_add.clone()); if let Some(item) = render_variant_lit(ctx, path_ctx, Some(local_name.clone()), var, None) { return item; } } _ => (), } let completion = ctx.completion; let cap = ctx.snippet_cap(); let db = completion.db; let config = completion.config; let requires_import = import_to_add.is_some(); let name = local_name.display_no_db().to_smolstr(); let mut item = render_resolution_simple_(ctx, &local_name, import_to_add, resolution); if local_name.is_escaped() { item.insert_text(local_name.display_no_db().to_smolstr()); } // Add `<>` for generic types let type_path_no_ty_args = matches!( path_ctx, PathCompletionCtx { kind: PathKind::Type { .. }, has_type_args: false, .. } ) && config.callable.is_some(); if type_path_no_ty_args { if let Some(cap) = cap { let has_non_default_type_params = match resolution { ScopeDef::ModuleDef(hir::ModuleDef::Adt(it)) => it.has_non_default_type_params(db), ScopeDef::ModuleDef(hir::ModuleDef::TypeAlias(it)) => { it.has_non_default_type_params(db) } _ => false, }; if has_non_default_type_params { cov_mark::hit!(inserts_angle_brackets_for_generics); item.lookup_by(name.clone()) .label(SmolStr::from_iter([&name, "<…>"])) .trigger_call_info() .insert_snippet(cap, format!("{}<$0>", local_name.display(db))); } } } let mut set_item_relevance = |ty: Type| { if !ty.is_unknown() { item.detail(ty.display(db).to_string()); } item.set_relevance(CompletionRelevance { type_match: compute_type_match(completion, &ty), exact_name_match: compute_exact_name_match(completion, &name), is_local: matches!(resolution, ScopeDef::Local(_)), requires_import, ..CompletionRelevance::default() }); path_ref_match(completion, path_ctx, &ty, &mut item); }; match resolution { ScopeDef::Local(local) => set_item_relevance(local.ty(db)), ScopeDef::ModuleDef(ModuleDef::Adt(adt)) | ScopeDef::AdtSelfType(adt) => { set_item_relevance(adt.ty(db)) } // Filtered out above ScopeDef::ModuleDef( ModuleDef::Function(_) | ModuleDef::Variant(_) | ModuleDef::Macro(_), ) => (), ScopeDef::ModuleDef(ModuleDef::Const(konst)) => set_item_relevance(konst.ty(db)), ScopeDef::ModuleDef(ModuleDef::Static(stat)) => set_item_relevance(stat.ty(db)), ScopeDef::ModuleDef(ModuleDef::BuiltinType(bt)) => set_item_relevance(bt.ty(db)), ScopeDef::ImplSelfType(imp) => set_item_relevance(imp.self_ty(db)), ScopeDef::GenericParam(_) | ScopeDef::Label(_) | ScopeDef::Unknown | ScopeDef::ModuleDef( ModuleDef::Trait(_) | ModuleDef::TraitAlias(_) | ModuleDef::Module(_) | ModuleDef::TypeAlias(_), ) => (), }; item } fn render_resolution_simple_( ctx: RenderContext<'_>, local_name: &hir::Name, import_to_add: Option, resolution: ScopeDef, ) -> Builder { let _p = tracing::info_span!("render_resolution_simple_").entered(); let db = ctx.db(); let ctx = ctx.import_to_add(import_to_add); let kind = res_to_kind(resolution); let mut item = CompletionItem::new( kind, ctx.source_range(), local_name.unescaped().display(db).to_smolstr(), ); item.set_relevance(ctx.completion_relevance()) .set_documentation(scope_def_docs(db, resolution)) .set_deprecated(scope_def_is_deprecated(&ctx, resolution)); if let Some(import_to_add) = ctx.import_to_add { item.add_import(import_to_add); } item.doc_aliases(ctx.doc_aliases); item } fn res_to_kind(resolution: ScopeDef) -> CompletionItemKind { use hir::ModuleDef::*; match resolution { ScopeDef::Unknown => CompletionItemKind::UnresolvedReference, ScopeDef::ModuleDef(Function(_)) => CompletionItemKind::SymbolKind(SymbolKind::Function), ScopeDef::ModuleDef(Variant(_)) => CompletionItemKind::SymbolKind(SymbolKind::Variant), ScopeDef::ModuleDef(Macro(_)) => CompletionItemKind::SymbolKind(SymbolKind::Macro), ScopeDef::ModuleDef(Module(..)) => CompletionItemKind::SymbolKind(SymbolKind::Module), ScopeDef::ModuleDef(Adt(adt)) => CompletionItemKind::SymbolKind(match adt { hir::Adt::Struct(_) => SymbolKind::Struct, hir::Adt::Union(_) => SymbolKind::Union, hir::Adt::Enum(_) => SymbolKind::Enum, }), ScopeDef::ModuleDef(Const(..)) => CompletionItemKind::SymbolKind(SymbolKind::Const), ScopeDef::ModuleDef(Static(..)) => CompletionItemKind::SymbolKind(SymbolKind::Static), ScopeDef::ModuleDef(Trait(..)) => CompletionItemKind::SymbolKind(SymbolKind::Trait), ScopeDef::ModuleDef(TraitAlias(..)) => { CompletionItemKind::SymbolKind(SymbolKind::TraitAlias) } ScopeDef::ModuleDef(TypeAlias(..)) => CompletionItemKind::SymbolKind(SymbolKind::TypeAlias), ScopeDef::ModuleDef(BuiltinType(..)) => CompletionItemKind::BuiltinType, ScopeDef::GenericParam(param) => CompletionItemKind::SymbolKind(match param { hir::GenericParam::TypeParam(_) => SymbolKind::TypeParam, hir::GenericParam::ConstParam(_) => SymbolKind::ConstParam, hir::GenericParam::LifetimeParam(_) => SymbolKind::LifetimeParam, }), ScopeDef::Local(..) => CompletionItemKind::SymbolKind(SymbolKind::Local), ScopeDef::Label(..) => CompletionItemKind::SymbolKind(SymbolKind::Label), ScopeDef::AdtSelfType(..) | ScopeDef::ImplSelfType(..) => { CompletionItemKind::SymbolKind(SymbolKind::SelfParam) } } } fn scope_def_docs(db: &RootDatabase, resolution: ScopeDef) -> Option { use hir::ModuleDef::*; match resolution { ScopeDef::ModuleDef(Module(it)) => it.docs(db), ScopeDef::ModuleDef(Adt(it)) => it.docs(db), ScopeDef::ModuleDef(Variant(it)) => it.docs(db), ScopeDef::ModuleDef(Const(it)) => it.docs(db), ScopeDef::ModuleDef(Static(it)) => it.docs(db), ScopeDef::ModuleDef(Trait(it)) => it.docs(db), ScopeDef::ModuleDef(TypeAlias(it)) => it.docs(db), _ => None, } } fn scope_def_is_deprecated(ctx: &RenderContext<'_>, resolution: ScopeDef) -> bool { match resolution { ScopeDef::ModuleDef(it) => ctx.is_deprecated_assoc_item(it), ScopeDef::GenericParam(it) => ctx.is_deprecated(it), ScopeDef::AdtSelfType(it) => ctx.is_deprecated(it), _ => false, } } // FIXME: This checks types without possible coercions which some completions might want to do fn match_types( ctx: &CompletionContext<'_>, ty1: &hir::Type, ty2: &hir::Type, ) -> Option { if ty1 == ty2 { Some(CompletionRelevanceTypeMatch::Exact) } else if ty1.could_unify_with(ctx.db, ty2) { Some(CompletionRelevanceTypeMatch::CouldUnify) } else { None } } fn compute_type_match( ctx: &CompletionContext<'_>, completion_ty: &hir::Type, ) -> Option { let expected_type = ctx.expected_type.as_ref()?; // We don't ever consider unit type to be an exact type match, since // nearly always this is not meaningful to the user. if expected_type.is_unit() { return None; } match_types(ctx, expected_type, completion_ty) } fn compute_exact_name_match(ctx: &CompletionContext<'_>, completion_name: &str) -> bool { ctx.expected_name.as_ref().map_or(false, |name| name.text() == completion_name) } fn compute_ref_match( ctx: &CompletionContext<'_>, completion_ty: &hir::Type, ) -> Option { let expected_type = ctx.expected_type.as_ref()?; if completion_ty != expected_type { let expected_type_without_ref = expected_type.remove_ref()?; if completion_ty.autoderef(ctx.db).any(|deref_ty| deref_ty == expected_type_without_ref) { cov_mark::hit!(suggest_ref); let mutability = if expected_type.is_mutable_reference() { hir::Mutability::Mut } else { hir::Mutability::Shared }; return Some(mutability); }; } None } fn path_ref_match( completion: &CompletionContext<'_>, path_ctx: &PathCompletionCtx, ty: &hir::Type, item: &mut Builder, ) { if let Some(original_path) = &path_ctx.original_path { // At least one char was typed by the user already, in that case look for the original path if let Some(original_path) = completion.sema.original_ast_node(original_path.clone()) { if let Some(ref_match) = compute_ref_match(completion, ty) { item.ref_match(ref_match, original_path.syntax().text_range().start()); } } } else { // completion requested on an empty identifier, there is no path here yet. // FIXME: This might create inconsistent completions where we show a ref match in macro inputs // as long as nothing was typed yet if let Some(ref_match) = compute_ref_match(completion, ty) { item.ref_match(ref_match, completion.position.offset); } } } #[cfg(test)] mod tests { use std::cmp; use expect_test::{expect, Expect}; use ide_db::SymbolKind; use itertools::Itertools; use crate::{ item::CompletionRelevanceTypeMatch, tests::{check_edit, do_completion, get_all_items, TEST_CONFIG}, CompletionItem, CompletionItemKind, CompletionRelevance, CompletionRelevancePostfixMatch, }; #[track_caller] fn check(ra_fixture: &str, kind: impl Into, expect: Expect) { let actual = do_completion(ra_fixture, kind.into()); expect.assert_debug_eq(&actual); } #[track_caller] fn check_kinds(ra_fixture: &str, kinds: &[CompletionItemKind], expect: Expect) { let actual: Vec<_> = kinds.iter().flat_map(|&kind| do_completion(ra_fixture, kind)).collect(); expect.assert_debug_eq(&actual); } #[track_caller] fn check_function_relevance(ra_fixture: &str, expect: Expect) { let actual: Vec<_> = do_completion(ra_fixture, CompletionItemKind::SymbolKind(SymbolKind::Method)) .into_iter() .map(|item| (item.detail.unwrap_or_default(), item.relevance.function)) .collect(); expect.assert_debug_eq(&actual); } #[track_caller] fn check_relevance_for_kinds(ra_fixture: &str, kinds: &[CompletionItemKind], expect: Expect) { let mut actual = get_all_items(TEST_CONFIG, ra_fixture, None); actual.retain(|it| kinds.contains(&it.kind)); actual.sort_by_key(|it| cmp::Reverse(it.relevance.score())); check_relevance_(actual, expect); } #[track_caller] fn check_relevance(ra_fixture: &str, expect: Expect) { let mut actual = get_all_items(TEST_CONFIG, ra_fixture, None); actual.retain(|it| it.kind != CompletionItemKind::Snippet); actual.retain(|it| it.kind != CompletionItemKind::Keyword); actual.retain(|it| it.kind != CompletionItemKind::BuiltinType); actual.sort_by_key(|it| cmp::Reverse(it.relevance.score())); check_relevance_(actual, expect); } #[track_caller] fn check_relevance_(actual: Vec, expect: Expect) { let actual = actual .into_iter() .flat_map(|it| { let mut items = vec![]; let tag = it.kind.tag(); let relevance = display_relevance(it.relevance); items.push(format!( "{tag} {}{} {relevance}\n", it.label, it.label_detail.clone().unwrap_or_default(), )); if let Some((label, _indel, relevance)) = it.ref_match() { let relevance = display_relevance(relevance); items.push(format!("{tag} {label} {relevance}\n")); } items }) .collect::(); expect.assert_eq(&actual); fn display_relevance(relevance: CompletionRelevance) -> String { let relevance_factors = vec![ (relevance.type_match == Some(CompletionRelevanceTypeMatch::Exact), "type"), ( relevance.type_match == Some(CompletionRelevanceTypeMatch::CouldUnify), "type_could_unify", ), (relevance.exact_name_match, "name"), (relevance.is_local, "local"), ( relevance.postfix_match == Some(CompletionRelevancePostfixMatch::Exact), "snippet", ), (relevance.is_op_method, "op_method"), (relevance.requires_import, "requires_import"), ] .into_iter() .filter_map(|(cond, desc)| if cond { Some(desc) } else { None }) .join("+"); format!("[{relevance_factors}]") } } #[test] fn set_struct_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub struct Struct {} } pub mod test_mod_a { pub struct Struct {} } //- /main.rs crate:main deps:dep fn test(input: dep::test_mod_b::Struct) { } fn main() { test(Struct$0); } "#, expect![[r#" st dep::test_mod_b::Struct {…} [type_could_unify] ex dep::test_mod_b::Struct { } [type_could_unify] st Struct (use dep::test_mod_b::Struct) [type_could_unify+requires_import] fn main() [] fn test(…) [] md dep [] st Struct (use dep::test_mod_a::Struct) [requires_import] "#]], ); } #[test] fn set_union_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub union Union { a: i32, b: i32 } } pub mod test_mod_a { pub enum Union { a: i32, b: i32 } } //- /main.rs crate:main deps:dep fn test(input: dep::test_mod_b::Union) { } fn main() { test(Union$0); } "#, expect![[r#" un Union (use dep::test_mod_b::Union) [type_could_unify+requires_import] fn main() [] fn test(…) [] md dep [] en Union (use dep::test_mod_a::Union) [requires_import] "#]], ); } #[test] fn set_enum_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub enum Enum { variant } } pub mod test_mod_a { pub enum Enum { variant } } //- /main.rs crate:main deps:dep fn test(input: dep::test_mod_b::Enum) { } fn main() { test(Enum$0); } "#, expect![[r#" ev dep::test_mod_b::Enum::variant [type_could_unify] ex dep::test_mod_b::Enum::variant [type_could_unify] en Enum (use dep::test_mod_b::Enum) [type_could_unify+requires_import] fn main() [] fn test(…) [] md dep [] en Enum (use dep::test_mod_a::Enum) [requires_import] "#]], ); } #[test] fn set_enum_variant_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub enum Enum { Variant } } pub mod test_mod_a { pub enum Enum { Variant } } //- /main.rs crate:main deps:dep fn test(input: dep::test_mod_b::Enum) { } fn main() { test(Variant$0); } "#, expect![[r#" ev dep::test_mod_b::Enum::Variant [type_could_unify] ex dep::test_mod_b::Enum::Variant [type_could_unify] fn main() [] fn test(…) [] md dep [] "#]], ); } #[test] fn set_fn_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub fn function(j: isize) -> i32 {} } pub mod test_mod_a { pub fn function(i: usize) -> i32 {} } //- /main.rs crate:main deps:dep fn test(input: fn(usize) -> i32) { } fn main() { test(function$0); } "#, expect![[r#" fn main() [] fn test(…) [] md dep [] fn function (use dep::test_mod_a::function) [requires_import] fn function(…) (use dep::test_mod_b::function) [requires_import] "#]], ); } #[test] fn set_const_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub const CONST: i32 = 1; } pub mod test_mod_a { pub const CONST: i64 = 2; } //- /main.rs crate:main deps:dep fn test(input: i32) { } fn main() { test(CONST$0); } "#, expect![[r#" ct CONST (use dep::test_mod_b::CONST) [type_could_unify+requires_import] fn main() [] fn test(…) [] md dep [] ct CONST (use dep::test_mod_a::CONST) [requires_import] "#]], ); } #[test] fn set_static_type_completion_info() { check_relevance( r#" //- /lib.rs crate:dep pub mod test_mod_b { pub static STATIC: i32 = 5; } pub mod test_mod_a { pub static STATIC: i64 = 5; } //- /main.rs crate:main deps:dep fn test(input: i32) { } fn main() { test(STATIC$0); } "#, expect![[r#" sc STATIC (use dep::test_mod_b::STATIC) [type_could_unify+requires_import] fn main() [] fn test(…) [] md dep [] sc STATIC (use dep::test_mod_a::STATIC) [requires_import] "#]], ); } #[test] fn set_self_type_completion_info_with_params() { check_relevance( r#" //- /lib.rs crate:dep pub struct Struct; impl Struct { pub fn Function(&self, input: i32) -> bool { false } } //- /main.rs crate:main deps:dep use dep::Struct; fn test(input: fn(&dep::Struct, i32) -> bool) { } fn main() { test(Struct::Function$0); } "#, expect![[r#" me Function [] "#]], ); } #[test] fn set_self_type_completion_info() { check_relevance( r#" //- /main.rs crate:main struct Struct; impl Struct { fn test(&self) { func(Self$0); } } fn func(input: Struct) { } "#, expect![[r#" st Struct [type] st Self [type] sp Self [type] st Struct [type] ex Struct [type] lc self [local] fn func(…) [] me self.test() [] "#]], ); } #[test] fn set_builtin_type_completion_info() { check_relevance( r#" //- /main.rs crate:main fn test(input: bool) { } pub Input: bool = false; fn main() { let input = false; let inputbad = 3; test(inp$0); } "#, expect![[r#" lc input [type+name+local] ex input [type] ex true [type] ex false [type] lc inputbad [local] fn main() [] fn test(…) [] "#]], ); } #[test] fn enum_detail_includes_record_fields() { check( r#" enum Foo { Foo { x: i32, y: i32 } } fn main() { Foo::Fo$0 } "#, SymbolKind::Variant, expect![[r#" [ CompletionItem { label: "Foo {…}", source_range: 54..56, delete: 54..56, insert: "Foo { x: ${1:()}, y: ${2:()} }$0", kind: SymbolKind( Variant, ), lookup: "Foo{}", detail: "Foo { x: i32, y: i32 }", trigger_call_info: true, }, ] "#]], ); } #[test] fn enum_detail_includes_tuple_fields() { check( r#" enum Foo { Foo (i32, i32) } fn main() { Foo::Fo$0 } "#, SymbolKind::Variant, expect![[r#" [ CompletionItem { label: "Foo(…)", source_range: 46..48, delete: 46..48, insert: "Foo(${1:()}, ${2:()})$0", kind: SymbolKind( Variant, ), lookup: "Foo()", detail: "Foo(i32, i32)", trigger_call_info: true, }, ] "#]], ); } #[test] fn fn_detail_includes_args_and_return_type() { check( r#" fn foo(a: u32, b: u32, t: T) -> (u32, T) { (a, t) } fn main() { fo$0 } "#, SymbolKind::Function, expect![[r#" [ CompletionItem { label: "foo(…)", source_range: 68..70, delete: 68..70, insert: "foo(${1:a}, ${2:b}, ${3:t})$0", kind: SymbolKind( Function, ), lookup: "foo", detail: "fn(u32, u32, T) -> (u32, T)", trigger_call_info: true, }, CompletionItem { label: "main()", source_range: 68..70, delete: 68..70, insert: "main()$0", kind: SymbolKind( Function, ), lookup: "main", detail: "fn()", }, ] "#]], ); } #[test] fn enum_detail_just_name_for_unit() { check( r#" enum Foo { Foo } fn main() { Foo::Fo$0 } "#, SymbolKind::Variant, expect![[r#" [ CompletionItem { label: "Foo", source_range: 35..37, delete: 35..37, insert: "Foo$0", kind: SymbolKind( Variant, ), detail: "Foo", trigger_call_info: true, }, ] "#]], ); } #[test] fn lookup_enums_by_two_qualifiers() { check_kinds( r#" mod m { pub enum Spam { Foo, Bar(i32) } } fn main() { let _: m::Spam = S$0 } "#, &[ CompletionItemKind::SymbolKind(SymbolKind::Function), CompletionItemKind::SymbolKind(SymbolKind::Module), CompletionItemKind::SymbolKind(SymbolKind::Variant), ], expect![[r#" [ CompletionItem { label: "main()", source_range: 75..76, delete: 75..76, insert: "main()$0", kind: SymbolKind( Function, ), lookup: "main", detail: "fn()", }, CompletionItem { label: "m", source_range: 75..76, delete: 75..76, insert: "m", kind: SymbolKind( Module, ), }, CompletionItem { label: "m::Spam::Bar(…)", source_range: 75..76, delete: 75..76, insert: "m::Spam::Bar(${1:()})$0", kind: SymbolKind( Variant, ), lookup: "Spam::Bar()", detail: "m::Spam::Bar(i32)", relevance: CompletionRelevance { exact_name_match: false, type_match: Some( Exact, ), is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, trigger_call_info: true, }, CompletionItem { label: "m::Spam::Foo", source_range: 75..76, delete: 75..76, insert: "m::Spam::Foo$0", kind: SymbolKind( Variant, ), lookup: "Spam::Foo", detail: "m::Spam::Foo", relevance: CompletionRelevance { exact_name_match: false, type_match: Some( Exact, ), is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, trigger_call_info: true, }, ] "#]], ) } #[test] fn sets_deprecated_flag_in_items() { check( r#" #[deprecated] fn something_deprecated() {} fn main() { som$0 } "#, SymbolKind::Function, expect![[r#" [ CompletionItem { label: "main()", source_range: 56..59, delete: 56..59, insert: "main()$0", kind: SymbolKind( Function, ), lookup: "main", detail: "fn()", }, CompletionItem { label: "something_deprecated()", source_range: 56..59, delete: 56..59, insert: "something_deprecated()$0", kind: SymbolKind( Function, ), lookup: "something_deprecated", detail: "fn()", deprecated: true, }, ] "#]], ); check( r#" struct A { #[deprecated] the_field: u32 } fn foo() { A { the$0 } } "#, SymbolKind::Field, expect![[r#" [ CompletionItem { label: "the_field", source_range: 57..60, delete: 57..60, insert: "the_field", kind: SymbolKind( Field, ), detail: "u32", deprecated: true, relevance: CompletionRelevance { exact_name_match: false, type_match: Some( CouldUnify, ), is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, }, ] "#]], ); } #[test] fn renders_docs() { check_kinds( r#" struct S { /// Field docs foo: } impl S { /// Method docs fn bar(self) { self.$0 } }"#, &[ CompletionItemKind::SymbolKind(SymbolKind::Method), CompletionItemKind::SymbolKind(SymbolKind::Field), ], expect![[r#" [ CompletionItem { label: "bar()", source_range: 94..94, delete: 94..94, insert: "bar()$0", kind: SymbolKind( Method, ), lookup: "bar", detail: "fn(self)", documentation: Documentation( "Method docs", ), relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Other, }, ), }, }, CompletionItem { label: "foo", source_range: 94..94, delete: 94..94, insert: "foo", kind: SymbolKind( Field, ), detail: "{unknown}", documentation: Documentation( "Field docs", ), }, ] "#]], ); check_kinds( r#" use self::my$0; /// mod docs mod my { } /// enum docs enum E { /// variant docs V } use self::E::*; "#, &[ CompletionItemKind::SymbolKind(SymbolKind::Module), CompletionItemKind::SymbolKind(SymbolKind::Variant), CompletionItemKind::SymbolKind(SymbolKind::Enum), ], expect![[r#" [ CompletionItem { label: "my", source_range: 10..12, delete: 10..12, insert: "my", kind: SymbolKind( Module, ), documentation: Documentation( "mod docs", ), }, CompletionItem { label: "V", source_range: 10..12, delete: 10..12, insert: "V$0", kind: SymbolKind( Variant, ), detail: "V", documentation: Documentation( "variant docs", ), trigger_call_info: true, }, CompletionItem { label: "E", source_range: 10..12, delete: 10..12, insert: "E", kind: SymbolKind( Enum, ), detail: "E", documentation: Documentation( "enum docs", ), }, ] "#]], ) } #[test] fn dont_render_attrs() { check( r#" struct S; impl S { #[inline] fn the_method(&self) { } } fn foo(s: S) { s.$0 } "#, CompletionItemKind::SymbolKind(SymbolKind::Method), expect![[r#" [ CompletionItem { label: "the_method()", source_range: 81..81, delete: 81..81, insert: "the_method()$0", kind: SymbolKind( Method, ), lookup: "the_method", detail: "fn(&self)", relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Other, }, ), }, }, ] "#]], ) } #[test] fn no_call_parens_if_fn_ptr_needed() { cov_mark::check!(no_call_parens_if_fn_ptr_needed); check_edit( "foo", r#" fn foo(foo: u8, bar: u8) {} struct ManualVtable { f: fn(u8, u8) } fn main() -> ManualVtable { ManualVtable { f: f$0 } } "#, r#" fn foo(foo: u8, bar: u8) {} struct ManualVtable { f: fn(u8, u8) } fn main() -> ManualVtable { ManualVtable { f: foo } } "#, ); check_edit( "type", r#" struct RawIdentTable { r#type: u32 } fn main() -> RawIdentTable { RawIdentTable { t$0: 42 } } "#, r#" struct RawIdentTable { r#type: u32 } fn main() -> RawIdentTable { RawIdentTable { r#type: 42 } } "#, ); } #[test] fn no_parens_in_use_item() { check_edit( "foo", r#" mod m { pub fn foo() {} } use crate::m::f$0; "#, r#" mod m { pub fn foo() {} } use crate::m::foo; "#, ); } #[test] fn no_parens_in_call() { check_edit( "foo", r#" fn foo(x: i32) {} fn main() { f$0(); } "#, r#" fn foo(x: i32) {} fn main() { foo(); } "#, ); check_edit( "foo", r#" struct Foo; impl Foo { fn foo(&self){} } fn f(foo: &Foo) { foo.f$0(); } "#, r#" struct Foo; impl Foo { fn foo(&self){} } fn f(foo: &Foo) { foo.foo(); } "#, ); } #[test] fn inserts_angle_brackets_for_generics() { cov_mark::check!(inserts_angle_brackets_for_generics); check_edit( "Vec", r#" struct Vec {} fn foo(xs: Ve$0) "#, r#" struct Vec {} fn foo(xs: Vec<$0>) "#, ); check_edit( "Vec", r#" type Vec = (T,); fn foo(xs: Ve$0) "#, r#" type Vec = (T,); fn foo(xs: Vec<$0>) "#, ); check_edit( "Vec", r#" struct Vec {} fn foo(xs: Ve$0) "#, r#" struct Vec {} fn foo(xs: Vec) "#, ); check_edit( "Vec", r#" struct Vec {} fn foo(xs: Ve$0) "#, r#" struct Vec {} fn foo(xs: Vec) "#, ); } #[test] fn active_param_relevance() { check_relevance( r#" struct S { foo: i64, bar: u32, baz: u32 } fn test(bar: u32) { } fn foo(s: S) { test(s.$0) } "#, expect![[r#" fd bar [type+name] fd baz [type] fd foo [] "#]], ); } #[test] fn record_field_relevances() { check_relevance( r#" struct A { foo: i64, bar: u32, baz: u32 } struct B { x: (), y: f32, bar: u32 } fn foo(a: A) { B { bar: a.$0 }; } "#, expect![[r#" fd bar [type+name] fd baz [type] fd foo [] "#]], ) } #[test] fn tuple_field_detail() { check( r#" struct S(i32); fn f() -> i32 { let s = S(0); s.0$0 } "#, SymbolKind::Field, expect![[r#" [ CompletionItem { label: "0", source_range: 56..57, delete: 56..57, insert: "0", kind: SymbolKind( Field, ), detail: "i32", relevance: CompletionRelevance { exact_name_match: false, type_match: Some( Exact, ), is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, }, ] "#]], ); } #[test] fn record_field_and_call_relevances() { check_relevance( r#" struct A { foo: i64, bar: u32, baz: u32 } struct B { x: (), y: f32, bar: u32 } fn f(foo: i64) { } fn foo(a: A) { B { bar: f(a.$0) }; } "#, expect![[r#" fd foo [type+name] fd bar [] fd baz [] "#]], ); check_relevance( r#" struct A { foo: i64, bar: u32, baz: u32 } struct B { x: (), y: f32, bar: u32 } fn f(foo: i64) { } fn foo(a: A) { f(B { bar: a.$0 }); } "#, expect![[r#" fd bar [type+name] fd baz [type] fd foo [] "#]], ); } #[test] fn prioritize_exact_ref_match() { check_relevance( r#" struct WorldSnapshot { _f: () }; fn go(world: &WorldSnapshot) { go(w$0) } "#, expect![[r#" lc world [type+name+local] ex world [type] st WorldSnapshot {…} [] st &WorldSnapshot {…} [type] st WorldSnapshot [] st &WorldSnapshot [type] fn go(…) [] "#]], ); } #[test] fn too_many_arguments() { cov_mark::check!(too_many_arguments); check_relevance( r#" struct Foo; fn f(foo: &Foo) { f(foo, w$0) } "#, expect![[r#" lc foo [local] st Foo [] fn f(…) [] "#]], ); } #[test] fn score_fn_type_and_name_match() { check_relevance( r#" struct A { bar: u8 } fn baz() -> u8 { 0 } fn bar() -> u8 { 0 } fn f() { A { bar: b$0 }; } "#, expect![[r#" fn bar() [type+name] fn baz() [type] ex baz() [type] ex bar() [type] st A [] fn f() [] "#]], ); } #[test] fn score_method_type_and_name_match() { check_relevance( r#" fn baz(aaa: u32){} struct Foo; impl Foo { fn aaa(&self) -> u32 { 0 } fn bbb(&self) -> u32 { 0 } fn ccc(&self) -> u64 { 0 } } fn f() { baz(Foo.$0 } "#, expect![[r#" me aaa() [type+name] me bbb() [type] me ccc() [] "#]], ); } #[test] fn score_method_name_match_only() { check_relevance( r#" fn baz(aaa: u32){} struct Foo; impl Foo { fn aaa(&self) -> u64 { 0 } } fn f() { baz(Foo.$0 } "#, expect![[r#" me aaa() [name] "#]], ); } #[test] fn suggest_ref_mut() { cov_mark::check!(suggest_ref); check_relevance( r#" struct S; fn foo(s: &mut S) {} fn main() { let mut s = S; foo($0); } "#, expect![[r#" lc s [name+local] lc &mut s [type+name+local] st S [] st &mut S [type] st S [] st &mut S [type] fn foo(…) [] fn main() [] "#]], ); check_relevance( r#" struct S; fn foo(s: &mut S) {} fn main() { let mut s = S; foo(&mut $0); } "#, expect![[r#" lc s [type+name+local] st S [type] st S [type] ex s [type] ex S [type] fn foo(…) [] fn main() [] "#]], ); check_relevance( r#" struct S; fn foo(s: &mut S) {} fn main() { let mut ssss = S; foo(&mut s$0); } "#, expect![[r#" lc ssss [type+local] st S [type] st S [type] ex ssss [type] ex S [type] fn foo(…) [] fn main() [] "#]], ); } #[test] fn suggest_deref() { check_relevance( r#" //- minicore: deref struct S; struct T(S); impl core::ops::Deref for T { type Target = S; fn deref(&self) -> &Self::Target { &self.0 } } fn foo(s: &S) {} fn main() { let t = T(S); let m = 123; foo($0); } "#, expect![[r#" ex core::ops::Deref::deref(&t) (use core::ops::Deref) [type_could_unify] lc m [local] lc t [local] lc &t [type+local] st S [] st &S [type] st S [] st &S [type] st T [] st &T [type] fn foo(…) [] fn main() [] md core [] "#]], ) } #[test] fn suggest_deref_mut() { check_relevance( r#" //- minicore: deref_mut struct S; struct T(S); impl core::ops::Deref for T { type Target = S; fn deref(&self) -> &Self::Target { &self.0 } } impl core::ops::DerefMut for T { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.0 } } fn foo(s: &mut S) {} fn main() { let t = T(S); let m = 123; foo($0); } "#, expect![[r#" ex core::ops::DerefMut::deref_mut(&mut t) (use core::ops::DerefMut) [type_could_unify] lc m [local] lc t [local] lc &mut t [type+local] st S [] st &mut S [type] st S [] st &mut S [type] st T [] st &mut T [type] fn foo(…) [] fn main() [] md core [] "#]], ) } #[test] fn locals() { check_relevance( r#" fn foo(bar: u32) { let baz = 0; f$0 } "#, expect![[r#" lc baz [local] lc bar [local] fn foo(…) [] "#]], ); } #[test] fn enum_owned() { check_relevance( r#" enum Foo { A, B } fn foo() { bar($0); } fn bar(t: Foo) {} "#, expect![[r#" ev Foo::A [type] ev Foo::B [type] en Foo [type] ex Foo::A [type] ex Foo::B [type] fn bar(…) [] fn foo() [] "#]], ); } #[test] fn enum_ref() { check_relevance( r#" enum Foo { A, B } fn foo() { bar($0); } fn bar(t: &Foo) {} "#, expect![[r#" ev Foo::A [] ev &Foo::A [type] ev Foo::B [] ev &Foo::B [type] en Foo [] en &Foo [type] fn bar(…) [] fn foo() [] "#]], ); } #[test] fn suggest_deref_fn_ret() { check_relevance( r#" //- minicore: deref struct S; struct T(S); impl core::ops::Deref for T { type Target = S; fn deref(&self) -> &Self::Target { &self.0 } } fn foo(s: &S) {} fn bar() -> T {} fn main() { foo($0); } "#, expect![[r#" ex core::ops::Deref::deref(&bar()) (use core::ops::Deref) [type_could_unify] st S [] st &S [type] st S [] st &S [type] st T [] st &T [type] fn bar() [] fn &bar() [type] fn foo(…) [] fn main() [] md core [] "#]], ) } #[test] fn op_function_relevances() { check_relevance( r#" #[lang = "sub"] trait Sub { fn sub(self, other: Self) -> Self { self } } impl Sub for u32 {} fn foo(a: u32) { a.$0 } "#, expect![[r#" me sub(…) (as Sub) [op_method] "#]], ); check_relevance( r#" struct Foo; impl Foo { fn new() -> Self {} } #[lang = "eq"] pub trait PartialEq { fn eq(&self, other: &Rhs) -> bool; fn ne(&self, other: &Rhs) -> bool; } impl PartialEq for Foo {} fn main() { Foo::$0 } "#, expect![[r#" fn new() [] me eq(…) (as PartialEq) [op_method] me ne(…) (as PartialEq) [op_method] "#]], ); } #[test] fn constructor_order_simple() { check_relevance( r#" struct Foo; struct Other; struct Option(T); impl Foo { fn fn_ctr() -> Foo { unimplemented!() } fn fn_another(n: u32) -> Other { unimplemented!() } fn fn_ctr_self() -> Option { unimplemented!() } } fn test() { let a = Foo::$0; } "#, expect![[r#" fn fn_ctr() [type_could_unify] fn fn_ctr_self() [type_could_unify] fn fn_another(…) [type_could_unify] "#]], ); } #[test] fn constructor_order_kind() { check_function_relevance( r#" struct Foo; struct Bar; struct Option(T); enum Result { Ok(T), Err(E) }; impl Foo { fn fn_ctr(&self) -> Foo { unimplemented!() } fn fn_ctr_with_args(&self, n: u32) -> Foo { unimplemented!() } fn fn_another(&self, n: u32) -> Bar { unimplemented!() } fn fn_ctr_wrapped(&self, ) -> Option { unimplemented!() } fn fn_ctr_wrapped_2(&self, ) -> Result { unimplemented!() } fn fn_ctr_wrapped_3(&self, ) -> Result { unimplemented!() } // Self is not the first type fn fn_ctr_wrapped_with_args(&self, m: u32) -> Option { unimplemented!() } fn fn_another_unit(&self) { unimplemented!() } } fn test() { let a = self::Foo::$0; } "#, expect![[r#" [ ( "fn(&self, u32) -> Bar", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Other, }, ), ), ( "fn(&self)", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Other, }, ), ), ( "fn(&self) -> Foo", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: DirectConstructor, }, ), ), ( "fn(&self, u32) -> Foo", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: DirectConstructor, }, ), ), ( "fn(&self) -> Option", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Constructor, }, ), ), ( "fn(&self) -> Result", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Constructor, }, ), ), ( "fn(&self) -> Result", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Constructor, }, ), ), ( "fn(&self, u32) -> Option", Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Constructor, }, ), ), ] "#]], ); } #[test] fn constructor_order_relevance() { check_relevance( r#" struct Foo; struct FooBuilder; struct Result(T); impl Foo { fn fn_no_ret(&self) {} fn fn_ctr_with_args(input: u32) -> Foo { unimplemented!() } fn fn_direct_ctr() -> Self { unimplemented!() } fn fn_ctr() -> Result { unimplemented!() } fn fn_other() -> Result { unimplemented!() } fn fn_builder() -> FooBuilder { unimplemented!() } } fn test() { let a = self::Foo::$0; } "#, // preference: // Direct Constructor // Direct Constructor with args // Builder // Constructor // Others expect![[r#" fn fn_direct_ctr() [type_could_unify] fn fn_ctr_with_args(…) [type_could_unify] fn fn_builder() [type_could_unify] fn fn_ctr() [type_could_unify] me fn_no_ret(…) [type_could_unify] fn fn_other() [type_could_unify] "#]], ); // } #[test] fn function_relevance_generic_1() { check_relevance( r#" struct Foo(T); struct FooBuilder; struct Option(T); enum Result{Ok(T), Err(E)}; impl Foo { fn fn_returns_unit(&self) {} fn fn_ctr_with_args(input: T) -> Foo { unimplemented!() } fn fn_direct_ctr() -> Self { unimplemented!() } fn fn_ctr_wrapped() -> Option { unimplemented!() } fn fn_ctr_wrapped_2() -> Result { unimplemented!() } fn fn_other() -> Option { unimplemented!() } fn fn_builder() -> FooBuilder { unimplemented!() } } fn test() { let a = self::Foo::::$0; } "#, expect![[r#" fn fn_direct_ctr() [type_could_unify] fn fn_ctr_with_args(…) [type_could_unify] fn fn_builder() [type_could_unify] fn fn_ctr_wrapped() [type_could_unify] fn fn_ctr_wrapped_2() [type_could_unify] me fn_returns_unit(…) [type_could_unify] fn fn_other() [type_could_unify] "#]], ); } #[test] fn function_relevance_generic_2() { // Generic 2 check_relevance( r#" struct Foo(T); struct FooBuilder; struct Option(T); enum Result{Ok(T), Err(E)}; impl Foo { fn fn_no_ret(&self) {} fn fn_ctr_with_args(input: T) -> Foo { unimplemented!() } fn fn_direct_ctr() -> Self { unimplemented!() } fn fn_ctr() -> Option { unimplemented!() } fn fn_ctr2() -> Result { unimplemented!() } fn fn_other() -> Option { unimplemented!() } fn fn_builder() -> FooBuilder { unimplemented!() } } fn test() { let a : Res> = Foo::$0; } "#, expect![[r#" fn fn_direct_ctr() [type_could_unify] fn fn_ctr_with_args(…) [type_could_unify] fn fn_builder() [type_could_unify] fn fn_ctr() [type_could_unify] fn fn_ctr2() [type_could_unify] me fn_no_ret(…) [type_could_unify] fn fn_other() [type_could_unify] "#]], ); } #[test] fn struct_field_method_ref() { check_kinds( r#" struct Foo { bar: u32, qux: fn() } impl Foo { fn baz(&self) -> u32 { 0 } } fn foo(f: Foo) { let _: &u32 = f.b$0 } "#, &[ CompletionItemKind::SymbolKind(SymbolKind::Method), CompletionItemKind::SymbolKind(SymbolKind::Field), ], expect![[r#" [ CompletionItem { label: "baz()", source_range: 109..110, delete: 109..110, insert: "baz()$0", kind: SymbolKind( Method, ), lookup: "baz", detail: "fn(&self) -> u32", relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: Some( CompletionRelevanceFn { has_params: true, has_self_param: true, return_type: Other, }, ), }, ref_match: "&@107", }, CompletionItem { label: "bar", source_range: 109..110, delete: 109..110, insert: "bar", kind: SymbolKind( Field, ), detail: "u32", ref_match: "&@107", }, CompletionItem { label: "qux", source_range: 109..110, text_edit: TextEdit { indels: [ Indel { insert: "(", delete: 107..107, }, Indel { insert: "qux)()", delete: 109..110, }, ], }, kind: SymbolKind( Field, ), detail: "fn()", }, ] "#]], ); } #[test] fn expected_fn_type_ref() { check_kinds( r#" struct S { field: fn() } fn foo() { let foo: fn() = S { fields: || {}}.fi$0; } "#, &[CompletionItemKind::SymbolKind(SymbolKind::Field)], expect![[r#" [ CompletionItem { label: "field", source_range: 76..78, delete: 76..78, insert: "field", kind: SymbolKind( Field, ), detail: "fn()", relevance: CompletionRelevance { exact_name_match: false, type_match: Some( Exact, ), is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, }, ] "#]], ) } #[test] fn qualified_path_ref() { check_kinds( r#" struct S; struct T; impl T { fn foo() -> S {} } fn bar(s: &S) {} fn main() { bar(T::$0); } "#, &[CompletionItemKind::SymbolKind(SymbolKind::Function)], expect![[r#" [ CompletionItem { label: "foo()", source_range: 95..95, delete: 95..95, insert: "foo()$0", kind: SymbolKind( Function, ), lookup: "foo", detail: "fn() -> S", relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: false, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: Some( CompletionRelevanceFn { has_params: false, has_self_param: false, return_type: Other, }, ), }, ref_match: "&@92", }, ] "#]], ); } #[test] fn generic_enum() { check_relevance( r#" enum Foo { A(T), B } // bar() should not be an exact type match // because the generic parameters are different fn bar() -> Foo { Foo::B } // FIXME baz() should be an exact type match // because the types could unify, but it currently // is not. This is due to the T here being // TyKind::Placeholder rather than TyKind::Missing. fn baz() -> Foo { Foo::B } fn foo() { let foo: Foo = Foo::B; let _: Foo = f$0; } "#, expect![[r#" lc foo [type+local] ex foo [type] ex Foo::B [type] ev Foo::A(…) [type_could_unify] ev Foo::B [type_could_unify] en Foo [type_could_unify] fn foo() [] fn bar() [] fn baz() [] "#]], ); } #[test] fn postfix_exact_match_is_high_priority() { cov_mark::check!(postfix_exact_match_is_high_priority); check_relevance_for_kinds( r#" mod ops { pub trait Not { type Output; fn not(self) -> Self::Output; } impl Not for bool { type Output = bool; fn not(self) -> bool { if self { false } else { true }} } } fn main() { let _: bool = (9 > 2).not$0; } "#, &[CompletionItemKind::Snippet, CompletionItemKind::SymbolKind(SymbolKind::Method)], expect![[r#" sn not [snippet] me not() (use ops::Not) [type_could_unify+requires_import] sn if [] sn while [] sn ref [] sn refm [] sn deref [] sn unsafe [] sn match [] sn box [] sn dbg [] sn dbgr [] sn call [] sn return [] "#]], ); } #[test] fn postfix_inexact_match_is_low_priority() { cov_mark::check!(postfix_inexact_match_is_low_priority); check_relevance_for_kinds( r#" struct S; impl S { fn f(&self) {} } fn main() { S.$0 } "#, &[CompletionItemKind::Snippet, CompletionItemKind::SymbolKind(SymbolKind::Method)], expect![[r#" me f() [] sn ref [] sn refm [] sn deref [] sn unsafe [] sn match [] sn box [] sn dbg [] sn dbgr [] sn call [] sn let [] sn letm [] sn return [] "#]], ); } #[test] fn flyimport_reduced_relevance() { check_relevance( r#" mod std { pub mod io { pub trait BufRead {} pub struct BufReader; pub struct BufWriter; } } struct Buffer; fn f() { Buf$0 } "#, expect![[r#" st Buffer [] fn f() [] md std [] tt BufRead (use std::io::BufRead) [requires_import] st BufReader (use std::io::BufReader) [requires_import] st BufWriter (use std::io::BufWriter) [requires_import] "#]], ); } #[test] fn completes_struct_with_raw_identifier() { check_edit( "type", r#" mod m { pub struct r#type {} } fn main() { let r#type = m::t$0; } "#, r#" mod m { pub struct r#type {} } fn main() { let r#type = m::r#type; } "#, ) } #[test] fn completes_fn_with_raw_identifier() { check_edit( "type", r#" mod m { pub fn r#type {} } fn main() { m::t$0 } "#, r#" mod m { pub fn r#type {} } fn main() { m::r#type()$0 } "#, ) } #[test] fn completes_macro_with_raw_identifier() { check_edit( "let!", r#" macro_rules! r#let { () => {} } fn main() { $0 } "#, r#" macro_rules! r#let { () => {} } fn main() { r#let!($0) } "#, ) } #[test] fn completes_variant_with_raw_identifier() { check_edit( "type", r#" enum A { r#type } fn main() { let a = A::t$0 } "#, r#" enum A { r#type } fn main() { let a = A::r#type$0 } "#, ) } #[test] fn completes_field_with_raw_identifier() { check_edit( "fn", r#" mod r#type { pub struct r#struct { pub r#fn: u32 } } fn main() { let a = r#type::r#struct {}; a.$0 } "#, r#" mod r#type { pub struct r#struct { pub r#fn: u32 } } fn main() { let a = r#type::r#struct {}; a.r#fn } "#, ) } #[test] fn completes_const_with_raw_identifier() { check_edit( "type", r#" struct r#struct {} impl r#struct { pub const r#type: u8 = 1; } fn main() { r#struct::t$0 } "#, r#" struct r#struct {} impl r#struct { pub const r#type: u8 = 1; } fn main() { r#struct::r#type } "#, ) } #[test] fn completes_type_alias_with_raw_identifier() { check_edit( "type type", r#" struct r#struct {} trait r#trait { type r#type; } impl r#trait for r#struct { type t$0 } "#, r#" struct r#struct {} trait r#trait { type r#type; } impl r#trait for r#struct { type r#type = $0; } "#, ) } #[test] fn field_access_includes_self() { check_edit( "length", r#" struct S { length: i32 } impl S { fn some_fn(&self) { let l = len$0 } } "#, r#" struct S { length: i32 } impl S { fn some_fn(&self) { let l = self.length } } "#, ) } #[test] fn notable_traits_method_relevance() { check_kinds( r#" #[doc(notable_trait)] trait Write { fn write(&self); fn flush(&self); } struct Writer; impl Write for Writer { fn write(&self) {} fn flush(&self) {} } fn main() { Writer.$0 } "#, &[ CompletionItemKind::SymbolKind(SymbolKind::Method), CompletionItemKind::SymbolKind(SymbolKind::Field), CompletionItemKind::SymbolKind(SymbolKind::Function), ], expect![[r#" [ CompletionItem { label: "flush()", source_range: 193..193, delete: 193..193, insert: "flush()$0", kind: SymbolKind( Method, ), lookup: "flush", detail: "fn(&self)", relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: true, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, }, CompletionItem { label: "write()", source_range: 193..193, delete: 193..193, insert: "write()$0", kind: SymbolKind( Method, ), lookup: "write", detail: "fn(&self)", relevance: CompletionRelevance { exact_name_match: false, type_match: None, is_local: false, is_item_from_trait: false, is_item_from_notable_trait: true, is_name_already_imported: false, requires_import: false, is_op_method: false, is_private_editable: false, postfix_match: None, is_definite: false, function: None, }, }, ] "#]], ); } }