Reaktoro  v2.13.0
A unified framework for modeling chemically reactive systems
Enumerate.hpp
1 // Reaktoro is a unified framework for modeling chemically reactive systems.
2 //
3 // Copyright © 2014-2024 Allan Leal
4 //
5 // This library is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU Lesser General Public
7 // License as published by the Free Software Foundation; either
8 // version 2.1 of the License, or (at your option) any later version.
9 //
10 // This library is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 // Lesser General Public License for more details.
14 //
15 // You should have received a copy of the GNU Lesser General Public License
16 // along with this library. If not, see <http://www.gnu.org/licenses/>.
17 
18 //===================================================================
19 // REFERENCE:
20 //-------------------------------------------------------------------
21 // https://www.reedbeta.com/blog/python-like-enumerate-in-cpp17/
22 //===================================================================
23 
24 #pragma once
25 
26 // C++ includes
27 #include <tuple>
28 #include <iterator>
29 
30 template <typename T,
31  typename TIter = decltype(std::begin(std::declval<T>())),
32  typename = decltype(std::end(std::declval<T>()))>
33 constexpr auto enumerate(T && iterable)
34 {
35  struct iterator
36  {
37  size_t i;
38  TIter iter;
39  bool operator != (const iterator& other) const { return iter != other.iter; }
40  void operator ++ () { ++i; ++iter; }
41  auto operator * () const { return std::tie(i, *iter); }
42  };
43 
44  struct iterable_wrapper
45  {
46  T iterable;
47  auto begin() { return iterator{ 0, std::begin(iterable) }; }
48  auto end() { return iterator{ 0, std::end(iterable) }; }
49  };
50  return iterable_wrapper{ std::forward<T>(iterable) };
51 }