1#[cfg(test)]
4mod tests;
5
6mod intra_doc_links;
7
8use std::ops::Range;
9
10use pulldown_cmark::{BrokenLink, CowStr, Event, InlineStr, LinkType, Options, Parser, Tag};
11use pulldown_cmark_to_cmark::{Options as CMarkOptions, cmark_resume_with_options};
12use stdx::format_to;
13use url::Url;
14
15use hir::{
16 Adt, AsAssocItem, AssocItem, AssocItemContainer, AttrsWithOwner, HasAttrs, db::HirDatabase, sym,
17};
18use ide_db::{
19 RootDatabase,
20 base_db::{CrateOrigin, LangCrateOrigin, ReleaseChannel, RootQueryDb},
21 defs::{Definition, NameClass, NameRefClass},
22 documentation::{DocsRangeMap, Documentation, HasDocs, docs_with_rangemap},
23 helpers::pick_best_token,
24};
25use syntax::{
26 AstNode, AstToken,
27 SyntaxKind::*,
28 SyntaxNode, SyntaxToken, T, TextRange, TextSize,
29 ast::{self, IsString},
30 match_ast,
31};
32
33use crate::{
34 FilePosition, Semantics,
35 doc_links::intra_doc_links::{parse_intra_doc_link, strip_prefixes_suffixes},
36};
37
38#[derive(Default, Debug, Clone, PartialEq, Eq)]
40pub struct DocumentationLinks {
41 pub web_url: Option<String>,
44 pub local_url: Option<String>,
47}
48
49const MARKDOWN_OPTIONS: Options =
50 Options::ENABLE_FOOTNOTES.union(Options::ENABLE_TABLES).union(Options::ENABLE_TASKLISTS);
51
52pub(crate) fn rewrite_links(
54 db: &RootDatabase,
55 markdown: &str,
56 definition: Definition,
57 range_map: Option<DocsRangeMap>,
58) -> String {
59 let mut cb = broken_link_clone_cb;
60 let doc = Parser::new_with_broken_link_callback(markdown, MARKDOWN_OPTIONS, Some(&mut cb))
61 .into_offset_iter();
62
63 let doc = map_links(doc, |target, title, range, link_type| {
64 if target.contains("://") {
68 (Some(LinkType::Inline), target.to_owned(), title.to_owned())
69 } else {
70 let text_range =
74 TextRange::new(range.start.try_into().unwrap(), range.end.try_into().unwrap());
75 let is_inner_doc = range_map
76 .as_ref()
77 .and_then(|range_map| range_map.map(text_range))
78 .map(|(_, attr_id)| attr_id.is_inner_attr())
79 .unwrap_or(false);
80 if let Some((target, title)) =
81 rewrite_intra_doc_link(db, definition, target, title, is_inner_doc, link_type)
82 {
83 (None, target, title)
84 } else if let Some(target) = rewrite_url_link(db, definition, target) {
85 (Some(LinkType::Inline), target, title.to_owned())
86 } else {
87 (None, target.to_owned(), title.to_owned())
88 }
89 }
90 });
91 let mut out = String::new();
92 cmark_resume_with_options(
93 doc,
94 &mut out,
95 None,
96 CMarkOptions { code_block_token_count: 3, ..Default::default() },
97 )
98 .ok();
99 out
100}
101
102pub(crate) fn remove_links(markdown: &str) -> String {
104 let mut drop_link = false;
105
106 let mut cb = |_: BrokenLink<'_>| {
107 let empty = InlineStr::try_from("").unwrap();
108 Some((CowStr::Inlined(empty), CowStr::Inlined(empty)))
109 };
110 let doc = Parser::new_with_broken_link_callback(markdown, MARKDOWN_OPTIONS, Some(&mut cb));
111 let doc = doc.filter_map(move |evt| match evt {
112 Event::Start(Tag::Link(link_type, target, title)) => {
113 if link_type == LinkType::Inline && target.contains("://") {
114 Some(Event::Start(Tag::Link(link_type, target, title)))
115 } else {
116 drop_link = true;
117 None
118 }
119 }
120 Event::End(_) if drop_link => {
121 drop_link = false;
122 None
123 }
124 _ => Some(evt),
125 });
126
127 let mut out = String::new();
128 cmark_resume_with_options(
129 doc,
130 &mut out,
131 None,
132 CMarkOptions { code_block_token_count: 3, ..Default::default() },
133 )
134 .ok();
135 out
136}
137
138pub(crate) fn external_docs(
149 db: &RootDatabase,
150 FilePosition { file_id, offset }: FilePosition,
151 target_dir: Option<&str>,
152 sysroot: Option<&str>,
153) -> Option<DocumentationLinks> {
154 let sema = &Semantics::new(db);
155 let file = sema.parse_guess_edition(file_id).syntax().clone();
156 let token = pick_best_token(file.token_at_offset(offset), |kind| match kind {
157 IDENT | INT_NUMBER | T![self] => 3,
158 T!['('] | T![')'] => 2,
159 kind if kind.is_trivia() => 0,
160 _ => 1,
161 })?;
162 let token = sema.descend_into_macros_single_exact(token);
163
164 let node = token.parent()?;
165 let definition = match_ast! {
166 match node {
167 ast::NameRef(name_ref) => match NameRefClass::classify(sema, &name_ref)? {
168 NameRefClass::Definition(def, _) => def,
169 NameRefClass::FieldShorthand { local_ref: _, field_ref, adt_subst: _ } => {
170 Definition::Field(field_ref)
171 }
172 NameRefClass::ExternCrateShorthand { decl, .. } => {
173 Definition::ExternCrateDecl(decl)
174 }
175 },
176 ast::Name(name) => match NameClass::classify(sema, &name)? {
177 NameClass::Definition(it) | NameClass::ConstReference(it) => it,
178 NameClass::PatFieldShorthand { local_def: _, field_ref, adt_subst: _ } => Definition::Field(field_ref),
179 },
180 _ => return None
181 }
182 };
183
184 Some(get_doc_links(db, definition, target_dir, sysroot))
185}
186
187pub(crate) fn extract_definitions_from_docs(
190 docs: &Documentation,
191) -> Vec<(TextRange, String, Option<hir::Namespace>)> {
192 Parser::new_with_broken_link_callback(
193 docs.as_str(),
194 MARKDOWN_OPTIONS,
195 Some(&mut broken_link_clone_cb),
196 )
197 .into_offset_iter()
198 .filter_map(|(event, range)| match event {
199 Event::Start(Tag::Link(_, target, _)) => {
200 let (link, ns) = parse_intra_doc_link(&target);
201 Some((
202 TextRange::new(range.start.try_into().ok()?, range.end.try_into().ok()?),
203 link.to_owned(),
204 ns,
205 ))
206 }
207 _ => None,
208 })
209 .collect()
210}
211
212pub(crate) fn resolve_doc_path_for_def(
213 db: &dyn HirDatabase,
214 def: Definition,
215 link: &str,
216 ns: Option<hir::Namespace>,
217 is_inner_doc: bool,
218) -> Option<Definition> {
219 match def {
220 Definition::Module(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
221 Definition::Crate(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
222 Definition::Function(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
223 Definition::Adt(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
224 Definition::Variant(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
225 Definition::Const(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
226 Definition::Static(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
227 Definition::Trait(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
228 Definition::TypeAlias(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
229 Definition::Macro(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
230 Definition::Field(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
231 Definition::SelfType(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
232 Definition::ExternCrateDecl(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
233 Definition::BuiltinAttr(_)
234 | Definition::BuiltinType(_)
235 | Definition::BuiltinLifetime(_)
236 | Definition::ToolModule(_)
237 | Definition::TupleField(_)
238 | Definition::Local(_)
239 | Definition::GenericParam(_)
240 | Definition::Label(_)
241 | Definition::DeriveHelper(_)
242 | Definition::InlineAsmRegOrRegClass(_)
243 | Definition::InlineAsmOperand(_) => None,
244 }
245 .map(Definition::from)
246}
247
248pub(crate) fn doc_attributes(
249 sema: &Semantics<'_, RootDatabase>,
250 node: &SyntaxNode,
251) -> Option<(hir::AttrsWithOwner, Definition)> {
252 match_ast! {
253 match node {
254 ast::SourceFile(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
255 ast::Module(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
256 ast::Fn(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
257 ast::Struct(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Struct(def)))),
258 ast::Union(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Union(def)))),
259 ast::Enum(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Enum(def)))),
260 ast::Variant(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
261 ast::Trait(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
262 ast::Static(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
263 ast::Const(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
264 ast::TypeAlias(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
265 ast::Impl(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
266 ast::RecordField(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
267 ast::TupleField(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
268 ast::Macro(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
269 ast::ExternCrate(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
270 _ => None
272 }
273 }
274}
275
276pub(crate) struct DocCommentToken {
277 doc_token: SyntaxToken,
278 prefix_len: TextSize,
279}
280
281pub(crate) fn token_as_doc_comment(doc_token: &SyntaxToken) -> Option<DocCommentToken> {
282 (match_ast! {
283 match doc_token {
284 ast::Comment(comment) => TextSize::try_from(comment.prefix().len()).ok(),
285 ast::String(string) => {
286 doc_token.parent_ancestors().find_map(ast::Attr::cast).filter(|attr| attr.simple_name().as_deref() == Some("doc"))?;
287 if doc_token.parent_ancestors().find_map(ast::MacroCall::cast).filter(|mac| mac.path().and_then(|p| p.segment()?.name_ref()).as_ref().map(|n| n.text()).as_deref() == Some("include_str")).is_some() {
288 return None;
289 }
290 string.open_quote_text_range().map(|it| it.len())
291 },
292 _ => None,
293 }
294 }).map(|prefix_len| DocCommentToken { prefix_len, doc_token: doc_token.clone() })
295}
296
297impl DocCommentToken {
298 pub(crate) fn get_definition_with_descend_at<T>(
299 self,
300 sema: &Semantics<'_, RootDatabase>,
301 offset: TextSize,
302 mut cb: impl FnMut(Definition, SyntaxNode, TextRange) -> Option<T>,
304 ) -> Option<T> {
305 let DocCommentToken { prefix_len, doc_token } = self;
306 let original_start = doc_token.text_range().start();
308 let relative_comment_offset = offset - original_start - prefix_len;
309
310 sema.descend_into_macros(doc_token).into_iter().find_map(|t| {
311 let (node, descended_prefix_len, is_inner) = match_ast!{
312 match t {
313 ast::Comment(comment) => {
314 (t.parent()?, TextSize::try_from(comment.prefix().len()).ok()?, comment.is_inner())
315 },
316 ast::String(string) => {
317 let attr = t.parent_ancestors().find_map(ast::Attr::cast)?;
318 let attr_is_inner = attr.excl_token().map(|excl| excl.kind() == BANG).unwrap_or(false);
319 (attr.syntax().parent()?, string.open_quote_text_range()?.len(), attr_is_inner)
320 },
321 _ => return None,
322 }
323 };
324 let token_start = t.text_range().start();
325 let abs_in_expansion_offset = token_start + relative_comment_offset + descended_prefix_len;
326 let (attributes, def) = Self::doc_attributes(sema, &node, is_inner)?;
327 let (docs, doc_mapping) = docs_with_rangemap(sema.db, &attributes)?;
328 let (in_expansion_range, link, ns, is_inner) =
329 extract_definitions_from_docs(&docs).into_iter().find_map(|(range, link, ns)| {
330 let (mapped, idx) = doc_mapping.map(range)?;
331 (mapped.value.contains(abs_in_expansion_offset)).then_some((mapped.value, link, ns, idx.is_inner_attr()))
332 })?;
333 let in_expansion_relative_range = in_expansion_range - descended_prefix_len - token_start;
335 let absolute_range = in_expansion_relative_range + original_start + prefix_len;
337 let def = resolve_doc_path_for_def(sema.db, def, &link, ns, is_inner)?;
338 cb(def, node, absolute_range)
339 })
340 }
341
342 fn doc_attributes(
352 sema: &Semantics<'_, RootDatabase>,
353 node: &SyntaxNode,
354 is_inner_doc: bool,
355 ) -> Option<(AttrsWithOwner, Definition)> {
356 if is_inner_doc && node.kind() != SOURCE_FILE {
357 let parent = node.parent()?;
358 doc_attributes(sema, &parent).or(doc_attributes(sema, &parent.parent()?))
359 } else {
360 doc_attributes(sema, node)
361 }
362 }
363}
364
365fn broken_link_clone_cb(link: BrokenLink<'_>) -> Option<(CowStr<'_>, CowStr<'_>)> {
366 Some((link.reference.clone(), link.reference))
367}
368
369fn get_doc_links(
377 db: &RootDatabase,
378 def: Definition,
379 target_dir: Option<&str>,
380 sysroot: Option<&str>,
381) -> DocumentationLinks {
382 let join_url = |base_url: Option<Url>, path: &str| -> Option<Url> {
383 base_url.and_then(|url| url.join(path).ok())
384 };
385
386 let Some((target, file, frag)) = filename_and_frag_for_def(db, def) else {
387 return Default::default();
388 };
389
390 let (mut web_url, mut local_url) = get_doc_base_urls(db, target, target_dir, sysroot);
391
392 let append_mod = !matches!(def, Definition::Macro(m) if m.is_macro_export(db));
393 if append_mod && let Some(path) = mod_path_of_def(db, target) {
394 web_url = join_url(web_url, &path);
395 local_url = join_url(local_url, &path);
396 }
397
398 web_url = join_url(web_url, &file);
399 local_url = join_url(local_url, &file);
400
401 if let Some(url) = web_url.as_mut() {
402 url.set_fragment(frag.as_deref())
403 }
404 if let Some(url) = local_url.as_mut() {
405 url.set_fragment(frag.as_deref())
406 }
407
408 DocumentationLinks {
409 web_url: web_url.map(|it| it.into()),
410 local_url: local_url.map(|it| it.into()),
411 }
412}
413
414fn rewrite_intra_doc_link(
415 db: &RootDatabase,
416 def: Definition,
417 target: &str,
418 title: &str,
419 is_inner_doc: bool,
420 link_type: LinkType,
421) -> Option<(String, String)> {
422 let (link, ns) = parse_intra_doc_link(target);
423
424 let (link, anchor) = match link.split_once('#') {
425 Some((new_link, anchor)) => (new_link, Some(anchor)),
426 None => (link, None),
427 };
428
429 let resolved = resolve_doc_path_for_def(db, def, link, ns, is_inner_doc)?;
430 let mut url = get_doc_base_urls(db, resolved, None, None).0?;
431
432 let (_, file, frag) = filename_and_frag_for_def(db, resolved)?;
433 if let Some(path) = mod_path_of_def(db, resolved) {
434 url = url.join(&path).ok()?;
435 }
436
437 let frag = anchor.or(frag.as_deref());
438
439 url = url.join(&file).ok()?;
440 url.set_fragment(frag);
441
442 let title = match link_type {
445 LinkType::Email
446 | LinkType::Autolink
447 | LinkType::Shortcut
448 | LinkType::Collapsed
449 | LinkType::Reference
450 | LinkType::Inline => title.to_owned(),
451 LinkType::ShortcutUnknown | LinkType::CollapsedUnknown | LinkType::ReferenceUnknown => {
452 strip_prefixes_suffixes(title).to_owned()
453 }
454 };
455
456 Some((url.into(), title))
457}
458
459fn rewrite_url_link(db: &RootDatabase, def: Definition, target: &str) -> Option<String> {
461 if !(target.contains('#') || target.contains(".html")) {
462 return None;
463 }
464
465 let mut url = get_doc_base_urls(db, def, None, None).0?;
466 let (def, file, frag) = filename_and_frag_for_def(db, def)?;
467
468 if let Some(path) = mod_path_of_def(db, def) {
469 url = url.join(&path).ok()?;
470 }
471
472 url = url.join(&file).ok()?;
473 url.set_fragment(frag.as_deref());
474 url.join(target).ok().map(Into::into)
475}
476
477fn mod_path_of_def(db: &RootDatabase, def: Definition) -> Option<String> {
478 def.canonical_module_path(db).map(|it| {
479 let mut path = String::new();
480 it.flat_map(|it| it.name(db)).for_each(|name| format_to!(path, "{}/", name.as_str()));
481 path
482 })
483}
484
485fn map_links<'e>(
487 events: impl Iterator<Item = (Event<'e>, Range<usize>)>,
488 callback: impl Fn(&str, &str, Range<usize>, LinkType) -> (Option<LinkType>, String, String),
489) -> impl Iterator<Item = Event<'e>> {
490 let mut in_link = false;
491 let mut end_link_target: Option<CowStr<'_>> = None;
493 let mut end_link_type: Option<LinkType> = None;
497
498 events.map(move |(evt, range)| match evt {
499 Event::Start(Tag::Link(link_type, ref target, _)) => {
500 in_link = true;
501 end_link_target = Some(target.clone());
502 end_link_type = Some(link_type);
503 evt
504 }
505 Event::End(Tag::Link(link_type, target, _)) => {
506 in_link = false;
507 Event::End(Tag::Link(
508 end_link_type.take().unwrap_or(link_type),
509 end_link_target.take().unwrap_or(target),
510 CowStr::Borrowed(""),
511 ))
512 }
513 Event::Text(s) if in_link => {
514 let (link_type, link_target_s, link_name) =
515 callback(&end_link_target.take().unwrap(), &s, range, end_link_type.unwrap());
516 end_link_target = Some(CowStr::Boxed(link_target_s.into()));
517 if !matches!(end_link_type, Some(LinkType::Autolink)) && link_type.is_some() {
518 end_link_type = link_type;
519 }
520 Event::Text(CowStr::Boxed(link_name.into()))
521 }
522 Event::Code(s) if in_link => {
523 let (link_type, link_target_s, link_name) =
524 callback(&end_link_target.take().unwrap(), &s, range, end_link_type.unwrap());
525 end_link_target = Some(CowStr::Boxed(link_target_s.into()));
526 if !matches!(end_link_type, Some(LinkType::Autolink)) && link_type.is_some() {
527 end_link_type = link_type;
528 }
529 Event::Code(CowStr::Boxed(link_name.into()))
530 }
531 _ => evt,
532 })
533}
534
535fn get_doc_base_urls(
544 db: &RootDatabase,
545 def: Definition,
546 target_dir: Option<&str>,
547 sysroot: Option<&str>,
548) -> (Option<Url>, Option<Url>) {
549 let local_doc = target_dir
550 .and_then(|path| Url::parse(&format!("file:///{path}/")).ok())
551 .and_then(|it| it.join("doc/").ok());
552 let system_doc = sysroot
553 .map(|sysroot| format!("file:///{sysroot}/share/doc/rust/html/"))
554 .and_then(|it| Url::parse(&it).ok());
555 let krate = def.krate(db);
556 let channel = krate
557 .and_then(|krate| db.toolchain_channel(krate.into()))
558 .unwrap_or(ReleaseChannel::Nightly)
559 .as_str();
560
561 if let Definition::BuiltinType(..) = def {
564 let web_link = Url::parse(&format!("https://doc.rust-lang.org/{channel}/core/")).ok();
565 let system_link = system_doc.and_then(|it| it.join("core/").ok());
566 return (web_link, system_link);
567 };
568
569 let Some(krate) = krate else { return Default::default() };
570 let Some(display_name) = krate.display_name(db) else { return Default::default() };
571 let (web_base, local_base) = match krate.origin(db) {
572 CrateOrigin::Lang(
575 origin @ (LangCrateOrigin::Alloc
576 | LangCrateOrigin::Core
577 | LangCrateOrigin::ProcMacro
578 | LangCrateOrigin::Std
579 | LangCrateOrigin::Test),
580 ) => {
581 let system_url = system_doc.and_then(|it| it.join(&format!("{origin}")).ok());
582 let web_url = format!("https://doc.rust-lang.org/{channel}/{origin}");
583 (Some(web_url), system_url)
584 }
585 CrateOrigin::Lang(_) => return (None, None),
586 CrateOrigin::Rustc { name: _ } => {
587 (Some(format!("https://doc.rust-lang.org/{channel}/nightly-rustc/")), None)
588 }
589 CrateOrigin::Local { repo: _, name: _ } => {
590 let weblink = krate.get_html_root_url(db).or_else(|| {
592 let version = krate.version(db);
593 Some(format!(
599 "https://docs.rs/{krate}/{version}/",
600 krate = display_name,
601 version = version.as_deref().unwrap_or("*")
602 ))
603 });
604 (weblink, local_doc)
605 }
606 CrateOrigin::Library { repo: _, name } => {
607 let weblink = krate.get_html_root_url(db).or_else(|| {
608 let version = krate.version(db);
609 Some(format!(
615 "https://docs.rs/{krate}/{version}/",
616 krate = name,
617 version = version.as_deref().unwrap_or("*")
618 ))
619 });
620 (weblink, local_doc)
621 }
622 };
623 let web_base = web_base
624 .and_then(|it| Url::parse(&it).ok())
625 .and_then(|it| it.join(&format!("{display_name}/")).ok());
626 let local_base = local_base.and_then(|it| it.join(&format!("{display_name}/")).ok());
627
628 (web_base, local_base)
629}
630
631fn filename_and_frag_for_def(
638 db: &dyn HirDatabase,
639 def: Definition,
640) -> Option<(Definition, String, Option<String>)> {
641 if let Some(assoc_item) = def.as_assoc_item(db) {
642 let def = match assoc_item.container(db) {
643 AssocItemContainer::Trait(t) => t.into(),
644 AssocItemContainer::Impl(i) => i.self_ty(db).as_adt()?.into(),
645 };
646 let (_, file, _) = filename_and_frag_for_def(db, def)?;
647 let frag = get_assoc_item_fragment(db, assoc_item)?;
648 return Some((def, file, Some(frag)));
649 }
650
651 let res = match def {
652 Definition::Adt(adt) => match adt {
653 Adt::Struct(s) => {
654 format!("struct.{}.html", s.name(db).as_str())
655 }
656 Adt::Enum(e) => format!("enum.{}.html", e.name(db).as_str()),
657 Adt::Union(u) => format!("union.{}.html", u.name(db).as_str()),
658 },
659 Definition::Crate(_) => String::from("index.html"),
660 Definition::Module(m) => match m.name(db) {
661 Some(name) => {
663 match m.attrs(db).by_key(sym::doc).find_string_value_in_tt(sym::keyword) {
664 Some(kw) => {
665 format!("keyword.{kw}.html")
666 }
667 None => format!("{}/index.html", name.as_str()),
668 }
669 }
670 None => String::from("index.html"),
671 },
672 Definition::Trait(t) => {
673 format!("trait.{}.html", t.name(db).as_str())
675 }
676 Definition::TypeAlias(t) => {
677 format!("type.{}.html", t.name(db).as_str())
678 }
679 Definition::BuiltinType(t) => {
680 format!("primitive.{}.html", t.name().as_str())
681 }
682 Definition::Function(f) => {
683 format!("fn.{}.html", f.name(db).as_str())
684 }
685 Definition::Variant(ev) => {
686 let def = Definition::Adt(ev.parent_enum(db).into());
687 let (_, file, _) = filename_and_frag_for_def(db, def)?;
688 return Some((def, file, Some(format!("variant.{}", ev.name(db).as_str()))));
689 }
690 Definition::Const(c) => {
691 format!("constant.{}.html", c.name(db)?.as_str())
692 }
693 Definition::Static(s) => {
694 format!("static.{}.html", s.name(db).as_str())
695 }
696 Definition::Macro(mac) => match mac.kind(db) {
697 hir::MacroKind::Declarative
698 | hir::MacroKind::AttrBuiltIn
699 | hir::MacroKind::DeclarativeBuiltIn
700 | hir::MacroKind::Attr
701 | hir::MacroKind::ProcMacro => {
702 format!("macro.{}.html", mac.name(db).as_str())
703 }
704 hir::MacroKind::Derive | hir::MacroKind::DeriveBuiltIn => {
705 format!("derive.{}.html", mac.name(db).as_str())
706 }
707 },
708 Definition::Field(field) => {
709 let def = match field.parent_def(db) {
710 hir::VariantDef::Struct(it) => Definition::Adt(it.into()),
711 hir::VariantDef::Union(it) => Definition::Adt(it.into()),
712 hir::VariantDef::Variant(it) => Definition::Variant(it),
713 };
714 let (_, file, _) = filename_and_frag_for_def(db, def)?;
715 return Some((def, file, Some(format!("structfield.{}", field.name(db).as_str()))));
716 }
717 Definition::SelfType(impl_) => {
718 let adt = impl_.self_ty(db).as_adt()?.into();
719 let (_, file, _) = filename_and_frag_for_def(db, adt)?;
720 return Some((adt, file, Some(String::from("impl"))));
722 }
723 Definition::ExternCrateDecl(it) => {
724 format!("{}/index.html", it.name(db).as_str())
725 }
726 Definition::Local(_)
727 | Definition::GenericParam(_)
728 | Definition::TupleField(_)
729 | Definition::Label(_)
730 | Definition::BuiltinAttr(_)
731 | Definition::BuiltinLifetime(_)
732 | Definition::ToolModule(_)
733 | Definition::DeriveHelper(_)
734 | Definition::InlineAsmRegOrRegClass(_)
735 | Definition::InlineAsmOperand(_) => return None,
736 };
737
738 Some((def, res, None))
739}
740
741fn get_assoc_item_fragment(db: &dyn HirDatabase, assoc_item: hir::AssocItem) -> Option<String> {
748 Some(match assoc_item {
749 AssocItem::Function(function) => {
750 let is_trait_method =
751 function.as_assoc_item(db).and_then(|assoc| assoc.container_trait(db)).is_some();
752 if is_trait_method && !function.has_body(db) {
756 format!("tymethod.{}", function.name(db).as_str())
757 } else {
758 format!("method.{}", function.name(db).as_str())
759 }
760 }
761 AssocItem::Const(constant) => {
762 format!("associatedconstant.{}", constant.name(db)?.as_str())
763 }
764 AssocItem::TypeAlias(ty) => {
765 format!("associatedtype.{}", ty.name(db).as_str())
766 }
767 })
768}