zola/components/utils/src/templates.rs

115 lines
4.2 KiB
Rust
Raw Normal View History

use std::collections::HashMap;
2017-08-23 10:17:24 +00:00
use tera::{Tera, Context};
use errors::Result;
2017-11-27 17:09:09 +00:00
static DEFAULT_TPL: &str = include_str!("default_tpl.html");
macro_rules! render_default_tpl {
($filename: expr, $url: expr) => {
{
let mut context = Context::new();
2018-09-09 17:43:14 +00:00
context.insert("filename", $filename);
context.insert("url", $url);
2017-11-27 17:09:09 +00:00
Tera::one_off(DEFAULT_TPL, &context, true).map_err(|e| e.into())
}
};
}
2017-08-23 10:17:24 +00:00
/// Renders the given template with the given context, but also ensures that, if the default file
/// is not found, it will look up for the equivalent template for the current theme if there is one.
/// Lastly, if it's a default template (index, section or page), it will just return an empty string
/// to avoid an error if there isn't a template with that name
pub fn render_template(name: &str, tera: &Tera, context: &Context, theme: &Option<String>) -> Result<String> {
2017-08-23 10:17:24 +00:00
if tera.templates.contains_key(name) {
return tera
.render(name, context)
.map_err(|e| e.into());
}
2018-05-11 11:54:16 +00:00
if let Some(ref t) = *theme {
2017-08-23 10:17:24 +00:00
return tera
.render(&format!("{}/templates/{}", t, name), context)
.map_err(|e| e.into());
}
2017-11-27 17:09:09 +00:00
// maybe it's a default one?
match name {
"index.html" | "section.html" => {
render_default_tpl!(name, "https://www.getgutenberg.io/documentation/templates/pages-sections/#section-variables")
2018-07-31 13:17:31 +00:00
}
2017-11-27 17:09:09 +00:00
"page.html" => {
render_default_tpl!(name, "https://www.getgutenberg.io/documentation/templates/pages-sections/#page-variables")
2018-07-31 13:17:31 +00:00
}
2018-07-16 08:54:05 +00:00
"single.html" | "list.html" => {
render_default_tpl!(name, "https://www.getgutenberg.io/documentation/templates/taxonomies/")
2018-07-31 13:17:31 +00:00
}
2017-11-27 17:09:09 +00:00
_ => bail!("Tried to render `{}` but the template wasn't found", name)
}
2017-08-23 10:17:24 +00:00
}
/// Rewrites the path from extend/macros of the theme used to ensure
/// that they will point to the right place (theme/templates/...)
/// Include is NOT supported as it would be a pain to add and using blocks
/// or macros is always better anyway for themes
/// This will also rename the shortcodes to NOT have the themes in the path
/// so themes shortcodes can be used.
pub fn rewrite_theme_paths(tera: &mut Tera, theme: &str) {
let mut shortcodes_to_move = vec![];
let mut templates = HashMap::new();
let old_templates = ::std::mem::replace(&mut tera.templates, HashMap::new());
// We want to match the paths in the templates to the new names
for (key, mut tpl) in old_templates{
tpl.name = format!("{}/templates/{}", theme, tpl.name);
// First the parent if there is none
if let Some(ref p) = tpl.parent.clone() {
tpl.parent = Some(format!("{}/templates/{}", theme, p));
}
// Next the macros import
let mut updated = vec![];
for &(ref filename, ref namespace) in &tpl.imported_macro_files {
updated.push((format!("{}/templates/{}", theme, filename), namespace.to_string()));
}
tpl.imported_macro_files = updated;
if tpl.name.starts_with(&format!("{}/templates/shortcodes", theme)) {
let new_name = tpl.name.replace(&format!("{}/templates/", theme), "");
shortcodes_to_move.push((key, new_name.clone()));
tpl.name = new_name;
}
templates.insert(tpl.name.clone(), tpl);
}
tera.templates = templates;
// and then replace shortcodes in the Tera instance using the new names
for (old_name, new_name) in shortcodes_to_move {
let tpl = tera.templates.remove(&old_name).unwrap();
tera.templates.insert(new_name, tpl);
}
}
#[cfg(test)]
mod tests {
use tera::Tera;
use super::rewrite_theme_paths;
#[test]
fn can_rewrite_all_paths_of_theme() {
2017-11-27 17:09:09 +00:00
let mut tera = Tera::parse("test-templates/*.html").unwrap();
rewrite_theme_paths(&mut tera, "hyde");
// special case to make the test work: we also rename the files to
// match the imports
for (key, val) in tera.templates.clone() {
tera.templates.insert(format!("hyde/templates/{}", key), val.clone());
}
tera.build_inheritance_chains().unwrap();
}
}