# `dyn-iter` [![continuous-integration-badge]][continuous-integration] [![code-coverage-badge]][code-coverage] [![crates.io-badge]][crates.io] [![license-badge]][license] [![documentation-badge]][documentation] [continuous-integration-badge]: https://img.shields.io/gitlab/pipeline-status/woshilapin/dyn-iter.svg?branch=main&color=fca121&logo=gitlab&style=flat [continuous-integration]: https://gitlab.com/woshilapin/dyn-iter/-/pipelines/main/latest [code-coverage-badge]: https://img.shields.io/codecov/c/gitlab/woshilapin/dyn-iter?style=flat&logo=codecov&color=f01f7a [code-coverage]: https://codecov.io/gl/woshilapin/dyn-iter [crates.io-badge]: https://img.shields.io/crates/v/dyn-iter?style=flat&logo=&color=e6ac38 [crates.io]: https://crates.io/crates/dyn-iter [license-badge]: https://img.shields.io/crates/l/dyn-iter?style=flat&logo=spdx&color=4398cc [license]: https://spdx.org/licenses/MIT.html [documentation-badge]: https://img.shields.io/badge/docs.rs-dyn--iter-4d76ae?style=flat&logo=&color=4d76ae [documentation]: https://docs.rs/dyn-iter/latest/dyn_iter/ This tiny crate should help you simplify your code when you need to wrap [`Iterator`] as trait-object. [`iterator`]: https://doc.rust-lang.org/std/iter/trait.Iterator.html Imagine for example a trait like the following. ```rust #[derive(Debug, Clone, Copy, PartialEq, Eq)] enum Color { Red, Green, Blue, White, Black, } trait Colors<'a> { type ColorsIter: Iterator + 'a; fn colors(&'a self) -> Self::ColorsIter; } ``` As an implementor, you have a `struct Flag` that looks like this. ```rust # use std::collections::HashSet; # #[derive(Debug, Clone, Copy, PartialEq, Eq)] # enum Color { # Red, # Green, # Blue, # White, # Black, # } struct Flag { pub primary_colors: HashSet, pub secondary_colors: HashSet, } ``` you might implement `Colors` that look like this ```rust,ignore # use std::collections::HashSet; # use dyn_iter::{DynIter, IntoDynIterator as _}; # #[derive(Debug, Clone, Copy, PartialEq, Eq)] # enum Color { # Red, # Green, # Blue, # White, # Black, # } # struct Flag { # pub primary_colors: HashSet, # pub secondary_colors: HashSet, # } # trait Colors<'a> { # type ColorsIter: Iterator + 'a; # fn colors(&'a self) -> Self::ColorsIter; # } impl<'a> Colors<'a> for Flag { type ColorsIter = ???; fn colors(&'a self) -> Self::ColorsIter { self.primary_colors .iter() .chain(&self.secondary_colors) .filter(|color| **color != Color::Black) .copied() } } ``` With the above implementation, defining the associated type `ColorsIter` might be difficult. `DynIter` should simplify your life because you can just write the following implementation. ```rust # use std::collections::HashSet; # use dyn_iter::{DynIter, IntoDynIterator as _}; # #[derive(Debug, Clone, Copy, PartialEq, Eq)] # enum Color { # Red, # Green, # Blue, # White, # Black, # } # struct Flag { # pub primary_colors: HashSet, # pub secondary_colors: HashSet, # } # trait Colors<'a> { # type ColorsIter: Iterator + 'a; # fn colors(&'a self) -> Self::ColorsIter; # } impl<'a> Colors<'a> for Flag { type ColorsIter = DynIter<'a, Color>; fn colors(&'a self) -> Self::ColorsIter { self.primary_colors .iter() .chain(&self.secondary_colors) .filter(|color| **color != Color::Black) .copied() .into_dyn_iter() } } ``` Behind the scene, `DynIter<'iter, V>` is only providing a wrapper around a `Box + 'iter>`. For more details about why this crate exists, read this [blog post]. [blog post]: https://hole.tuziwo.info/dyn-iterator.html