indexmap/
macros.rs

1/// Create an [`IndexMap`][crate::IndexMap] from a list of key-value pairs
2/// and a `BuildHasherDefault`-wrapped custom hasher.
3///
4/// ## Example
5///
6/// ```
7/// use indexmap::indexmap_with_default;
8/// use fnv::FnvHasher;
9///
10/// let map = indexmap_with_default!{
11///     FnvHasher;
12///     "a" => 1,
13///     "b" => 2,
14/// };
15/// assert_eq!(map["a"], 1);
16/// assert_eq!(map["b"], 2);
17/// assert_eq!(map.get("c"), None);
18///
19/// // "a" is the first key
20/// assert_eq!(map.keys().next(), Some(&"a"));
21/// ```
22#[macro_export]
23macro_rules! indexmap_with_default {
24    ($H:ty; $($key:expr => $value:expr,)+) => { $crate::indexmap_with_default!($H; $($key => $value),+) };
25    ($H:ty; $($key:expr => $value:expr),*) => {{
26        let builder = ::core::hash::BuildHasherDefault::<$H>::default();
27        const CAP: usize = <[()]>::len(&[$({ stringify!($key); }),*]);
28        #[allow(unused_mut)]
29        // Specify your custom `H` (must implement Default + Hasher) as the hasher:
30        let mut map = $crate::IndexMap::with_capacity_and_hasher(CAP, builder);
31        $(
32            map.insert($key, $value);
33        )*
34        map
35    }};
36}
37
38#[cfg(feature = "std")]
39#[cfg_attr(docsrs, doc(cfg(feature = "std")))]
40#[macro_export]
41/// Create an [`IndexMap`][crate::IndexMap] from a list of key-value pairs
42///
43/// ## Example
44///
45/// ```
46/// use indexmap::indexmap;
47///
48/// let map = indexmap!{
49///     "a" => 1,
50///     "b" => 2,
51/// };
52/// assert_eq!(map["a"], 1);
53/// assert_eq!(map["b"], 2);
54/// assert_eq!(map.get("c"), None);
55///
56/// // "a" is the first key
57/// assert_eq!(map.keys().next(), Some(&"a"));
58/// ```
59macro_rules! indexmap {
60    ($($key:expr => $value:expr,)+) => { $crate::indexmap!($($key => $value),+) };
61    ($($key:expr => $value:expr),*) => {
62        {
63            // Note: `stringify!($key)` is just here to consume the repetition,
64            // but we throw away that string literal during constant evaluation.
65            const CAP: usize = <[()]>::len(&[$({ stringify!($key); }),*]);
66            let mut map = $crate::IndexMap::with_capacity(CAP);
67            $(
68                map.insert($key, $value);
69            )*
70            map
71        }
72    };
73}
74
75/// Create an [`IndexSet`][crate::IndexSet] from a list of values
76/// and a `BuildHasherDefault`-wrapped custom hasher.
77///
78/// ## Example
79///
80/// ```
81/// use indexmap::indexset_with_default;
82/// use fnv::FnvHasher;
83///
84/// let set = indexset_with_default!{
85///     FnvHasher;
86///     "a",
87///     "b",
88/// };
89/// assert!(set.contains("a"));
90/// assert!(set.contains("b"));
91/// assert!(!set.contains("c"));
92///
93/// // "a" is the first value
94/// assert_eq!(set.iter().next(), Some(&"a"));
95/// ```
96#[macro_export]
97macro_rules! indexset_with_default {
98    ($H:ty; $($value:expr,)+) => { $crate::indexset_with_default!($H; $($value),+) };
99    ($H:ty; $($value:expr),*) => {{
100        let builder = ::core::hash::BuildHasherDefault::<$H>::default();
101        const CAP: usize = <[()]>::len(&[$({ stringify!($value); }),*]);
102        #[allow(unused_mut)]
103        // Specify your custom `H` (must implement Default + Hash) as the hasher:
104        let mut set = $crate::IndexSet::with_capacity_and_hasher(CAP, builder);
105        $(
106            set.insert($value);
107        )*
108        set
109    }};
110}
111
112#[cfg(feature = "std")]
113#[cfg_attr(docsrs, doc(cfg(feature = "std")))]
114#[macro_export]
115/// Create an [`IndexSet`][crate::IndexSet] from a list of values
116///
117/// ## Example
118///
119/// ```
120/// use indexmap::indexset;
121///
122/// let set = indexset!{
123///     "a",
124///     "b",
125/// };
126/// assert!(set.contains("a"));
127/// assert!(set.contains("b"));
128/// assert!(!set.contains("c"));
129///
130/// // "a" is the first value
131/// assert_eq!(set.iter().next(), Some(&"a"));
132/// ```
133macro_rules! indexset {
134    ($($value:expr,)+) => { $crate::indexset!($($value),+) };
135    ($($value:expr),*) => {
136        {
137            // Note: `stringify!($value)` is just here to consume the repetition,
138            // but we throw away that string literal during constant evaluation.
139            const CAP: usize = <[()]>::len(&[$({ stringify!($value); }),*]);
140            let mut set = $crate::IndexSet::with_capacity(CAP);
141            $(
142                set.insert($value);
143            )*
144            set
145        }
146    };
147}
148
149// generate all the Iterator methods by just forwarding to the underlying
150// self.iter and mapping its element.
151macro_rules! iterator_methods {
152    // $map_elt is the mapping function from the underlying iterator's element
153    // same mapping function for both options and iterators
154    ($map_elt:expr) => {
155        fn next(&mut self) -> Option<Self::Item> {
156            self.iter.next().map($map_elt)
157        }
158
159        fn size_hint(&self) -> (usize, Option<usize>) {
160            self.iter.size_hint()
161        }
162
163        fn count(self) -> usize {
164            self.iter.len()
165        }
166
167        fn nth(&mut self, n: usize) -> Option<Self::Item> {
168            self.iter.nth(n).map($map_elt)
169        }
170
171        fn last(mut self) -> Option<Self::Item> {
172            self.next_back()
173        }
174
175        fn collect<C>(self) -> C
176        where
177            C: FromIterator<Self::Item>,
178        {
179            // NB: forwarding this directly to standard iterators will
180            // allow it to leverage unstable traits like `TrustedLen`.
181            self.iter.map($map_elt).collect()
182        }
183    };
184}
185
186macro_rules! double_ended_iterator_methods {
187    // $map_elt is the mapping function from the underlying iterator's element
188    // same mapping function for both options and iterators
189    ($map_elt:expr) => {
190        fn next_back(&mut self) -> Option<Self::Item> {
191            self.iter.next_back().map($map_elt)
192        }
193
194        fn nth_back(&mut self, n: usize) -> Option<Self::Item> {
195            self.iter.nth_back(n).map($map_elt)
196        }
197    };
198}
199
200// generate `ParallelIterator` methods by just forwarding to the underlying
201// self.entries and mapping its elements.
202#[cfg(feature = "rayon")]
203macro_rules! parallel_iterator_methods {
204    // $map_elt is the mapping function from the underlying iterator's element
205    ($map_elt:expr) => {
206        fn drive_unindexed<C>(self, consumer: C) -> C::Result
207        where
208            C: UnindexedConsumer<Self::Item>,
209        {
210            self.entries
211                .into_par_iter()
212                .map($map_elt)
213                .drive_unindexed(consumer)
214        }
215
216        // NB: This allows indexed collection, e.g. directly into a `Vec`, but the
217        // underlying iterator must really be indexed.  We should remove this if we
218        // start having tombstones that must be filtered out.
219        fn opt_len(&self) -> Option<usize> {
220            Some(self.entries.len())
221        }
222    };
223}
224
225// generate `IndexedParallelIterator` methods by just forwarding to the underlying
226// self.entries and mapping its elements.
227#[cfg(feature = "rayon")]
228macro_rules! indexed_parallel_iterator_methods {
229    // $map_elt is the mapping function from the underlying iterator's element
230    ($map_elt:expr) => {
231        fn drive<C>(self, consumer: C) -> C::Result
232        where
233            C: Consumer<Self::Item>,
234        {
235            self.entries.into_par_iter().map($map_elt).drive(consumer)
236        }
237
238        fn len(&self) -> usize {
239            self.entries.len()
240        }
241
242        fn with_producer<CB>(self, callback: CB) -> CB::Output
243        where
244            CB: ProducerCallback<Self::Item>,
245        {
246            self.entries
247                .into_par_iter()
248                .map($map_elt)
249                .with_producer(callback)
250        }
251    };
252}