mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 21:54:42 +00:00
Merge #3707
3707: Add ItemScope::visibility_of r=matklad a=edwin0cheng ~This PR implements `HasVisibility` for various constructs and change `Definition::search_scope` to use `Visibility` directly instead of depends on ad-hoc string parsing.~ This PR added `visibility_of` in `ItemScope` and `Module` and use it directly directly instead of depends on ad-hoc string parsing. And also add a FIXME to indicate that there is a bug which do not search child-submodules in other files recursively in `Definition::search_scope`. I will submit another PR to fix that bug after this is merged. cc @flodiebold Co-authored-by: Edwin Cheng <edwin0cheng@gmail.com>
This commit is contained in:
commit
8a73a8937d
5 changed files with 50 additions and 59 deletions
|
@ -234,6 +234,10 @@ impl Module {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn visibility_of(self, db: &dyn HirDatabase, def: &ModuleDef) -> Option<Visibility> {
|
||||||
|
db.crate_def_map(self.id.krate)[self.id.local_id].scope.visbility_of(def.clone().into())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) {
|
pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) {
|
||||||
let _p = profile("Module::diagnostics");
|
let _p = profile("Module::diagnostics");
|
||||||
let crate_def_map = db.crate_def_map(self.id.krate);
|
let crate_def_map = db.crate_def_map(self.id.krate);
|
||||||
|
|
|
@ -54,7 +54,7 @@ pub use crate::{
|
||||||
Adt, AsAssocItem, AssocItem, AssocItemContainer, AttrDef, Const, Crate, CrateDependency,
|
Adt, AsAssocItem, AssocItem, AssocItemContainer, AttrDef, Const, Crate, CrateDependency,
|
||||||
DefWithBody, Docs, Enum, EnumVariant, FieldSource, Function, GenericDef, HasAttrs,
|
DefWithBody, Docs, Enum, EnumVariant, FieldSource, Function, GenericDef, HasAttrs,
|
||||||
HasVisibility, ImplDef, Local, MacroDef, Module, ModuleDef, ScopeDef, Static, Struct,
|
HasVisibility, ImplDef, Local, MacroDef, Module, ModuleDef, ScopeDef, Static, Struct,
|
||||||
StructField, Trait, Type, TypeAlias, TypeParam, Union, VariantDef,
|
StructField, Trait, Type, TypeAlias, TypeParam, Union, VariantDef, Visibility,
|
||||||
},
|
},
|
||||||
has_source::HasSource,
|
has_source::HasSource,
|
||||||
semantics::{original_range, PathResolution, Semantics, SemanticsScope},
|
semantics::{original_range, PathResolution, Semantics, SemanticsScope},
|
||||||
|
|
|
@ -68,6 +68,12 @@ impl ItemScope {
|
||||||
self.impls.iter().copied()
|
self.impls.iter().copied()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn visbility_of(&self, def: ModuleDefId) -> Option<Visibility> {
|
||||||
|
self.name_of(ItemInNs::Types(def))
|
||||||
|
.or_else(|| self.name_of(ItemInNs::Values(def)))
|
||||||
|
.map(|(_, v)| v)
|
||||||
|
}
|
||||||
|
|
||||||
/// Iterate over all module scoped macros
|
/// Iterate over all module scoped macros
|
||||||
pub(crate) fn macros<'a>(&'a self) -> impl Iterator<Item = (&'a Name, MacroDefId)> + 'a {
|
pub(crate) fn macros<'a>(&'a self) -> impl Iterator<Item = (&'a Name, MacroDefId)> + 'a {
|
||||||
self.visible.iter().filter_map(|(name, def)| def.take_macros().map(|macro_| (name, macro_)))
|
self.visible.iter().filter_map(|(name, def)| def.take_macros().map(|macro_| (name, macro_)))
|
||||||
|
|
|
@ -6,12 +6,12 @@
|
||||||
// FIXME: this badly needs rename/rewrite (matklad, 2020-02-06).
|
// FIXME: this badly needs rename/rewrite (matklad, 2020-02-06).
|
||||||
|
|
||||||
use hir::{
|
use hir::{
|
||||||
Adt, FieldSource, HasSource, ImplDef, Local, MacroDef, Module, ModuleDef, Name, PathResolution,
|
HasVisibility, ImplDef, Local, MacroDef, Module, ModuleDef, Name, PathResolution, Semantics,
|
||||||
Semantics, StructField, TypeParam,
|
StructField, TypeParam, Visibility,
|
||||||
};
|
};
|
||||||
use ra_prof::profile;
|
use ra_prof::profile;
|
||||||
use ra_syntax::{
|
use ra_syntax::{
|
||||||
ast::{self, AstNode, VisibilityOwner},
|
ast::{self, AstNode},
|
||||||
match_ast,
|
match_ast,
|
||||||
};
|
};
|
||||||
use test_utils::tested_by;
|
use test_utils::tested_by;
|
||||||
|
@ -41,28 +41,13 @@ impl Definition {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn visibility(&self, db: &RootDatabase) -> Option<ast::Visibility> {
|
pub fn visibility(&self, db: &RootDatabase) -> Option<Visibility> {
|
||||||
|
let module = self.module(db);
|
||||||
|
|
||||||
match self {
|
match self {
|
||||||
Definition::Macro(_) => None,
|
Definition::Macro(_) => None,
|
||||||
Definition::StructField(sf) => match sf.source(db).value {
|
Definition::StructField(sf) => Some(sf.visibility(db)),
|
||||||
FieldSource::Named(it) => it.visibility(),
|
Definition::ModuleDef(def) => module?.visibility_of(db, def),
|
||||||
FieldSource::Pos(it) => it.visibility(),
|
|
||||||
},
|
|
||||||
Definition::ModuleDef(def) => match def {
|
|
||||||
ModuleDef::Module(it) => it.declaration_source(db)?.value.visibility(),
|
|
||||||
ModuleDef::Function(it) => it.source(db).value.visibility(),
|
|
||||||
ModuleDef::Adt(adt) => match adt {
|
|
||||||
Adt::Struct(it) => it.source(db).value.visibility(),
|
|
||||||
Adt::Union(it) => it.source(db).value.visibility(),
|
|
||||||
Adt::Enum(it) => it.source(db).value.visibility(),
|
|
||||||
},
|
|
||||||
ModuleDef::Const(it) => it.source(db).value.visibility(),
|
|
||||||
ModuleDef::Static(it) => it.source(db).value.visibility(),
|
|
||||||
ModuleDef::Trait(it) => it.source(db).value.visibility(),
|
|
||||||
ModuleDef::TypeAlias(it) => it.source(db).value.visibility(),
|
|
||||||
ModuleDef::EnumVariant(_) => None,
|
|
||||||
ModuleDef::BuiltinType(_) => None,
|
|
||||||
},
|
|
||||||
Definition::SelfType(_) => None,
|
Definition::SelfType(_) => None,
|
||||||
Definition::Local(_) => None,
|
Definition::Local(_) => None,
|
||||||
Definition::TypeParam(_) => None,
|
Definition::TypeParam(_) => None,
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
|
|
||||||
use std::mem;
|
use std::mem;
|
||||||
|
|
||||||
use hir::{DefWithBody, HasSource, ModuleSource, Semantics};
|
use hir::{DefWithBody, HasSource, Module, ModuleSource, Semantics, Visibility};
|
||||||
use once_cell::unsync::Lazy;
|
use once_cell::unsync::Lazy;
|
||||||
use ra_db::{FileId, FileRange, SourceDatabaseExt};
|
use ra_db::{FileId, FileRange, SourceDatabaseExt};
|
||||||
use ra_prof::profile;
|
use ra_prof::profile;
|
||||||
|
@ -123,51 +123,47 @@ impl Definition {
|
||||||
return SearchScope::new(res);
|
return SearchScope::new(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
let vis = self.visibility(db).as_ref().map(|v| v.syntax().to_string()).unwrap_or_default();
|
let vis = self.visibility(db);
|
||||||
|
|
||||||
if vis.as_str() == "pub(super)" {
|
// FIXME:
|
||||||
if let Some(parent_module) = module.parent(db) {
|
// The following logic are wrong that it does not search
|
||||||
let mut res = FxHashMap::default();
|
// for submodules within other files recursively.
|
||||||
let parent_src = parent_module.definition_source(db);
|
|
||||||
let file_id = parent_src.file_id.original_file(db);
|
|
||||||
|
|
||||||
match parent_src.value {
|
if let Some(Visibility::Module(module)) = vis.and_then(|it| it.into()) {
|
||||||
ModuleSource::Module(m) => {
|
let module: Module = module.into();
|
||||||
let range = Some(m.syntax().text_range());
|
let mut res = FxHashMap::default();
|
||||||
res.insert(file_id, range);
|
let src = module.definition_source(db);
|
||||||
}
|
let file_id = src.file_id.original_file(db);
|
||||||
ModuleSource::SourceFile(_) => {
|
|
||||||
res.insert(file_id, None);
|
match src.value {
|
||||||
res.extend(parent_module.children(db).map(|m| {
|
ModuleSource::Module(m) => {
|
||||||
let src = m.definition_source(db);
|
let range = Some(m.syntax().text_range());
|
||||||
(src.file_id.original_file(db), None)
|
res.insert(file_id, range);
|
||||||
}));
|
}
|
||||||
}
|
ModuleSource::SourceFile(_) => {
|
||||||
|
res.insert(file_id, None);
|
||||||
|
res.extend(module.children(db).map(|m| {
|
||||||
|
let src = m.definition_source(db);
|
||||||
|
(src.file_id.original_file(db), None)
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
return SearchScope::new(res);
|
|
||||||
}
|
}
|
||||||
|
return SearchScope::new(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
if vis.as_str() != "" {
|
if let Some(Visibility::Public) = vis {
|
||||||
let source_root_id = db.file_source_root(file_id);
|
let source_root_id = db.file_source_root(file_id);
|
||||||
let source_root = db.source_root(source_root_id);
|
let source_root = db.source_root(source_root_id);
|
||||||
let mut res = source_root.walk().map(|id| (id, None)).collect::<FxHashMap<_, _>>();
|
let mut res = source_root.walk().map(|id| (id, None)).collect::<FxHashMap<_, _>>();
|
||||||
|
|
||||||
// FIXME: add "pub(in path)"
|
let krate = module.krate();
|
||||||
|
for rev_dep in krate.reverse_dependencies(db) {
|
||||||
if vis.as_str() == "pub(crate)" {
|
let root_file = rev_dep.root_file(db);
|
||||||
return SearchScope::new(res);
|
let source_root_id = db.file_source_root(root_file);
|
||||||
}
|
let source_root = db.source_root(source_root_id);
|
||||||
if vis.as_str() == "pub" {
|
res.extend(source_root.walk().map(|id| (id, None)));
|
||||||
let krate = module.krate();
|
|
||||||
for rev_dep in krate.reverse_dependencies(db) {
|
|
||||||
let root_file = rev_dep.root_file(db);
|
|
||||||
let source_root_id = db.file_source_root(root_file);
|
|
||||||
let source_root = db.source_root(source_root_id);
|
|
||||||
res.extend(source_root.walk().map(|id| (id, None)));
|
|
||||||
}
|
|
||||||
return SearchScope::new(res);
|
|
||||||
}
|
}
|
||||||
|
return SearchScope::new(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut res = FxHashMap::default();
|
let mut res = FxHashMap::default();
|
||||||
|
|
Loading…
Reference in a new issue