mod block;
mod inline;
#[cfg(test)]
mod tests;
use anyhow::Error;
use pest::iterators::Pairs;
use document_tree::{
Element, HasChildren, attribute_types as at, element_categories as c, elements as e,
};
use crate::pest_rst::Rule;
fn ssubel_to_section_unchecked_mut(ssubel: &mut c::StructuralSubElement) -> &mut e::Section {
match ssubel {
c::StructuralSubElement::SubStructure(b) => match b.as_mut() {
c::SubStructure::Section(s) => s,
_ => unreachable!(),
},
_ => unreachable!(),
}
}
fn get_level<'tl>(
toplevel: &'tl mut Vec<c::StructuralSubElement>,
section_idxs: &[Option<usize>],
) -> &'tl mut Vec<c::StructuralSubElement> {
let mut level = toplevel;
for i in section_idxs.iter().flatten().copied() {
level = ssubel_to_section_unchecked_mut(&mut level[i]).children_mut();
}
level
}
pub fn convert_document(pairs: Pairs<Rule>) -> Result<e::Document, Error> {
use self::block::TitleOrSsubel::{Ssubel, Title};
let mut toplevel: Vec<c::StructuralSubElement> = vec![];
let mut kinds: Vec<block::TitleKind> = vec![];
let mut section_idxs: Vec<Option<usize>> = vec![];
for pair in pairs {
if let Some(ssubel) = block::convert_ssubel(pair)? {
match ssubel {
Title(title, kind) => {
match kinds.iter().position(|k| k == &kind) {
Some(idx) => {
section_idxs.truncate(idx);
while section_idxs.len() < idx {
section_idxs.push(None);
}
}
None => kinds.push(kind),
}
let super_level = get_level(&mut toplevel, §ion_idxs);
let slug = title
.names()
.iter()
.next()
.map(|at::NameToken(name)| at::ID(name.to_owned()));
let mut section = e::Section::with_children(vec![title.into()]);
section.ids_mut().extend(slug.into_iter());
super_level.push(section.into());
section_idxs.push(Some(super_level.len() - 1));
}
Ssubel(elem) => get_level(&mut toplevel, §ion_idxs).push(elem),
}
}
}
Ok(e::Document::with_children(toplevel))
}
pub fn whitespace_normalize_name(name: &str) -> String {
let split_iter = name
.split(|ch: char| ch.is_whitespace() || ('\x1C'..='\x1F').contains(&ch))
.filter(|split| !split.is_empty());
let mut ret = String::new();
for split in split_iter {
if !ret.is_empty() {
ret.push(' ');
}
ret.push_str(split);
}
ret
}