#pragma once #include #include #include namespace NIteratorPrivate { template constexpr bool HasRandomAccess() { return std::is_same_v::iterator_category, std::random_access_iterator_tag>; } } template class TMappedIterator { protected: using TSelf = TMappedIterator; using TSrcPointerType = typename std::iterator_traits::reference; using TInvokeResult = std::invoke_result_t; using TValue = std::remove_reference_t; public: using difference_type = std::ptrdiff_t; using value_type = TValue; using reference = TValue&; using const_reference = const TValue&; using pointer = std::remove_reference_t*; using iterator_category = std::conditional_t(), std::random_access_iterator_tag, std::input_iterator_tag>; TMappedIterator(TIterator it, TMapper mapper) : Iter(it) , Mapper(std::move(mapper)) { } TSelf& operator++() { ++Iter; return *this; } TSelf& operator--() { --Iter; return *this; } TInvokeResult operator*() { return Mapper((*Iter)); } TInvokeResult operator*() const { return Mapper((*Iter)); } pointer operator->() const { return &(Mapper((*Iter))); } TInvokeResult operator[](difference_type n) const { return Mapper(*(Iter + n)); } TSelf& operator+=(difference_type n) { Iter += n; return *this; } TSelf& operator-=(difference_type n) { Iter -= n; return *this; } TSelf operator+(difference_type n) const { return TSelf(Iter + n, Mapper); } TSelf operator-(difference_type n) const { return TSelf(Iter - n, Mapper); } difference_type operator-(const TSelf& other) const { return Iter - other.Iter; } bool operator==(const TSelf& other) const { return Iter == other.Iter; } bool operator!=(const TSelf& other) const { return Iter != other.Iter; } bool operator>(const TSelf& other) const { return Iter > other.Iter; } bool operator<(const TSelf& other) const { return Iter < other.Iter; } private: TIterator Iter; TMapper Mapper; }; template class TInputMappedRange { protected: using TContainerStorage = TAutoEmbedOrPtrPolicy; using TMapperStorage = TAutoEmbedOrPtrPolicy; using TMapperWrapper = std::reference_wrapper>; using TInternalIterator = decltype(std::begin(std::declval())); using TIterator = TMappedIterator; public: using iterator = TIterator; using const_iterator = TIterator; using value_type = typename TIterator::value_type; using reference = typename TIterator::reference; using const_reference = typename TIterator::const_reference; TInputMappedRange(TContainer&& container, TMapper&& mapper) : Container(std::forward(container)) , Mapper(std::forward(mapper)) { } TIterator begin() const { return {std::begin(*Container.Ptr()), {*Mapper.Ptr()}}; } TIterator end() const { return {std::end(*Container.Ptr()), {*Mapper.Ptr()}}; } bool empty() const { return std::begin(*Container.Ptr()) == std::end(*Container.Ptr()); } protected: mutable TContainerStorage Container; mutable TMapperStorage Mapper; }; template class TRandomAccessMappedRange : public TInputMappedRange { using TBase = TInputMappedRange; using TInternalIterator = typename TBase::TInternalIterator; using TIterator = typename TBase::TIterator; public: using iterator = typename TBase::iterator; using const_iterator = typename TBase::const_iterator; using value_type = typename TBase::value_type; using reference = typename TBase::reference; using const_reference = typename TBase::const_reference; using difference_type = typename std::iterator_traits::difference_type; using size_type = std::size_t; TRandomAccessMappedRange(TContainer&& container, TMapper&& mapper) : TBase(std::forward(container), std::forward(mapper)) { } using TBase::begin; using TBase::end; using TBase::empty; size_type size() const { return std::end(*this->Container.Ptr()) - std::begin(*this->Container.Ptr()); } const_reference operator[](size_t at) const { Y_ASSERT(at < this->size()); return *(this->begin() + at); } reference operator[](size_t at) { Y_ASSERT(at < this->size()); return *(this->begin() + at); } }; template TMappedIterator MakeMappedIterator(TIterator iter, TMapper mapper) { return {iter, mapper}; } template auto MakeMappedRange(TIterator begin, TIterator end, TMapper mapper) { return MakeIteratorRange(MakeMappedIterator(begin, mapper), MakeMappedIterator(end, mapper)); } template auto MakeMappedRange(TContainer&& container, TMapper&& mapper) { if constexpr (NIteratorPrivate::HasRandomAccess()) { return TRandomAccessMappedRange(std::forward(container), std::forward(mapper)); } else { return TInputMappedRange(std::forward(container), std::forward(mapper)); } }