core/iter/adapters/
map_while.rs
1use crate::fmt;
2use crate::iter::InPlaceIterable;
3use crate::iter::adapters::SourceIter;
4use crate::num::NonZero;
5use crate::ops::{ControlFlow, Try};
6
7#[must_use = "iterators are lazy and do nothing unless consumed"]
15#[stable(feature = "iter_map_while", since = "1.57.0")]
16#[derive(Clone)]
17pub struct MapWhile<I, P> {
18 iter: I,
19 predicate: P,
20}
21
22impl<I, P> MapWhile<I, P> {
23 pub(in crate::iter) fn new(iter: I, predicate: P) -> MapWhile<I, P> {
24 MapWhile { iter, predicate }
25 }
26}
27
28#[stable(feature = "iter_map_while", since = "1.57.0")]
29impl<I: fmt::Debug, P> fmt::Debug for MapWhile<I, P> {
30 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
31 f.debug_struct("MapWhile").field("iter", &self.iter).finish()
32 }
33}
34
35#[stable(feature = "iter_map_while", since = "1.57.0")]
36impl<B, I: Iterator, P> Iterator for MapWhile<I, P>
37where
38 P: FnMut(I::Item) -> Option<B>,
39{
40 type Item = B;
41
42 #[inline]
43 fn next(&mut self) -> Option<B> {
44 let x = self.iter.next()?;
45 (self.predicate)(x)
46 }
47
48 #[inline]
49 fn size_hint(&self) -> (usize, Option<usize>) {
50 let (_, upper) = self.iter.size_hint();
51 (0, upper) }
53
54 #[inline]
55 fn try_fold<Acc, Fold, R>(&mut self, init: Acc, mut fold: Fold) -> R
56 where
57 Self: Sized,
58 Fold: FnMut(Acc, Self::Item) -> R,
59 R: Try<Output = Acc>,
60 {
61 let Self { iter, predicate } = self;
62 iter.try_fold(init, |acc, x| match predicate(x) {
63 Some(item) => ControlFlow::from_try(fold(acc, item)),
64 None => ControlFlow::Break(try { acc }),
65 })
66 .into_try()
67 }
68
69 impl_fold_via_try_fold! { fold -> try_fold }
70}
71
72#[unstable(issue = "none", feature = "inplace_iteration")]
73unsafe impl<I, P> SourceIter for MapWhile<I, P>
74where
75 I: SourceIter,
76{
77 type Source = I::Source;
78
79 #[inline]
80 unsafe fn as_inner(&mut self) -> &mut I::Source {
81 unsafe { SourceIter::as_inner(&mut self.iter) }
83 }
84}
85
86#[unstable(issue = "none", feature = "inplace_iteration")]
87unsafe impl<I: InPlaceIterable, P> InPlaceIterable for MapWhile<I, P> {
88 const EXPAND_BY: Option<NonZero<usize>> = I::EXPAND_BY;
89 const MERGE_BY: Option<NonZero<usize>> = I::MERGE_BY;
90}