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