1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
pub struct EnumerateSplit<Iter, T>
where Iter: Iterator<Item = T>, T: PartialEq {
iter: Iter,
split_value: T,
block_number: usize,
block_position: usize
}
impl<Iter, T> EnumerateSplit<Iter, T>
where Iter: Iterator<Item = T>, T: PartialEq {
#[inline]
fn nudge_block(&mut self) {
self.block_number += 1;
self.block_position = 0;
}
#[inline]
fn nudge_position(&mut self) {
self.block_position += 1;
}
#[inline]
fn get_position(&self) -> (usize, usize) {
(self.block_number, self.block_position)
}
}
pub fn enumerate_split<Iter, T>(iter: Iter, split_value: T) -> EnumerateSplit<Iter, T>
where Iter: Iterator<Item = T>, T: PartialEq {
EnumerateSplit {
iter: iter,
split_value: split_value,
block_number: 0,
block_position: 0
}
}
impl<Iter, T> Iterator for EnumerateSplit<Iter, T>
where Iter: Iterator<Item = T>, T: PartialEq {
type Item = (T, (usize, usize));
fn next(&mut self) -> Option<(T, (usize, usize))> {
match self.iter.next() {
Some(t) => {
let split = t == self.split_value;
let res = Some((t, self.get_position()));
if split {
self.nudge_block();
} else {
self.nudge_position();
}
res
}
None => None
}
}
}
#[cfg(test)]
mod tests {
use std::io::prelude::*;
#[test]
fn enumerate_split() {
let mut input = super::enumerate_split("Some \n\nstring with a newline".chars(), '\n');
assert_eq!(input.next(), Some(('S', (0, 0))));
assert_eq!(input.next(), Some(('o', (0, 1))));
assert_eq!(input.next(), Some(('m', (0, 2))));
assert_eq!(input.next(), Some(('e', (0, 3))));
assert_eq!(input.next(), Some((' ', (0, 4))));
assert_eq!(input.next(), Some(('\n', (0, 5))));
assert_eq!(input.next(), Some(('\n', (1, 0))));
assert_eq!(input.next(), Some(('s', (2, 0))));
}
}