From 68626e4ef5acfea05812b68f41efa2bcd5bea448 Mon Sep 17 00:00:00 2001 From: Kirill Bulatov Date: Sun, 17 Jan 2021 00:53:15 +0200 Subject: Draft the working completion --- crates/completion/src/completions/flyimport.rs | 149 +++++++++++++++++++++++-- crates/ide_db/src/helpers/import_assets.rs | 30 +++-- crates/ide_db/src/imports_locator.rs | 16 ++- 3 files changed, 168 insertions(+), 27 deletions(-) (limited to 'crates') diff --git a/crates/completion/src/completions/flyimport.rs b/crates/completion/src/completions/flyimport.rs index 9101e405c..a40843669 100644 --- a/crates/completion/src/completions/flyimport.rs +++ b/crates/completion/src/completions/flyimport.rs @@ -20,11 +20,14 @@ //! # pub mod std { pub mod marker { pub struct PhantomData { } } } //! ``` //! +//! Also completes associated items, that require trait imports. +//! //! .Fuzzy search details //! //! To avoid an excessive amount of the results returned, completion input is checked for inclusion in the names only //! (i.e. in `HashMap` in the `std::collections::HashMap` path). -//! For the same reasons, avoids searching for any imports for inputs with their length less that 2 symbols. +//! For the same reasons, avoids searching for any path imports for inputs with their length less that 2 symbols +//! (but shows all associated items for any input length). //! //! .Import configuration //! @@ -46,8 +49,11 @@ //! capability enabled. use hir::{ModPath, ScopeDef}; -use ide_db::helpers::{import_assets::ImportAssets, insert_use::ImportScope}; -use syntax::AstNode; +use ide_db::helpers::{ + import_assets::{ImportAssets, ImportCandidate}, + insert_use::ImportScope, +}; +use syntax::{AstNode, SyntaxNode, T}; use test_utils::mark; use crate::{ @@ -65,16 +71,24 @@ pub(crate) fn import_on_the_fly(acc: &mut Completions, ctx: &CompletionContext) if ctx.attribute_under_caret.is_some() || ctx.mod_declaration_under_caret.is_some() { return None; } - let potential_import_name = ctx.token.to_string(); - if potential_import_name.len() < 2 { - return None; - } + let potential_import_name = { + let token_kind = ctx.token.kind(); + if token_kind == T![.] || token_kind == T![::] { + String::new() + } else { + ctx.token.to_string() + } + }; + let _p = profile::span("import_on_the_fly").detail(|| potential_import_name.to_string()); - let import_scope = - ImportScope::find_insert_use_container(ctx.name_ref_syntax.as_ref()?.syntax(), &ctx.sema)?; let user_input_lowercased = potential_import_name.to_lowercase(); - let mut all_mod_paths = import_assets(ctx, potential_import_name)? + let import_assets = import_assets(ctx, potential_import_name)?; + let import_scope = ImportScope::find_insert_use_container( + position_for_import(ctx, import_assets.import_candidate())?, + &ctx.sema, + )?; + let mut all_mod_paths = import_assets .search_for_relative_paths(&ctx.sema) .into_iter() .map(|(mod_path, item_in_ns)| { @@ -108,6 +122,17 @@ pub(crate) fn import_on_the_fly(acc: &mut Completions, ctx: &CompletionContext) Some(()) } +fn position_for_import<'a>( + ctx: &'a CompletionContext, + import_candidate: &ImportCandidate, +) -> Option<&'a SyntaxNode> { + Some(match import_candidate { + ImportCandidate::Path(_) => ctx.name_ref_syntax.as_ref()?.syntax(), + ImportCandidate::TraitAssocItem(_) => ctx.path_qual.as_ref()?.syntax(), + ImportCandidate::TraitMethod(_) => ctx.dot_receiver.as_ref()?.syntax(), + }) +} + fn import_assets(ctx: &CompletionContext, fuzzy_name: String) -> Option { let current_module = ctx.scope.module()?; if let Some(dot_receiver) = &ctx.dot_receiver { @@ -117,7 +142,22 @@ fn import_assets(ctx: &CompletionContext, fuzzy_name: String) -> Option| { + // TODO kb process all traits at once instead? trait_candidates.clear(); match &self.import_candidate { ImportCandidate::TraitAssocItem(trait_candidate) => { @@ -191,6 +192,11 @@ impl ImportAssets { None, |_, assoc| { if canidate_assoc_item == assoc { + if let AssocItem::Function(f) = assoc { + if f.self_param(db).is_some() { + return None; + } + } Some(assoc_to_module_def(assoc)) } else { None @@ -238,17 +244,21 @@ impl ImportAssets { // see https://github.com/rust-analyzer/rust-analyzer/pull/7293#issuecomment-761585032 // and https://rust-lang.zulipchat.com/#narrow/stream/185405-t-compiler.2Fwg-rls-2.2E0/topic/Blanket.20trait.20impls.20lookup // for the details - NameToImport::Fuzzy(fuzzy_name) => imports_locator::find_similar_imports( - sema, - current_crate, - fuzzy_name.clone(), - match self.import_candidate { + NameToImport::Fuzzy(fuzzy_name) => { + let (assoc_item_search, limit) = match self.import_candidate { ImportCandidate::TraitAssocItem(_) | ImportCandidate::TraitMethod(_) => { - AssocItemSearch::AssocItemsOnly + (AssocItemSearch::AssocItemsOnly, None) } - _ => AssocItemSearch::Exclude, - }, - ), + _ => (AssocItemSearch::Exclude, Some(DEFAULT_QUERY_SEARCH_LIMIT)), + }; + imports_locator::find_similar_imports( + sema, + current_crate, + fuzzy_name.clone(), + assoc_item_search, + limit, + ) + } }; let mut res = unfiltered_imports diff --git a/crates/ide_db/src/imports_locator.rs b/crates/ide_db/src/imports_locator.rs index d69e65960..502e8281a 100644 --- a/crates/ide_db/src/imports_locator.rs +++ b/crates/ide_db/src/imports_locator.rs @@ -15,7 +15,7 @@ use crate::{ use either::Either; use rustc_hash::FxHashSet; -const QUERY_SEARCH_LIMIT: usize = 40; +pub(crate) const DEFAULT_QUERY_SEARCH_LIMIT: usize = 40; pub fn find_exact_imports<'a>( sema: &Semantics<'a, RootDatabase>, @@ -29,11 +29,11 @@ pub fn find_exact_imports<'a>( { let mut local_query = symbol_index::Query::new(name_to_import.clone()); local_query.exact(); - local_query.limit(QUERY_SEARCH_LIMIT); + local_query.limit(DEFAULT_QUERY_SEARCH_LIMIT); local_query }, import_map::Query::new(name_to_import) - .limit(QUERY_SEARCH_LIMIT) + .limit(DEFAULT_QUERY_SEARCH_LIMIT) .name_only() .search_mode(import_map::SearchMode::Equals) .case_sensitive(), @@ -51,13 +51,13 @@ pub fn find_similar_imports<'a>( krate: Crate, fuzzy_search_string: String, assoc_item_search: AssocItemSearch, + limit: Option, ) -> Box> + 'a> { let _p = profile::span("find_similar_imports"); let mut external_query = import_map::Query::new(fuzzy_search_string.clone()) .search_mode(import_map::SearchMode::Fuzzy) - .name_only() - .limit(QUERY_SEARCH_LIMIT); + .name_only(); match assoc_item_search { AssocItemSearch::Include => {} @@ -70,7 +70,11 @@ pub fn find_similar_imports<'a>( } let mut local_query = symbol_index::Query::new(fuzzy_search_string); - local_query.limit(QUERY_SEARCH_LIMIT); + + if let Some(limit) = limit { + external_query = external_query.limit(limit); + local_query.limit(limit); + } let db = sema.db; Box::new(find_imports(sema, krate, local_query, external_query).filter( -- cgit v1.2.3