use crate::IdxSize;
#[derive(Clone, Debug)]
#[must_use = "iterators are lazy and do nothing unless consumed"]
pub struct EnumerateIdx<I> {
iter: I,
count: IdxSize,
}
impl<I> Iterator for EnumerateIdx<I>
where
I: Iterator,
{
type Item = (IdxSize, <I as Iterator>::Item);
#[inline]
fn next(&mut self) -> Option<(IdxSize, <I as Iterator>::Item)> {
let a = self.iter.next()?;
let i = self.count;
self.count += 1;
Some((i, a))
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
self.iter.size_hint()
}
#[inline]
fn nth(&mut self, n: usize) -> Option<(IdxSize, I::Item)> {
let a = self.iter.nth(n)?;
let i = self.count + (n as IdxSize);
self.count = i + 1;
Some((i, a))
}
#[inline]
fn count(self) -> usize {
self.iter.count()
}
}
impl<I> DoubleEndedIterator for EnumerateIdx<I>
where
I: ExactSizeIterator + DoubleEndedIterator,
{
#[inline]
fn next_back(&mut self) -> Option<(IdxSize, <I as Iterator>::Item)> {
let a = self.iter.next_back()?;
let len = self.iter.len();
Some((self.count + len as IdxSize, a))
}
#[inline]
fn nth_back(&mut self, n: usize) -> Option<(IdxSize, <I as Iterator>::Item)> {
let a = self.iter.nth_back(n)?;
let len = self.iter.len();
Some((self.count + len as IdxSize, a))
}
}
impl<I> ExactSizeIterator for EnumerateIdx<I>
where
I: ExactSizeIterator,
{
fn len(&self) -> usize {
self.iter.len()
}
}
pub trait EnumerateIdxTrait: Iterator {
fn enumerate_idx(self) -> EnumerateIdx<Self>
where
Self: Sized,
{
EnumerateIdx {
iter: self,
count: 0,
}
}
}
impl<T: ?Sized> EnumerateIdxTrait for T where T: Iterator {}