#ifndef SPROUT_SUB_ARRAY_HPP #define SPROUT_SUB_ARRAY_HPP #include #include #include #include #include #include #include #include #include #include #include HDR_ALGORITHM_SSCRISK_CEL_OR_SPROUT #include HDR_ITERATOR_SSCRISK_CEL_OR_SPROUT namespace sprout { namespace detail { struct is_non_reference_array_tag {}; struct is_not_non_reference_array_tag {}; template class sub_array_impl { protected: typedef Container container_type; typedef typename std::remove_reference::type internal_type; protected: SPROUT_STATIC_CONSTEXPR bool is_reference = std::is_reference::value; SPROUT_STATIC_CONSTEXPR bool is_const = std::is_const::value; protected: typedef typename sprout::container_traits::value_type value_type; typedef typename std::conditional< is_const, typename sprout::container_traits::const_iterator, typename sprout::container_traits::iterator >::type iterator; typedef typename sprout::container_traits::const_iterator const_iterator; typedef typename std::conditional< is_const, typename sprout::container_traits::const_reference, typename sprout::container_traits::reference >::type reference; typedef typename sprout::container_traits::const_reference const_reference; typedef typename sprout::container_traits::size_type size_type; typedef typename sprout::container_traits::difference_type difference_type; typedef typename std::conditional< is_const, typename sprout::container_traits::const_pointer, typename sprout::container_traits::pointer >::type pointer; typedef typename sprout::container_traits::const_pointer const_pointer; protected: SPROUT_STATIC_CONSTEXPR size_type static_size = sprout::container_traits::static_size; protected: typedef typename std::conditional< is_reference, internal_type*, typename std::remove_const::type >::type holder_type; typedef typename std::conditional< is_reference, internal_type&, internal_type const& >::type param_type; typedef internal_type const& const_param_type; protected: typedef typename std::conditional< std::is_array::value, sprout::detail::is_non_reference_array_tag, sprout::detail::is_not_non_reference_array_tag >::type array_tag; protected: template static SPROUT_CONSTEXPR typename std::enable_if< std::is_reference::value, holder_type >::type to_holder(param_type arr) { return &arr; } template static SPROUT_CONSTEXPR typename std::enable_if< !std::is_reference::value, holder_type const& >::type to_holder(param_type arr) { return arr; } template static SPROUT_CONSTEXPR typename std::enable_if< std::is_reference::value, param_type >::type to_param(holder_type arr) { return *arr; } template static SPROUT_CONSTEXPR typename std::enable_if< !std::is_reference::value, param_type >::type to_param(holder_type& arr) { return arr; } template static SPROUT_CONSTEXPR typename std::enable_if< !std::is_reference::value, param_type >::type to_param(holder_type const& arr) { return arr; } template static SPROUT_CONSTEXPR typename std::enable_if< std::is_reference::value, const_param_type >::type to_const_param(holder_type arr) { return *arr; } template static SPROUT_CONSTEXPR typename std::enable_if< !std::is_reference::value, const_param_type >::type to_const_param(holder_type const& arr) { return arr; } protected: holder_type array_; difference_type first_; difference_type last_; public: sub_array_impl() = default; protected: template SPROUT_CONSTEXPR sub_array_impl( ContainerTag, param_type arr, sprout::index_tuple, const_iterator first, const_iterator last, typename std::enable_if::value>::type* = 0 ) : array_{to_holder(arr)[Indexes]...} , first_(NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::cbegin(arr), first)) , last_(NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::cbegin(arr), last)) {} template SPROUT_CONSTEXPR sub_array_impl( ContainerTag, param_type arr, sprout::index_tuple, const_iterator first, const_iterator last, typename std::enable_if::value>::type* = 0 ) : array_(to_holder(arr)) , first_(NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::cbegin(arr), first)) , last_(NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::cbegin(arr), last)) {} template SPROUT_CONSTEXPR sub_array_impl( ContainerTag, param_type arr, sprout::index_tuple, difference_type first, difference_type last, typename std::enable_if::value>::type* = 0 ) : array_{to_holder(arr)[Indexes]...} , first_(first) , last_(last) {} template SPROUT_CONSTEXPR sub_array_impl( ContainerTag, param_type arr, sprout::index_tuple, difference_type first, difference_type last, typename std::enable_if::value>::type* = 0 ) : array_(to_holder(arr)) , first_(first) , last_(last) {} }; } // namespace detail // // sub_array // template class sub_array : private sprout::detail::sub_array_impl { private: typedef sprout::detail::sub_array_impl impl_type; public: typedef typename impl_type::container_type container_type; typedef typename impl_type::internal_type internal_type; public: SPROUT_STATIC_CONSTEXPR bool is_reference = impl_type::is_reference; SPROUT_STATIC_CONSTEXPR bool is_const = impl_type::is_const; public: typedef typename impl_type::value_type value_type; typedef typename impl_type::iterator iterator; typedef typename impl_type::const_iterator const_iterator; typedef typename impl_type::reference reference; typedef typename impl_type::const_reference const_reference; typedef typename impl_type::size_type size_type; typedef typename impl_type::difference_type difference_type; typedef typename impl_type::pointer pointer; typedef typename impl_type::const_pointer const_pointer; public: SPROUT_STATIC_CONSTEXPR size_type static_size = impl_type::static_size; public: typedef typename impl_type::holder_type holder_type; typedef typename impl_type::param_type param_type; typedef typename impl_type::const_param_type const_param_type; private: typedef typename impl_type::array_tag array_tag; private: using impl_type::array_; using impl_type::first_; using impl_type::last_; public: // construct/copy/destroy: sub_array() = default; SPROUT_CONSTEXPR sub_array(param_type arr, const_iterator first, const_iterator last) : impl_type( array_tag(), arr, typename sprout::index_range<0, static_size>::type(), first, last ) {} SPROUT_CONSTEXPR sub_array(param_type arr, difference_type first, difference_type last) : impl_type( array_tag(), arr, typename sprout::index_range<0, static_size>::type(), first, last ) {} SPROUT_CONSTEXPR sub_array(sub_array const& other, const_iterator first, const_iterator last) : impl_type( array_tag(), impl_type::template to_param(other.array_), typename sprout::index_range<0, static_size>::type(), NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::begin(other.get_array()), first), NS_SSCRISK_CEL_OR_SPROUT::distance(sprout::begin(other.get_array()), last) ) {} SPROUT_CONSTEXPR sub_array(sub_array const& other, difference_type first, difference_type last) : impl_type( array_tag(), impl_type::template to_param(other.array_), typename sprout::index_range<0, static_size>::type(), first + other.first_, last + other.first_ ) {} void fill(const_reference value) { std::fill_n(begin(), size(), value); } template void swap(sub_array& other) { using std::swap; swap(other.array_, array_); swap(other.first_, first_); swap(other.last_, last_); } // iterators: iterator begin() { return sprout::next(sprout::begin(get_array()), first_); } SPROUT_CONSTEXPR const_iterator begin() const { return sprout::next(sprout::begin(get_array()), first_); } iterator end() { return sprout::next(sprout::begin(get_array()), last_); } SPROUT_CONSTEXPR const_iterator end() const { return sprout::next(sprout::begin(get_array()), last_); } SPROUT_CONSTEXPR const_iterator cbegin() const { return sprout::next(sprout::begin(get_array()), first_); } SPROUT_CONSTEXPR const_iterator cend() const { return sprout::next(sprout::begin(get_array()), last_); } // capacity: SPROUT_CONSTEXPR size_type size() const { return last_ - first_; } SPROUT_CONSTEXPR size_type max_size() const { return size(); } SPROUT_CONSTEXPR bool empty() const { return first_ == last_; } // element access: reference operator[](size_type i) { return *sprout::next(sprout::begin(get_array()), first_ + i); } SPROUT_CONSTEXPR const_reference operator[](size_type i) const { return *sprout::next(sprout::begin(get_array()), first_ + i); } reference at(size_type i) { return i < size() ? *sprout::next(sprout::begin(get_array()), first_ + i) : (throw std::out_of_range("sub_array<>: index out of range"), *sprout::next(sprout::begin(get_array()), first_ + i)) ; } SPROUT_CONSTEXPR const_reference at(size_type i) const { return i < size() ? *sprout::next(sprout::begin(get_array()), first_ + i) : (throw std::out_of_range("sub_array<>: index out of range"), *sprout::next(sprout::begin(get_array()), first_ + i)) ; } reference front() { return *sprout::next(sprout::begin(get_array()), first_); } SPROUT_CONSTEXPR const_reference front() const { return *sprout::next(sprout::begin(get_array()), first_); } reference back() { return *sprout::next(sprout::begin(get_array()), last_ - 1); } SPROUT_CONSTEXPR const_reference back() const { return *sprout::next(sprout::begin(get_array()), last_ - 1); } pointer data() { return get_array().data() + first_; } SPROUT_CONSTEXPR const_pointer data() const { return get_array().data() + first_; } // others: template sub_array& operator=(sub_array const& rhs) { array_ = rhs.array_; first_ = rhs.first_; last_ = rhs.last_; return *this; } template sub_array& operator=(sub_array&& rhs) { array_ = std::move(rhs.array_); first_ = std::move(rhs.first_); last_ = std::move(rhs.last_); return *this; } pointer c_array() { return data(); } void assign(const_reference value) { fill(value); } void rangecheck(size_type i) const { if (i >= size()) { throw std::out_of_range("sub_array<>: index out of range"); } } param_type get_internal() { return impl_type::template to_param(array_); } SPROUT_CONSTEXPR const_param_type get_internal() const { return impl_type::template to_const_param(array_); } param_type get_array() { return impl_type::template to_param(array_); } SPROUT_CONSTEXPR const_param_type get_array() const { return impl_type::template to_const_param(array_); } }; template SPROUT_CONSTEXPR typename sprout::sub_array::size_type sprout::sub_array::static_size; // // operator== // operator!= // operator< // operator> // operator<= // operator>= // template SPROUT_CONSTEXPR inline bool operator==(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return NS_SSCRISK_CEL_OR_SPROUT::equal(sprout::begin(lhs), sprout::end(lhs), sprout::begin(rhs)); } template SPROUT_CONSTEXPR inline bool operator!=(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return !(lhs == rhs); } template SPROUT_CONSTEXPR inline bool operator<(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return NS_SSCRISK_CEL_OR_SPROUT::lexicographical_compare(sprout::begin(lhs), sprout::end(lhs), sprout::begin(rhs), sprout::end(rhs)); } template SPROUT_CONSTEXPR inline bool operator>(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return rhs < lhs; } template SPROUT_CONSTEXPR inline bool operator<=(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return !(rhs < lhs); } template SPROUT_CONSTEXPR inline bool operator>=(sprout::sub_array const& lhs, sprout::sub_array const& rhs) { return !(lhs < rhs); } // // swap // template inline void swap(sprout::sub_array& lhs, sprout::sub_array& rhs) { lhs.swap(rhs); } // // container_construct_traits // template struct container_construct_traits > { private: typedef typename sprout::sub_array::internal_type internal_type; typedef typename sprout::container_construct_traits::copied_type internal_copied_type; public: typedef sprout::sub_array copied_type; private: static SPROUT_CONSTEXPR copied_type make_impl(internal_copied_type const& internal_copied) { return copied_type(internal_copied, sprout::begin(internal_copied), sprout::end(internal_copied)); } template static SPROUT_CONSTEXPR copied_type remake_impl( Cont&& cont, typename sprout::container_traits >::difference_type size, internal_copied_type const& internal_copied ) { return copied_type( internal_copied, sprout::next(sprout::begin(internal_copied), sprout::internal_begin_offset(cont)), sprout::next(sprout::begin(internal_copied), sprout::internal_begin_offset(cont) + size) ); } public: template static SPROUT_CONSTEXPR copied_type deep_copy(Cont&& cont) { return copied_type( sprout::deep_copy(sprout::get_internal(cont)), sprout::internal_begin_offset(cont), sprout::internal_end_offset(cont) ); } template static SPROUT_CONSTEXPR copied_type make(Args&&... args) { return make_impl( sprout::make(sprout::forward(args)...) ); } template static SPROUT_CONSTEXPR copied_type remake( Cont&& cont, typename sprout::container_traits >::difference_type size, Args&&... args ) { return remake_impl( sprout::forward(cont), size, sprout::make(sprout::forward(args)...) ); } }; // // container_transform_traits // template struct container_transform_traits > { public: template >::size_type Size> struct rebind_size { public: typedef sprout::sub_array< typename sprout::container_transform_traits< typename std::remove_reference::type >::template rebind_size::type > type; }; }; // // sub_container_traits // template struct sub_container_traits > { private: static typename sprout::sub_array::param_type call(sprout::sub_array& cont) { return cont.get_internal(); } static SPROUT_CONSTEXPR typename sprout::sub_array::const_param_type call(sprout::sub_array const& cont) { return cont.get_internal(); } public: template struct internal { public: typedef decltype(call(std::declval())) type; }; public: template static SPROUT_CONSTEXPR typename internal::type get_internal(Cont&& cont) { return call(sprout::forward(cont)); } }; namespace detail { template struct is_sub_array_impl : public std::false_type {}; template struct is_sub_array_impl< T, typename std::enable_if< std::is_same< T, sprout::sub_array >::value >::type > : public std::true_type {}; } // namespace detail // // is_sub_array // template struct is_sub_array : public sprout::detail::is_sub_array_impl {}; // // sub // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container& arr, typename sprout::container_traits >::const_iterator first, typename sprout::container_traits >::const_iterator last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container& arr, typename sprout::container_traits >::difference_type first, typename sprout::container_traits >::difference_type last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container& arr, typename sprout::container_traits >::const_iterator first ) { return sprout::sub(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container& arr, typename sprout::container_traits >::difference_type first ) { return sprout::sub(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container& arr ) { return sprout::sub(arr, sprout::begin(arr), sprout::end(arr)); } // // sub // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container const& arr, typename sprout::container_traits >::const_iterator first, typename sprout::container_traits >::const_iterator last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container const& arr, typename sprout::container_traits >::difference_type first, typename sprout::container_traits >::difference_type last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container const& arr, typename sprout::container_traits >::const_iterator first ) { return sprout::sub(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container const& arr, typename sprout::container_traits >::difference_type first ) { return sprout::sub(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub( Container const& arr ) { return sprout::sub(arr, sprout::begin(arr), sprout::end(arr)); } // // sub // template SPROUT_CONSTEXPR inline typename std::enable_if::value, Container>::type sub( Container const& arr, typename sprout::container_traits::const_iterator first, typename sprout::container_traits::const_iterator last ) { return Container(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, Container>::type sub( Container const& arr, typename sprout::container_traits::difference_type first, typename sprout::container_traits::difference_type last ) { return Container(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, Container>::type sub( Container const& arr, typename sprout::container_traits::const_iterator first ) { return sprout::sub(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, Container>::type sub( Container const& arr, typename sprout::container_traits::difference_type first ) { return sprout::sub(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, Container>::type sub( Container const& arr ) { return sprout::sub(arr, sprout::begin(arr), sprout::end(arr)); } // // csub // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits >::const_iterator first, typename sprout::container_traits >::const_iterator last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits >::difference_type first, typename sprout::container_traits >::difference_type last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits >::const_iterator first ) { return sprout::csub(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits >::difference_type first ) { return sprout::csub(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr ) { return sprout::csub(arr, sprout::begin(arr), sprout::end(arr)); } // // csub // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits::const_iterator first, typename sprout::container_traits::const_iterator last ) { return sprout::sub_array(arr.get_array(), first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits::difference_type first, typename sprout::container_traits::difference_type last ) { return sprout::sub_array( arr.get_array(), sprout::next(sprout::begin(arr), first), sprout::next(sprout::begin(arr), last) ); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits::const_iterator first ) { return sprout::csub(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr, typename sprout::container_traits::difference_type first ) { return sprout::csub(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type csub( Container const& arr ) { return sprout::csub(arr, sprout::begin(arr), sprout::end(arr)); } // // sub_copy // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::const_iterator first, typename sprout::container_traits::const_iterator last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::difference_type first, typename sprout::container_traits::difference_type last ) { return sprout::sub_array(arr, first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::const_iterator first ) { return sprout::sub_copy(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::difference_type first ) { return sprout::sub_copy(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr ) { return sprout::sub_copy(arr, sprout::begin(arr), sprout::end(arr)); } // // sub_copy // template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::const_iterator first, typename sprout::container_traits::const_iterator last ) { return sprout::sub_array(arr.get_array(), first, last); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::difference_type first, typename sprout::container_traits::difference_type last ) { return sprout::sub_array( arr.get_array(), sprout::next(sprout::begin(arr), first), sprout::next(sprout::begin(arr), last) ); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::const_iterator first ) { return sprout::sub_copy(arr, first, sprout::end(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr, typename sprout::container_traits::difference_type first ) { return sprout::sub_copy(arr, first, sprout::size(arr)); } template SPROUT_CONSTEXPR inline typename std::enable_if::value, sprout::sub_array >::type sub_copy( Container const& arr ) { return sprout::sub_copy(arr, sprout::begin(arr), sprout::end(arr)); } } // namespace sprout namespace std { // // tuple_size // template struct tuple_size > : public std::tuple_size::type> {}; // // tuple_element // template struct tuple_element > : public std::tuple_element::type> {}; } // namespace std #endif // #ifndef SPROUT_SUB_ARRAY_HPP