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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
use iter::SortedIterator;
use std::collections::BTreeMap;
use std::iter::Peekable;
pub struct TransitiveIterator<I, J, F>
where
I: SortedIterator,
I::Item: Ord + Clone,
J: SortedIterator<Item = I::Item>,
F: Fn(&J::Item) -> J,
{
iter: Peekable<I>,
iters: BTreeMap<I::Item, Peekable<J>>,
f: F,
}
impl<I, J, F> TransitiveIterator<I, J, F>
where
I: SortedIterator,
I::Item: Ord + Clone,
J: SortedIterator<Item = I::Item>,
J::Item: Ord + Clone,
F: Fn(&J::Item) -> J,
{
pub fn new(iter: I, f: F) -> TransitiveIterator<I, J, F> {
TransitiveIterator {
iter: iter.peekable(),
iters: BTreeMap::new(),
f,
}
}
fn min_next(&mut self) -> Option<usize> {
let mut min = self.iter.peek();
let mut min_pos = if min.is_some() { Some(0) } else { None };
for (pos, item) in self.iters.iter_mut().enumerate() {
match (item.1.peek(), min) {
(Some(m), Some(n)) if m < n => {
min_pos = Some(pos);
min = Some(m)
}
(Some(m), None) => {
min_pos = Some(pos);
min = Some(m)
}
_ => {}
}
}
min_pos
}
fn take_next(&mut self, pos: usize) -> Option<I::Item> {
let mut iters = self.iters.iter_mut();
let r = if pos == 0 {
self.iter.next()
} else {
iters.nth(pos - 1).unwrap().1.next()
};
for i in iters {
if i.1.peek() == r.as_ref() {
i.1.next();
}
}
r
}
fn add_iterator(&mut self, item: &I::Item) {
let present_and_empty = if let Some(i) = self.iters.get_mut(item) {
if i.peek().is_some() {
return;
}
true
} else {
false
};
if present_and_empty {
self.iters.remove(item);
return;
}
let mut new_iterator = (self.f)(item).peekable();
loop {
match new_iterator.peek() {
Some(i) if i > item => break,
None => return,
_ => {}
}
new_iterator.next();
}
self.iters.insert(item.clone(), new_iterator);
}
}
impl<I, J, F> Iterator for TransitiveIterator<I, J, F>
where
I: SortedIterator,
I::Item: Ord + Clone,
J: SortedIterator<Item = I::Item>,
J::Item: Ord + Clone,
F: Fn(&J::Item) -> J,
{
type Item = I::Item;
fn next(&mut self) -> Option<Self::Item> {
if let Some(next) = self.min_next() {
let next = self.take_next(next);
self.add_iterator(next.as_ref().unwrap());
next
} else {
None
}
}
}
impl<I, J, F> SortedIterator for TransitiveIterator<I, J, F>
where
I: SortedIterator,
I::Item: Ord + Clone,
J: SortedIterator<Item = I::Item>,
J::Item: Ord + Clone,
F: Fn(&J::Item) -> J,
{
}