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
129
130
131
132
133
134
135
136
137
138
139
140
//! Implemented iterators for tuples


use core::iter::{empty, once, Empty, FusedIterator, Once};
use core::mem::MaybeUninit;
use core::ops::Range;

pub trait TupleIter<'a> {
    type Iter: Iterator + 'a;

    fn iter(&'a self) -> Self::Iter;
}

pub trait TupleIntoIter {
    type Iter: Iterator;

    fn into_iter(self) -> Self::Iter;
}

pub trait TupleFromIter<T> {
    /// Like `Iter<T> -> (T, T, T)` with panic on failure

    fn from_iter<I: IntoIterator<Item = T>>(iter: I) -> Self;
}

pub trait TupleTryFromIter<T>: Sized {
    /// Like `Iter<T> -> Option<(T, T, T)>`

    fn try_from_iter<I: IntoIterator<Item = T>>(iter: I) -> Option<Self>;
}

pub trait TupleFromIterTry<T> {
    type OutTuple;

    /// Like `Iter<T> -> (Option<T>, Option<T>, Option<T>)`

    fn from_iter_try<I: IntoIterator<Item = T>>(iter: I) -> Self::OutTuple;
}

impl<'a> TupleIter<'a> for () {
    type Iter = Empty<()>;

    #[inline]
    fn iter(&'a self) -> Self::Iter {
        empty()
    }
}

impl TupleIntoIter for () {
    type Iter = Empty<()>;

    #[inline]
    fn into_iter(self) -> Self::Iter {
        empty()
    }
}

impl<T> TupleFromIter<T> for () {
    fn from_iter<I: IntoIterator<Item = T>>(_: I) -> Self {
        ()
    }
}

impl<T> TupleTryFromIter<T> for () {
    fn try_from_iter<I: IntoIterator<Item = T>>(_: I) -> Option<Self> {
        Some(())
    }
}

impl<T> TupleFromIterTry<T> for () {
    type OutTuple = ();

    fn from_iter_try<I: IntoIterator<Item = T>>(_: I) -> Self {
        ()
    }
}

impl<'a, T: 'a> TupleIter<'a> for (T,) {
    type Iter = Once<&'a T>;

    #[inline]
    fn iter(&'a self) -> Self::Iter {
        once(&self.0)
    }
}

impl<T> TupleIntoIter for (T,) {
    type Iter = Once<T>;

    #[inline]
    fn into_iter(self) -> Self::Iter {
        once(self.0)
    }
}

impl<T> TupleFromIter<T> for (T,) {
    fn from_iter<I: IntoIterator<Item = T>>(iter: I) -> Self {
        let mut iter = iter.into_iter();
        (iter.next().unwrap(),)
    }
}

impl<T> TupleTryFromIter<T> for (T,) {
    fn try_from_iter<I: IntoIterator<Item = T>>(iter: I) -> Option<Self> {
        let mut iter = iter.into_iter();
        Some((iter.next()?,))
    }
}

impl<T> TupleFromIterTry<T> for (T,) {
    type OutTuple = (Option<T>,);
    fn from_iter_try<I: IntoIterator<Item = T>>(iter: I) -> Self::OutTuple {
        let mut iter = iter.into_iter();
        (iter.next(),)
    }
}

include!("./gen/tuple_iter.rs");

pub trait TupleCollect<T> {
    /// Like `Iter<T> -> (T, T, T)` with panic on failure

    fn collect_tuple<B: TupleFromIter<T>>(self) -> B;
    /// Like `Iter<T> -> Option<(T, T, T)>`

    fn try_collect_tuple<B: TupleTryFromIter<T>>(self) -> Option<B>;
    /// Like `Iter<T> -> (Option<T>, Option<T>, Option<T>)`

    fn collect_tuple_try<B: TupleFromIterTry<T>>(self) -> B::OutTuple;
}

impl<I: IntoIterator<Item = T>, T> TupleCollect<T> for I {
    fn collect_tuple<B: TupleFromIter<T>>(self) -> B {
        let iter = self.into_iter();
        B::from_iter(iter)
    }

    fn try_collect_tuple<B: TupleTryFromIter<T>>(self) -> Option<B> {
        let iter = self.into_iter();
        B::try_from_iter(iter)
    }

    fn collect_tuple_try<B: TupleFromIterTry<T>>(self) -> B::OutTuple {
        let iter = self.into_iter();
        B::from_iter_try(iter)
    }
}