qxLib
base_iterator.h
Go to the documentation of this file.
1 /**
2 
3  @file base_iterator.h
4  @author Khrapov
5  @date 2.12.2023
6  @copyright © Nick Khrapov, 2023. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/meta/concepts.h>
12 
13 #include <iterator>
14 #include <ranges>
15 
16 namespace qx
17 {
18 
19 /**
20 
21  @class base_iterator
22  @brief Base contiguous iterator with a container pointer and an index
23  @details ~
24  @tparam container_t - container satisfying std::ranges::contiguous_range concept
25  @author Khrapov
26  @date 2.12.2023
27 
28 **/
29 template<class container_t>
30 /* requires std::ranges::contiguous_range<container_t> */
32 {
33 public:
34  template<class container_t_, class derived_t_>
35  friend class base_forward_iterator;
36 
37  template<class container_t_, class derived_t_>
38  friend class base_reverse_iterator;
39 
40  using difference_type = typename container_t::difference_type;
41  using size_type = typename container_t::size_type;
42  using value_type = typename container_t::value_type;
43  using pointer = typename container_t::pointer;
44  using const_pointer = typename container_t::const_pointer;
45  using reference = typename container_t::reference;
46  using const_reference = typename container_t::const_reference;
47 
48  using iterator_category = std::contiguous_iterator_tag;
49  using iterator_concept = std::contiguous_iterator_tag;
50 
51 public:
52  constexpr base_iterator() noexcept = default;
53  constexpr base_iterator(container_t* pContainer, size_type nIndex) noexcept
54  : m_nIndex(nIndex)
55  , m_pContainer(pContainer)
56  {
57  }
58  constexpr operator void*() noexcept
59  {
60  return &(*m_pContainer).at(m_nIndex);
61  }
62 
63 protected:
64  void shift(difference_type nShift) noexcept
65  {
66  m_nIndex += nShift;
67  }
68  auto&& get(difference_type nShift = 0) const noexcept
69  {
70  return (*m_pContainer).at(m_nIndex + nShift);
71  }
72  size_type get_index() const noexcept
73  {
74  return m_nIndex;
75  }
76 
77 private:
78  size_type m_nIndex = 0u;
79  container_t* m_pContainer = nullptr;
80 };
81 
82 /**
83 
84  @class base_forward_iterator
85  @brief Basic contiguous forward iterator, meaning that incrementing an iterator will lead it moving forward
86  @details ~
87  @tparam container_t - container satisfying std::ranges::contiguous_range concept
88  @tparam derived_t - derived iterator type
89  @author Khrapov
90  @date 2.12.2023
91 
92 **/
93 template<class container_t, class derived_t>
94 class base_forward_iterator : public base_iterator<container_t>
95 {
97  using decayed_derived_type = std::decay_t<derived_t>;
98 
99 public:
100  using super_type::super_type;
101 
102  template<class iterator_t>
103  requires derived_from_template_c<iterator_t, base_iterator>
104  constexpr typename super_type::difference_type operator-(const iterator_t& r) const noexcept
105  {
106  return static_cast<typename super_type::difference_type>(super_type::get_index() - r.get_index());
107  }
108  constexpr bool operator<(const base_forward_iterator& r) const noexcept
109  {
110  return super_type::get_index() < r.get_index();
111  }
112  constexpr bool operator<=(const base_forward_iterator& r) const noexcept
113  {
114  return super_type::get_index() <= r.get_index();
115  }
116  constexpr bool operator>(const base_forward_iterator& r) const noexcept
117  {
118  return super_type::get_index() > r.get_index();
119  }
120  constexpr bool operator>=(const base_forward_iterator& r) const noexcept
121  {
122  return super_type::get_index() >= r.get_index();
123  }
124  constexpr bool operator!=(const base_forward_iterator& r) const noexcept
125  {
126  return super_type::get_index() != r.get_index();
127  }
128  constexpr bool operator==(const base_forward_iterator& r) const noexcept
129  {
130  return super_type::get_index() == r.get_index();
131  }
132  constexpr decayed_derived_type& operator++() noexcept
133  {
134  super_type::shift(1);
135  return static_cast<decayed_derived_type&>(*this);
136  }
137  constexpr decayed_derived_type& operator--() noexcept
138  {
139  super_type::shift(-1);
140  return static_cast<decayed_derived_type&>(*this);
141  }
142  [[nodiscard]] constexpr decayed_derived_type operator++(int) noexcept
143  {
144  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
145  super_type::shift(1);
146  return r;
147  }
148  [[nodiscard]] constexpr decayed_derived_type operator--(int) noexcept
149  {
150  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
151  super_type::shift(-1);
152  return r;
153  }
154  constexpr decayed_derived_type& operator+=(typename super_type::difference_type n) noexcept
155  {
156  super_type::shift(n);
157  return static_cast<decayed_derived_type&>(*this);
158  }
159  constexpr decayed_derived_type& operator-=(typename super_type::difference_type n) noexcept
160  {
161  super_type::shift(-n);
162  return static_cast<decayed_derived_type&>(*this);
163  }
164  constexpr decayed_derived_type operator+(typename super_type::difference_type n) const noexcept
165  {
166  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
167  return r += n;
168  }
169  constexpr decayed_derived_type operator-(typename super_type::difference_type n) const noexcept
170  {
171  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
172  return r -= n;
173  }
174 };
175 
176 template<class iterator_t>
177  requires derived_from_template_c<iterator_t, base_forward_iterator>
178 constexpr iterator_t operator+(typename iterator_t::size_type n, iterator_t it) noexcept
179 {
180  return it += n;
181 }
182 
183 /**
184 
185  @class base_reverse_iterator
186  @brief Basic contiguous reverse iterator, meaning that incrementing an iterator will lead it moving backward
187  @details ~
188  @tparam container_t - container satisfying std::ranges::contiguous_range concept
189  @tparam derived_t - derived iterator type
190  @author Khrapov
191  @date 2.12.2023
192 
193 **/
194 template<class container_t, class derived_t>
195 class base_reverse_iterator : public base_iterator<container_t>
196 {
198  using decayed_derived_type = std::decay_t<derived_t>;
199 
200 public:
201  using super_type::super_type;
202 
203  template<class iterator_t>
204  requires derived_from_template_c<iterator_t, base_iterator>
205  constexpr typename super_type::difference_type operator-(const iterator_t& r) const noexcept
206  {
207  return static_cast<typename super_type::difference_type>(r.get_index() - super_type::get_index());
208  }
209  // + 1 is necessary, as `m_nIndex = -1 (aka size_type::max)`
210  // may be used as end (past-the-last) element
211  constexpr bool operator<(const base_reverse_iterator& r) const noexcept
212  {
213  return super_type::get_index() + 1 > r.get_index() + 1;
214  }
215  constexpr bool operator<=(const base_reverse_iterator& r) const noexcept
216  {
217  return super_type::get_index() + 1 >= r.get_index() + 1;
218  }
219  constexpr bool operator>(const base_reverse_iterator& r) const noexcept
220  {
221  return super_type::get_index() + 1 < r.get_index() + 1;
222  }
223  constexpr bool operator>=(const base_reverse_iterator& r) const noexcept
224  {
225  return super_type::get_index() + 1 <= r.get_index() + 1;
226  }
227  constexpr bool operator!=(const base_reverse_iterator& r) const noexcept
228  {
229  return super_type::get_index() + 1 != r.get_index() + 1;
230  }
231  constexpr bool operator==(const base_reverse_iterator& r) const noexcept
232  {
233  return super_type::get_index() + 1 == r.get_index() + 1;
234  }
235  constexpr decayed_derived_type& operator++() noexcept
236  {
237  super_type::shift(-1);
238  return static_cast<decayed_derived_type&>(*this);
239  }
240  constexpr decayed_derived_type& operator--() noexcept
241  {
242  super_type::shift(1);
243  return static_cast<decayed_derived_type&>(*this);
244  }
245  [[nodiscard]] constexpr decayed_derived_type operator++(int) noexcept
246  {
247  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
248  super_type::shift(-1);
249  return r;
250  }
251  [[nodiscard]] constexpr decayed_derived_type operator--(int) noexcept
252  {
253  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
254  super_type::shift(1);
255  return r;
256  }
257  constexpr decayed_derived_type& operator+=(typename super_type::difference_type n) noexcept
258  {
259  super_type::shift(-n);
260  return static_cast<decayed_derived_type&>(*this);
261  }
262  constexpr decayed_derived_type& operator-=(typename super_type::difference_type n) noexcept
263  {
264  super_type::shift(n);
265  return static_cast<decayed_derived_type&>(*this);
266  }
267  constexpr decayed_derived_type operator+(typename super_type::difference_type n) const noexcept
268  {
269  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
270  return r += n;
271  }
272  constexpr decayed_derived_type operator-(typename super_type::difference_type n) const noexcept
273  {
274  decayed_derived_type r(static_cast<const decayed_derived_type&>(*this));
275  return r -= n;
276  }
277 };
278 
279 template<class iterator_t>
280  requires derived_from_template_c<iterator_t, base_reverse_iterator>
281 constexpr iterator_t operator+(typename iterator_t::size_type n, iterator_t it) noexcept
282 {
283  return it += n;
284 }
285 
286 } // namespace qx
Basic contiguous forward iterator, meaning that incrementing an iterator will lead it moving forward.
Definition: base_iterator.h:95
Base contiguous iterator with a container pointer and an index.
Definition: base_iterator.h:32
Basic contiguous reverse iterator, meaning that incrementing an iterator will lead it moving backward...