diff --git a/crates/rustdoc_to_markdown/src/markdown_writer.rs b/crates/rustdoc_to_markdown/src/markdown_writer.rs index cd530b0a4d..2764ebe17f 100644 --- a/crates/rustdoc_to_markdown/src/markdown_writer.rs +++ b/crates/rustdoc_to_markdown/src/markdown_writer.rs @@ -224,10 +224,13 @@ impl MarkdownWriter { return StartTagOutcome::Skip; } - if tag.attrs.borrow().iter().any(|attr| { - attr.name.local.to_string() == "class" && attr.value.to_string() == "item-name" - }) { - self.push_str("`"); + if self.is_inside_item_name() { + if tag.attrs.borrow().iter().any(|attr| { + attr.name.local.to_string() == "class" + && attr.value.split(' ').any(|class| class.trim() == "stab") + }) { + self.push_str(" ["); + } } } _ => {} @@ -265,11 +268,20 @@ impl MarkdownWriter { "table" => { self.current_table_columns = 0; } - "div" => { + "div" | "span" => { if tag.attrs.borrow().iter().any(|attr| { attr.name.local.to_string() == "class" && attr.value.to_string() == "item-name" }) { - self.push_str("`: "); + self.push_str(": "); + } + + if self.is_inside_item_name() { + if tag.attrs.borrow().iter().any(|attr| { + attr.name.local.to_string() == "class" + && attr.value.split(' ').any(|class| class.trim() == "stab") + }) { + self.push_str("]"); + } } } _ => {} @@ -283,8 +295,24 @@ impl MarkdownWriter { } let trimmed_text = text.trim_matches(|char| char == '\n' || char == '\r' || char == '§'); + + if self.is_inside_item_name() && !self.is_inside("span") && !self.is_inside("code") { + self.push_str(&format!("`{trimmed_text}`")); + return Ok(()); + } + self.push_str(trimmed_text); Ok(()) } + + /// Returns whether we're currently inside of an `.item-name` element, which + /// rustdoc uses to display Rust items in a list. + fn is_inside_item_name(&self) -> bool { + self.current_element_stack.iter().any(|element| { + element.attrs.borrow().iter().any(|attr| { + attr.name.local.to_string() == "class" && attr.value.to_string() == "item-name" + }) + }) + } } diff --git a/crates/rustdoc_to_markdown/src/rustdoc_to_markdown.rs b/crates/rustdoc_to_markdown/src/rustdoc_to_markdown.rs index c4743c00b8..b900c8e795 100644 --- a/crates/rustdoc_to_markdown/src/rustdoc_to_markdown.rs +++ b/crates/rustdoc_to_markdown/src/rustdoc_to_markdown.rs @@ -139,6 +139,42 @@ mod tests { ) } + #[test] + fn test_item_table() { + let html = indoc! {r##" +

Structs§

+ +

Functions§

+ + "##}; + let expected = indoc! {r#" + ## Structs + + - `Error`: Errors that can happen when using axum. + - `Extension`: Extractor and response for extensions. + - `Form` [`form`]: URL encoded extractor and response. + - `Json` [`json`]: JSON Extractor / Response. + - `Router`: The router type for composing handlers and services. + + ## Functions + + - `serve` [`tokio` and (`http1` or `http2`)]: Serve the service with the supplied listener. + "#} + .trim(); + + assert_eq!( + convert_rustdoc_to_markdown(html.as_bytes()).unwrap(), + expected + ) + } + #[test] fn test_table() { let html = indoc! {r##"