Restrict fuzzy qualifiers for now

This commit is contained in:
Kirill Bulatov 2021-03-05 00:11:07 +02:00
parent 6ca6f101c1
commit 84c575a212
3 changed files with 26 additions and 47 deletions

View file

@ -1094,27 +1094,4 @@ mod tests {
expect![[r#""#]], expect![[r#""#]],
); );
} }
#[test]
fn search_with_path() {
check_search(
r#"
//- /main.rs crate:main deps:dep
//- /dep.rs crate:dep
pub mod foo {
pub mod bar {
pub mod baz {
pub trait Display {
fn fmt();
}
}
}
}"#,
"main",
Query::new("baz::fmt".to_string()).search_mode(SearchMode::Fuzzy),
expect![[r#"
dep::foo::bar::baz::Display::fmt (a)
"#]],
);
}
} }

View file

@ -21,8 +21,9 @@
//! ``` //! ```
//! //!
//! Also completes associated items, that require trait imports. //! Also completes associated items, that require trait imports.
//! If any unresolved and/or partially-qualified path predeces the input, it will be taken into account: only the items with import string //! If any unresolved and/or partially-qualified path predeces the input, it will be taken into account.
//! containing this whole path will be considered and the corresponding path import will be added: //! Currently, only the imports with their import path ending with the whole qialifier will be proposed
//! (no fuzzy matching for qualifier).
//! //!
//! ``` //! ```
//! mod foo { //! mod foo {
@ -187,7 +188,6 @@ fn import_assets<'a>(ctx: &'a CompletionContext, fuzzy_name: String) -> Option<I
ctx.scope.clone(), ctx.scope.clone(),
)?; )?;
// TODO kb bad: with the path prefix, the "min 3 symbols" limit applies. Fix in a separate PR on the symbol_index level
if matches!(assets_for_path.import_candidate(), ImportCandidate::Path(_)) if matches!(assets_for_path.import_candidate(), ImportCandidate::Path(_))
&& fuzzy_name_length < 2 && fuzzy_name_length < 2
{ {
@ -937,7 +937,6 @@ mod foo {
} }
fn main() { fn main() {
let zz = "sdsd";
bar::Ass$0 bar::Ass$0
}"#, }"#,
expect![[]], expect![[]],

View file

@ -314,19 +314,21 @@ fn import_for_item(
let import_path_candidate = mod_path(original_item_candidate)?; let import_path_candidate = mod_path(original_item_candidate)?;
let import_path_string = import_path_candidate.to_string(); let import_path_string = import_path_candidate.to_string();
let expected_import_end = if item_as_assoc(db, original_item).is_some() {
unresolved_qualifier.to_string()
} else {
format!("{}::{}", unresolved_qualifier, item_name(db, original_item)?)
};
if !import_path_string.contains(unresolved_first_segment) if !import_path_string.contains(unresolved_first_segment)
|| !import_path_string.contains(unresolved_qualifier) || !import_path_string.ends_with(&expected_import_end)
{ {
return None; return None;
} }
let segment_import = let segment_import =
find_import_for_segment(db, original_item_candidate, &unresolved_first_segment)?; find_import_for_segment(db, original_item_candidate, &unresolved_first_segment)?;
let trait_item_to_import = original_item let trait_item_to_import = item_as_assoc(db, original_item)
.as_module_def_id() .and_then(|assoc| assoc.containing_trait(db))
.and_then(|module_def_id| {
ModuleDef::from(module_def_id).as_assoc_item(db)?.containing_trait(db)
})
.map(|trait_| ItemInNs::from(ModuleDef::from(trait_))); .map(|trait_| ItemInNs::from(ModuleDef::from(trait_)));
Some(match (segment_import == original_item_candidate, trait_item_to_import) { Some(match (segment_import == original_item_candidate, trait_item_to_import) {
(true, Some(_)) => { (true, Some(_)) => {
@ -358,10 +360,7 @@ fn import_for_item(
fn item_for_path_search(db: &RootDatabase, item: ItemInNs) -> Option<ItemInNs> { fn item_for_path_search(db: &RootDatabase, item: ItemInNs) -> Option<ItemInNs> {
Some(match item { Some(match item {
ItemInNs::Types(module_def_id) | ItemInNs::Values(module_def_id) => { ItemInNs::Types(_) | ItemInNs::Values(_) => match item_as_assoc(db, item) {
let module_def = ModuleDef::from(module_def_id);
match module_def.as_assoc_item(db) {
Some(assoc_item) => match assoc_item.container(db) { Some(assoc_item) => match assoc_item.container(db) {
AssocItemContainer::Trait(trait_) => ItemInNs::from(ModuleDef::from(trait_)), AssocItemContainer::Trait(trait_) => ItemInNs::from(ModuleDef::from(trait_)),
AssocItemContainer::Impl(impl_) => { AssocItemContainer::Impl(impl_) => {
@ -369,8 +368,7 @@ fn item_for_path_search(db: &RootDatabase, item: ItemInNs) -> Option<ItemInNs> {
} }
}, },
None => item, None => item,
} },
}
ItemInNs::Macros(_) => item, ItemInNs::Macros(_) => item,
}) })
} }
@ -427,7 +425,7 @@ fn trait_applicable_items(
let trait_candidates = items_with_candidate_name let trait_candidates = items_with_candidate_name
.into_iter() .into_iter()
.filter_map(|input| ModuleDef::from(input.as_module_def_id()?).as_assoc_item(db)) .filter_map(|input| item_as_assoc(db, input))
.filter_map(|assoc| { .filter_map(|assoc| {
let assoc_item_trait = assoc.containing_trait(db)?; let assoc_item_trait = assoc.containing_trait(db)?;
required_assoc_items.insert(assoc); required_assoc_items.insert(assoc);
@ -583,3 +581,8 @@ fn path_import_candidate(
None => ImportCandidate::Path(PathImportCandidate { qualifier: Qualifier::Absent, name }), None => ImportCandidate::Path(PathImportCandidate { qualifier: Qualifier::Absent, name }),
}) })
} }
fn item_as_assoc(db: &RootDatabase, item: ItemInNs) -> Option<AssocItem> {
item.as_module_def_id()
.and_then(|module_def_id| ModuleDef::from(module_def_id).as_assoc_item(db))
}