/* * Copyright 2022 Axel Waggershauser */ // SPDX-License-Identifier: Apache-2.0 #pragma once #include "ZXAlgorithms.h" #include namespace ZXing { template struct StrideIter { Iterator pos; int stride; using iterator_category = std::random_access_iterator_tag; using difference_type = typename std::iterator_traits::difference_type; using value_type = typename std::iterator_traits::value_type; using pointer = Iterator; using reference = typename std::iterator_traits::reference; auto operator*() const { return *pos; } auto operator[](int i) const { return *(pos + i * stride); } StrideIter& operator++() { return pos += stride, *this; } StrideIter operator++(int) { auto temp = *this; ++*this; return temp; } bool operator!=(const StrideIter& rhs) const { return pos != rhs.pos; } StrideIter operator+(int i) const { return {pos + i * stride, stride}; } StrideIter operator-(int i) const { return {pos - i * stride, stride}; } int operator-(const StrideIter& rhs) const { return narrow_cast((pos - rhs.pos) / stride); } }; template StrideIter(const Iterator&, int) -> StrideIter; template struct Range { Iterator _begin, _end; Range(Iterator b, Iterator e) : _begin(b), _end(e) {} template Range(const C& c) : _begin(std::begin(c)), _end(std::end(c)) {} Iterator begin() const noexcept { return _begin; } Iterator end() const noexcept { return _end; } explicit operator bool() const { return begin() < end(); } int size() const { return narrow_cast(end() - begin()); } }; template Range(const C&) -> Range; } // namespace ZXing