aboutsummaryrefslogtreecommitdiff
path: root/code/src/markdown/mod.rs
blob: 7ee18a2da6e3bc5e52dc961d79a2d7183dce3a3d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
use pest::{
    iterators::{Pair, Pairs},
    Parser,
};
use pest_derive::Parser;

use crate::{html::escape, syntax_highlight::syntax_highlight};

#[derive(Parser)]
#[grammar = "src/markdown/parser.pest"]
struct Grammar;

pub fn render(s: &str) -> String {
    match Grammar::parse(Rule::file, s) {
        Ok(pairs) => {
            eprintln!("{pairs:#?}");
            render_pairs(pairs)
        }
        Err(e) => panic!("{e}"),
    }
}

pub fn render_pairs(p: Pairs<Rule>) -> String {
    p.map(|p| render_ast(p)).collect::<Vec<_>>().join("")
}

pub fn render_ast(p: Pair<Rule>) -> String {
    match p.as_rule() {
        Rule::block => render_pairs(p.into_inner()),
        Rule::header => {
            let mut level = 0;
            while p.as_str()[level..].starts_with("#") {
                level += 1
            }
            format!("<h{level}>{}</h{level}>", render_pairs(p.into_inner()))
        }
        Rule::paragraph => format!("<p>{}</p>", render_pairs(p.into_inner())),
        Rule::style_italic => format!("<i>{}</i>", render_pairs(p.into_inner())),
        Rule::style_bold => format!("<b>{}</b>", render_pairs(p.into_inner())),
        Rule::style_code => format!("<code>{}</code>", p.into_inner().next().unwrap().as_str()),
        Rule::unordered_list => format!("<ul>{}</ul>", render_pairs(p.into_inner())),
        Rule::ordered_list => format!("<ol>{}</ol>", render_pairs(p.into_inner())),
        Rule::unordered_list_item | Rule::ordered_list_item => {
            format!("<li>{}</li>", render_pairs(p.into_inner()))
        }
        Rule::hyperlink => {
            let k = p.into_inner().collect::<Vec<_>>();
            let label = k[0].as_str();
            let target = k[1].as_str();
            format!("<a href=\"{}\">{}</a>", escape(target), escape(label))
        }
        Rule::span => render_pairs(p.into_inner()),
        Rule::EOI => "".to_string(),
        Rule::code_block => {
            let k = p.into_inner().collect::<Vec<_>>();
            let lang = k[0].as_str();
            let inner = k[1].as_str();
            format!(
                "<pre>{}</pre>",
                syntax_highlight(lang, &inner).unwrap_or_else(|| escape(&inner))
            )
        }
        Rule::inline_latex => fix_katex(
            &katex::render_with_opts(
                &p.into_inner().as_str(),
                &katex::OptsBuilder::default().build().unwrap(),
            )
            .unwrap(),
        ),
        Rule::latex_block => fix_katex(
            &katex::render_with_opts(
                &p.into_inner().as_str(),
                &katex::OptsBuilder::default()
                    .display_mode(true)
                    .build()
                    .unwrap(),
            )
            .unwrap(),
        ),

        Rule::text => escape(p.as_str()),

        _ => todo!("{:?}", p.as_rule()),
    }
}

// TODO this is *really* bad fix
fn fix_katex<'a>(s: &str) -> String {
    let e = s.find("<span class=\"katex-html\"").unwrap();
    s[0..e].replace(
        "<mspace linebreak=\"newline\"></mspace>",
        "</mrow></semantics></math><math xmlns=\"http://www.w3.org/1998/Math/MathML\" display=\"block\"><semantics><mrow>",
    )
}