#pragma once /* * Copyright (C) 2024 Brett Terpstra * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef BLT_ITERATOR_H #define BLT_ITERATOR_H #include #include #include #include #include namespace blt { // forward declare useful types template> class enumerator; template> class enumerator_rev; namespace iterator { template> class enumerate_wrapper; // struct which is returned by the enumerator template struct enumerate_item { blt::size_t index; T value; }; template class pair_iterator_base { public: explicit pair_iterator_base(Iter1 iter1, Iter2 iter2): m_iter1(std::move(iter1)), m_iter2(std::move(iter2)) {} friend bool operator==(const pair_iterator_base& a, const pair_iterator_base& b) { return a.m_iter1 == b.m_iter1; } friend bool operator!=(const pair_iterator_base& a, const pair_iterator_base& b) { return a.m_iter1 != b.m_iter1; } auto iter1() const { return m_iter1; } auto iter2() const { return m_iter2; } protected: Iter1 m_iter1; Iter2 m_iter2; }; template class enumerate_iterator_base { public: explicit enumerate_iterator_base(Iter iter, blt::size_t place = 0): iter(std::move(iter)), index(place) {} enumerate_item> operator*() const { return {index, *this->iter}; } friend bool operator==(const enumerate_iterator_base& a, const enumerate_iterator_base& b) { return a.iter == b.iter; } friend bool operator!=(const enumerate_iterator_base& a, const enumerate_iterator_base& b) { return a.iter != b.iter; } auto base() const { return iter; } auto get_index() const { return index; } protected: Iter iter; blt::size_t index; }; template class enumerate_forward_iterator : public enumerate_iterator_base { public: using enumerate_iterator_base::enumerate_iterator_base; enumerate_forward_iterator& operator++() { ++this->iter; ++this->index; return *this; } enumerate_forward_iterator operator++(int) { auto tmp = *this; ++*this; return tmp; } }; template class enumerate_bidirectional_iterator : public enumerate_forward_iterator { public: using enumerate_forward_iterator::enumerate_forward_iterator; enumerate_bidirectional_iterator& operator--() { --this->iter; --this->index; return *this; } enumerate_bidirectional_iterator operator--(int) { auto tmp = *this; --*this; return tmp; } }; template class enumerate_wrapper, std::void_t>> : public enumerate_forward_iterator { public: using iterator_category = std::forward_iterator_tag; using value_type = enumerate_item>; using difference_type = typename std::iterator_traits::difference_type; using pointer = value_type; using reference = value_type; using iterator_type = Iter; using enumerate_forward_iterator::enumerate_forward_iterator; }; template class enumerate_wrapper, std::void_t>> : public enumerate_bidirectional_iterator { public: using iterator_category = typename std::iterator_traits::iterator_category; using value_type = enumerate_item>; using difference_type = typename std::iterator_traits::difference_type; using pointer = value_type; using reference = value_type; using iterator_type = Iter; using enumerate_bidirectional_iterator::enumerate_bidirectional_iterator; }; template class enumerator_base { public: explicit enumerator_base(Iter begin, Iter end): begin_(std::move(begin)), end_(std::move(end)) {} explicit enumerator_base(IterWrapper begin, IterWrapper end): begin_(std::move(begin)), end_(std::move(end)) {} auto begin() { return begin_; } auto end() { return end_; } protected: IterWrapper begin_; IterWrapper end_; }; template class enumerator_reversible : public enumerator_base { public: explicit enumerator_reversible(Iter begin, Iter end, blt::size_t container_size): enumerator_base{IterWrapper{enumerate_wrapper{std::move(begin), 0}}, IterWrapper{enumerate_wrapper{std::move(end), container_size}}}, container_size(container_size) {} explicit enumerator_reversible(Iter begin, Iter end, blt::size_t begin_index, blt::size_t end_index, blt::size_t container_size): enumerator_base(IterWrapper{enumerate_wrapper{std::move(begin), begin_index}}, IterWrapper{enumerate_wrapper{std::move(end), end_index}}), container_size(container_size) {} auto rev() const { return enumerator_rev{this->end_.base(), this->begin_.base(), this->end_.get_index(), this->begin_.get_index(), this->container_size}; } auto skip(blt::size_t offset) { auto begin = this->begin(); for (blt::size_t i = 0; i < offset; i++) ++begin; return enumerator{begin.base(), this->end_.base(), begin.get_index(), this->end_.get_index(), this->container_size}; } protected: blt::size_t container_size; }; template class enumerator_reversible_rev : public enumerator_reversible { public: using enumerator_reversible::enumerator_reversible; auto rev() const { return enumerator{this->end_.base().base(), this->begin_.base().base(), this->end_.base().get_index(), this->begin_.base().get_index(), this->container_size}; } auto skip(blt::size_t offset) { auto begin = this->begin(); for (blt::size_t i = 0; i < offset; i++) --begin; return enumerator{begin.base(), this->end_.base(), begin.get_index(), this->end_.get_index(), this->container_size}; } }; } template class enumerator, std::void_t>> : public iterator::enumerator_base> { public: using iterator::enumerator_base>::enumerator_base; }; template class enumerator, std::void_t>> : public iterator::enumerator_reversible> { public: using iterator::enumerator_reversible>::enumerator_reversible; }; template class enumerator, std::void_t>> : public iterator::enumerator_reversible> { public: using iterator::enumerator_reversible>::enumerator_reversible; auto skip(blt::size_t offset) { return enumerator{this->begin_.base() + offset, this->end_.base(), this->begin_.get_index() + offset, this->end_.get_index(), this->container_size}; } }; template class enumerator_rev, std::void_t>> : public iterator::enumerator_reversible_rev>> { public: using iterator::enumerator_reversible_rev>>::enumerator_reversible_rev; }; template class enumerator_rev, std::void_t>> : public iterator::enumerator_reversible_rev>> { public: using iterator::enumerator_reversible_rev>>::enumerator_reversible_rev; auto skip(blt::size_t offset) { return enumerator_rev{this->begin_.base().base() - offset, this->end_.base().base(), this->begin_.base().get_index() - offset, this->end_.base().get_index(), this->container_size}; } }; template enumerator(Iter, Iter) -> enumerator; template enumerator(Iter, Iter, blt::size_t) -> enumerator; template enumerator(Iter, Iter, blt::size_t, blt::size_t) -> enumerator; template static inline auto enumerate(const T& container) { return enumerator{container.begin(), container.end(), container.size()}; } template static inline auto enumerate(const T(& container)[size]) { return enumerator{&container[0], &container[size], size}; } template static inline auto enumerate(T(& container)[size]) { return enumerator{&container[0], &container[size], size}; } template static inline auto enumerate(T& container) { return enumerator{container.begin(), container.end(), container.size()}; } template static inline auto enumerate(T&& container) { return enumerator{container.begin(), container.end(), container.size()}; } } #endif //BLT_ITERATOR_H