use std::cell::{Ref, RefCell};
use std::rc::Rc;
use super::{MjRaw, MjRawChild, NAME};
use crate::helper::size::Pixel;
use crate::prelude::render::{Error, Header, Options, Render, Renderable};
impl<'r, 'e: 'r, 'h: 'r> Renderable<'r, 'e, 'h> for MjRawChild {
fn renderer(&'e self, header: Rc<RefCell<Header<'h>>>) -> Box<dyn Render<'h> + 'r> {
match self {
Self::Comment(elt) => elt.renderer(header),
Self::Node(elt) => elt.renderer(header),
Self::Text(elt) => elt.renderer(header),
}
}
}
struct MjRawRender<'e, 'h> {
header: Rc<RefCell<Header<'h>>>,
element: &'e MjRaw,
container_width: Option<Pixel>,
}
impl<'e, 'h> Render<'h> for MjRawRender<'e, 'h> {
fn tag(&self) -> Option<&str> {
Some(NAME)
}
fn header(&self) -> Ref<Header<'h>> {
self.header.borrow()
}
fn set_container_width(&mut self, width: Option<Pixel>) {
self.container_width = width;
}
fn render(&self, opts: &Options) -> Result<String, Error> {
let siblings = self.element.children.len();
self.element.children.iter().enumerate().try_fold(
String::default(),
|res, (index, child)| {
let mut renderer = child.renderer(Rc::clone(&self.header));
renderer.set_index(index);
renderer.set_siblings(siblings);
renderer.set_raw_siblings(siblings);
renderer.set_container_width(self.container_width.clone());
Ok(res + &renderer.render(opts)?)
},
)
}
}
impl<'r, 'e: 'r, 'h: 'r> Renderable<'r, 'e, 'h> for MjRaw {
fn renderer(&'e self, header: Rc<RefCell<Header<'h>>>) -> Box<dyn Render<'h> + 'r> {
Box::new(MjRawRender::<'e, 'h> {
element: self,
header,
container_width: None,
})
}
}
#[cfg(test)]
mod tests {
crate::should_render!(basic, "mj-raw");
crate::should_render!(in_head, "mj-raw-head");
}