1use std::fmt;
4
5use rustc_hir::def::{CtorOf, DefKind, MacroKinds};
6use rustc_hir::def_id::DefId;
7use rustc_middle::ty::TyCtxt;
8use rustc_span::hygiene::MacroKind;
9use serde::{Deserialize, Deserializer, Serialize, Serializer, de};
10
11use crate::clean;
12
13macro_rules! item_type {
14 ($($variant:ident = $number:literal,)+) => {
15
16#[derive(Copy, PartialEq, Eq, Hash, Clone, Debug, PartialOrd, Ord)]
35#[repr(u8)]
36pub(crate) enum ItemType {
37 $($variant = $number,)+
38}
39
40impl Serialize for ItemType {
41 fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
42 where
43 S: Serializer,
44 {
45 (*self as u8).serialize(serializer)
46 }
47}
48
49impl<'de> Deserialize<'de> for ItemType {
50 fn deserialize<D>(deserializer: D) -> Result<ItemType, D::Error>
51 where
52 D: Deserializer<'de>,
53 {
54 struct ItemTypeVisitor;
55 impl<'de> de::Visitor<'de> for ItemTypeVisitor {
56 type Value = ItemType;
57 fn expecting(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
58 write!(formatter, "an integer between 0 and 27")
59 }
60 fn visit_u64<E: de::Error>(self, v: u64) -> Result<ItemType, E> {
61 Ok(match v {
62 $($number => ItemType::$variant,)+
63 _ => return Err(E::missing_field("unknown number for `ItemType` enum")),
64 })
65 }
66 }
67 deserializer.deserialize_any(ItemTypeVisitor)
68 }
69}
70
71 }
72}
73
74item_type! {
75 Keyword = 0,
76 Primitive = 1,
77 Module = 2,
78 ExternCrate = 3,
79 Import = 4,
80 Struct = 5,
81 Enum = 6,
82 Function = 7,
83 TypeAlias = 8,
84 Static = 9,
85 Trait = 10,
86 Impl = 11,
87 TyMethod = 12,
88 Method = 13,
89 StructField = 14,
90 Variant = 15,
91 Macro = 16,
92 AssocType = 17,
93 Constant = 18,
94 AssocConst = 19,
95 Union = 20,
96 ForeignType = 21,
97 ProcAttribute = 23,
99 ProcDerive = 24,
100 TraitAlias = 25,
101 Attribute = 27,
104}
105
106impl<'a> From<&'a clean::Item> for ItemType {
107 fn from(item: &'a clean::Item) -> ItemType {
108 let kind = match &item.kind {
109 clean::StrippedItem(box item) => item,
110 kind => kind,
111 };
112
113 match kind {
114 clean::ModuleItem(..) => ItemType::Module,
115 clean::ExternCrateItem { .. } => ItemType::ExternCrate,
116 clean::ImportItem(..) => ItemType::Import,
117 clean::StructItem(..) => ItemType::Struct,
118 clean::UnionItem(..) => ItemType::Union,
119 clean::EnumItem(..) => ItemType::Enum,
120 clean::FunctionItem(..) => ItemType::Function,
121 clean::TypeAliasItem(..) => ItemType::TypeAlias,
122 clean::StaticItem(..) => ItemType::Static,
123 clean::ConstantItem(..) => ItemType::Constant,
124 clean::TraitItem(..) => ItemType::Trait,
125 clean::ImplItem(..) => ItemType::Impl,
126 clean::RequiredMethodItem(..) => ItemType::TyMethod,
127 clean::MethodItem(..) => ItemType::Method,
128 clean::StructFieldItem(..) => ItemType::StructField,
129 clean::VariantItem(..) => ItemType::Variant,
130 clean::ForeignFunctionItem(..) => ItemType::Function, clean::ForeignStaticItem(..) => ItemType::Static, clean::MacroItem(..) => ItemType::Macro,
133 clean::PrimitiveItem(..) => ItemType::Primitive,
134 clean::RequiredAssocConstItem(..)
135 | clean::ProvidedAssocConstItem(..)
136 | clean::ImplAssocConstItem(..) => ItemType::AssocConst,
137 clean::RequiredAssocTypeItem(..) | clean::AssocTypeItem(..) => ItemType::AssocType,
138 clean::ForeignTypeItem => ItemType::ForeignType,
139 clean::KeywordItem => ItemType::Keyword,
140 clean::AttributeItem => ItemType::Attribute,
141 clean::TraitAliasItem(..) => ItemType::TraitAlias,
142 clean::ProcMacroItem(mac) => match mac.kind {
143 MacroKind::Bang => ItemType::Macro,
144 MacroKind::Attr => ItemType::ProcAttribute,
145 MacroKind::Derive => ItemType::ProcDerive,
146 },
147 clean::StrippedItem(..) => unreachable!(),
148 }
149 }
150}
151
152impl ItemType {
153 pub(crate) fn from_def_id(def_id: DefId, tcx: TyCtxt<'_>) -> Self {
154 let def_kind = tcx.def_kind(def_id);
155 match def_kind {
156 DefKind::Enum => Self::Enum,
157 DefKind::Fn => Self::Function,
158 DefKind::Mod => Self::Module,
159 DefKind::Const => Self::Constant,
160 DefKind::Static { .. } => Self::Static,
161 DefKind::Struct => Self::Struct,
162 DefKind::Union => Self::Union,
163 DefKind::Trait => Self::Trait,
164 DefKind::TyAlias => Self::TypeAlias,
165 DefKind::TraitAlias => Self::TraitAlias,
166 DefKind::Macro(MacroKinds::BANG) => ItemType::Macro,
167 DefKind::Macro(MacroKinds::ATTR) => ItemType::ProcAttribute,
168 DefKind::Macro(MacroKinds::DERIVE) => ItemType::ProcDerive,
169 DefKind::Macro(_) => todo!("Handle macros with multiple kinds"),
170 DefKind::ForeignTy => Self::ForeignType,
171 DefKind::Variant => Self::Variant,
172 DefKind::Field => Self::StructField,
173 DefKind::AssocTy => Self::AssocType,
174 DefKind::AssocFn => {
175 if tcx.associated_item(def_id).defaultness(tcx).has_value() {
176 Self::Method
177 } else {
178 Self::TyMethod
179 }
180 }
181 DefKind::Ctor(CtorOf::Struct, _) => Self::Struct,
182 DefKind::Ctor(CtorOf::Variant, _) => Self::Variant,
183 DefKind::AssocConst => Self::AssocConst,
184 DefKind::TyParam
185 | DefKind::ConstParam
186 | DefKind::ExternCrate
187 | DefKind::Use
188 | DefKind::ForeignMod
189 | DefKind::AnonConst
190 | DefKind::InlineConst
191 | DefKind::OpaqueTy
192 | DefKind::LifetimeParam
193 | DefKind::GlobalAsm
194 | DefKind::Impl { .. }
195 | DefKind::Closure
196 | DefKind::SyntheticCoroutineBody => Self::ForeignType,
197 }
198 }
199
200 pub(crate) fn as_str(&self) -> &'static str {
201 match self {
202 ItemType::Module => "mod",
203 ItemType::ExternCrate => "externcrate",
204 ItemType::Import => "import",
205 ItemType::Struct => "struct",
206 ItemType::Union => "union",
207 ItemType::Enum => "enum",
208 ItemType::Function => "fn",
209 ItemType::TypeAlias => "type",
210 ItemType::Static => "static",
211 ItemType::Trait => "trait",
212 ItemType::Impl => "impl",
213 ItemType::TyMethod => "tymethod",
214 ItemType::Method => "method",
215 ItemType::StructField => "structfield",
216 ItemType::Variant => "variant",
217 ItemType::Macro => "macro",
218 ItemType::Primitive => "primitive",
219 ItemType::AssocType => "associatedtype",
220 ItemType::Constant => "constant",
221 ItemType::AssocConst => "associatedconstant",
222 ItemType::ForeignType => "foreigntype",
223 ItemType::Keyword => "keyword",
224 ItemType::ProcAttribute => "attr",
225 ItemType::ProcDerive => "derive",
226 ItemType::TraitAlias => "traitalias",
227 ItemType::Attribute => "attribute",
228 }
229 }
230 pub(crate) fn is_method(&self) -> bool {
231 matches!(self, ItemType::Method | ItemType::TyMethod)
232 }
233 pub(crate) fn is_adt(&self) -> bool {
234 matches!(self, ItemType::Struct | ItemType::Union | ItemType::Enum)
235 }
236 pub(crate) fn is_fn_like(&self) -> bool {
238 matches!(self, ItemType::Function | ItemType::Method | ItemType::TyMethod)
239 }
240}
241
242impl fmt::Display for ItemType {
243 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
244 f.write_str(self.as_str())
245 }
246}