From 192e2bbb0e5c772d43ec61d36de56a0f7062610e Mon Sep 17 00:00:00 2001 From: Aleksey Kladov Date: Tue, 27 Nov 2018 19:56:03 +0300 Subject: resolve_path works with DefIds --- .../src/completion/reference_completion.rs | 7 +++++- crates/ra_analysis/src/descriptors/mod.rs | 26 ++++++++++++++++++++-- crates/ra_analysis/src/descriptors/module/mod.rs | 24 +++++++++++++------- 3 files changed, 46 insertions(+), 11 deletions(-) (limited to 'crates/ra_analysis/src') diff --git a/crates/ra_analysis/src/completion/reference_completion.rs b/crates/ra_analysis/src/completion/reference_completion.rs index a0af59178..858b52e76 100644 --- a/crates/ra_analysis/src/completion/reference_completion.rs +++ b/crates/ra_analysis/src/completion/reference_completion.rs @@ -13,6 +13,7 @@ use crate::{ descriptors::{ module::{ModuleDescriptor}, function::FnScopes, + Def, Path, }, Cancelable @@ -156,10 +157,14 @@ fn complete_path( return Ok(()); } path.segments.pop(); - let target_module = match module.resolve_path(db, path)? { + let def_id = match module.resolve_path(db, path)? { None => return Ok(()), Some(it) => it, }; + let target_module = match def_id.resolve(db)? { + Def::Module(it) => it, + Def::Item => return Ok(()), + }; let module_scope = target_module.scope(db)?; let completions = module_scope.entries().map(|(name, _res)| CompletionItem { label: name.to_string(), diff --git a/crates/ra_analysis/src/descriptors/mod.rs b/crates/ra_analysis/src/descriptors/mod.rs index 98094d9ee..7a1bcf447 100644 --- a/crates/ra_analysis/src/descriptors/mod.rs +++ b/crates/ra_analysis/src/descriptors/mod.rs @@ -13,9 +13,12 @@ use crate::{ FileId, db::SyntaxDatabase, descriptors::function::{resolve_local_name, FnId, FnScopes}, - descriptors::module::{ModuleId, ModuleTree, ModuleSource, nameres::{ItemMap, InputModuleItems, FileItems}}, + descriptors::module::{ + ModuleId, ModuleTree, ModuleSource, ModuleDescriptor, + nameres::{ItemMap, InputModuleItems, FileItems} + }, input::SourceRootId, - loc2id::IdDatabase, + loc2id::{IdDatabase, DefId, DefLoc}, syntax_ptr::LocalSyntaxPtr, Cancelable, }; @@ -67,6 +70,25 @@ salsa::query_group! { } } +pub(crate) enum Def { + Module(ModuleDescriptor), + Item, +} + +impl DefId { + pub(crate) fn resolve(self, db: &impl DescriptorDatabase) -> Cancelable { + let loc = db.id_maps().def_loc(self); + let res = match loc { + DefLoc::Module { id, source_root } => { + let descr = ModuleDescriptor::new(db, source_root, id)?; + Def::Module(descr) + } + DefLoc::Item { .. } => Def::Item, + }; + Ok(res) + } +} + #[derive(Debug)] pub struct ReferenceDescriptor { pub range: TextRange, diff --git a/crates/ra_analysis/src/descriptors/module/mod.rs b/crates/ra_analysis/src/descriptors/module/mod.rs index 764e19ce0..78911d5d9 100644 --- a/crates/ra_analysis/src/descriptors/module/mod.rs +++ b/crates/ra_analysis/src/descriptors/module/mod.rs @@ -77,7 +77,7 @@ impl ModuleDescriptor { Ok(res) } - fn new( + pub(super) fn new( db: &impl DescriptorDatabase, source_root_id: SourceRootId, module_id: ModuleId, @@ -132,6 +132,14 @@ impl ModuleDescriptor { Some(link.name(&self.tree)) } + pub fn def_id(&self, db: &impl DescriptorDatabase) -> DefId { + let def_loc = DefLoc::Module { + id: self.module_id, + source_root: self.source_root_id, + }; + db.id_maps().def_id(def_loc) + } + /// Finds a child module with the specified name. pub fn child(&self, name: &str) -> Option { let child_id = self.module_id.child(&self.tree, name)?; @@ -152,23 +160,23 @@ impl ModuleDescriptor { &self, db: &impl DescriptorDatabase, path: Path, - ) -> Cancelable> { + ) -> Cancelable> { let mut curr = match path.kind { PathKind::Crate => self.crate_root(), PathKind::Self_ | PathKind::Plain => self.clone(), PathKind::Super => ctry!(self.parent()), - }; + } + .def_id(db); let segments = path.segments; - for name in segments { - let scope = curr.scope(db)?; - let def_id = ctry!(ctry!(scope.get(&name)).def_id); - curr = match db.id_maps().def_loc(def_id) { + for name in segments.iter() { + let module = match db.id_maps().def_loc(curr) { DefLoc::Module { id, source_root } => ModuleDescriptor::new(db, source_root, id)?, _ => return Ok(None), }; + let scope = module.scope(db)?; + curr = ctry!(ctry!(scope.get(&name)).def_id); } - Ok(Some(curr)) } -- cgit v1.2.3