diff options
author | 2019-01-05 20:18:59 +0900 | |
---|---|---|
committer | 2019-01-08 15:59:45 +0100 | |
commit | df07f7f5e3345ced96decb0cb4c4f1bdfcdb1b12 (patch) | |
tree | e6619644eec17048143e1bca34ee37dc0701dae8 /askama | |
parent | c5c37f56a1232ff924a6bfda6c1c05e64a6241b1 (diff) | |
download | askama-df07f7f5e3345ced96decb0cb4c4f1bdfcdb1b12.tar.gz askama-df07f7f5e3345ced96decb0cb4c4f1bdfcdb1b12.tar.bz2 askama-df07f7f5e3345ced96decb0cb4c4f1bdfcdb1b12.zip |
Add support for loop.last
Diffstat (limited to '')
-rw-r--r-- | askama/src/lib.rs | 1 | ||||
-rw-r--r-- | askama_derive/src/generator.rs | 15 | ||||
-rw-r--r-- | askama_shared/src/helpers/mod.rs | 48 | ||||
-rw-r--r-- | askama_shared/src/lib.rs | 1 |
4 files changed, 62 insertions, 3 deletions
diff --git a/askama/src/lib.rs b/askama/src/lib.rs index d1bb639..725c3de 100644 --- a/askama/src/lib.rs +++ b/askama/src/lib.rs @@ -433,6 +433,7 @@ pub trait Template { } pub use crate::shared::filters; +pub use crate::shared::helpers; pub use crate::shared::{read_config_file, Error, MarkupDisplay, Result}; pub use askama_derive::*; diff --git a/askama_derive/src/generator.rs b/askama_derive/src/generator.rs index 67ec2c2..6c02a26 100644 --- a/askama_derive/src/generator.rs +++ b/askama_derive/src/generator.rs @@ -467,11 +467,17 @@ impl<'a> Generator<'a> { let expr_code = self.visit_expr_root(iter); self.write_buf_writable(buf); - buf.write("for (_loop_index, "); + buf.write("for (_loop_index, _loop_last, "); self.visit_target(buf, var); match iter { - Expr::Range(_, _, _) => buf.writeln(&format!(") in ({}).enumerate() {{", expr_code)), - _ => buf.writeln(&format!(") in (&{}).into_iter().enumerate() {{", expr_code)), + Expr::Range(_, _, _) => buf.writeln(&format!( + ") in ::askama::helpers::enumerate({}) {{", + expr_code + )), + _ => buf.writeln(&format!( + ") in ::askama::helpers::enumerate((&{}).into_iter()) {{", + expr_code + )), }; self.handle(ctx, body, buf, AstLevel::Nested); @@ -915,6 +921,9 @@ impl<'a> Generator<'a> { } else if attr == "first" { buf.write("(_loop_index == 0)"); return DisplayWrap::Unwrapped; + } else if attr == "last" { + buf.write("_loop_last"); + return DisplayWrap::Unwrapped; } else { panic!("unknown loop variable"); } diff --git a/askama_shared/src/helpers/mod.rs b/askama_shared/src/helpers/mod.rs new file mode 100644 index 0000000..1a3fd7b --- /dev/null +++ b/askama_shared/src/helpers/mod.rs @@ -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, + } +} diff --git a/askama_shared/src/lib.rs b/askama_shared/src/lib.rs index 850cb8b..a0c9321 100644 --- a/askama_shared/src/lib.rs +++ b/askama_shared/src/lib.rs @@ -16,6 +16,7 @@ pub use askama_escape::MarkupDisplay; use std::collections::BTreeMap; pub mod filters; +pub mod helpers; #[derive(Debug)] pub struct Config<'a> { |