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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
use super::*;

impl Display for MathOperator {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write_tag_start(f, self)?;
        safe_html_str(f, &self.operator)?;
        write_tag_close(f, self)
    }
}

impl Display for MathSpace {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write_tag_self_close(f, self)
    }
}

impl Display for MathSqrt {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        match &self.surd {
            Some(power) => {
                write!(f, "<mroot>{}{}</mroot>", self.base, power)
            }
            None => {
                write!(f, "<msqrt>{}</msqrt>", self.base)
            }
        }
    }
}

impl Display for MathFenced {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        let last = self.separators.chars().last().unwrap_or(',');
        let mut separators = self.separators.chars().chain(repeat(last));
        f.write_str("<mrow><mo stretchy=\"true\" form=\"prefix\">")?;
        safe_html_char(f, self.open)?;
        f.write_str("</mo>")?;
        for (i, item) in self.base.iter().enumerate() {
            if i == 0 {
                write!(f, "{}", item)?;
            }
            else {
                // SAFETY: `separators` is infinite
                let split = unsafe { separators.next().unwrap_unchecked() };
                f.write_str("<mo>")?;
                safe_html_char(f, split)?;
                f.write_str("</mo>")?;
                write!(f, "{}", item)?;
            }
        }
        f.write_str("<mo stretchy=\"true\" form=\"postfix\">")?;
        safe_html_char(f, self.close)?;
        f.write_str("</mo></mrow>")
    }
}

impl Display for MathMultiScript {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write_tag_start(f, self)?;
        Display::fmt(&self.base, f)?;
        if self.is_sub_super_script() {
            // SAFETY: rd, ru has only one element
            unsafe {
                Display::fmt(self.rd.get_unchecked(0), f)?;
                Display::fmt(self.ru.get_unchecked(0), f)?;
            }
        }
        else if self.is_sub_script() {
            // SAFETY: rd has only one element
            unsafe {
                Display::fmt(self.rd.get_unchecked(0), f)?;
            }
        }
        else if self.is_super_script() {
            // SAFETY: ru has only one element
            unsafe {
                Display::fmt(self.ru.get_unchecked(0), f)?;
            }
        }
        else {
            let r_max = self.ru.len().max(self.rd.len());
            for i in 0..r_max {
                match self.rd.get(i) {
                    Some(s) => write!(f, "{}", s)?,
                    None => f.write_str("<mrow/>")?,
                }
                match self.ru.get(i) {
                    Some(s) => write!(f, "{}", s)?,
                    None => f.write_str("<mrow/>")?,
                }
            }
            f.write_str("<mprescripts/>")?;
            let l_max = self.lu.len().max(self.ld.len());
            for i in 0..l_max {
                match self.ld.get(i) {
                    Some(s) => write!(f, "{}", s)?,
                    None => f.write_str("<mrow/>")?,
                }
                match self.lu.get(i) {
                    Some(s) => write!(f, "{}", s)?,
                    None => f.write_str("<mrow/>")?,
                }
            }
        }
        write_tag_close(f, self)
    }
}

impl Display for MathUnderOver {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        write_tag_start(f, self)?;
        write!(f, "{}", self.base)?;
        if let Some(under) = &self.under {
            write!(f, "{}", under)?;
        }
        if let Some(over) = &self.over {
            write!(f, "{}", over)?;
        }
        write_tag_close(f, self)
    }
}