General structure to the rendering code.
This commit is contained in:
parent
680fc167ea
commit
da6655d4b6
@ -1,12 +1,20 @@
|
|||||||
use crate::parser::template;
|
use crate::parser::template;
|
||||||
|
use crate::parser::Body;
|
||||||
|
use crate::parser::DustTag;
|
||||||
|
use crate::parser::PartialNameElement;
|
||||||
use crate::parser::Path;
|
use crate::parser::Path;
|
||||||
use crate::parser::Template;
|
use crate::parser::Template;
|
||||||
|
use crate::parser::TemplateElement;
|
||||||
use crate::renderer::breadcrumb_tree::BreadcrumbTree;
|
use crate::renderer::breadcrumb_tree::BreadcrumbTree;
|
||||||
use crate::renderer::breadcrumb_tree::BreadcrumbTreeElement;
|
use crate::renderer::breadcrumb_tree::BreadcrumbTreeElement;
|
||||||
use crate::renderer::context_element::ContextElement;
|
use crate::renderer::context_element::ContextElement;
|
||||||
use crate::renderer::context_element::IntoContextElement;
|
use crate::renderer::context_element::IntoContextElement;
|
||||||
use crate::renderer::errors::CompileError;
|
use crate::renderer::errors::CompileError;
|
||||||
|
use crate::renderer::errors::RenderError;
|
||||||
|
use crate::renderer::inline_partial_tree::extract_inline_partials;
|
||||||
|
use crate::renderer::inline_partial_tree::InlinePartialTreeElement;
|
||||||
use crate::renderer::tree_walking::walk_path;
|
use crate::renderer::tree_walking::walk_path;
|
||||||
|
use std::borrow::Borrow;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
@ -32,6 +40,103 @@ impl<'a> DustRenderer<'a> {
|
|||||||
self.templates.insert(name, template);
|
self.templates.insert(name, template);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn render<C>(&'a self, name: &str, context: Option<&C>) -> Result<String, RenderError>
|
||||||
|
where
|
||||||
|
C: IntoContextElement,
|
||||||
|
{
|
||||||
|
let breadcrumbs =
|
||||||
|
context.map(|ctx| BreadcrumbTree::new(None, BreadcrumbTreeElement::Borrowed(ctx)));
|
||||||
|
self.render_template(name, breadcrumbs.as_ref(), None)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn render_template(
|
||||||
|
&'a self,
|
||||||
|
name: &str,
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree>,
|
||||||
|
blocks: Option<&'a InlinePartialTreeElement<'a>>,
|
||||||
|
) -> Result<String, RenderError> {
|
||||||
|
let main_template = match self.templates.get(name) {
|
||||||
|
Some(tmpl) => tmpl,
|
||||||
|
None => {
|
||||||
|
return Err(RenderError::TemplateNotFound(name.to_owned()));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let extracted_inline_partials = extract_inline_partials(main_template);
|
||||||
|
let new_blocks = InlinePartialTreeElement::new(blocks, extracted_inline_partials);
|
||||||
|
let new_block_context = BlockContext {
|
||||||
|
breadcrumbs: breadcrumbs,
|
||||||
|
blocks: &new_blocks,
|
||||||
|
};
|
||||||
|
self.render_body(&main_template.contents, breadcrumbs, &new_block_context)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_maybe_body(
|
||||||
|
&'a self,
|
||||||
|
body: &'a Option<Body>,
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree>,
|
||||||
|
blocks: &'a BlockContext<'a>,
|
||||||
|
) -> Result<String, RenderError> {
|
||||||
|
match body {
|
||||||
|
None => Ok("".to_owned()),
|
||||||
|
Some(body) => Ok(self.render_body(body, breadcrumbs, blocks)?),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_body(
|
||||||
|
&'a self,
|
||||||
|
body: &'a Body,
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree>,
|
||||||
|
blocks: &'a BlockContext<'a>,
|
||||||
|
) -> Result<String, RenderError> {
|
||||||
|
let mut output = String::new();
|
||||||
|
for elem in &body.elements {
|
||||||
|
match elem {
|
||||||
|
TemplateElement::TEIgnoredWhitespace(_) => {}
|
||||||
|
TemplateElement::TESpan(span) => output.push_str(span.contents),
|
||||||
|
TemplateElement::TETag(dt) => {
|
||||||
|
output.push_str(&self.render_tag(dt, breadcrumbs, blocks)?);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(output)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// For rendering a dynamic partial's name or an rvalue template
|
||||||
|
pub fn render_partial_name(
|
||||||
|
&'a self,
|
||||||
|
body: &'a Vec<PartialNameElement>,
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree>,
|
||||||
|
) -> Result<String, RenderError> {
|
||||||
|
let converted_to_template_elements: Vec<TemplateElement<'a>> =
|
||||||
|
body.into_iter().map(|e| e.into()).collect();
|
||||||
|
// Simple templates like partial names and reference rvalues
|
||||||
|
// cannot contain blocks or inline partials, so we use a blank
|
||||||
|
// BlockContext.
|
||||||
|
let empty_block_context = BlockContext {
|
||||||
|
breadcrumbs: None,
|
||||||
|
blocks: &InlinePartialTreeElement::new(None, HashMap::new()),
|
||||||
|
};
|
||||||
|
self.render_body(
|
||||||
|
&Body {
|
||||||
|
elements: converted_to_template_elements,
|
||||||
|
},
|
||||||
|
breadcrumbs,
|
||||||
|
&empty_block_context,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_tag(
|
||||||
|
&'a self,
|
||||||
|
tag: &'a DustTag,
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree>,
|
||||||
|
blocks: &'a BlockContext<'a>,
|
||||||
|
) -> Result<String, RenderError> {
|
||||||
|
match tag {
|
||||||
|
_ => panic!("Unsupported tag"),
|
||||||
|
}
|
||||||
|
Ok("".to_owned())
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns a option of a tuple of (parent, new_node_elements)
|
/// Returns a option of a tuple of (parent, new_node_elements)
|
||||||
/// which can then be formed into new BreadcrumbTreeNodes
|
/// which can then be formed into new BreadcrumbTreeNodes
|
||||||
///
|
///
|
||||||
@ -42,8 +147,8 @@ impl<'a> DustRenderer<'a> {
|
|||||||
/// explicit contexts) and the additional node elements (which may
|
/// explicit contexts) and the additional node elements (which may
|
||||||
/// be empty) should be combined into a final BreadcrumbTreeNode
|
/// be empty) should be combined into a final BreadcrumbTreeNode
|
||||||
fn new_breadcrumbs_section<'b>(
|
fn new_breadcrumbs_section<'b>(
|
||||||
&self,
|
&'b self,
|
||||||
maybe_breadcrumbs: Option<&'a BreadcrumbTree>,
|
maybe_breadcrumbs: Option<&'b BreadcrumbTree>,
|
||||||
index_context: Option<&'b dyn IntoContextElement>,
|
index_context: Option<&'b dyn IntoContextElement>,
|
||||||
injected_context: Option<&'b dyn IntoContextElement>,
|
injected_context: Option<&'b dyn IntoContextElement>,
|
||||||
explicit_context: &Option<Path<'b>>,
|
explicit_context: &Option<Path<'b>>,
|
||||||
@ -87,6 +192,15 @@ impl<'a> DustRenderer<'a> {
|
|||||||
Some((parent, new_nodes))
|
Some((parent, new_nodes))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns a option of a tuple of (parent, new_node_elements)
|
||||||
|
/// which can then be formed into new BreadcrumbTreeNodes
|
||||||
|
///
|
||||||
|
/// If None is returned, then it is a signal to simply re-use the
|
||||||
|
/// existing breadcrumbs.
|
||||||
|
///
|
||||||
|
/// Otherwise, the parent (which may be None, especially for
|
||||||
|
/// explicit contexts) and the additional node elements (which may
|
||||||
|
/// be empty) should be combined into a final BreadcrumbTreeNode
|
||||||
fn new_breadcrumbs_partial<'b>(
|
fn new_breadcrumbs_partial<'b>(
|
||||||
&'b self,
|
&'b self,
|
||||||
maybe_breadcrumbs: Option<&'b BreadcrumbTree>,
|
maybe_breadcrumbs: Option<&'b BreadcrumbTree>,
|
||||||
@ -168,3 +282,9 @@ impl<'a> DustRenderer<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct BlockContext<'a> {
|
||||||
|
/// The breadcrumbs at the time of entering the current partial
|
||||||
|
breadcrumbs: Option<&'a BreadcrumbTree<'a>>,
|
||||||
|
blocks: &'a InlinePartialTreeElement<'a>,
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user