mirror of
https://github.com/askama-rs/askama.git
synced 2025-10-01 06:51:15 +00:00
Add support for loop.last
This commit is contained in:
parent
c5c37f56a1
commit
df07f7f5e3
@ -433,6 +433,7 @@ pub trait Template {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub use crate::shared::filters;
|
pub use crate::shared::filters;
|
||||||
|
pub use crate::shared::helpers;
|
||||||
pub use crate::shared::{read_config_file, Error, MarkupDisplay, Result};
|
pub use crate::shared::{read_config_file, Error, MarkupDisplay, Result};
|
||||||
pub use askama_derive::*;
|
pub use askama_derive::*;
|
||||||
|
|
||||||
|
@ -467,11 +467,17 @@ impl<'a> Generator<'a> {
|
|||||||
let expr_code = self.visit_expr_root(iter);
|
let expr_code = self.visit_expr_root(iter);
|
||||||
|
|
||||||
self.write_buf_writable(buf);
|
self.write_buf_writable(buf);
|
||||||
buf.write("for (_loop_index, ");
|
buf.write("for (_loop_index, _loop_last, ");
|
||||||
self.visit_target(buf, var);
|
self.visit_target(buf, var);
|
||||||
match iter {
|
match iter {
|
||||||
Expr::Range(_, _, _) => buf.writeln(&format!(") in ({}).enumerate() {{", expr_code)),
|
Expr::Range(_, _, _) => buf.writeln(&format!(
|
||||||
_ => buf.writeln(&format!(") in (&{}).into_iter().enumerate() {{", expr_code)),
|
") in ::askama::helpers::enumerate({}) {{",
|
||||||
|
expr_code
|
||||||
|
)),
|
||||||
|
_ => buf.writeln(&format!(
|
||||||
|
") in ::askama::helpers::enumerate((&{}).into_iter()) {{",
|
||||||
|
expr_code
|
||||||
|
)),
|
||||||
};
|
};
|
||||||
|
|
||||||
self.handle(ctx, body, buf, AstLevel::Nested);
|
self.handle(ctx, body, buf, AstLevel::Nested);
|
||||||
@ -915,6 +921,9 @@ impl<'a> Generator<'a> {
|
|||||||
} else if attr == "first" {
|
} else if attr == "first" {
|
||||||
buf.write("(_loop_index == 0)");
|
buf.write("(_loop_index == 0)");
|
||||||
return DisplayWrap::Unwrapped;
|
return DisplayWrap::Unwrapped;
|
||||||
|
} else if attr == "last" {
|
||||||
|
buf.write("_loop_last");
|
||||||
|
return DisplayWrap::Unwrapped;
|
||||||
} else {
|
} else {
|
||||||
panic!("unknown loop variable");
|
panic!("unknown loop variable");
|
||||||
}
|
}
|
||||||
|
48
askama_shared/src/helpers/mod.rs
Normal file
48
askama_shared/src/helpers/mod.rs
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
use std::iter::Peekable;
|
||||||
|
|
||||||
|
pub struct Enumerate<I>
|
||||||
|
where
|
||||||
|
I: Iterator,
|
||||||
|
{
|
||||||
|
iter: Peekable<I>,
|
||||||
|
count: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<I> Iterator for Enumerate<I>
|
||||||
|
where
|
||||||
|
I: Iterator,
|
||||||
|
{
|
||||||
|
type Item = (usize, bool, <I as Iterator>::Item);
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn next(&mut self) -> Option<(usize, bool, <I as Iterator>::Item)> {
|
||||||
|
self.iter.next().map(|a| {
|
||||||
|
let last = self.iter.peek().is_none();
|
||||||
|
let ret = (self.count, last, a);
|
||||||
|
// Possible undefined overflow.
|
||||||
|
self.count += 1;
|
||||||
|
ret
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
self.iter.size_hint()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn count(self) -> usize {
|
||||||
|
self.iter.count()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn enumerate<I>(iter: I) -> Enumerate<I>
|
||||||
|
where
|
||||||
|
I: Iterator,
|
||||||
|
{
|
||||||
|
Enumerate {
|
||||||
|
iter: iter.peekable(),
|
||||||
|
count: 0,
|
||||||
|
}
|
||||||
|
}
|
@ -16,6 +16,7 @@ pub use askama_escape::MarkupDisplay;
|
|||||||
use std::collections::BTreeMap;
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
pub mod filters;
|
pub mod filters;
|
||||||
|
pub mod helpers;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Config<'a> {
|
pub struct Config<'a> {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{% for s in 0..2 -%}
|
{% for s in 0..2 -%}
|
||||||
foo
|
foo{% if loop.first %} (first){% endif %}{% if loop.last %} (last){% endif %}
|
||||||
{% endfor -%}
|
{% endfor -%}
|
||||||
|
|
||||||
{% for s in init..1 -%}
|
{% for s in init..1 -%}
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
{% for s in strings %}
|
{% for s in strings %}
|
||||||
{{- loop.index0 }}. {{ s }}{{ 2 * loop.index }}{% if !loop.first %}{% else %} (first){% endif %}
|
{{- loop.index0 }}. {{ s }}{{ 2 * loop.index }}{% if !loop.first %}{% else %} (first){% endif %}{% if loop.last %} (last){% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -46,7 +46,10 @@ fn test_precedence_for() {
|
|||||||
let s = PrecedenceTemplate {
|
let s = PrecedenceTemplate {
|
||||||
strings: vec!["A", "alfa", "1"],
|
strings: vec!["A", "alfa", "1"],
|
||||||
};
|
};
|
||||||
assert_eq!(s.render().unwrap(), "0. A2 (first)\n1. alfa4\n2. 16\n");
|
assert_eq!(
|
||||||
|
s.render().unwrap(),
|
||||||
|
"0. A2 (first)\n1. alfa4\n2. 16 (last)\n"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -59,5 +62,8 @@ struct ForRangeTemplate {
|
|||||||
#[test]
|
#[test]
|
||||||
fn test_for_range() {
|
fn test_for_range() {
|
||||||
let s = ForRangeTemplate { init: -1, end: 1 };
|
let s = ForRangeTemplate { init: -1, end: 1 };
|
||||||
assert_eq!(s.render().unwrap(), "foo\nfoo\nbar\nbar\nfoo\nbar\nbar\n");
|
assert_eq!(
|
||||||
|
s.render().unwrap(),
|
||||||
|
"foo (first)\nfoo (last)\nbar\nbar\nfoo\nbar\nbar\n"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user