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
use alloc::{vec, vec::Vec};
#[cfg(test)]
mod tests;
#[derive(Debug, Eq, PartialEq)]
#[must_use]
pub struct ClassifyIT<'a, TT: 'a, TC, FnT, IT: ?Sized> {
inner: &'a mut IT,
fnx: FnT,
edge: (Option<TC>, Option<TT>),
}
impl<'a, TT: 'a, TC, FnT, IT> ClassifyIT<'a, TT, TC, FnT, IT>
where
TC: Default,
FnT: FnMut(&TT) -> TC,
IT: Iterator<Item = TT>,
{
#[inline]
pub fn new(inner: &'a mut IT, fnx: FnT) -> Self {
Self {
inner,
fnx,
edge: (Some(<_>::default()), None),
}
}
}
impl<TT, TC, FnT, IT> Iterator for ClassifyIT<'_, TT, TC, FnT, IT>
where
TC: PartialEq,
FnT: FnMut(&TT) -> TC,
IT: Iterator<Item = TT>,
{
type Item = (TC, Vec<TT>);
fn next(&mut self) -> Option<Self::Item> {
let mut ccl = self.edge.0.take()?;
let mut last = self.edge.1.take().map_or_else(Vec::new, |x| vec![x]);
for x in &mut self.inner {
let old_ccl = core::mem::replace(&mut ccl, (self.fnx)(&x));
if ccl == old_ccl || last.is_empty() {
last.push(x);
} else {
self.edge = (Some(ccl), Some(x));
return Some((old_ccl, last));
}
}
if last.is_empty() {
None
} else {
Some((ccl, last))
}
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<TT, TC, FnT, IT> core::iter::FusedIterator for ClassifyIT<'_, TT, TC, FnT, IT>
where
TC: PartialEq,
FnT: FnMut(&TT) -> TC,
IT: Iterator<Item = TT>,
{
}
pub trait Classify<'a, TT: 'a>: Iterator<Item = TT> + 'a {
fn classify<TC, FnT>(&'a mut self, fnx: FnT) -> ClassifyIT<'a, TT, TC, FnT, Self>
where
TC: Default + PartialEq,
FnT: FnMut(&TT) -> TC;
}
impl<'a, IT, TT: 'a> Classify<'a, TT> for IT
where
IT: Iterator<Item = TT> + 'a,
{
#[inline]
fn classify<TC, FnT>(&'a mut self, fnx: FnT) -> ClassifyIT<'a, TT, TC, FnT, Self>
where
TC: Default + PartialEq,
FnT: FnMut(&TT) -> TC,
{
ClassifyIT::new(self, fnx)
}
}
#[inline]
pub fn classify<Input, TT, TC, TRes>(input: Input, fnx: impl FnMut(&TT) -> TC) -> TRes
where
Input: IntoIterator<Item = TT>,
TC: Default + PartialEq,
TRes: core::iter::FromIterator<(TC, Vec<TT>)>,
{
input.into_iter().classify(fnx).collect()
}
#[inline]
pub fn classify_as_vec<Input, TT, TC>(
input: Input,
fnx: impl FnMut(&TT) -> TC,
) -> Vec<(TC, Vec<TT>)>
where
Input: IntoIterator<Item = TT>,
TC: Default + PartialEq,
{
classify(input, fnx)
}