1use markdown_it::{generics::inline::emph_pair, MarkdownIt, Node, NodeValue, Renderer};
18
19#[derive(Debug)]
20pub struct Sup;
21
22impl NodeValue for Sup {
23 fn render(&self, node: &Node, fmt: &mut dyn Renderer) {
24 fmt.open("sup", &node.attrs);
25 fmt.contents(&node.children);
26 fmt.close("sup");
27 }
28}
29
30pub fn add(md: &mut MarkdownIt) {
32 emph_pair::add_with::<'^', 1, true>(md, || Node::new(Sup));
33}
34
35#[cfg(test)]
36mod test {
37 use super::add;
38 use markdown_it::{
39 plugins::{cmark, extra},
40 MarkdownIt,
41 };
42 use rstest::rstest;
43 use std::sync::LazyLock;
44
45 static MARKDOWN_PARSER: LazyLock<MarkdownIt> = LazyLock::new(|| {
46 let mut parser = MarkdownIt::new();
47 cmark::add(&mut parser);
48 extra::add(&mut parser);
49 add(&mut parser);
50
51 parser
52 });
53
54 #[rstest]
55 #[case("^foo^", "<p><sup>foo</sup></p>\n")]
56 #[case("foo^", "<p>foo^</p>\n")]
57 #[case("^foo", "<p>^foo</p>\n")]
58 #[case("foo^bar^", "<p>foo<sup>bar</sup></p>\n")]
59 #[case("\\^foo^", "<p>^foo^</p>\n")]
60 #[case("^foo\\^", "<p>^foo^</p>\n")]
61 fn test(#[case] md_str: &str, #[case] expected: &str) {
62 let result = MARKDOWN_PARSER.parse(md_str).xrender();
63
64 assert_eq!(result, String::from(expected));
65 }
66}