rust-analyzer/crates/hir/src/attrs.rs

181 lines
6 KiB
Rust
Raw Normal View History

2020-08-25 10:57:15 +00:00
//! Attributes & documentation for hir types.
use hir_def::{
attr::{AttrsWithOwner, Documentation},
item_scope::ItemInNs,
2020-12-07 17:49:03 +00:00
path::ModPath,
2021-01-19 15:43:06 +00:00
per_ns::PerNs,
2020-12-07 17:49:03 +00:00
resolver::HasResolver,
2021-01-01 23:42:07 +00:00
AttrDefId, GenericParamId, ModuleDefId,
};
2022-03-08 22:51:48 +00:00
use hir_expand::hygiene::Hygiene;
use hir_ty::db::HirDatabase;
2021-12-28 12:48:07 +00:00
use syntax::{ast, AstNode};
use crate::{
Adt, AssocItem, Const, ConstParam, Enum, Field, Function, GenericParam, Impl, LifetimeParam,
2023-03-03 15:24:07 +00:00
Macro, Module, ModuleDef, Static, Struct, Trait, TraitAlias, TypeAlias, TypeParam, Union,
Variant,
};
pub trait HasAttrs {
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner;
fn docs(self, db: &dyn HirDatabase) -> Option<Documentation>;
fn resolve_doc_path(
self,
db: &dyn HirDatabase,
link: &str,
ns: Option<Namespace>,
2022-03-08 22:51:48 +00:00
) -> Option<ModuleDef>;
}
#[derive(PartialEq, Eq, Hash, Copy, Clone, Debug)]
pub enum Namespace {
Types,
Values,
Macros,
}
2020-08-25 12:44:15 +00:00
macro_rules! impl_has_attrs {
($(($def:ident, $def_id:ident),)*) => {$(
impl HasAttrs for $def {
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
2020-08-25 12:44:15 +00:00
let def = AttrDefId::$def_id(self.into());
db.attrs(def)
}
fn docs(self, db: &dyn HirDatabase) -> Option<Documentation> {
let def = AttrDefId::$def_id(self.into());
2020-12-07 17:06:46 +00:00
db.attrs(def).docs()
2020-08-25 12:44:15 +00:00
}
2022-03-08 22:51:48 +00:00
fn resolve_doc_path(self, db: &dyn HirDatabase, link: &str, ns: Option<Namespace>) -> Option<ModuleDef> {
let def = AttrDefId::$def_id(self.into());
2022-03-08 22:51:48 +00:00
resolve_doc_path(db, def, link, ns).map(ModuleDef::from)
}
2020-08-25 12:44:15 +00:00
}
)*};
}
2020-08-25 12:44:15 +00:00
impl_has_attrs![
(Field, FieldId),
2020-12-20 07:05:24 +00:00
(Variant, EnumVariantId),
2020-08-25 12:44:15 +00:00
(Static, StaticId),
(Const, ConstId),
(Trait, TraitId),
2023-03-03 15:24:07 +00:00
(TraitAlias, TraitAliasId),
2020-08-25 12:44:15 +00:00
(TypeAlias, TypeAliasId),
2022-03-08 22:51:48 +00:00
(Macro, MacroId),
2020-08-25 12:44:15 +00:00
(Function, FunctionId),
(Adt, AdtId),
(Module, ModuleId),
2021-01-01 23:42:07 +00:00
(GenericParam, GenericParamId),
(Impl, ImplId),
2020-08-25 12:44:15 +00:00
];
macro_rules! impl_has_attrs_enum {
($($variant:ident),* for $enum:ident) => {$(
impl HasAttrs for $variant {
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
$enum::$variant(self).attrs(db)
2020-08-25 12:44:15 +00:00
}
fn docs(self, db: &dyn HirDatabase) -> Option<Documentation> {
$enum::$variant(self).docs(db)
2020-08-25 12:44:15 +00:00
}
2022-03-08 22:51:48 +00:00
fn resolve_doc_path(self, db: &dyn HirDatabase, link: &str, ns: Option<Namespace>) -> Option<ModuleDef> {
$enum::$variant(self).resolve_doc_path(db, link, ns)
}
2020-08-25 12:44:15 +00:00
}
)*};
}
impl_has_attrs_enum![Struct, Union, Enum for Adt];
impl_has_attrs_enum![TypeParam, ConstParam, LifetimeParam for GenericParam];
2020-08-25 12:44:15 +00:00
impl HasAttrs for AssocItem {
fn attrs(self, db: &dyn HirDatabase) -> AttrsWithOwner {
match self {
AssocItem::Function(it) => it.attrs(db),
AssocItem::Const(it) => it.attrs(db),
AssocItem::TypeAlias(it) => it.attrs(db),
}
}
fn docs(self, db: &dyn HirDatabase) -> Option<Documentation> {
match self {
AssocItem::Function(it) => it.docs(db),
AssocItem::Const(it) => it.docs(db),
AssocItem::TypeAlias(it) => it.docs(db),
}
}
fn resolve_doc_path(
self,
db: &dyn HirDatabase,
link: &str,
ns: Option<Namespace>,
2022-03-08 22:51:48 +00:00
) -> Option<ModuleDef> {
match self {
AssocItem::Function(it) => it.resolve_doc_path(db, link, ns),
AssocItem::Const(it) => it.resolve_doc_path(db, link, ns),
AssocItem::TypeAlias(it) => it.resolve_doc_path(db, link, ns),
}
}
}
fn resolve_doc_path(
db: &dyn HirDatabase,
def: AttrDefId,
link: &str,
ns: Option<Namespace>,
2022-03-08 22:51:48 +00:00
) -> Option<ModuleDefId> {
let resolver = match def {
AttrDefId::ModuleId(it) => it.resolver(db.upcast()),
AttrDefId::FieldId(it) => it.parent.resolver(db.upcast()),
AttrDefId::AdtId(it) => it.resolver(db.upcast()),
AttrDefId::FunctionId(it) => it.resolver(db.upcast()),
AttrDefId::EnumVariantId(it) => it.parent.resolver(db.upcast()),
AttrDefId::StaticId(it) => it.resolver(db.upcast()),
AttrDefId::ConstId(it) => it.resolver(db.upcast()),
AttrDefId::TraitId(it) => it.resolver(db.upcast()),
2023-03-03 15:24:07 +00:00
AttrDefId::TraitAliasId(it) => it.resolver(db.upcast()),
AttrDefId::TypeAliasId(it) => it.resolver(db.upcast()),
AttrDefId::ImplId(it) => it.resolver(db.upcast()),
2021-12-07 16:31:26 +00:00
AttrDefId::ExternBlockId(it) => it.resolver(db.upcast()),
2022-03-08 22:51:48 +00:00
AttrDefId::MacroId(it) => it.resolver(db.upcast()),
2021-01-01 23:42:07 +00:00
AttrDefId::GenericParamId(it) => match it {
2021-12-29 13:35:59 +00:00
GenericParamId::TypeParamId(it) => it.parent(),
GenericParamId::ConstParamId(it) => it.parent(),
2021-01-01 23:42:07 +00:00
GenericParamId::LifetimeParamId(it) => it.parent,
}
.resolver(db.upcast()),
};
2021-12-28 12:48:07 +00:00
let modpath = {
2022-03-06 18:11:05 +00:00
// FIXME: this is not how we should get a mod path here
let ast_path = ast::SourceFile::parse(&format!("type T = {link};"))
2021-12-28 12:48:07 +00:00
.syntax_node()
.descendants()
.find_map(ast::Path::cast)?;
if ast_path.to_string() != link {
return None;
}
ModPath::from_src(db.upcast(), ast_path, &Hygiene::new_unhygienic())?
};
let resolved = resolver.resolve_module_path_in_items(db.upcast(), &modpath);
let resolved = if resolved == PerNs::none() {
resolver.resolve_module_path_in_trait_assoc_items(db.upcast(), &modpath)?
} else {
resolved
};
match ns {
2022-03-08 22:51:48 +00:00
Some(Namespace::Types) => resolved.take_types(),
Some(Namespace::Values) => resolved.take_values(),
Some(Namespace::Macros) => resolved.take_macros().map(ModuleDefId::MacroId),
None => resolved.iter_items().next().map(|it| match it {
2022-03-08 22:51:48 +00:00
ItemInNs::Types(it) => it,
ItemInNs::Values(it) => it,
ItemInNs::Macros(it) => ModuleDefId::MacroId(it),
}),
}
}