use super::StructuredPrinter;
use super::TagHandler;
use markup5ever_rcdom::Handle;
fn list_hierarchy(printer: &mut StructuredPrinter) -> Vec<&String> {
printer
.parent_chain
.iter()
.rev()
.filter(|&tag| tag == "ul" || tag == "ol" || tag == "menu")
.collect()
}
#[derive(Default)]
pub struct ListHandler;
impl TagHandler for ListHandler {
fn handle(&mut self, _tag: &Handle, printer: &mut StructuredPrinter) {
printer.insert_newline();
if list_hierarchy(printer).is_empty() {
printer.insert_newline();
}
}
fn after_handle(&mut self, printer: &mut StructuredPrinter) {
printer.insert_newline();
}
}
#[derive(Default)]
pub struct ListItemHandler {
start_pos: usize,
list_type: String,
}
impl TagHandler for ListItemHandler {
fn handle(&mut self, _tag: &Handle, printer: &mut StructuredPrinter) {
{
let parent_lists = list_hierarchy(printer);
let nearest_parent_list = parent_lists.first();
if nearest_parent_list.is_none() {
return;
}
if let Some(s) = nearest_parent_list {
self.list_type = s.to_string();
}
}
if !printer.data.ends_with('\n') {
printer.insert_newline();
}
let current_depth = printer.parent_chain.len();
let order = if printer.siblings.len() >= current_depth {
printer.siblings[¤t_depth].len() + 1
} else {
0
};
match self.list_type.as_ref() {
"ul" | "menu" => printer.append_str("* "), "ol" => printer.append_str(&(order.to_string() + ". ")), _ => (), }
self.start_pos = printer.data.len();
}
fn after_handle(&mut self, printer: &mut StructuredPrinter) {
let padding = match self.list_type.as_ref() {
"ul" => 2,
"ol" => 3,
_ => 4,
};
let index = self.start_pos;
while index < printer.data.len() {
if printer.data.as_bytes().get(index).copied() == Some(b'\n')
|| printer.data.as_bytes().get(index).copied() == Some(b' ')
{
printer.data.remove(index);
} else {
break;
}
}
let mut index = printer.data.len();
while index > self.start_pos {
if printer.data.as_bytes().get(index).copied() == Some(b'\n') {
printer.insert_str(index + 1, &" ".repeat(padding));
}
index -= 1;
}
}
}