30#ifndef _GLIBCXX_RANGES
31#define _GLIBCXX_RANGES 1
33#if __cplusplus > 201703L
36#pragma GCC system_header
50#if __cplusplus > 202002L
58#define __glibcxx_want_algorithm_default_value_type
59#define __glibcxx_want_ranges
60#define __glibcxx_want_ranges_as_const
61#define __glibcxx_want_ranges_as_rvalue
62#define __glibcxx_want_ranges_cache_latest
63#define __glibcxx_want_ranges_cartesian_product
64#define __glibcxx_want_ranges_concat
65#define __glibcxx_want_ranges_chunk
66#define __glibcxx_want_ranges_chunk_by
67#define __glibcxx_want_ranges_enumerate
68#define __glibcxx_want_ranges_filter
69#define __glibcxx_want_ranges_indices
70#define __glibcxx_want_ranges_join_with
71#define __glibcxx_want_ranges_repeat
72#define __glibcxx_want_ranges_slide
73#define __glibcxx_want_ranges_stride
74#define __glibcxx_want_ranges_to_container
75#define __glibcxx_want_ranges_as_input
76#define __glibcxx_want_ranges_zip
79#ifdef __glibcxx_generator
80# include <bits/elements_of.h>
89namespace std _GLIBCXX_VISIBILITY(default)
91_GLIBCXX_BEGIN_NAMESPACE_VERSION
106 template<
typename _Tp>
requires is_object_v<_Tp>
111 static constexpr _Tp* begin()
noexcept {
return nullptr; }
112 static constexpr _Tp* end()
noexcept {
return nullptr; }
113 static constexpr _Tp* data()
noexcept {
return nullptr; }
114 static constexpr size_t size()
noexcept {
return 0; }
115 static constexpr bool empty()
noexcept {
return true; }
118 template<
typename _Tp>
119 inline constexpr bool enable_borrowed_range<empty_view<_Tp>> =
true;
123#if __cpp_lib_ranges >= 202207L
125 template<
typename _Tp>
128 template<
typename _Tp>
132 template<__boxable _Tp>
133 struct __box : std::optional<_Tp>
135 using std::optional<_Tp>::optional;
139 noexcept(is_nothrow_default_constructible_v<_Tp>)
141 :
std::optional<_Tp>{std::in_place}
144 __box(
const __box&) =
default;
145 __box(__box&&) =
default;
147 using std::optional<_Tp>::operator=;
153 operator=(
const __box& __that)
154 noexcept(is_nothrow_copy_constructible_v<_Tp>)
155 requires (!copyable<_Tp>) && copy_constructible<_Tp>
160 this->emplace(*__that);
168 operator=(__box&& __that)
169 noexcept(is_nothrow_move_constructible_v<_Tp>)
170 requires (!movable<_Tp>)
183 template<
typename _Tp>
184 concept __boxable_copyable
185 = copy_constructible<_Tp>
186 && (copyable<_Tp> || (is_nothrow_move_constructible_v<_Tp>
187 && is_nothrow_copy_constructible_v<_Tp>));
188 template<
typename _Tp>
189 concept __boxable_movable
190 = (!copy_constructible<_Tp>)
191 && (movable<_Tp> || is_nothrow_move_constructible_v<_Tp>);
197 template<__boxable _Tp>
198 requires __boxable_copyable<_Tp> || __boxable_movable<_Tp>
202 [[no_unique_address]] _Tp _M_value = _Tp();
205 __box()
requires default_initializable<_Tp> = default;
208 __box(const _Tp& __t)
209 noexcept(is_nothrow_copy_constructible_v<_Tp>)
210 requires copy_constructible<_Tp>
216 noexcept(is_nothrow_move_constructible_v<_Tp>)
217 : _M_value(std::move(__t))
220 template<
typename... _Args>
221 requires constructible_from<_Tp, _Args...>
223 __box(in_place_t, _Args&&... __args)
224 noexcept(is_nothrow_constructible_v<_Tp, _Args...>)
225 : _M_value(std::
forward<_Args>(__args)...)
228 __box(
const __box&) =
default;
229 __box(__box&&) =
default;
230 __box& operator=(
const __box&)
requires copyable<_Tp> =
default;
231 __box& operator=(__box&&)
requires movable<_Tp> = default;
236 operator=(const __box& __that) noexcept
237 requires (!copyable<_Tp>) && copy_constructible<_Tp>
239 static_assert(is_nothrow_copy_constructible_v<_Tp>);
250 operator=(__box&& __that)
noexcept
251 requires (!movable<_Tp>)
253 static_assert(is_nothrow_move_constructible_v<_Tp>);
263 has_value() const noexcept
278 constexpr const _Tp&&
283 operator->() noexcept
287 operator->() const noexcept
291 namespace __func_handle
293 template<
typename _Fn>
296 _Inplace() =
default;
299 _Inplace(_Fn __func) noexcept
303 template<
typename... _Iters>
304 constexpr decltype(
auto)
305 _M_call_deref(
const _Iters&... __iters)
const
306 noexcept(
noexcept(_M_fn(*__iters...)))
307 {
return _M_fn(*__iters...); }
309 template<
typename _DistType,
typename... _Iters>
310 constexpr decltype(
auto)
311 _M_call_subscript(
const _DistType __n,
const _Iters&... __iters)
const
312 noexcept(
noexcept(_M_fn(__iters[iter_difference_t<_Iters>(__n)]...)))
313 {
return _M_fn(__iters[iter_difference_t<_Iters>(__n)]...); }
316 [[no_unique_address]] _Fn _M_fn = _Fn();
319 template<
typename _Fn>
320 struct _InplaceMemPtr
322 _InplaceMemPtr() =
default;
325 _InplaceMemPtr(_Fn __func) noexcept
329 template<
typename... _Iters>
330 constexpr decltype(
auto)
331 _M_call_deref(
const _Iters&... __iters)
const
335 template<
typename _DistType,
typename... _Iters>
336 constexpr decltype(
auto)
337 _M_call_subscript(
const _DistType __n,
const _Iters&... __iters)
const
338 noexcept(
noexcept(
std::__invoke(_M_ptr, __iters[iter_difference_t<_Iters>(__n)]...)))
339 {
return std::__invoke(_M_ptr, __iters[iter_difference_t<_Iters>(__n)]...); }
342 _Fn _M_ptr =
nullptr;
345 template<
typename _Fn>
348 _ViaPointer() =
default;
351 _ViaPointer(_Fn& __func) noexcept
355 template<
typename _Un>
356 requires (!is_const_v<_Un>) && is_same_v<const _Un, _Fn>
358 _ViaPointer(_ViaPointer<_Un> __other) noexcept
359 : _M_ptr(__other._M_ptr)
362 template<
typename... _Iters>
363 constexpr decltype(
auto)
364 _M_call_deref(
const _Iters&... __iters)
const
365 noexcept(
noexcept((*_M_ptr)(*__iters...)))
366 {
return (*_M_ptr)(*__iters...); }
368 template<
typename _DistType,
typename... _Iters>
369 constexpr decltype(
auto)
370 _M_call_subscript(
const _DistType __n,
const _Iters&... __iters)
const
371 noexcept(
noexcept((*_M_ptr)(__iters[iter_difference_t<_Iters>(__n)]...)))
372 {
return (*_M_ptr)(__iters[iter_difference_t<_Iters>(__n)]...); }
375 _Fn* _M_ptr =
nullptr;
378 friend struct _ViaPointer;
381 template<
typename _Fn>
384 _StaticCall() =
default;
387 _StaticCall(
const _Fn&)
noexcept
390 template<
typename... _Iters>
391 static constexpr decltype(
auto)
392 _M_call_deref(
const _Iters&... __iters)
393 noexcept(
noexcept(_Fn::operator()(*__iters...)))
394 {
return _Fn::operator()(*__iters...); }
396 template<
typename _DistType,
typename... _Iters>
397 static constexpr decltype(
auto)
398 _M_call_subscript(_DistType __n,
const _Iters&... __iters)
399 noexcept(
noexcept(_Fn::operator()(__iters[iter_difference_t<_Iters>(__n)]...)))
400 {
return _Fn::operator()(__iters[iter_difference_t<_Iters>(__n)]...); }
403 template<
typename _Fn,
typename... _Iters>
407 using _Fd = remove_cv_t<_Fn>;
408 if constexpr (is_member_pointer_v<_Fd>)
409 return __func_handle::_InplaceMemPtr<_Fd>();
410 else if constexpr (is_function_v<remove_pointer_t<_Fd>>)
411 return __func_handle::_Inplace<_Fd>();
412 else if constexpr (__is_std_op_wrapper<_Fd>)
413 return __func_handle::_Inplace<_Fd>();
414 else if constexpr (
requires (
const _Iters&... __iters)
415 { _Fd::operator()(*__iters...); })
416 return __func_handle::_StaticCall<_Fd>();
418 return __func_handle::_ViaPointer<_Fn>();
422 template<
typename _Fn,
typename... _Iters>
423 using __func_handle_t =
decltype(__func_handle::__select<_Fn, _Iters...>());
427#if __cpp_lib_ranges >= 202207L
428 template<move_constructible _Tp>
430 template<copy_constructible _Tp>
432 requires is_object_v<_Tp>
439 single_view(
const _Tp& __t)
440 noexcept(is_nothrow_copy_constructible_v<_Tp>)
446 single_view(_Tp&& __t)
447 noexcept(is_nothrow_move_constructible_v<_Tp>)
453 template<
typename... _Args>
456 single_view(in_place_t, _Args&&... __args)
457 noexcept(is_nothrow_constructible_v<_Tp, _Args...>)
466 begin()
const noexcept
471 {
return data() + 1; }
475 {
return data() + 1; }
479 static constexpr bool
483 static constexpr size_t
489 {
return _M_value.operator->(); }
492 data()
const noexcept
493 {
return _M_value.operator->(); }
496 [[no_unique_address]] __detail::__box<_Tp> _M_value;
499 template<
typename _Tp>
504 template<
typename _Wp>
505 constexpr auto __to_signed_like(_Wp __w)
noexcept
507 if constexpr (!integral<_Wp>)
508 return iter_difference_t<_Wp>();
509 else if constexpr (
sizeof(iter_difference_t<_Wp>) >
sizeof(_Wp))
510 return iter_difference_t<_Wp>(__w);
511 else if constexpr (
sizeof(ptrdiff_t) >
sizeof(_Wp))
512 return ptrdiff_t(__w);
513 else if constexpr (
sizeof(
long long) >
sizeof(_Wp))
514 return (
long long)(__w);
515#ifdef __SIZEOF_INT128__
516 else if constexpr (__SIZEOF_INT128__ >
sizeof(_Wp))
517 return __int128(__w);
520 return __max_diff_type(__w);
523 template<
typename _Wp>
526 template<
typename _It>
527 concept __decrementable = incrementable<_It>
530 { --__i } -> same_as<_It&>;
531 { __i-- } -> same_as<_It>;
534 template<
typename _It>
535 concept __advanceable = __decrementable<_It> && totally_ordered<_It>
536 &&
requires( _It __i,
const _It __j,
const __iota_diff_t<_It> __n)
538 { __i += __n } -> same_as<_It&>;
539 { __i -= __n } -> same_as<_It&>;
543 { __j - __j } -> convertible_to<__iota_diff_t<_It>>;
546 template<
typename _Winc>
547 struct __iota_view_iter_cat
550 template<incrementable _Winc>
551 struct __iota_view_iter_cat<_Winc>
552 {
using iterator_category = input_iterator_tag; };
555 template<weakly_incrementable _Winc,
556 semiregular _Bound = unreachable_sentinel_t>
557 requires std::__detail::__weakly_eq_cmp_with<_Winc, _Bound>
564 struct _Iterator : __detail::__iota_view_iter_cat<_Winc>
570 using namespace __detail;
571 if constexpr (__advanceable<_Winc>)
572 return random_access_iterator_tag{};
573 else if constexpr (__decrementable<_Winc>)
574 return bidirectional_iterator_tag{};
575 else if constexpr (incrementable<_Winc>)
576 return forward_iterator_tag{};
578 return input_iterator_tag{};
582 using iterator_concept =
decltype(_S_iter_concept());
584 using value_type = _Winc;
585 using difference_type = __detail::__iota_diff_t<_Winc>;
587 _Iterator()
requires default_initializable<_Winc> = default;
590 _Iterator(_Winc __value)
591 : _M_value(__value) { }
594 operator*() const noexcept(is_nothrow_copy_constructible_v<_Winc>)
609 operator++(
int)
requires incrementable<_Winc>
617 operator--()
requires __detail::__decrementable<_Winc>
624 operator--(
int)
requires __detail::__decrementable<_Winc>
632 operator+=(difference_type __n)
requires __detail::__advanceable<_Winc>
634 using __detail::__is_integer_like;
635 using __detail::__is_signed_integer_like;
636 if constexpr (__is_integer_like<_Winc>
637 && !__is_signed_integer_like<_Winc>)
639 if (__n >= difference_type(0))
640 _M_value +=
static_cast<_Winc
>(__n);
642 _M_value -=
static_cast<_Winc
>(-__n);
650 operator-=(difference_type __n)
requires __detail::__advanceable<_Winc>
652 using __detail::__is_integer_like;
653 using __detail::__is_signed_integer_like;
654 if constexpr (__is_integer_like<_Winc>
655 && !__is_signed_integer_like<_Winc>)
657 if (__n >= difference_type(0))
658 _M_value -=
static_cast<_Winc
>(__n);
660 _M_value +=
static_cast<_Winc
>(-__n);
668 operator[](difference_type __n)
const
669 requires __detail::__advanceable<_Winc>
670 {
return _Winc(_M_value + __n); }
672 friend constexpr bool
673 operator==(
const _Iterator& __x,
const _Iterator& __y)
674 requires equality_comparable<_Winc>
675 {
return __x._M_value == __y._M_value; }
677 friend constexpr bool
678 operator<(
const _Iterator& __x,
const _Iterator& __y)
679 requires totally_ordered<_Winc>
680 {
return __x._M_value < __y._M_value; }
682 friend constexpr bool
683 operator>(
const _Iterator& __x,
const _Iterator& __y)
684 requires totally_ordered<_Winc>
685 {
return __y < __x; }
687 friend constexpr bool
688 operator<=(
const _Iterator& __x,
const _Iterator& __y)
689 requires totally_ordered<_Winc>
690 {
return !(__y < __x); }
692 friend constexpr bool
693 operator>=(
const _Iterator& __x,
const _Iterator& __y)
694 requires totally_ordered<_Winc>
695 {
return !(__x < __y); }
697#ifdef __cpp_lib_three_way_comparison
698 friend constexpr auto
699 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
700 requires totally_ordered<_Winc> && three_way_comparable<_Winc>
701 {
return __x._M_value <=> __y._M_value; }
704 friend constexpr _Iterator
705 operator+(_Iterator __i, difference_type __n)
706 requires __detail::__advanceable<_Winc>
712 friend constexpr _Iterator
713 operator+(difference_type __n, _Iterator __i)
714 requires __detail::__advanceable<_Winc>
715 {
return __i += __n; }
717 friend constexpr _Iterator
718 operator-(_Iterator __i, difference_type __n)
719 requires __detail::__advanceable<_Winc>
725 friend constexpr difference_type
726 operator-(
const _Iterator& __x,
const _Iterator& __y)
727 requires __detail::__advanceable<_Winc>
729 using __detail::__is_integer_like;
730 using __detail::__is_signed_integer_like;
731 using _Dt = difference_type;
732 if constexpr (__is_integer_like<_Winc>)
734 if constexpr (__is_signed_integer_like<_Winc>)
735 return _Dt(_Dt(__x._M_value) - _Dt(__y._M_value));
737 return (__y._M_value > __x._M_value)
738 ? _Dt(-_Dt(__y._M_value - __x._M_value))
739 : _Dt(__x._M_value - __y._M_value);
742 return __x._M_value - __y._M_value;
746 _Winc _M_value = _Winc();
755 _Bound _M_bound = _Bound();
758 _Sentinel() =
default;
761 _Sentinel(_Bound __bound)
762 : _M_bound(__bound) { }
764 friend constexpr bool
765 operator==(
const _Iterator& __x,
const _Sentinel& __y)
766 {
return __x._M_value == __y._M_bound; }
768 friend constexpr iter_difference_t<_Winc>
769 operator-(
const _Iterator& __x,
const _Sentinel& __y)
770 requires sized_sentinel_for<_Bound, _Winc>
771 {
return -(__y._M_bound - __x._M_value); }
773 friend constexpr iter_difference_t<_Winc>
774 operator-(
const _Sentinel& __x,
const _Iterator& __y)
775 requires sized_sentinel_for<_Bound, _Winc>
776 {
return __x._M_bound - __y._M_value; }
781 _Winc _M_value = _Winc();
782 [[no_unique_address]] _Bound _M_bound = _Bound();
785 iota_view()
requires default_initializable<_Winc> = default;
788 iota_view(_Winc __value)
793 iota_view(type_identity_t<_Winc> __value,
794 type_identity_t<_Bound> __bound)
795 : _M_value(__value), _M_bound(__bound)
797 if constexpr (totally_ordered_with<_Winc, _Bound>)
798 __glibcxx_assert(
bool(__value <= __bound) );
802 iota_view(_Iterator __first, _Iterator __last)
803 requires same_as<_Winc, _Bound>
804 : iota_view(__first._M_value, __last._M_value)
808 iota_view(_Iterator __first, unreachable_sentinel_t __last)
809 requires same_as<_Bound, unreachable_sentinel_t>
810 : iota_view(__first._M_value, __last)
814 iota_view(_Iterator __first, _Sentinel __last)
815 requires (!same_as<_Winc, _Bound>) && (!same_as<_Bound, unreachable_sentinel_t>)
816 : iota_view(__first._M_value, __last._M_bound)
820 begin()
const {
return _Iterator{_M_value}; }
825 if constexpr (same_as<_Bound, unreachable_sentinel_t>)
826 return unreachable_sentinel;
828 return _Sentinel{_M_bound};
832 end() const requires same_as<_Winc, _Bound>
833 {
return _Iterator{_M_bound}; }
839 {
return _M_value == _M_bound; }
843 requires (same_as<_Winc, _Bound> && __detail::__advanceable<_Winc>)
844 || (integral<_Winc> && integral<_Bound>)
845 || sized_sentinel_for<_Bound, _Winc>
847 using __detail::__is_integer_like;
848 using __detail::__to_unsigned_like;
849 if constexpr (integral<_Winc> && integral<_Bound>)
852 return _Up(_M_bound) - _Up(_M_value);
854 else if constexpr (__is_integer_like<_Winc>)
855 return __to_unsigned_like(_M_bound) - __to_unsigned_like(_M_value);
857 return __to_unsigned_like(_M_bound - _M_value);
861 template<
typename _Winc,
typename _Bound>
862 requires (!__detail::__is_integer_like<_Winc>
863 || !__detail::__is_integer_like<_Bound>
864 || (__detail::__is_signed_integer_like<_Winc>
865 == __detail::__is_signed_integer_like<_Bound>))
866 iota_view(_Winc, _Bound) -> iota_view<_Winc, _Bound>;
868 template<
typename _Winc,
typename _Bound>
869 inline constexpr bool
870 enable_borrowed_range<iota_view<_Winc, _Bound>> =
true;
874 template<
typename _Tp>
875 inline constexpr empty_view<_Tp> empty{};
879 template<
typename _Tp>
880 concept __can_single_view
886 template<__detail::__can_single_view _Tp>
888 operator() [[nodiscard]] (_Tp&& __e)
const
893 inline constexpr _Single single{};
897 template<
typename... _Args>
903 template<__detail::__can_iota_view _Tp>
905 operator() [[nodiscard]] (_Tp&& __e)
const
908 template<
typename _Tp,
typename _Up>
909 requires __detail::__can_iota_view<_Tp, _Up>
911 operator() [[nodiscard]] (_Tp&& __e, _Up&& __f)
const
915 inline constexpr _Iota iota{};
917#ifdef __cpp_lib_ranges_indices
920 template<ranges::__detail::__is_
integer_like _Tp>
921 requires __detail::__can_iota_view<_Tp>
922 [[nodiscard]]
constexpr auto
923 operator() (_Tp __e)
const noexcept
924 {
return iota(_Tp{}, __e); }
927 inline constexpr _Indices indices{};
934 template<
typename _Val,
typename _CharT,
typename _Traits>
935 concept __stream_extractable
936 =
requires(basic_istream<_CharT, _Traits>& is, _Val& t) { is >> t; };
939 template<movable _Val,
typename _CharT,
940 typename _Traits = char_traits<_CharT>>
941 requires default_initializable<_Val>
942 && __detail::__stream_extractable<_Val, _CharT, _Traits>
943 class basic_istream_view
944 :
public view_interface<basic_istream_view<_Val, _CharT, _Traits>>
948 basic_istream_view(basic_istream<_CharT, _Traits>& __stream)
955 *_M_stream >> _M_object;
956 return _Iterator{
this};
959 constexpr default_sentinel_t
964 basic_istream<_CharT, _Traits>* _M_stream;
965 _Val _M_object = _Val();
970 using iterator_concept = input_iterator_tag;
971 using difference_type = ptrdiff_t;
972 using value_type = _Val;
975 _Iterator(basic_istream_view* __parent) noexcept
976 : _M_parent(__parent)
979 _Iterator(
const _Iterator&) =
delete;
980 _Iterator(_Iterator&&) =
default;
981 _Iterator& operator=(
const _Iterator&) =
delete;
982 _Iterator& operator=(_Iterator&&) =
default;
987 *_M_parent->_M_stream >> _M_parent->_M_object;
997 {
return _M_parent->_M_object; }
1000 operator==(
const _Iterator& __x, default_sentinel_t)
1001 {
return !*__x._M_parent->_M_stream; }
1004 basic_istream_view* _M_parent;
1010 template<
typename _Val>
1011 using istream_view = basic_istream_view<_Val, char>;
1013 template<
typename _Val>
1014 using wistream_view = basic_istream_view<_Val, wchar_t>;
1020 template<
typename _Tp,
typename _Up>
1021 concept __can_istream_view =
requires (_Up __e) {
1022 basic_istream_view<_Tp, typename _Up::char_type, typename _Up::traits_type>(__e);
1026 template<
typename _Tp>
1029 template<
typename _CharT,
typename _Traits>
1031 operator() [[nodiscard]] (basic_istream<_CharT, _Traits>& __e)
const
1033 {
return basic_istream_view<_Tp, _CharT, _Traits>(__e); }
1036 template<
typename _Tp>
1037 inline constexpr _Istream<_Tp> istream;
1045 template<
typename _Tp,
int _Disc>
1054 template<
bool _Present,
typename _Tp,
int _Disc = 0>
1055 using __maybe_present_t = __conditional_t<_Present, _Tp, _Absent<_Tp, _Disc>>;
1058 template<
bool _Const,
typename _Tp>
1059 using __maybe_const_t = __conditional_t<_Const, const _Tp, _Tp>;
1064using __detail::__maybe_const_t;
1066namespace views::__adaptor
1069 template<
typename _Adaptor,
typename... _Args>
1070 concept __adaptor_invocable
1075 template<
typename _Adaptor,
typename... _Args>
1076 concept __adaptor_partial_app_viable = (_Adaptor::_S_arity > 1)
1077 && (
sizeof...(_Args) == _Adaptor::_S_arity - 1)
1078 && (constructible_from<decay_t<_Args>, _Args> && ...);
1080 template<
typename _Adaptor,
typename... _Args>
1083 template<
typename _Lhs,
typename _Rhs>
1091 template<
typename _Derived>
1092 struct _RangeAdaptorClosure;
1094 template<
typename _Tp,
typename _Up>
1095 requires (!same_as<_Tp, _RangeAdaptorClosure<_Up>>)
1096 void __is_range_adaptor_closure_fn
1097 (
const _Tp&,
const _RangeAdaptorClosure<_Up>&);
1099 template<
typename _Tp>
1100 concept __is_range_adaptor_closure
1101 =
requires (_Tp __t) { __adaptor::__is_range_adaptor_closure_fn(__t, __t); };
1103#pragma GCC diagnostic push
1104#pragma GCC diagnostic ignored "-Wdangling-reference"
1106 template<
typename _Self,
typename _Range>
1107 requires __is_range_adaptor_closure<_Self>
1108 && __adaptor_invocable<_Self, _Range>
1110 operator|(_Range&& __r, _Self&& __self)
1115 template<
typename _Lhs,
typename _Rhs>
1116 requires __is_range_adaptor_closure<_Lhs>
1117 && __is_range_adaptor_closure<_Rhs>
1119 operator|(_Lhs&& __lhs, _Rhs&& __rhs)
1124#pragma GCC diagnostic pop
1126 template<
typename _Derived>
1127 struct _RangeAdaptorClosure
1133 template<
typename _Self,
typename _Range>
1134 requires __is_range_adaptor_closure<_Self>
1135 && __adaptor_invocable<_Self, _Range>
1136 friend constexpr auto
1137 operator|(_Range&& __r, _Self&& __self);
1139 template<
typename _Lhs,
typename _Rhs>
1140 requires __is_range_adaptor_closure<_Lhs>
1141 && __is_range_adaptor_closure<_Rhs>
1142 friend constexpr auto
1143 operator|(_Lhs&& __lhs, _Rhs&& __rhs);
1157 template<
typename _Derived>
1158 struct _RangeAdaptor
1162 template<
typename... _Args>
1163 requires __adaptor_partial_app_viable<_Derived, _Args...>
1165 operator()(_Args&&... __args)
const
1174 template<
typename _Adaptor>
1175 concept __closure_has_simple_call_op = _Adaptor::_S_has_simple_call_op;
1179 template<
typename _Adaptor,
typename... _Args>
1180 concept __adaptor_has_simple_extra_args = _Adaptor::_S_has_simple_extra_args
1181 || _Adaptor::template _S_has_simple_extra_args<_Args...>;
1185 template<
typename _Adaptor,
typename... _Args>
1186 struct _Partial : _RangeAdaptorClosure<_Partial<_Adaptor, _Args...>>
1188 using _Binder = _Bind_back_t<_Adaptor, _Args...>;
1189 [[no_unique_address]] _Binder _M_binder;
1193 template<
typename... _Ts>
1195 _Partial(
int, _Ts&&... __args)
1196 : _M_binder(0, _Adaptor(), std::
forward<_Ts>(__args)...)
1201#if _GLIBCXX_EXPLICIT_THIS_PARAMETER
1202# pragma GCC diagnostic push
1203# pragma GCC diagnostic ignored "-Wc++23-extensions"
1204 template<
typename _Self,
typename _Range>
1205 requires __adaptor_invocable<_Adaptor, _Range, __like_t<_Self, _Args>...>
1207 operator()(
this _Self&& __self, _Range&& __r)
1209 return _Binder::_S_call(__like_t<_Self, _Partial>(__self)._M_binder,
1212# pragma GCC diagnostic pop
1214 template<
typename _Range>
1215 requires __adaptor_invocable<_Adaptor, _Range,
const _Args&...>
1217 operator()(_Range&& __r)
const &
1220 template<
typename _Range>
1221 requires __adaptor_invocable<_Adaptor, _Range, _Args...>
1223 operator()(_Range&& __r) &&
1226 template<
typename _Range>
1228 operator()(_Range&& __r)
const && =
delete;
1236 template<
typename _Adaptor,
typename... _Args>
1237 requires __adaptor_has_simple_extra_args<_Adaptor, _Args...>
1238 && (is_trivially_copy_constructible_v<_Args> && ...)
1239 struct _Partial<_Adaptor, _Args...> : _RangeAdaptorClosure<_Partial<_Adaptor, _Args...>>
1241 using _Binder = _Bind_back_t<_Adaptor, _Args...>;
1242 [[no_unique_address]] _Binder _M_binder;
1244 template<
typename... _Ts>
1246 _Partial(
int, _Ts&&... __args)
1247 : _M_binder(0, _Adaptor(), std::
forward<_Ts>(__args)...)
1252 template<
typename _Range>
1253 requires __adaptor_invocable<_Adaptor, _Range,
const _Args&...>
1255 operator()(_Range&& __r)
const
1258 static constexpr bool _S_has_simple_call_op =
true;
1261 template<
typename _Lhs,
typename _Rhs,
typename _Range>
1262 concept __pipe_invocable
1267 template<
typename _Lhs,
typename _Rhs>
1268 struct _Pipe : _RangeAdaptorClosure<_Pipe<_Lhs, _Rhs>>
1270 [[no_unique_address]] _Lhs _M_lhs;
1271 [[no_unique_address]] _Rhs _M_rhs;
1273 template<
typename _Tp,
typename _Up>
1275 _Pipe(_Tp&& __lhs, _Up&& __rhs)
1276 : _M_lhs(std::
forward<_Tp>(__lhs)), _M_rhs(std::
forward<_Up>(__rhs))
1281#if _GLIBCXX_EXPLICIT_THIS_PARAMETER
1282# pragma GCC diagnostic push
1283# pragma GCC diagnostic ignored "-Wc++23-extensions"
1284 template<
typename _Self,
typename _Range>
1285 requires __pipe_invocable<__like_t<_Self, _Lhs>, __like_t<_Self, _Rhs>, _Range>
1287 operator()(
this _Self&& __self, _Range&& __r)
1289 return (__like_t<_Self, _Pipe>(__self)._M_rhs
1290 (__like_t<_Self, _Pipe>(__self)._M_lhs
1293# pragma GCC diagnostic pop
1295 template<
typename _Range>
1296 requires __pipe_invocable<const _Lhs&, const _Rhs&, _Range>
1298 operator()(_Range&& __r)
const &
1301 template<
typename _Range>
1302 requires __pipe_invocable<_Lhs, _Rhs, _Range>
1304 operator()(_Range&& __r) &&
1307 template<
typename _Range>
1309 operator()(_Range&& __r)
const && =
delete;
1317 template<
typename _Lhs,
typename _Rhs>
1318 requires __closure_has_simple_call_op<_Lhs>
1319 && __closure_has_simple_call_op<_Rhs>
1320 struct _Pipe<_Lhs, _Rhs> : _RangeAdaptorClosure<_Pipe<_Lhs, _Rhs>>
1322 [[no_unique_address]] _Lhs _M_lhs;
1323 [[no_unique_address]] _Rhs _M_rhs;
1325 template<
typename _Tp,
typename _Up>
1327 _Pipe(_Tp&& __lhs, _Up&& __rhs)
1328 : _M_lhs(std::
forward<_Tp>(__lhs)), _M_rhs(std::
forward<_Up>(__rhs))
1331 template<
typename _Range>
1332 requires __pipe_invocable<const _Lhs&, const _Rhs&, _Range>
1334 operator()(_Range&& __r)
const
1337 static constexpr bool _S_has_simple_call_op =
true;
1341#if __cpp_lib_ranges >= 202202L
1343 template<
typename _Derived>
1344 requires is_class_v<_Derived> && same_as<_Derived, remove_cv_t<_Derived>>
1345 class range_adaptor_closure
1346 :
public views::__adaptor::_RangeAdaptorClosure<_Derived>
1350 template<range _Range>
requires is_object_v<_Range>
1356 static void _S_fun(_Range&);
1357 static void _S_fun(_Range&&) =
delete;
1360 template<__detail::__different_from<ref_view> _Tp>
1361 requires convertible_to<_Tp, _Range&>
1373 constexpr iterator_t<_Range>
1375 {
return ranges::begin(*_M_r); }
1377 constexpr sentinel_t<_Range>
1379 {
return ranges::end(*_M_r); }
1382 empty() const requires requires { ranges::empty(*_M_r); }
1383 {
return ranges::empty(*_M_r); }
1386 size() const requires sized_range<_Range>
1387 {
return ranges::size(*_M_r); }
1390 data() const requires contiguous_range<_Range>
1391 {
return ranges::data(*_M_r); }
1394 template<
typename _Range>
1395 ref_view(_Range&) -> ref_view<_Range>;
1397 template<
typename _Tp>
1398 inline constexpr bool enable_borrowed_range<ref_view<_Tp>> =
true;
1400 template<range _Range>
1401 requires movable<_Range>
1402 && (!__detail::__is_initializer_list<remove_cv_t<_Range>>)
1406 _Range _M_r = _Range();
1409 owning_view()
requires default_initializable<_Range> = default;
1412 owning_view(_Range&& __t)
1413 noexcept(is_nothrow_move_constructible_v<_Range>)
1414 : _M_r(std::move(__t))
1417 owning_view(owning_view&&) =
default;
1418 owning_view& operator=(owning_view&&) =
default;
1424 constexpr const _Range&
1425 base() const& noexcept
1432 constexpr const _Range&&
1433 base() const&& noexcept
1436 constexpr iterator_t<_Range>
1438 {
return ranges::begin(_M_r); }
1440 constexpr sentinel_t<_Range>
1442 {
return ranges::end(_M_r); }
1446 {
return ranges::begin(_M_r); }
1449 end() const requires
range<const _Range>
1450 {
return ranges::end(_M_r); }
1453 empty()
requires requires { ranges::empty(_M_r); }
1454 {
return ranges::empty(_M_r); }
1457 empty() const requires requires { ranges::empty(_M_r); }
1458 {
return ranges::empty(_M_r); }
1462 {
return ranges::size(_M_r); }
1466 {
return ranges::size(_M_r); }
1470 {
return ranges::data(_M_r); }
1474 {
return ranges::data(_M_r); }
1477 template<
typename _Tp>
1478 inline constexpr bool enable_borrowed_range<owning_view<_Tp>>
1479 = enable_borrowed_range<_Tp>;
1485 template<
typename _Range>
1488 template<
typename _Range>
1492 struct _All : __adaptor::_RangeAdaptorClosure<_All>
1494 template<
typename _Range>
1495 static constexpr bool
1498 if constexpr (view<decay_t<_Range>>)
1500 else if constexpr (__detail::__can_ref_view<_Range>)
1506 template<viewable_range _Range>
1507 requires view<decay_t<_Range>>
1508 || __detail::__can_ref_view<_Range>
1509 || __detail::__can_owning_view<_Range>
1511 operator() [[nodiscard]] (_Range&& __r)
const
1512 noexcept(_S_noexcept<_Range>())
1514 if constexpr (view<decay_t<_Range>>)
1516 else if constexpr (__detail::__can_ref_view<_Range>)
1522 static constexpr bool _S_has_simple_call_op =
true;
1525 inline constexpr _All all;
1527 template<viewable_range _Range>
1533 template<
typename _Tp>
1534 struct __non_propagating_cache
1542 template<
typename _Tp>
1543 requires is_object_v<_Tp>
1544 struct __non_propagating_cache<_Tp>
1545 :
protected _Optional_base<_Tp>
1547 __non_propagating_cache() =
default;
1550 __non_propagating_cache(
const __non_propagating_cache&)
noexcept
1554 __non_propagating_cache(__non_propagating_cache&& __other)
noexcept
1555 { __other._M_reset(); }
1557 constexpr __non_propagating_cache&
1558 operator=(
const __non_propagating_cache& __other)
noexcept
1565 constexpr __non_propagating_cache&
1566 operator=(__non_propagating_cache&& __other)
noexcept
1573 constexpr __non_propagating_cache&
1574 operator=(_Tp __val)
1577 this->_M_payload._M_construct(
std::move(__val));
1582 operator bool() const noexcept
1583 {
return this->_M_is_engaged(); }
1587 {
return this->_M_get(); }
1589 constexpr const _Tp&
1591 {
return this->_M_get(); }
1593 template<
typename _Iter>
1595 _M_emplace_deref(
const _Iter& __i)
1598 auto __f = [] (
auto& __x) {
return *__x; };
1599 this->_M_payload._M_apply(_Optional_func{__f}, __i);
1600 return this->_M_get();
1603 using _Optional_base<_Tp>::_M_reset;
1606 template<range _Range>
1607 struct _CachedPosition
1610 _M_has_value()
const
1613 constexpr iterator_t<_Range>
1614 _M_get(
const _Range&)
const
1616 __glibcxx_assert(
false);
1617 __builtin_unreachable();
1621 _M_set(
const _Range&,
const iterator_t<_Range>&)
const
1625 template<forward_range _Range>
1626 struct _CachedPosition<_Range>
1627 :
protected __non_propagating_cache<iterator_t<_Range>>
1630 _M_has_value()
const
1631 {
return this->_M_is_engaged(); }
1633 constexpr iterator_t<_Range>
1634 _M_get(
const _Range&)
const
1636 __glibcxx_assert(_M_has_value());
1641 _M_set(
const _Range&,
const iterator_t<_Range>& __it)
1643 __glibcxx_assert(!_M_has_value());
1646 this->_M_payload._M_engaged =
true;
1650 template<random_access_range _Range>
1651 struct _CachedPosition<_Range>
1654 range_difference_t<_Range> _M_offset = -1;
1657 _CachedPosition() =
default;
1660 _CachedPosition(
const _CachedPosition&) =
default;
1663 _CachedPosition(_CachedPosition&& __other)
noexcept
1666 constexpr _CachedPosition&
1667 operator=(
const _CachedPosition&) =
default;
1669 constexpr _CachedPosition&
1670 operator=(_CachedPosition&& __other)
noexcept
1673 _M_offset = __other._M_offset;
1674 __other._M_offset = -1;
1679 _M_has_value()
const
1680 {
return _M_offset >= 0; }
1682 constexpr iterator_t<_Range>
1683 _M_get(_Range& __r)
const
1685 __glibcxx_assert(_M_has_value());
1686 return ranges::begin(__r) + _M_offset;
1690 _M_set(_Range& __r,
const iterator_t<_Range>& __it)
1692 __glibcxx_assert(!_M_has_value());
1693 _M_offset = __it - ranges::begin(__r);
1700 template<
typename _Base>
1701 struct __filter_view_iter_cat
1704 template<forward_range _Base>
1705 struct __filter_view_iter_cat<_Base>
1711 using _Cat =
typename iterator_traits<iterator_t<_Base>>::iterator_category;
1712 if constexpr (derived_from<_Cat, bidirectional_iterator_tag>)
1713 return bidirectional_iterator_tag{};
1714 else if constexpr (derived_from<_Cat, forward_iterator_tag>)
1715 return forward_iterator_tag{};
1720 using iterator_category =
decltype(_S_iter_cat());
1725 indirect_unary_predicate<iterator_t<_Vp>> _Pred>
1726 requires view<_Vp> && is_object_v<_Pred>
1727 class filter_view :
public view_interface<filter_view<_Vp, _Pred>>
1730 template<
bool _Const>
1733 template<
bool _Const>
1734 struct _Iterator : __detail::__filter_view_iter_cat<_Vp>
1737 static constexpr auto
1740 if constexpr (_Const)
1741 return input_iterator_tag{};
1742 else if constexpr (bidirectional_range<_Vp>)
1743 return bidirectional_iterator_tag{};
1744 else if constexpr (forward_range<_Vp>)
1745 return forward_iterator_tag{};
1747 return input_iterator_tag{};
1751 friend _Iterator<!_Const>;
1753 using _Parent = __maybe_const_t<_Const, filter_view>;
1754 using _Base = __maybe_const_t<_Const, _Vp>;
1755 using _Base_iter = iterator_t<_Base>;
1757 _Base_iter _M_current = _Base_iter();
1758 _Parent* _M_parent =
nullptr;
1761 using iterator_concept =
decltype(_S_iter_concept());
1763 using value_type = range_value_t<_Base>;
1764 using difference_type = range_difference_t<_Base>;
1766 _Iterator()
requires default_initializable<_Base_iter> = default;
1769 _Iterator(_Parent* __parent, _Base_iter __current)
1770 : _M_current(std::move(__current)),
1775 _Iterator(_Iterator<!_Const> __i)
1776 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
1777 : _M_current(
std::move(__i._M_current)),
1781 constexpr const _Base_iter&
1782 base() const & noexcept
1783 {
return _M_current; }
1785 constexpr _Base_iter
1789 constexpr range_reference_t<_Base>
1791 {
return *_M_current; }
1793 constexpr _Base_iter
1795 requires __detail::__has_arrow<_Base_iter>
1796 && copyable<_Base_iter>
1797 {
return _M_current; }
1799 constexpr _Iterator&
1802 _M_current = ranges::find_if(
std::move(++_M_current),
1803 ranges::end(_M_parent->_M_base),
1813 operator++(
int)
requires forward_range<_Base>
1820 constexpr _Iterator&
1821 operator--()
requires bidirectional_range<_Base>
1830 operator--(
int)
requires bidirectional_range<_Base>
1837 friend constexpr bool
1838 operator==(
const _Iterator& __x,
const _Iterator& __y)
1839 requires equality_comparable<_Base_iter>
1840 {
return __x._M_current == __y._M_current; }
1842 friend constexpr range_rvalue_reference_t<_Base>
1843 iter_move(
const _Iterator& __i)
1844 noexcept(
noexcept(ranges::iter_move(__i._M_current)))
1845 {
return ranges::iter_move(__i._M_current); }
1847 friend constexpr void
1848 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
1849 noexcept(
noexcept(ranges::iter_swap(__x._M_current, __y._M_current)))
1850 requires indirectly_swappable<_Base_iter>
1851 { ranges::iter_swap(__x._M_current, __y._M_current); }
1854 template<
bool _Const>
1858 using _Parent = __maybe_const_t<_Const, filter_view>;
1859 using _Base = __maybe_const_t<_Const, _Vp>;
1860 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
1862 friend _Sentinel<!_Const>;
1865 _Sentinel() =
default;
1868 _Sentinel(_Parent* __parent)
1869 : _M_end(ranges::end(__parent->_M_base))
1873 _Sentinel(_Sentinel<!_Const> __i)
1874 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
1878 constexpr sentinel_t<_Base>
1882 template<
bool _Const2>
1883 requires sentinel_for<sentinel_t<_Base>,
1884 iterator_t<__maybe_const_t<_Const2, _Vp>>>
1885 friend constexpr bool
1886 operator==(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
1887 {
return __x._M_current == __y._M_end; }
1890 _Vp _M_base = _Vp();
1891 [[no_unique_address]] __detail::__box<_Pred> _M_pred;
1892 [[no_unique_address]] __detail::_CachedPosition<_Vp> _M_cached_begin;
1895 filter_view()
requires (default_initializable<_Vp>
1896 && default_initializable<_Pred>)
1900 filter_view(_Vp __base, _Pred __pred)
1901 : _M_base(std::move(
__base)), _M_pred(std::move(__pred))
1905 base() const& requires copy_constructible<_Vp>
1912 constexpr const _Pred&
1914 {
return *_M_pred; }
1916 constexpr _Iterator<false>
1919 if (_M_cached_begin._M_has_value())
1920 return {
this, _M_cached_begin._M_get(_M_base)};
1922 __glibcxx_assert(_M_pred.has_value());
1923 auto __it = ranges::find_if(ranges::begin(_M_base),
1924 ranges::end(_M_base),
1926 _M_cached_begin._M_set(_M_base, __it);
1930 constexpr _Iterator<true>
1932 requires (input_range<const _Vp> && !forward_range<const _Vp>
1933 && indirect_unary_predicate<const _Pred, iterator_t<const _Vp>>)
1935 __glibcxx_assert(_M_pred.has_value());
1936 auto __it = ranges::find_if(ranges::begin(_M_base),
1937 ranges::end(_M_base),
1945 if constexpr (common_range<_Vp>)
1946 return _Iterator<false>{
this, ranges::end(_M_base)};
1948 return _Sentinel<false>{
this};
1951 constexpr _Sentinel<true>
1953 requires (input_range<const _Vp> && !forward_range<const _Vp>
1954 && indirect_unary_predicate<const _Pred, iterator_t<const _Vp>>)
1955 {
return _Sentinel<true>{
this}; }
1958 template<
typename _Range,
typename _Pred>
1959 filter_view(_Range&&, _Pred) -> filter_view<views::all_t<_Range>, _Pred>;
1965 template<
typename _Range,
typename _Pred>
1966 concept __can_filter_view
1970 struct _Filter : __adaptor::_RangeAdaptor<_Filter>
1972 template<viewable_range _Range,
typename _Pred>
1973 requires __detail::__can_filter_view<_Range, _Pred>
1975 operator() [[nodiscard]] (_Range&& __r, _Pred&& __p)
const
1980 using _RangeAdaptor<_Filter>::operator();
1981 static constexpr int _S_arity = 2;
1982 static constexpr bool _S_has_simple_extra_args =
true;
1985 inline constexpr _Filter filter;
1988#if __cpp_lib_ranges >= 202207L
1989 template<input_range _Vp, move_constructible _Fp>
1991 template<input_range _Vp, copy_constructible _Fp>
1994 && regular_invocable<_Fp&, range_reference_t<_Vp>>
1995 && std::__detail::__can_reference<invoke_result_t<_Fp&,
1996 range_reference_t<_Vp>>>
1997 class transform_view :
public view_interface<transform_view<_Vp, _Fp>>
2000 template<
bool _Const>
2001 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
2003 template<
bool _Const>
2007 template<
bool _Const>
2008 requires forward_range<_Base<_Const>>
2009 struct __iter_cat<_Const>
2018 using _Base = transform_view::_Base<_Const>;
2019 using _Res = invoke_result_t<__maybe_const_t<_Const, _Fp>&,
2020 range_reference_t<_Base>>;
2023 if constexpr (is_reference_v<_Res>)
2026 =
typename iterator_traits<iterator_t<_Base>>::iterator_category;
2027 if constexpr (derived_from<_Cat, contiguous_iterator_tag>)
2028 return random_access_iterator_tag{};
2033 return input_iterator_tag{};
2036 using iterator_category =
decltype(_S_iter_cat());
2039 template<
bool _Const>
2042 template<
bool _Const>
2043 struct _Iterator : __iter_cat<_Const>
2046 using _Parent = __detail::__maybe_const_t<_Const, transform_view>;
2047 using _Base = transform_view::_Base<_Const>;
2048 using _Base_iter = iterator_t<_Base>;
2049 using _Func_handle = __detail::__func_handle_t<
2050 __detail::__maybe_const_t<_Const, _Fp>,
2056 if constexpr (random_access_range<_Base>)
2057 return random_access_iterator_tag{};
2058 else if constexpr (bidirectional_range<_Base>)
2059 return bidirectional_iterator_tag{};
2060 else if constexpr (forward_range<_Base>)
2061 return forward_iterator_tag{};
2063 return input_iterator_tag{};
2066 _Base_iter _M_current = _Base_iter();
2067 [[no_unique_address]] _Func_handle _M_fun;
2070 using iterator_concept =
decltype(_S_iter_concept());
2073 = remove_cvref_t<invoke_result_t<__maybe_const_t<_Const, _Fp>&,
2074 range_reference_t<_Base>>>;
2075 using difference_type = range_difference_t<_Base>;
2077 _Iterator()
requires default_initializable<_Base_iter> = default;
2080 _Iterator(_Func_handle __fun, _Base_iter __current)
2081 : _M_current(std::move(__current)), _M_fun(__fun)
2085 _Iterator(_Parent* __parent, _Base_iter __current)
2086 : _M_current(std::move(__current)), _M_fun(*__parent->_M_fun)
2090 _Iterator(_Iterator<!_Const> __i)
2092 && convertible_to<iterator_t<_Vp>, _Base_iter>
2093 : _M_current(
std::move(__i._M_current)), _M_fun(__i._M_fun)
2096 constexpr const _Base_iter&
2097 base() const & noexcept
2098 {
return _M_current; }
2100 constexpr _Base_iter
2104 constexpr decltype(
auto)
2106 noexcept(
noexcept(_M_fun._M_call_deref(_M_current)))
2107 {
return _M_fun._M_call_deref(_M_current); }
2109 constexpr _Iterator&
2121 operator++(
int)
requires forward_range<_Base>
2128 constexpr _Iterator&
2129 operator--()
requires bidirectional_range<_Base>
2136 operator--(
int)
requires bidirectional_range<_Base>
2143 constexpr _Iterator&
2144 operator+=(difference_type __n)
requires random_access_range<_Base>
2150 constexpr _Iterator&
2151 operator-=(difference_type __n)
requires random_access_range<_Base>
2157 constexpr decltype(
auto)
2158 operator[](difference_type __n)
const
2159 requires random_access_range<_Base>
2160 {
return _M_fun._M_call_subscript(__n, _M_current); }
2162 friend constexpr bool
2163 operator==(
const _Iterator& __x,
const _Iterator& __y)
2164 requires equality_comparable<_Base_iter>
2165 {
return __x._M_current == __y._M_current; }
2167 friend constexpr bool
2168 operator<(
const _Iterator& __x,
const _Iterator& __y)
2169 requires random_access_range<_Base>
2170 {
return __x._M_current < __y._M_current; }
2172 friend constexpr bool
2173 operator>(
const _Iterator& __x,
const _Iterator& __y)
2174 requires random_access_range<_Base>
2175 {
return __y < __x; }
2177 friend constexpr bool
2178 operator<=(
const _Iterator& __x,
const _Iterator& __y)
2179 requires random_access_range<_Base>
2180 {
return !(__y < __x); }
2182 friend constexpr bool
2183 operator>=(
const _Iterator& __x,
const _Iterator& __y)
2184 requires random_access_range<_Base>
2185 {
return !(__x < __y); }
2187#ifdef __cpp_lib_three_way_comparison
2188 friend constexpr auto
2189 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
2190 requires random_access_range<_Base>
2191 && three_way_comparable<_Base_iter>
2192 {
return __x._M_current <=> __y._M_current; }
2195 friend constexpr _Iterator
2196 operator+(_Iterator __i, difference_type __n)
2197 requires random_access_range<_Base>
2198 {
return {__i._M_fun, __i._M_current + __n}; }
2200 friend constexpr _Iterator
2201 operator+(difference_type __n, _Iterator __i)
2202 requires random_access_range<_Base>
2203 {
return {__i._M_fun, __i._M_current + __n}; }
2205 friend constexpr _Iterator
2206 operator-(_Iterator __i, difference_type __n)
2207 requires random_access_range<_Base>
2208 {
return {__i._M_fun, __i._M_current - __n}; }
2212 friend constexpr difference_type
2213 operator-(
const _Iterator& __x,
const _Iterator& __y)
2214 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
2215 {
return __x._M_current - __y._M_current; }
2217 friend constexpr decltype(
auto)
2218 iter_move(
const _Iterator& __i)
noexcept(
noexcept(*__i))
2220 if constexpr (is_lvalue_reference_v<
decltype(*__i)>)
2226 friend _Iterator<!_Const>;
2227 template<
bool>
friend struct _Sentinel;
2230 template<
bool _Const>
2234 using _Parent = __detail::__maybe_const_t<_Const, transform_view>;
2235 using _Base = transform_view::_Base<_Const>;
2237 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
2240 _Sentinel() =
default;
2243 _Sentinel(sentinel_t<_Base> __end)
2248 _Sentinel(_Sentinel<!_Const> __i)
2250 && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
2254 constexpr sentinel_t<_Base>
2258 template<
bool _Const2>
2259 requires sentinel_for<sentinel_t<_Base>,
2260 iterator_t<__detail::__maybe_const_t<_Const2, _Vp>>>
2261 friend constexpr bool
2262 operator==(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
2263 {
return __x._M_current == __y._M_end; }
2265 template<
bool _Const2,
2266 typename _Base2 = __detail::__maybe_const_t<_Const2, _Vp>>
2267 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
2268 friend constexpr range_difference_t<_Base2>
2269 operator-(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
2270 {
return -(__y._M_end - __x._M_current); }
2272 template<
bool _Const2,
2273 typename _Base2 = __detail::__maybe_const_t<_Const2, _Vp>>
2274 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
2275 friend constexpr range_difference_t<_Base2>
2276 operator-(
const _Sentinel& __y,
const _Iterator<_Const2>& __x)
2277 {
return __y._M_end - __x._M_current; }
2279 friend _Sentinel<!_Const>;
2282 _Vp _M_base = _Vp();
2283 [[no_unique_address]] __detail::__box<_Fp> _M_fun;
2286 transform_view()
requires (default_initializable<_Vp>
2287 && default_initializable<_Fp>)
2291 transform_view(_Vp __base, _Fp __fun)
2292 : _M_base(std::move(
__base)), _M_fun(std::move(__fun))
2296 base() const& requires copy_constructible<_Vp>
2297 {
return _M_base ; }
2303 constexpr _Iterator<false>
2305 {
return _Iterator<false>{
this, ranges::begin(_M_base)}; }
2307 constexpr _Iterator<true>
2309 requires range<const _Vp>
2310 && regular_invocable<const _Fp&, range_reference_t<const _Vp>>
2311 {
return _Iterator<true>{
this, ranges::begin(_M_base)}; }
2313 constexpr _Sentinel<false>
2315 {
return _Sentinel<false>{ranges::end(_M_base)}; }
2317 constexpr _Iterator<false>
2318 end()
requires common_range<_Vp>
2319 {
return _Iterator<false>{
this, ranges::end(_M_base)}; }
2321 constexpr _Sentinel<true>
2323 requires range<const _Vp>
2324 && regular_invocable<const _Fp&, range_reference_t<const _Vp>>
2325 {
return _Sentinel<true>{ranges::end(_M_base)}; }
2327 constexpr _Iterator<true>
2329 requires common_range<const _Vp>
2330 && regular_invocable<const _Fp&, range_reference_t<const _Vp>>
2331 {
return _Iterator<true>{
this, ranges::end(_M_base)}; }
2334 size()
requires sized_range<_Vp>
2335 {
return ranges::size(_M_base); }
2338 size() const requires sized_range<const _Vp>
2339 {
return ranges::size(_M_base); }
2342 template<
typename _Range,
typename _Fp>
2343 transform_view(_Range&&, _Fp) -> transform_view<views::all_t<_Range>, _Fp>;
2349 template<
typename _Range,
typename _Fp>
2350 concept __can_transform_view
2354 struct _Transform : __adaptor::_RangeAdaptor<_Transform>
2356 template<viewable_range _Range,
typename _Fp>
2357 requires __detail::__can_transform_view<_Range, _Fp>
2359 operator() [[nodiscard]] (_Range&& __r, _Fp&& __f)
const
2364 using _RangeAdaptor<_Transform>::operator();
2365 static constexpr int _S_arity = 2;
2366 static constexpr bool _S_has_simple_extra_args =
true;
2369 inline constexpr _Transform transform;
2376 template<
bool _Const>
2377 using _CI = counted_iterator<
2378 iterator_t<__detail::__maybe_const_t<_Const, _Vp>>>;
2380 template<
bool _Const>
2384 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
2385 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
2388 _Sentinel() =
default;
2391 _Sentinel(sentinel_t<_Base> __end)
2396 _Sentinel(_Sentinel<!_Const> __s)
2397 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
2401 constexpr sentinel_t<_Base>
2405 friend constexpr bool
2406 operator==(
const _CI<_Const>& __y,
const _Sentinel& __x)
2407 {
return __y.count() == 0 || __y.base() == __x._M_end; }
2409 template<
bool _OtherConst = !_Const,
2410 typename _Base2 = __detail::__maybe_const_t<_OtherConst, _Vp>>
2411 requires sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
2412 friend constexpr bool
2413 operator==(
const _CI<_OtherConst>& __y,
const _Sentinel& __x)
2414 {
return __y.count() == 0 || __y.base() == __x._M_end; }
2416 friend _Sentinel<!_Const>;
2419 _Vp _M_base = _Vp();
2420 range_difference_t<_Vp> _M_count = 0;
2423 take_view()
requires default_initializable<_Vp> = default;
2426 take_view(_Vp __base, range_difference_t<_Vp> __count)
2427 : _M_base(std::move(__base)), _M_count(std::move(__count))
2431 base() const& requires copy_constructible<_Vp>
2439 begin()
requires (!__detail::__simple_view<_Vp>)
2441 if constexpr (sized_range<_Vp>)
2443 if constexpr (random_access_range<_Vp>)
2444 return ranges::begin(_M_base);
2448 return counted_iterator(ranges::begin(_M_base), __sz);
2452 return counted_iterator(ranges::begin(_M_base), _M_count);
2456 begin() const requires range<const _Vp>
2458 if constexpr (sized_range<const _Vp>)
2460 if constexpr (random_access_range<const _Vp>)
2461 return ranges::begin(_M_base);
2465 return counted_iterator(ranges::begin(_M_base), __sz);
2469 return counted_iterator(ranges::begin(_M_base), _M_count);
2473 end()
requires (!__detail::__simple_view<_Vp>)
2475 if constexpr (sized_range<_Vp>)
2477 if constexpr (random_access_range<_Vp>)
2478 return ranges::begin(_M_base) + size();
2483 return _Sentinel<false>{ranges::end(_M_base)};
2487 end() const requires range<const _Vp>
2489 if constexpr (sized_range<const _Vp>)
2491 if constexpr (random_access_range<const _Vp>)
2492 return ranges::begin(_M_base) + size();
2497 return _Sentinel<true>{ranges::end(_M_base)};
2501 size()
requires sized_range<_Vp>
2503 auto __n = ranges::size(_M_base);
2504 return std::min(__n,
static_cast<decltype(__n)
>(_M_count));
2508 size() const requires sized_range<const _Vp>
2510 auto __n = ranges::size(_M_base);
2511 return std::min(__n,
static_cast<decltype(__n)
>(_M_count));
2518 template<
typename _Range>
2519 take_view(_Range&&, range_difference_t<_Range>)
2520 -> take_view<views::all_t<_Range>>;
2522 template<
typename _Tp>
2523 inline constexpr bool enable_borrowed_range<take_view<_Tp>>
2524 = enable_borrowed_range<_Tp>;
2530 template<
typename _Range>
2531 inline constexpr bool __is_empty_view =
false;
2533 template<
typename _Tp>
2534 inline constexpr bool __is_empty_view<empty_view<_Tp>> =
true;
2536 template<
typename _Range>
2537 inline constexpr bool __is_basic_string_view =
false;
2539 template<
typename _CharT,
typename _Traits>
2540 inline constexpr bool __is_basic_string_view<basic_string_view<_CharT, _Traits>>
2543 using ranges::__detail::__is_subrange;
2545 template<
typename _Range>
2546 inline constexpr bool __is_iota_view =
false;
2548 template<
typename _Winc,
typename _Bound>
2549 inline constexpr bool __is_iota_view<iota_view<_Winc, _Bound>> =
true;
2551 template<
typename _Range>
2552 inline constexpr bool __is_repeat_view =
false;
2554 template<
typename _Range>
2556 __take_of_repeat_view(_Range&&, range_difference_t<_Range>);
2558 template<
typename _Range,
typename _Dp>
2559 concept __can_take_view
2563 struct _Take : __adaptor::_RangeAdaptor<_Take>
2565 template<viewable_range _Range,
typename _Dp = range_difference_t<_Range>>
2566 requires __detail::__can_take_view<_Range, _Dp>
2568 operator() [[nodiscard]] (_Range&& __r, type_identity_t<_Dp> __n)
const
2570 using _Tp = remove_cvref_t<_Range>;
2571 if constexpr (__detail::__is_empty_view<_Tp>)
2573#ifdef __cpp_lib_optional_range_support
2574 else if constexpr (__is_optional_v<_Tp> && view<_Tp>)
2577 else if constexpr (random_access_range<_Tp>
2579 && (std::__detail::__is_span<_Tp>
2580 || __detail::__is_basic_string_view<_Tp>
2581 || __detail::__is_subrange<_Tp>
2582 || __detail::__is_iota_view<_Tp>))
2585 auto __begin = ranges::begin(__r);
2586 auto __end = __begin + __n;
2587 if constexpr (std::__detail::__is_span<_Tp>)
2588 return span<typename _Tp::element_type>(__begin, __end);
2589 else if constexpr (__detail::__is_basic_string_view<_Tp>)
2590 return _Tp(__begin, __end);
2591 else if constexpr (__detail::__is_subrange<_Tp>)
2592 return subrange<iterator_t<_Tp>>(__begin, __end);
2594 return iota_view(*__begin, *__end);
2596 else if constexpr (__detail::__is_repeat_view<_Tp>)
2602 using _RangeAdaptor<_Take>::operator();
2603 static constexpr int _S_arity = 2;
2607 template<
typename _Tp>
2608 static constexpr bool _S_has_simple_extra_args
2609 = ranges::__detail::__is_integer_like<_Tp>;
2612 inline constexpr _Take take;
2615 template<view _Vp,
typename _Pred>
2617 && indirect_unary_predicate<const _Pred, iterator_t<_Vp>>
2618 class take_while_view :
public view_interface<take_while_view<_Vp, _Pred>>
2620 template<
bool _Const>
2624 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
2626 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
2627 const _Pred* _M_pred =
nullptr;
2630 _Sentinel() =
default;
2633 _Sentinel(sentinel_t<_Base> __end,
const _Pred* __pred)
2634 : _M_end(__end), _M_pred(__pred)
2638 _Sentinel(_Sentinel<!_Const> __s)
2639 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
2640 : _M_end(__s._M_end), _M_pred(__s._M_pred)
2643 constexpr sentinel_t<_Base>
2644 base()
const {
return _M_end; }
2646 friend constexpr bool
2647 operator==(
const iterator_t<_Base>& __x,
const _Sentinel& __y)
2648 {
return __y._M_end == __x || !
std::__invoke(*__y._M_pred, *__x); }
2650 template<
bool _OtherConst = !_Const,
2651 typename _Base2 = __detail::__maybe_const_t<_OtherConst, _Vp>>
2652 requires sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
2653 friend constexpr bool
2654 operator==(
const iterator_t<_Base2>& __x,
const _Sentinel& __y)
2655 {
return __y._M_end == __x || !
std::__invoke(*__y._M_pred, *__x); }
2657 friend _Sentinel<!_Const>;
2660 _Vp _M_base = _Vp();
2661 [[no_unique_address]] __detail::__box<_Pred> _M_pred;
2664 take_while_view()
requires (default_initializable<_Vp>
2665 && default_initializable<_Pred>)
2669 take_while_view(_Vp __base, _Pred __pred)
2670 : _M_base(std::move(
__base)), _M_pred(std::move(__pred))
2674 base() const& requires copy_constructible<_Vp>
2681 constexpr const _Pred&
2683 {
return *_M_pred; }
2686 begin()
requires (!__detail::__simple_view<_Vp>)
2687 {
return ranges::begin(_M_base); }
2690 begin() const requires range<const _Vp>
2691 && indirect_unary_predicate<const _Pred, iterator_t<const _Vp>>
2692 {
return ranges::begin(_M_base); }
2695 end()
requires (!__detail::__simple_view<_Vp>)
2696 {
return _Sentinel<false>(ranges::end(_M_base),
2700 end() const requires range<const _Vp>
2701 && indirect_unary_predicate<const _Pred, iterator_t<const _Vp>>
2702 {
return _Sentinel<true>(ranges::end(_M_base),
2706 template<
typename _Range,
typename _Pred>
2707 take_while_view(_Range&&, _Pred)
2708 -> take_while_view<views::all_t<_Range>, _Pred>;
2714 template<
typename _Range,
typename _Pred>
2715 concept __can_take_while_view
2719 struct _TakeWhile : __adaptor::_RangeAdaptor<_TakeWhile>
2721 template<viewable_range _Range,
typename _Pred>
2722 requires __detail::__can_take_while_view<_Range, _Pred>
2724 operator() [[nodiscard]] (_Range&& __r, _Pred&& __p)
const
2729 using _RangeAdaptor<_TakeWhile>::operator();
2730 static constexpr int _S_arity = 2;
2731 static constexpr bool _S_has_simple_extra_args =
true;
2734 inline constexpr _TakeWhile take_while;
2741 _Vp _M_base = _Vp();
2742 range_difference_t<_Vp> _M_count = 0;
2746 static constexpr bool _S_needs_cached_begin
2747 = !(random_access_range<const _Vp> && sized_range<const _Vp>);
2748 [[no_unique_address]]
2749 __detail::__maybe_present_t<_S_needs_cached_begin,
2750 __detail::_CachedPosition<_Vp>>
2754 drop_view()
requires default_initializable<_Vp> = default;
2757 drop_view(_Vp __base, range_difference_t<_Vp> __count)
2758 : _M_base(std::move(__base)), _M_count(__count)
2759 { __glibcxx_assert(__count >= 0); }
2762 base() const& requires copy_constructible<_Vp>
2772 requires (!(__detail::__simple_view<_Vp>
2773 && random_access_range<const _Vp>
2774 && sized_range<const _Vp>))
2776 if constexpr (_S_needs_cached_begin)
2777 if (_M_cached_begin._M_has_value())
2778 return _M_cached_begin._M_get(_M_base);
2780 auto __it = ranges::next(ranges::begin(_M_base),
2781 _M_count, ranges::end(_M_base));
2782 if constexpr (_S_needs_cached_begin)
2783 _M_cached_begin._M_set(_M_base, __it);
2791 requires random_access_range<const _Vp> && sized_range<const _Vp>
2793 return ranges::begin(_M_base) + ranges::min(ranges::distance(_M_base),
2798 end()
requires (!__detail::__simple_view<_Vp>)
2799 {
return ranges::end(_M_base); }
2802 end() const requires range<const _Vp>
2803 {
return ranges::end(_M_base); }
2806 size()
requires sized_range<_Vp>
2808 const auto __s = ranges::size(_M_base);
2809 const auto __c =
static_cast<decltype(__s)
>(_M_count);
2810 return __s < __c ? 0 : __s - __c;
2814 size() const requires sized_range<const _Vp>
2816 const auto __s = ranges::size(_M_base);
2817 const auto __c =
static_cast<decltype(__s)
>(_M_count);
2818 return __s < __c ? 0 : __s - __c;
2822 template<
typename _Range>
2823 drop_view(_Range&&, range_difference_t<_Range>)
2824 -> drop_view<views::all_t<_Range>>;
2826 template<
typename _Tp>
2827 inline constexpr bool enable_borrowed_range<drop_view<_Tp>>
2828 = enable_borrowed_range<_Tp>;
2834 template<
typename _Range>
2836 __drop_of_repeat_view(_Range&&, range_difference_t<_Range>);
2838 template<
typename _Range,
typename _Dp>
2839 concept __can_drop_view
2843 struct _Drop : __adaptor::_RangeAdaptor<_Drop>
2845 template<viewable_range _Range,
typename _Dp = range_difference_t<_Range>>
2846 requires __detail::__can_drop_view<_Range, _Dp>
2848 operator() [[nodiscard]] (_Range&& __r, type_identity_t<_Dp> __n)
const
2850 using _Tp = remove_cvref_t<_Range>;
2851 if constexpr (__detail::__is_empty_view<_Tp>)
2853#ifdef __cpp_lib_optional_range_support
2854 else if constexpr (__is_optional_v<_Tp> && view<_Tp>)
2857 else if constexpr (random_access_range<_Tp>
2859 && (std::__detail::__is_span<_Tp>
2860 || __detail::__is_basic_string_view<_Tp>
2861 || __detail::__is_iota_view<_Tp>
2862 || __detail::__is_subrange<_Tp>))
2865 auto __begin = ranges::begin(__r) + __n;
2866 auto __end = ranges::end(__r);
2867 if constexpr (std::__detail::__is_span<_Tp>)
2868 return span<typename _Tp::element_type>(__begin, __end);
2869 else if constexpr (__detail::__is_subrange<_Tp>)
2871 if constexpr (_Tp::_S_store_size)
2873 using ranges::__detail::__to_unsigned_like;
2874 auto __m = ranges::distance(__r) - __n;
2875 return _Tp(__begin, __end, __to_unsigned_like(__m));
2878 return _Tp(__begin, __end);
2881 return _Tp(__begin, __end);
2883 else if constexpr (__detail::__is_repeat_view<_Tp>)
2889 using _RangeAdaptor<_Drop>::operator();
2890 static constexpr int _S_arity = 2;
2891 template<
typename _Tp>
2892 static constexpr bool _S_has_simple_extra_args
2893 = _Take::_S_has_simple_extra_args<_Tp>;
2896 inline constexpr _Drop drop;
2899 template<view _Vp,
typename _Pred>
2901 && indirect_unary_predicate<const _Pred, iterator_t<_Vp>>
2902 class drop_while_view :
public view_interface<drop_while_view<_Vp, _Pred>>
2905 _Vp _M_base = _Vp();
2906 [[no_unique_address]] __detail::__box<_Pred> _M_pred;
2907 [[no_unique_address]] __detail::_CachedPosition<_Vp> _M_cached_begin;
2910 drop_while_view()
requires (default_initializable<_Vp>
2911 && default_initializable<_Pred>)
2915 drop_while_view(_Vp __base, _Pred __pred)
2916 : _M_base(std::move(
__base)), _M_pred(std::move(__pred))
2920 base() const& requires copy_constructible<_Vp>
2927 constexpr const _Pred&
2929 {
return *_M_pred; }
2934 if (_M_cached_begin._M_has_value())
2935 return _M_cached_begin._M_get(_M_base);
2937 __glibcxx_assert(_M_pred.has_value());
2938 auto __it = ranges::find_if_not(ranges::begin(_M_base),
2939 ranges::end(_M_base),
2941 _M_cached_begin._M_set(_M_base, __it);
2947 {
return ranges::end(_M_base); }
2950 template<
typename _Range,
typename _Pred>
2951 drop_while_view(_Range&&, _Pred)
2952 -> drop_while_view<views::all_t<_Range>, _Pred>;
2954 template<
typename _Tp,
typename _Pred>
2955 inline constexpr bool enable_borrowed_range<drop_while_view<_Tp, _Pred>>
2956 = enable_borrowed_range<_Tp>;
2962 template<
typename _Range,
typename _Pred>
2963 concept __can_drop_while_view
2967 struct _DropWhile : __adaptor::_RangeAdaptor<_DropWhile>
2969 template<viewable_range _Range,
typename _Pred>
2970 requires __detail::__can_drop_while_view<_Range, _Pred>
2972 operator() [[nodiscard]] (_Range&& __r, _Pred&& __p)
const
2978 using _RangeAdaptor<_DropWhile>::operator();
2979 static constexpr int _S_arity = 2;
2980 static constexpr bool _S_has_simple_extra_args =
true;
2983 inline constexpr _DropWhile drop_while;
2988 template<
typename _Tp>
2990 __as_lvalue(_Tp&& __t)
2991 {
return static_cast<_Tp&
>(__t); }
2994 template<input_range _Vp>
2999 using _InnerRange = range_reference_t<_Vp>;
3001 template<
bool _Const>
3002 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
3004 template<
bool _Const>
3005 using _Outer_iter = iterator_t<_Base<_Const>>;
3007 template<
bool _Const>
3008 using _Inner_iter = iterator_t<range_reference_t<_Base<_Const>>>;
3010 template<
bool _Const>
3011 static constexpr bool _S_ref_is_glvalue
3012 = is_reference_v<range_reference_t<_Base<_Const>>>;
3014 template<
bool _Const>
3018 template<
bool _Const>
3019 requires _S_ref_is_glvalue<_Const>
3020 && forward_range<_Base<_Const>>
3021 && forward_range<range_reference_t<_Base<_Const>>>
3022 struct __iter_cat<_Const>
3025 static constexpr auto
3028 using _Outer_iter = join_view::_Outer_iter<_Const>;
3029 using _Inner_iter = join_view::_Inner_iter<_Const>;
3030 using _OuterCat =
typename iterator_traits<_Outer_iter>::iterator_category;
3031 using _InnerCat =
typename iterator_traits<_Inner_iter>::iterator_category;
3032 if constexpr (derived_from<_OuterCat, bidirectional_iterator_tag>
3033 && derived_from<_InnerCat, bidirectional_iterator_tag>
3034 && common_range<range_reference_t<_Base<_Const>>>)
3035 return bidirectional_iterator_tag{};
3036 else if constexpr (derived_from<_OuterCat, forward_iterator_tag>
3037 && derived_from<_InnerCat, forward_iterator_tag>)
3038 return forward_iterator_tag{};
3040 return input_iterator_tag{};
3043 using iterator_category =
decltype(_S_iter_cat());
3046 template<
bool _Const>
3049 template<
bool _Const>
3050 struct _Iterator : __iter_cat<_Const>
3053 using _Parent = __detail::__maybe_const_t<_Const, join_view>;
3054 using _Base = join_view::_Base<_Const>;
3058 static constexpr bool _S_ref_is_glvalue
3059 = join_view::_S_ref_is_glvalue<_Const>;
3064 auto __update_inner = [
this] (
const iterator_t<_Base>& __x) ->
auto&& {
3065 if constexpr (_S_ref_is_glvalue)
3068 return _M_parent->_M_inner._M_emplace_deref(__x);
3071 _Outer_iter& __outer = _M_get_outer();
3072 for (; __outer != ranges::end(_M_parent->_M_base); ++__outer)
3074 auto&& __inner = __update_inner(__outer);
3075 _M_inner = ranges::begin(__inner);
3076 if (_M_inner != ranges::end(__inner))
3080 if constexpr (_S_ref_is_glvalue)
3082 if constexpr (forward_iterator<_Inner_iter>)
3083 _M_inner = _Inner_iter();
3089 static constexpr auto
3092 if constexpr (_S_ref_is_glvalue
3093 && bidirectional_range<_Base>
3094 && bidirectional_range<range_reference_t<_Base>>
3095 && common_range<range_reference_t<_Base>>)
3096 return bidirectional_iterator_tag{};
3097 else if constexpr (_S_ref_is_glvalue
3098 && forward_range<_Base>
3099 && forward_range<range_reference_t<_Base>>)
3100 return forward_iterator_tag{};
3102 return input_iterator_tag{};
3105 using _Outer_iter = join_view::_Outer_iter<_Const>;
3106 using _Inner_iter = join_view::_Inner_iter<_Const>;
3108 constexpr _Outer_iter&
3111 if constexpr (forward_range<_Base>)
3114 return *_M_parent->_M_outer;
3117 constexpr const _Outer_iter&
3118 _M_get_outer()
const
3120 if constexpr (forward_range<_Base>)
3123 return *_M_parent->_M_outer;
3126 constexpr _Inner_iter&
3127 _M_get_inner() noexcept
3129 if constexpr (forward_iterator<_Inner_iter>)
3135 constexpr const _Inner_iter&
3136 _M_get_inner() const noexcept
3138 if constexpr (forward_iterator<_Inner_iter>)
3145 _Iterator(_Parent* __parent, _Outer_iter __outer)
requires forward_range<_Base>
3146 : _M_outer(
std::move(__outer)), _M_parent(__parent)
3150 _Iterator(_Parent* __parent)
requires (!forward_range<_Base>)
3151 : _M_parent(__parent)
3154 [[no_unique_address]]
3155 __detail::__maybe_present_t<forward_range<_Base>, _Outer_iter> _M_outer
3156 =
decltype(_M_outer)();
3157 __conditional_t<forward_iterator<_Inner_iter>,
3158 _Inner_iter, optional<_Inner_iter>> _M_inner
3159 =
decltype(_M_inner)();
3160 _Parent* _M_parent =
nullptr;
3163 using iterator_concept =
decltype(_S_iter_concept());
3165 using value_type = range_value_t<range_reference_t<_Base>>;
3166 using difference_type
3168 range_difference_t<range_reference_t<_Base>>>;
3170 _Iterator() =
default;
3173 _Iterator(_Iterator<!_Const> __i)
3175 && convertible_to<iterator_t<_Vp>, _Outer_iter>
3176 && convertible_to<iterator_t<_InnerRange>, _Inner_iter>
3178 _M_parent(__i._M_parent)
3181 constexpr decltype(
auto)
3183 {
return *_M_get_inner(); }
3187 constexpr _Inner_iter
3189 requires __detail::__has_arrow<_Inner_iter>
3190 && copyable<_Inner_iter>
3191 {
return _M_get_inner(); }
3193 constexpr _Iterator&
3196 auto&& __inner_range = [
this] () ->
auto&& {
3197 if constexpr (_S_ref_is_glvalue)
3198 return *_M_get_outer();
3200 return *_M_parent->_M_inner;
3202 if (++_M_get_inner() == ranges::end(__inner_range))
3216 requires _S_ref_is_glvalue && forward_range<_Base>
3217 && forward_range<range_reference_t<_Base>>
3224 constexpr _Iterator&
3226 requires _S_ref_is_glvalue && bidirectional_range<_Base>
3227 && bidirectional_range<range_reference_t<_Base>>
3228 && common_range<range_reference_t<_Base>>
3230 if (_M_outer == ranges::end(_M_parent->_M_base))
3231 _M_inner = ranges::end(__detail::__as_lvalue(*--_M_outer));
3232 while (_M_get_inner() == ranges::begin(__detail::__as_lvalue(*_M_outer)))
3233 _M_get_inner() = ranges::end(__detail::__as_lvalue(*--_M_outer));
3240 requires _S_ref_is_glvalue && bidirectional_range<_Base>
3241 && bidirectional_range<range_reference_t<_Base>>
3242 && common_range<range_reference_t<_Base>>
3249 friend constexpr bool
3250 operator==(
const _Iterator& __x,
const _Iterator& __y)
3251 requires _S_ref_is_glvalue
3252 && forward_range<_Base>
3253 && equality_comparable<_Inner_iter>
3255 return (__x._M_outer == __y._M_outer
3256 && __x._M_inner == __y._M_inner);
3259 friend constexpr decltype(
auto)
3260 iter_move(
const _Iterator& __i)
3261 noexcept(
noexcept(ranges::iter_move(__i._M_get_inner())))
3262 {
return ranges::iter_move(__i._M_get_inner()); }
3264 friend constexpr void
3265 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
3266 noexcept(
noexcept(ranges::iter_swap(__x._M_get_inner(), __y._M_get_inner())))
3267 requires indirectly_swappable<_Inner_iter>
3268 {
return ranges::iter_swap(__x._M_get_inner(), __y._M_get_inner()); }
3270 friend _Iterator<!_Const>;
3271 template<
bool>
friend struct _Sentinel;
3274 template<
bool _Const>
3278 using _Parent = __detail::__maybe_const_t<_Const, join_view>;
3279 using _Base = join_view::_Base<_Const>;
3281 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
3284 _Sentinel() =
default;
3287 _Sentinel(_Parent* __parent)
3288 : _M_end(ranges::end(__parent->_M_base))
3292 _Sentinel(_Sentinel<!_Const> __s)
3293 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
3297 template<
bool _Const2>
3298 requires sentinel_for<sentinel_t<_Base>,
3299 iterator_t<__detail::__maybe_const_t<_Const2, _Vp>>>
3300 friend constexpr bool
3301 operator==(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
3302 {
return __x._M_get_outer() == __y._M_end; }
3304 friend _Sentinel<!_Const>;
3307 _Vp _M_base = _Vp();
3308 [[no_unique_address]]
3309 __detail::__maybe_present_t<!forward_range<_Vp>,
3310 __detail::__non_propagating_cache<iterator_t<_Vp>>> _M_outer;
3311 [[no_unique_address]]
3312 __detail::__non_propagating_cache<remove_cv_t<_InnerRange>> _M_inner;
3315 join_view()
requires default_initializable<_Vp> = default;
3318 join_view(_Vp __base)
3319 : _M_base(std::move(__base))
3323 base() const& requires copy_constructible<_Vp>
3333 if constexpr (forward_range<_Vp>)
3335 constexpr bool __use_const
3336 = (__detail::__simple_view<_Vp>
3337 && is_reference_v<range_reference_t<_Vp>>);
3338 return _Iterator<__use_const>{
this, ranges::begin(_M_base)};
3342 _M_outer = ranges::begin(_M_base);
3343 return _Iterator<false>{
this};
3349 requires forward_range<const _Vp>
3350 && is_reference_v<range_reference_t<const _Vp>>
3351 && input_range<range_reference_t<const _Vp>>
3353 return _Iterator<true>{
this, ranges::begin(_M_base)};
3359 if constexpr (forward_range<_Vp> && is_reference_v<_InnerRange>
3360 && forward_range<_InnerRange>
3361 && common_range<_Vp> && common_range<_InnerRange>)
3362 return _Iterator<__detail::__simple_view<_Vp>>{
this,
3363 ranges::end(_M_base)};
3365 return _Sentinel<__detail::__simple_view<_Vp>>{
this};
3370 requires forward_range<const _Vp>
3371 && is_reference_v<range_reference_t<const _Vp>>
3372 && input_range<range_reference_t<const _Vp>>
3374 if constexpr (is_reference_v<range_reference_t<const _Vp>>
3375 && forward_range<range_reference_t<const _Vp>>
3376 && common_range<const _Vp>
3377 && common_range<range_reference_t<const _Vp>>)
3378 return _Iterator<true>{
this, ranges::end(_M_base)};
3380 return _Sentinel<true>{
this};
3384 template<
typename _Range>
3385 explicit join_view(_Range&&) -> join_view<views::all_t<_Range>>;
3391 template<
typename _Range>
3392 concept __can_join_view
3396 struct _Join : __adaptor::_RangeAdaptorClosure<_Join>
3398 template<viewable_range _Range>
3399 requires __detail::__can_join_view<_Range>
3401 operator() [[nodiscard]] (_Range&& __r)
const
3408 static constexpr bool _S_has_simple_call_op =
true;
3411 inline constexpr _Join join;
3417 struct __require_constant;
3419 template<
typename _Range>
3420 concept __tiny_range = sized_range<_Range>
3422 {
typename __require_constant<remove_reference_t<_Range>::size()>; }
3423 && (remove_reference_t<_Range>::size() <= 1);
3425 template<
typename _Base>
3426 struct __lazy_split_view_outer_iter_cat
3429 template<forward_range _Base>
3430 struct __lazy_split_view_outer_iter_cat<_Base>
3431 {
using iterator_category = input_iterator_tag; };
3433 template<
typename _Base>
3434 struct __lazy_split_view_inner_iter_cat
3437 template<forward_range _Base>
3438 struct __lazy_split_view_inner_iter_cat<_Base>
3441 static constexpr auto
3444 using _Cat =
typename iterator_traits<iterator_t<_Base>>::iterator_category;
3445 if constexpr (derived_from<_Cat, forward_iterator_tag>)
3446 return forward_iterator_tag{};
3451 using iterator_category =
decltype(_S_iter_cat());
3455 template<input_range _Vp, forward_range _Pattern>
3457 && indirectly_comparable<iterator_t<_Vp>, iterator_t<_Pattern>,
3460 class lazy_split_view : public
view_interface<lazy_split_view<_Vp, _Pattern>>
3463 template<
bool _Const>
3464 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
3466 template<
bool _Const>
3469 template<
bool _Const>
3471 : __detail::__lazy_split_view_outer_iter_cat<_Base<_Const>>
3474 using _Parent = __detail::__maybe_const_t<_Const, lazy_split_view>;
3475 using _Base = lazy_split_view::_Base<_Const>;
3482 __current() noexcept
3484 if constexpr (forward_range<_Vp>)
3487 return *_M_parent->_M_current;
3491 __current() const noexcept
3493 if constexpr (forward_range<_Vp>)
3496 return *_M_parent->_M_current;
3499 _Parent* _M_parent =
nullptr;
3501 [[no_unique_address]]
3502 __detail::__maybe_present_t<forward_range<_Vp>,
3503 iterator_t<_Base>> _M_current
3504 =
decltype(_M_current)();
3505 bool _M_trailing_empty =
false;
3508 using iterator_concept = __conditional_t<forward_range<_Base>,
3509 forward_iterator_tag,
3510 input_iterator_tag>;
3512 using difference_type = range_difference_t<_Base>;
3514 struct value_type : view_interface<value_type>
3517 _OuterIter _M_i = _OuterIter();
3523 value_type(_OuterIter __i)
3524 : _M_i(std::move(__i))
3530 constexpr _InnerIter<_Const>
3532 {
return _InnerIter<_Const>{_M_i}; }
3534 constexpr default_sentinel_t
3535 end() const noexcept
3539 _OuterIter() =
default;
3542 _OuterIter(_Parent* __parent)
requires (!forward_range<_Base>)
3543 : _M_parent(__parent)
3547 _OuterIter(_Parent* __parent, iterator_t<_Base> __current)
3548 requires forward_range<_Base>
3549 : _M_parent(__parent),
3554 _OuterIter(_OuterIter<!_Const> __i)
3556 && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
3557 : _M_parent(__i._M_parent), _M_current(
std::move(__i._M_current)),
3558 _M_trailing_empty(__i._M_trailing_empty)
3561 constexpr value_type
3563 {
return value_type{*
this}; }
3565 constexpr _OuterIter&
3570 const auto __end = ranges::end(_M_parent->_M_base);
3571 if (__current() == __end)
3573 _M_trailing_empty =
false;
3576 const auto [__pbegin, __pend] = subrange{_M_parent->_M_pattern};
3577 if (__pbegin == __pend)
3579 else if constexpr (__detail::__tiny_range<_Pattern>)
3581 __current() = ranges::find(
std::move(__current()), __end,
3583 if (__current() != __end)
3586 if (__current() == __end)
3587 _M_trailing_empty =
true;
3594 = ranges::mismatch(__current(), __end, __pbegin, __pend);
3598 if (__current() == __end)
3599 _M_trailing_empty =
true;
3602 }
while (++__current() != __end);
3606 constexpr decltype(
auto)
3609 if constexpr (forward_range<_Base>)
3619 friend constexpr bool
3620 operator==(
const _OuterIter& __x,
const _OuterIter& __y)
3621 requires forward_range<_Base>
3623 return __x._M_current == __y._M_current
3624 && __x._M_trailing_empty == __y._M_trailing_empty;
3627 friend constexpr bool
3628 operator==(
const _OuterIter& __x, default_sentinel_t)
3630 return __x.__current() == ranges::end(__x._M_parent->_M_base)
3631 && !__x._M_trailing_empty;
3634 friend _OuterIter<!_Const>;
3635 friend _InnerIter<_Const>;
3638 template<
bool _Const>
3640 : __detail::__lazy_split_view_inner_iter_cat<_Base<_Const>>
3643 using _Base = lazy_split_view::_Base<_Const>;
3648 auto [__pcur, __pend] = subrange{_M_i._M_parent->_M_pattern};
3649 auto __end = ranges::end(_M_i._M_parent->_M_base);
3650 if constexpr (__detail::__tiny_range<_Pattern>)
3652 const auto& __cur = _M_i_current();
3655 if (__pcur == __pend)
3656 return _M_incremented;
3657 return *__cur == *__pcur;
3661 auto __cur = _M_i_current();
3664 if (__pcur == __pend)
3665 return _M_incremented;
3668 if (*__cur != *__pcur)
3670 if (++__pcur == __pend)
3672 }
while (++__cur != __end);
3678 _M_i_current() noexcept
3679 {
return _M_i.__current(); }
3682 _M_i_current() const noexcept
3683 {
return _M_i.__current(); }
3685 _OuterIter<_Const> _M_i = _OuterIter<_Const>();
3686 bool _M_incremented =
false;
3689 using iterator_concept
3690 =
typename _OuterIter<_Const>::iterator_concept;
3692 using value_type = range_value_t<_Base>;
3693 using difference_type = range_difference_t<_Base>;
3695 _InnerIter() =
default;
3698 _InnerIter(_OuterIter<_Const> __i)
3699 : _M_i(std::move(__i))
3702 constexpr const iterator_t<_Base>&
3703 base() const& noexcept
3704 {
return _M_i_current(); }
3706 constexpr iterator_t<_Base>
3707 base() &&
requires forward_range<_Vp>
3710 constexpr decltype(
auto)
3712 {
return *_M_i_current(); }
3714 constexpr _InnerIter&
3717 _M_incremented =
true;
3718 if constexpr (!forward_range<_Base>)
3719 if constexpr (_Pattern::size() == 0)
3725 constexpr decltype(
auto)
3728 if constexpr (forward_range<_Base>)
3738 friend constexpr bool
3739 operator==(
const _InnerIter& __x,
const _InnerIter& __y)
3740 requires forward_range<_Base>
3741 {
return __x._M_i == __y._M_i; }
3743 friend constexpr bool
3744 operator==(
const _InnerIter& __x, default_sentinel_t)
3745 {
return __x.__at_end(); }
3747 friend constexpr decltype(
auto)
3748 iter_move(
const _InnerIter& __i)
3749 noexcept(
noexcept(ranges::iter_move(__i._M_i_current())))
3750 {
return ranges::iter_move(__i._M_i_current()); }
3752 friend constexpr void
3753 iter_swap(
const _InnerIter& __x,
const _InnerIter& __y)
3754 noexcept(
noexcept(ranges::iter_swap(__x._M_i_current(),
3755 __y._M_i_current())))
3756 requires indirectly_swappable<iterator_t<_Base>>
3757 { ranges::iter_swap(__x._M_i_current(), __y._M_i_current()); }
3760 _Vp _M_base = _Vp();
3761 _Pattern _M_pattern = _Pattern();
3762 [[no_unique_address]]
3763 __detail::__maybe_present_t<!forward_range<_Vp>,
3764 __detail::__non_propagating_cache<iterator_t<_Vp>>> _M_current;
3768 lazy_split_view()
requires (default_initializable<_Vp>
3769 && default_initializable<_Pattern>)
3773 lazy_split_view(_Vp __base, _Pattern __pattern)
3774 : _M_base(std::move(
__base)), _M_pattern(std::move(__pattern))
3777 template<input_range _Range>
3778 requires constructible_from<_Vp, views::all_t<_Range>>
3779 && constructible_from<_Pattern, single_view<range_value_t<_Range>>>
3781 lazy_split_view(_Range&& __r, range_value_t<_Range> __e)
3782 : _M_base(views::all(std::
forward<_Range>(__r))),
3783 _M_pattern(views::single(std::move(__e)))
3787 base() const& requires copy_constructible<_Vp>
3797 if constexpr (forward_range<_Vp>)
3799 constexpr bool __simple
3800 = __detail::__simple_view<_Vp> && __detail::__simple_view<_Pattern>;
3801 return _OuterIter<__simple>{
this, ranges::begin(_M_base)};
3805 _M_current = ranges::begin(_M_base);
3806 return _OuterIter<false>{
this};
3814 begin() const requires forward_range<_Vp> && forward_range<const _Vp>
3815 && forward_range<const _Pattern>
3817 return _OuterIter<true>{
this, ranges::begin(_M_base)};
3821 end()
requires forward_range<_Vp> && common_range<_Vp>
3823 constexpr bool __simple
3824 = __detail::__simple_view<_Vp> && __detail::__simple_view<_Pattern>;
3825 return _OuterIter<__simple>{
this, ranges::end(_M_base)};
3831 if constexpr (forward_range<_Vp>
3832 && forward_range<const _Vp>
3833 && common_range<const _Vp>
3834 && forward_range<const _Pattern>)
3835 return _OuterIter<true>{
this, ranges::end(_M_base)};
3841 template<
typename _Range,
typename _Pattern>
3842 lazy_split_view(_Range&&, _Pattern&&)
3843 -> lazy_split_view<views::all_t<_Range>, views::all_t<_Pattern>>;
3845 template<input_range _Range>
3846 lazy_split_view(_Range&&, range_value_t<_Range>)
3853 template<
typename _Range,
typename _Pattern>
3854 concept __can_lazy_split_view
3858 struct _LazySplit : __adaptor::_RangeAdaptor<_LazySplit>
3860 template<viewable_range _Range,
typename _Pattern>
3861 requires __detail::__can_lazy_split_view<_Range, _Pattern>
3863 operator() [[nodiscard]] (_Range&& __r, _Pattern&& __f)
const
3868 using _RangeAdaptor<_LazySplit>::operator();
3869 static constexpr int _S_arity = 2;
3874 template<
typename _Pattern>
3875 static constexpr bool _S_has_simple_extra_args
3876 = is_scalar_v<_Pattern> || (view<_Pattern>
3877 && copy_constructible<_Pattern>);
3880 inline constexpr _LazySplit lazy_split;
3883 template<forward_range _Vp, forward_range _Pattern>
3885 && indirectly_comparable<iterator_t<_Vp>, iterator_t<_Pattern>,
3887 class split_view :
public view_interface<split_view<_Vp, _Pattern>>
3890 _Vp _M_base = _Vp();
3891 _Pattern _M_pattern = _Pattern();
3892 __detail::__non_propagating_cache<subrange<iterator_t<_Vp>>> _M_cached_begin;
3898 split_view()
requires (default_initializable<_Vp>
3899 && default_initializable<_Pattern>)
3903 split_view(_Vp __base, _Pattern __pattern)
3904 : _M_base(std::move(
__base)), _M_pattern(std::move(__pattern))
3907 template<forward_range _Range>
3908 requires constructible_from<_Vp, views::all_t<_Range>>
3909 && constructible_from<_Pattern, single_view<range_value_t<_Range>>>
3911 split_view(_Range&& __r, range_value_t<_Range> __e)
3912 : _M_base(views::all(std::
forward<_Range>(__r))),
3913 _M_pattern(views::single(std::move(__e)))
3917 base() const& requires copy_constructible<_Vp>
3927 if (!_M_cached_begin)
3928 _M_cached_begin = _M_find_next(ranges::begin(_M_base));
3929 return {
this, ranges::begin(_M_base), *_M_cached_begin};
3935 if constexpr (common_range<_Vp>)
3936 return _Iterator{
this, ranges::end(_M_base), {}};
3938 return _Sentinel{
this};
3941 constexpr subrange<iterator_t<_Vp>>
3942 _M_find_next(iterator_t<_Vp> __it)
3944 auto [__b, __e] = ranges::search(subrange(__it, ranges::end(_M_base)), _M_pattern);
3945 if (__b != ranges::end(_M_base) && ranges::empty(_M_pattern))
3957 split_view* _M_parent =
nullptr;
3958 iterator_t<_Vp> _M_cur = iterator_t<_Vp>();
3959 subrange<iterator_t<_Vp>> _M_next = subrange<iterator_t<_Vp>>();
3960 bool _M_trailing_empty =
false;
3962 friend struct _Sentinel;
3965 using iterator_concept = forward_iterator_tag;
3966 using iterator_category = input_iterator_tag;
3967 using value_type = subrange<iterator_t<_Vp>>;
3968 using difference_type = range_difference_t<_Vp>;
3970 _Iterator() =
default;
3973 _Iterator(split_view* __parent,
3974 iterator_t<_Vp> __current,
3975 subrange<iterator_t<_Vp>> __next)
3976 : _M_parent(__parent),
3977 _M_cur(std::move(__current)),
3978 _M_next(std::move(__next))
3981 constexpr iterator_t<_Vp>
3985 constexpr value_type
3987 {
return {_M_cur, _M_next.begin()}; }
3989 constexpr _Iterator&
3992 _M_cur = _M_next.begin();
3993 if (_M_cur != ranges::end(_M_parent->_M_base))
3995 _M_cur = _M_next.end();
3996 if (_M_cur == ranges::end(_M_parent->_M_base))
3998 _M_trailing_empty =
true;
3999 _M_next = {_M_cur, _M_cur};
4002 _M_next = _M_parent->_M_find_next(_M_cur);
4005 _M_trailing_empty =
false;
4017 friend constexpr bool
4018 operator==(
const _Iterator& __x,
const _Iterator& __y)
4020 return __x._M_cur == __y._M_cur
4021 && __x._M_trailing_empty == __y._M_trailing_empty;
4028 sentinel_t<_Vp> _M_end = sentinel_t<_Vp>();
4031 _Sentinel() =
default;
4034 _Sentinel(split_view* __parent)
4035 : _M_end(ranges::end(__parent->_M_base))
4038 friend constexpr bool
4039 operator==(
const _Iterator& __x,
const _Sentinel& __y)
4040 {
return __x._M_cur == __y._M_end && !__x._M_trailing_empty; }
4044 template<
typename _Range,
typename _Pattern>
4045 split_view(_Range&&, _Pattern&&)
4046 -> split_view<views::all_t<_Range>, views::all_t<_Pattern>>;
4048 template<forward_range _Range>
4049 split_view(_Range&&, range_value_t<_Range>)
4056 template<
typename _Range,
typename _Pattern>
4057 concept __can_split_view
4061 struct _Split : __adaptor::_RangeAdaptor<_Split>
4063 template<viewable_range _Range,
typename _Pattern>
4064 requires __detail::__can_split_view<_Range, _Pattern>
4066 operator() [[nodiscard]] (_Range&& __r, _Pattern&& __f)
const
4071 using _RangeAdaptor<_Split>::operator();
4072 static constexpr int _S_arity = 2;
4073 template<
typename _Pattern>
4074 static constexpr bool _S_has_simple_extra_args
4075 = _LazySplit::_S_has_simple_extra_args<_Pattern>;
4078 inline constexpr _Split split;
4085 template<input_or_output_iterator _Iter>
4087 operator() [[nodiscard]] (_Iter __i, iter_difference_t<_Iter> __n)
const
4089 if constexpr (contiguous_iterator<_Iter>)
4091 else if constexpr (random_access_iterator<_Iter>)
4092 return subrange(__i, __i + __n);
4094 return subrange(counted_iterator(
std::move(__i), __n),
4099 inline constexpr _Counted counted{};
4107 _Vp _M_base = _Vp();
4110 common_view()
requires default_initializable<_Vp> = default;
4113 common_view(_Vp __r)
4114 : _M_base(std::move(__r))
4118 base() const& requires copy_constructible<_Vp>
4128 begin()
requires (!__detail::__simple_view<_Vp>)
4131 return ranges::begin(_M_base);
4133 return common_iterator<iterator_t<_Vp>, sentinel_t<_Vp>>
4134 (ranges::begin(_M_base));
4141 return ranges::begin(_M_base);
4143 return common_iterator<iterator_t<const _Vp>, sentinel_t<const _Vp>>
4144 (ranges::begin(_M_base));
4148 end()
requires (!__detail::__simple_view<_Vp>)
4151 return ranges::begin(_M_base) + ranges::size(_M_base);
4153 return common_iterator<iterator_t<_Vp>, sentinel_t<_Vp>>
4154 (ranges::end(_M_base));
4158 end() const requires
range<const _Vp>
4161 return ranges::begin(_M_base) + ranges::size(_M_base);
4163 return common_iterator<iterator_t<const _Vp>, sentinel_t<const _Vp>>
4164 (ranges::end(_M_base));
4169 {
return ranges::size(_M_base); }
4173 {
return ranges::size(_M_base); }
4176 template<
typename _Range>
4177 common_view(_Range&&) -> common_view<views::all_t<_Range>>;
4179 template<
typename _Tp>
4180 inline constexpr bool enable_borrowed_range<common_view<_Tp>>
4181 = enable_borrowed_range<_Tp>;
4187 template<
typename _Range>
4188 concept __already_common = common_range<_Range>
4191 template<
typename _Range>
4192 concept __can_common_view
4196 struct _Common : __adaptor::_RangeAdaptorClosure<_Common>
4198 template<viewable_range _Range>
4199 requires __detail::__already_common<_Range>
4200 || __detail::__can_common_view<_Range>
4202 operator() [[nodiscard]] (_Range&& __r)
const
4204 if constexpr (__detail::__already_common<_Range>)
4210 static constexpr bool _S_has_simple_call_op =
true;
4213 inline constexpr _Common common;
4221 static constexpr bool _S_needs_cached_begin
4222 = !common_range<_Vp> && !(random_access_range<_Vp>
4223 && sized_sentinel_for<sentinel_t<_Vp>,
4226 _Vp _M_base = _Vp();
4227 [[no_unique_address]]
4228 __detail::__maybe_present_t<_S_needs_cached_begin,
4229 __detail::_CachedPosition<_Vp>>
4233 reverse_view()
requires default_initializable<_Vp> = default;
4236 reverse_view(_Vp __r)
4237 : _M_base(std::move(__r))
4241 base() const& requires copy_constructible<_Vp>
4248 constexpr reverse_iterator<iterator_t<_Vp>>
4251 if constexpr (_S_needs_cached_begin)
4252 if (_M_cached_begin._M_has_value())
4255 auto __it = ranges::next(ranges::begin(_M_base), ranges::end(_M_base));
4256 if constexpr (_S_needs_cached_begin)
4257 _M_cached_begin._M_set(_M_base, __it);
4262 begin()
requires common_range<_Vp>
4266 begin() const requires common_range<const _Vp>
4269 constexpr reverse_iterator<iterator_t<_Vp>>
4274 end() const requires common_range<const _Vp>
4278 size()
requires sized_range<_Vp>
4279 {
return ranges::size(_M_base); }
4282 size() const requires sized_range<const _Vp>
4283 {
return ranges::size(_M_base); }
4286 template<
typename _Range>
4287 reverse_view(_Range&&) -> reverse_view<views::all_t<_Range>>;
4289 template<
typename _Tp>
4290 inline constexpr bool enable_borrowed_range<reverse_view<_Tp>>
4291 = enable_borrowed_range<_Tp>;
4298 inline constexpr bool __is_reversible_subrange =
false;
4300 template<
typename _Iter, subrange_kind _Kind>
4301 inline constexpr bool
4302 __is_reversible_subrange<subrange<reverse_iterator<_Iter>,
4303 reverse_iterator<_Iter>,
4307 inline constexpr bool __is_reverse_view =
false;
4309 template<
typename _Vp>
4310 inline constexpr bool __is_reverse_view<reverse_view<_Vp>> =
true;
4312 template<
typename _Range>
4313 concept __can_reverse_view
4317 struct _Reverse : __adaptor::_RangeAdaptorClosure<_Reverse>
4319 template<viewable_range _Range>
4320 requires __detail::__is_reverse_view<remove_cvref_t<_Range>>
4321 || __detail::__is_reversible_subrange<remove_cvref_t<_Range>>
4322 || __detail::__can_reverse_view<_Range>
4324 operator() [[nodiscard]] (_Range&& __r)
const
4326 using _Tp = remove_cvref_t<_Range>;
4327 if constexpr (__detail::__is_reverse_view<_Tp>)
4329#ifdef __cpp_lib_optional_range_support
4330 else if constexpr (__is_optional_v<_Tp> && view<_Tp>)
4333 else if constexpr (__detail::__is_reversible_subrange<_Tp>)
4335 using _Iter =
decltype(ranges::begin(__r).base());
4336 if constexpr (sized_range<_Tp>)
4337 return subrange<_Iter, _Iter, subrange_kind::sized>
4338 {__r.end().base(), __r.begin().base(), __r.size()};
4340 return subrange<_Iter, _Iter, subrange_kind::unsized>
4341 {__r.end().base(), __r.begin().base()};
4347 static constexpr bool _S_has_simple_call_op =
true;
4350 inline constexpr _Reverse reverse;
4355#if __cpp_lib_tuple_like
4356 template<
typename _Tp,
size_t _Nm>
4357 concept __has_tuple_element = __tuple_like<_Tp> && _Nm < tuple_size_v<_Tp>;
4359 template<
typename _Tp,
size_t _Nm>
4360 concept __has_tuple_element =
requires(_Tp __t)
4362 typename tuple_size<_Tp>::type;
4363 requires _Nm < tuple_size_v<_Tp>;
4364 typename tuple_element_t<_Nm, _Tp>;
4365 { std::get<_Nm>(__t) }
4366 -> convertible_to<const tuple_element_t<_Nm, _Tp>&>;
4370 template<
typename _Tp,
size_t _Nm>
4371 concept __returnable_element
4372 = is_reference_v<_Tp> || move_constructible<tuple_element_t<_Nm, _Tp>>;
4375 template<input_range _Vp,
size_t _Nm>
4377 && __detail::__has_tuple_element<range_value_t<_Vp>, _Nm>
4378 && __detail::__has_tuple_element<remove_reference_t<range_reference_t<_Vp>>,
4380 && __detail::__returnable_element<range_reference_t<_Vp>, _Nm>
4381 class elements_view :
public view_interface<elements_view<_Vp, _Nm>>
4384 elements_view()
requires default_initializable<_Vp> = default;
4387 elements_view(_Vp __base)
4388 : _M_base(std::move(__base))
4392 base() const& requires copy_constructible<_Vp>
4400 begin()
requires (!__detail::__simple_view<_Vp>)
4401 {
return _Iterator<false>(ranges::begin(_M_base)); }
4404 begin() const requires range<const _Vp>
4405 {
return _Iterator<true>(ranges::begin(_M_base)); }
4408 end()
requires (!__detail::__simple_view<_Vp> && !common_range<_Vp>)
4409 {
return _Sentinel<false>{ranges::end(_M_base)}; }
4412 end()
requires (!__detail::__simple_view<_Vp> && common_range<_Vp>)
4413 {
return _Iterator<false>{ranges::end(_M_base)}; }
4416 end() const requires range<const _Vp>
4417 {
return _Sentinel<true>{ranges::end(_M_base)}; }
4420 end() const requires common_range<const _Vp>
4421 {
return _Iterator<true>{ranges::end(_M_base)}; }
4424 size()
requires sized_range<_Vp>
4425 {
return ranges::size(_M_base); }
4428 size() const requires sized_range<const _Vp>
4429 {
return ranges::size(_M_base); }
4432 template<
bool _Const>
4433 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
4435 template<
bool _Const>
4439 template<
bool _Const>
4440 requires forward_range<_Base<_Const>>
4441 struct __iter_cat<_Const>
4444 static auto _S_iter_cat()
4446 using _Base = elements_view::_Base<_Const>;
4447 using _Cat =
typename iterator_traits<iterator_t<_Base>>::iterator_category;
4448 using _Res =
decltype((std::get<_Nm>(*
std::declval<iterator_t<_Base>>())));
4449 if constexpr (!is_lvalue_reference_v<_Res>)
4450 return input_iterator_tag{};
4451 else if constexpr (derived_from<_Cat, random_access_iterator_tag>)
4452 return random_access_iterator_tag{};
4457 using iterator_category =
decltype(_S_iter_cat());
4460 template<
bool _Const>
4463 template<
bool _Const>
4464 struct _Iterator : __iter_cat<_Const>
4467 using _Base = elements_view::_Base<_Const>;
4469 iterator_t<_Base> _M_current = iterator_t<_Base>();
4471 static constexpr decltype(
auto)
4472 _S_get_element(
const iterator_t<_Base>& __i)
4474 if constexpr (is_reference_v<range_reference_t<_Base>>)
4475 return std::get<_Nm>(*__i);
4478 using _Et = remove_cv_t<tuple_element_t<_Nm, range_reference_t<_Base>>>;
4479 return static_cast<_Et
>(std::get<_Nm>(*__i));
4486 if constexpr (random_access_range<_Base>)
4487 return random_access_iterator_tag{};
4488 else if constexpr (bidirectional_range<_Base>)
4489 return bidirectional_iterator_tag{};
4490 else if constexpr (forward_range<_Base>)
4491 return forward_iterator_tag{};
4493 return input_iterator_tag{};
4496 friend _Iterator<!_Const>;
4499 using iterator_concept =
decltype(_S_iter_concept());
4502 = remove_cvref_t<tuple_element_t<_Nm, range_value_t<_Base>>>;
4503 using difference_type = range_difference_t<_Base>;
4505 _Iterator()
requires default_initializable<iterator_t<_Base>> = default;
4508 _Iterator(iterator_t<_Base> __current)
4509 : _M_current(std::move(__current))
4513 _Iterator(_Iterator<!_Const> __i)
4514 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
4518 constexpr const iterator_t<_Base>&
4519 base() const& noexcept
4520 {
return _M_current; }
4522 constexpr iterator_t<_Base>
4526 constexpr decltype(
auto)
4528 {
return _S_get_element(_M_current); }
4530 constexpr _Iterator&
4542 operator++(
int)
requires forward_range<_Base>
4549 constexpr _Iterator&
4550 operator--()
requires bidirectional_range<_Base>
4557 operator--(
int)
requires bidirectional_range<_Base>
4564 constexpr _Iterator&
4565 operator+=(difference_type __n)
4566 requires random_access_range<_Base>
4572 constexpr _Iterator&
4573 operator-=(difference_type __n)
4574 requires random_access_range<_Base>
4580 constexpr decltype(
auto)
4581 operator[](difference_type __n)
const
4582 requires random_access_range<_Base>
4583 {
return _S_get_element(_M_current + __n); }
4585 friend constexpr bool
4586 operator==(
const _Iterator& __x,
const _Iterator& __y)
4587 requires equality_comparable<iterator_t<_Base>>
4588 {
return __x._M_current == __y._M_current; }
4590 friend constexpr bool
4591 operator<(
const _Iterator& __x,
const _Iterator& __y)
4592 requires random_access_range<_Base>
4593 {
return __x._M_current < __y._M_current; }
4595 friend constexpr bool
4596 operator>(
const _Iterator& __x,
const _Iterator& __y)
4597 requires random_access_range<_Base>
4598 {
return __y._M_current < __x._M_current; }
4600 friend constexpr bool
4601 operator<=(
const _Iterator& __x,
const _Iterator& __y)
4602 requires random_access_range<_Base>
4603 {
return !(__y._M_current > __x._M_current); }
4605 friend constexpr bool
4606 operator>=(
const _Iterator& __x,
const _Iterator& __y)
4607 requires random_access_range<_Base>
4608 {
return !(__x._M_current > __y._M_current); }
4610#ifdef __cpp_lib_three_way_comparison
4611 friend constexpr auto
4612 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
4613 requires random_access_range<_Base>
4614 && three_way_comparable<iterator_t<_Base>>
4615 {
return __x._M_current <=> __y._M_current; }
4618 friend constexpr _Iterator
4619 operator+(
const _Iterator& __x, difference_type __y)
4620 requires random_access_range<_Base>
4621 {
return _Iterator{__x} += __y; }
4623 friend constexpr _Iterator
4624 operator+(difference_type __x,
const _Iterator& __y)
4625 requires random_access_range<_Base>
4626 {
return __y + __x; }
4628 friend constexpr _Iterator
4629 operator-(
const _Iterator& __x, difference_type __y)
4630 requires random_access_range<_Base>
4631 {
return _Iterator{__x} -= __y; }
4635 friend constexpr difference_type
4636 operator-(
const _Iterator& __x,
const _Iterator& __y)
4637 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
4638 {
return __x._M_current - __y._M_current; }
4640 template <
bool>
friend struct _Sentinel;
4643 template<
bool _Const>
4647 using _Base = elements_view::_Base<_Const>;
4648 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
4651 _Sentinel() =
default;
4654 _Sentinel(sentinel_t<_Base> __end)
4655 : _M_end(std::move(__end))
4659 _Sentinel(_Sentinel<!_Const> __other)
4661 && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
4665 constexpr sentinel_t<_Base>
4669 template<
bool _Const2>
4670 requires sentinel_for<sentinel_t<_Base>,
4671 iterator_t<__detail::__maybe_const_t<_Const2, _Vp>>>
4672 friend constexpr bool
4673 operator==(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
4674 {
return __x._M_current == __y._M_end; }
4676 template<
bool _Const2,
4677 typename _Base2 = __detail::__maybe_const_t<_Const2, _Vp>>
4678 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
4679 friend constexpr range_difference_t<_Base2>
4680 operator-(
const _Iterator<_Const2>& __x,
const _Sentinel& __y)
4681 {
return -(__y._M_end - __x._M_current); }
4683 template<
bool _Const2,
4684 typename _Base2 = __detail::__maybe_const_t<_Const2, _Vp>>
4685 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base2>>
4686 friend constexpr range_difference_t<_Base2>
4687 operator-(
const _Sentinel& __x,
const _Iterator<_Const2>& __y)
4688 {
return __x._M_end - __y._M_current; }
4690 friend _Sentinel<!_Const>;
4693 _Vp _M_base = _Vp();
4696 template<
typename _Tp,
size_t _Nm>
4697 inline constexpr bool enable_borrowed_range<elements_view<_Tp, _Nm>>
4698 = enable_borrowed_range<_Tp>;
4702 template<
typename _Range>
4703 using keys_view = elements_view<_Range, 0>;
4705 template<
typename _Range>
4706 using values_view = elements_view<_Range, 1>;
4712 template<
size_t _Nm,
typename _Range>
4713 concept __can_elements_view
4717 template<
size_t _Nm>
4718 struct _Elements : __adaptor::_RangeAdaptorClosure<_Elements<_Nm>>
4720 template<viewable_range _Range>
4721 requires __detail::__can_elements_view<_Nm, _Range>
4723 operator() [[nodiscard]] (_Range&& __r)
const
4728 static constexpr bool _S_has_simple_call_op =
true;
4731 template<
size_t _Nm>
4732 inline constexpr _Elements<_Nm> elements;
4733 inline constexpr auto keys = elements<0>;
4734 inline constexpr auto values = elements<1>;
4737#ifdef __cpp_lib_ranges_zip
4740 template<
typename... _Rs>
4741 concept __zip_is_common = (
sizeof...(_Rs) == 1 && (common_range<_Rs> && ...))
4742 || (!(bidirectional_range<_Rs> && ...) && (common_range<_Rs> && ...))
4743 || ((random_access_range<_Rs> && ...) && (sized_range<_Rs> && ...));
4745 template<
typename _Fp,
typename _Tuple>
4747 __tuple_transform(_Fp&& __f, _Tuple&& __tuple)
4749 return std::apply([&]<
typename... _Ts>(_Ts&&... __elts) {
4750 return tuple<invoke_result_t<_Fp&, _Ts>...>
4755 template<
typename _Fp,
typename _Tuple>
4757 __tuple_for_each(_Fp&& __f, _Tuple&& __tuple)
4759 std::apply([&]<
typename... _Ts>(_Ts&&... __elts) {
4766 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0)
4769 tuple<_Vs...> _M_views;
4771 template<
bool>
class _Iterator;
4772 template<
bool>
class _Sentinel;
4775 zip_view() =
default;
4778 zip_view(_Vs... __views)
4779 : _M_views(std::move(__views)...)
4783 begin()
requires (!(__detail::__simple_view<_Vs> && ...))
4784 {
return _Iterator<false>(__detail::__tuple_transform(ranges::begin, _M_views)); }
4787 begin() const requires (
range<const _Vs> && ...)
4788 {
return _Iterator<true>(__detail::__tuple_transform(ranges::begin, _M_views)); }
4791 end()
requires (!(__detail::__simple_view<_Vs> && ...))
4793 if constexpr (!__detail::__zip_is_common<_Vs...>)
4794 return _Sentinel<false>(__detail::__tuple_transform(ranges::end, _M_views));
4796 return begin() + iter_difference_t<_Iterator<false>>(
size());
4798 return _Iterator<false>(__detail::__tuple_transform(ranges::end, _M_views));
4802 end() const requires (
range<const _Vs> && ...)
4804 if constexpr (!__detail::__zip_is_common<
const _Vs...>)
4805 return _Sentinel<true>(__detail::__tuple_transform(ranges::end, _M_views));
4807 return begin() + iter_difference_t<_Iterator<true>>(
size());
4809 return _Iterator<true>(__detail::__tuple_transform(ranges::end, _M_views));
4815 return std::apply([](
auto... __sizes) {
4816 using _CT = __detail::__make_unsigned_like_t<
common_type_t<
decltype(__sizes)...>>;
4817 return ranges::min({_CT(__sizes)...});
4818 }, __detail::__tuple_transform(ranges::size, _M_views));
4824 return std::apply([](
auto... __sizes) {
4825 using _CT = __detail::__make_unsigned_like_t<
common_type_t<
decltype(__sizes)...>>;
4826 return ranges::min({_CT(__sizes)...});
4827 }, __detail::__tuple_transform(ranges::size, _M_views));
4831 template<
typename... _Rs>
4832 zip_view(_Rs&&...) -> zip_view<views::all_t<_Rs>...>;
4834 template<
typename... _Views>
4835 inline constexpr bool enable_borrowed_range<zip_view<_Views...>>
4836 = (enable_borrowed_range<_Views> && ...);
4840 template<
bool _Const,
typename... _Vs>
4841 concept __all_random_access
4842 = (random_access_range<__maybe_const_t<_Const, _Vs>> && ...);
4844 template<
bool _Const,
typename... _Vs>
4845 concept __all_bidirectional
4846 = (bidirectional_range<__maybe_const_t<_Const, _Vs>> && ...);
4848 template<
bool _Const,
typename... _Vs>
4849 concept __all_forward
4850 = (forward_range<__maybe_const_t<_Const, _Vs>> && ...);
4852 template<
bool _Const,
typename... _Views>
4853 struct __zip_view_iter_cat
4856 template<
bool _Const,
typename... _Views>
4857 requires __all_forward<_Const, _Views...>
4858 struct __zip_view_iter_cat<_Const, _Views...>
4859 {
using iterator_category = input_iterator_tag; };
4863 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0)
4864 template<bool _Const>
4865 class zip_view<_Vs...>::_Iterator
4866 : public __detail::__zip_view_iter_cat<_Const, _Vs...>
4868#ifdef _GLIBCXX_CLANG
4871 tuple<iterator_t<__detail::__maybe_const_t<_Const, _Vs>>...> _M_current;
4874 _Iterator(
decltype(_M_current) __current)
4875 : _M_current(std::move(__current))
4881 if constexpr (__detail::__all_random_access<_Const, _Vs...>)
4882 return random_access_iterator_tag{};
4883 else if constexpr (__detail::__all_bidirectional<_Const, _Vs...>)
4884 return bidirectional_iterator_tag{};
4885 else if constexpr (__detail::__all_forward<_Const, _Vs...>)
4886 return forward_iterator_tag{};
4888 return input_iterator_tag{};
4891#ifndef _GLIBCXX_CLANG
4892 template<move_constructible _Fp,
input_range... _Ws>
4893 requires (
view<_Ws> && ...) && (
sizeof...(_Ws) > 0) && is_object_v<_Fp>
4894 && regular_invocable<_Fp&, range_reference_t<_Ws>...>
4895 && std::__detail::__can_reference<invoke_result_t<_Fp&, range_reference_t<_Ws>...>>
4896 friend class zip_transform_view;
4901 using iterator_concept =
decltype(_S_iter_concept());
4903 = tuple<range_value_t<__detail::__maybe_const_t<_Const, _Vs>>...>;
4904 using difference_type
4907 _Iterator() =
default;
4910 _Iterator(_Iterator<!_Const> __i)
4912 && (convertible_to<iterator_t<_Vs>,
4913 iterator_t<__detail::__maybe_const_t<_Const, _Vs>>> && ...)
4914 : _M_current(std::move(__i._M_current))
4920 auto __f = [](
auto& __i) ->
decltype(
auto) {
4923 return __detail::__tuple_transform(__f, _M_current);
4926 constexpr _Iterator&
4929 __detail::__tuple_for_each([](
auto& __i) { ++__i; }, _M_current);
4939 requires __detail::__all_forward<_Const, _Vs...>
4946 constexpr _Iterator&
4948 requires __detail::__all_bidirectional<_Const, _Vs...>
4950 __detail::__tuple_for_each([](
auto& __i) { --__i; }, _M_current);
4956 requires __detail::__all_bidirectional<_Const, _Vs...>
4963 constexpr _Iterator&
4964 operator+=(difference_type __x)
4965 requires __detail::__all_random_access<_Const, _Vs...>
4967 auto __f = [&]<
typename _It>(_It& __i) {
4968 __i += iter_difference_t<_It>(__x);
4970 __detail::__tuple_for_each(__f, _M_current);
4974 constexpr _Iterator&
4975 operator-=(difference_type __x)
4976 requires __detail::__all_random_access<_Const, _Vs...>
4978 auto __f = [&]<
typename _It>(_It& __i) {
4979 __i -= iter_difference_t<_It>(__x);
4981 __detail::__tuple_for_each(__f, _M_current);
4986 operator[](difference_type __n)
const
4987 requires __detail::__all_random_access<_Const, _Vs...>
4989 auto __f = [&]<
typename _It>(_It& __i) ->
decltype(
auto) {
4990 return __i[iter_difference_t<_It>(__n)];
4992 return __detail::__tuple_transform(__f, _M_current);
4995 friend constexpr bool
4996 operator==(
const _Iterator& __x,
const _Iterator& __y)
4997 requires (equality_comparable<iterator_t<__detail::__maybe_const_t<_Const, _Vs>>> && ...)
4999 if constexpr (__detail::__all_bidirectional<_Const, _Vs...>)
5000 return __x._M_current == __y._M_current;
5003 return ((std::get<_Is>(__x._M_current) == std::get<_Is>(__y._M_current)) || ...);
5007 friend constexpr auto
5008 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
5009 requires __detail::__all_random_access<_Const, _Vs...>
5010 {
return __x._M_current <=> __y._M_current; }
5012 friend constexpr _Iterator
5013 operator+(
const _Iterator& __i, difference_type __n)
5014 requires __detail::__all_random_access<_Const, _Vs...>
5021 friend constexpr _Iterator
5022 operator+(difference_type __n,
const _Iterator& __i)
5023 requires __detail::__all_random_access<_Const, _Vs...>
5030 friend constexpr _Iterator
5031 operator-(
const _Iterator& __i, difference_type __n)
5032 requires __detail::__all_random_access<_Const, _Vs...>
5039 friend constexpr difference_type
5040 operator-(
const _Iterator& __x,
const _Iterator& __y)
5041 requires (sized_sentinel_for<iterator_t<__detail::__maybe_const_t<_Const, _Vs>>,
5042 iterator_t<__detail::__maybe_const_t<_Const, _Vs>>> && ...)
5045 return ranges::min({difference_type(std::get<_Is>(__x._M_current)
5046 - std::get<_Is>(__y._M_current))...},
5048 [](difference_type __i) {
5049 return __detail::__to_unsigned_like(__i < 0 ? -__i : __i);
5054 friend constexpr auto
5055 iter_move(
const _Iterator& __i)
5056 {
return __detail::__tuple_transform(ranges::iter_move, __i._M_current); }
5058 friend constexpr void
5059 iter_swap(
const _Iterator& __l,
const _Iterator& __r)
5060 requires (indirectly_swappable<iterator_t<__detail::__maybe_const_t<_Const, _Vs>>> && ...)
5063 (ranges::iter_swap(std::get<_Is>(__l._M_current), std::get<_Is>(__r._M_current)), ...);
5067 friend class zip_view;
5071 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0)
5072 template<bool _Const>
5073 class zip_view<_Vs...>::_Sentinel
5075 tuple<sentinel_t<__detail::__maybe_const_t<_Const, _Vs>>...> _M_end;
5078 _Sentinel(
decltype(_M_end) __end)
5082 friend class zip_view;
5085 _Sentinel() =
default;
5088 _Sentinel(_Sentinel<!_Const> __i)
5090 && (convertible_to<sentinel_t<_Vs>,
5091 sentinel_t<__detail::__maybe_const_t<_Const, _Vs>>> && ...)
5092 : _M_end(std::move(__i._M_end))
5095 template<
bool _OtherConst>
5096 requires (sentinel_for<sentinel_t<__detail::__maybe_const_t<_Const, _Vs>>,
5097 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vs>>> && ...)
5098 friend constexpr bool
5099 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5102 return ((std::get<_Is>(__x._M_current) == std::get<_Is>(__y._M_end)) || ...);
5106 template<
bool _OtherConst>
5107 requires (sized_sentinel_for<sentinel_t<__detail::__maybe_const_t<_Const, _Vs>>,
5108 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vs>>> && ...)
5109 friend constexpr auto
5110 operator-(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5115 return ranges::min({_Ret(std::get<_Is>(__x._M_current) - std::get<_Is>(__y._M_end))...},
5118 return __detail::__to_unsigned_like(__i < 0 ? -__i : __i);
5123 template<
bool _OtherConst>
5124 requires (sized_sentinel_for<sentinel_t<__detail::__maybe_const_t<_Const, _Vs>>,
5125 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vs>>> && ...)
5126 friend constexpr auto
5127 operator-(
const _Sentinel& __y,
const _Iterator<_OtherConst>& __x)
5128 {
return -(__x - __y); }
5135 template<
typename... _Ts>
5136 concept __can_zip_view
5142 template<
typename... _Ts>
5143 requires (
sizeof...(_Ts) == 0 || __detail::__can_zip_view<_Ts...>)
5145 operator() [[nodiscard]] (_Ts&&... __ts)
const
5147 if constexpr (
sizeof...(_Ts) == 0)
5148 return views::empty<tuple<>>;
5154 inline constexpr _Zip zip;
5159 template<
typename _Range,
bool _Const>
5160 using __range_iter_cat
5161 =
typename iterator_traits<iterator_t<__maybe_const_t<_Const, _Range>>>::iterator_category;
5164 template<move_constructible _Fp,
input_range... _Vs>
5165 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0) && is_object_v<_Fp>
5166 && regular_invocable<_Fp&, range_reference_t<_Vs>...>
5167 && std::__detail::__can_reference<invoke_result_t<_Fp&, range_reference_t<_Vs>...>>
5168 class zip_transform_view : public
view_interface<zip_transform_view<_Fp, _Vs...>>
5170 [[no_unique_address]] __detail::__box<_Fp> _M_fun;
5171 zip_view<_Vs...> _M_zip;
5173 using _InnerView = zip_view<_Vs...>;
5175 template<
bool _Const>
5176 using __ziperator = iterator_t<__detail::__maybe_const_t<_Const, _InnerView>>;
5178 template<
bool _Const>
5179 using __zentinel = sentinel_t<__detail::__maybe_const_t<_Const, _InnerView>>;
5181 template<
bool _Const>
5182 using _Base = __detail::__maybe_const_t<_Const, _InnerView>;
5184 template<
bool _Const>
5188 template<
bool _Const>
5190 struct __iter_cat<_Const>
5196 using __detail::__maybe_const_t;
5197 using __detail::__range_iter_cat;
5198 using _Res = invoke_result_t<__maybe_const_t<_Const, _Fp>&,
5199 range_reference_t<__maybe_const_t<_Const, _Vs>>...>;
5202 if constexpr (!is_reference_v<_Res>)
5203 return input_iterator_tag{};
5204 else if constexpr ((derived_from<__range_iter_cat<_Vs, _Const>,
5205 random_access_iterator_tag> && ...))
5206 return random_access_iterator_tag{};
5207 else if constexpr ((derived_from<__range_iter_cat<_Vs, _Const>,
5208 bidirectional_iterator_tag> && ...))
5209 return bidirectional_iterator_tag{};
5210 else if constexpr ((derived_from<__range_iter_cat<_Vs, _Const>,
5211 forward_iterator_tag> && ...))
5212 return forward_iterator_tag{};
5214 return input_iterator_tag{};
5217 using iterator_category =
decltype(_S_iter_cat());
5220 template<
bool>
class _Iterator;
5221 template<
bool>
class _Sentinel;
5224 zip_transform_view() =
default;
5227 zip_transform_view(_Fp __fun, _Vs... __views)
5228 : _M_fun(std::move(__fun)), _M_zip(std::move(__views)...)
5233 {
return _Iterator<false>(*
this, _M_zip.begin()); }
5237 requires
range<const _InnerView>
5238 && regular_invocable<const _Fp&, range_reference_t<const _Vs>...>
5239 {
return _Iterator<true>(*
this, _M_zip.begin()); }
5245 return _Iterator<false>(*
this, _M_zip.end());
5247 return _Sentinel<false>(_M_zip.end());
5252 requires
range<const _InnerView>
5253 && regular_invocable<const _Fp&, range_reference_t<const _Vs>...>
5256 return _Iterator<true>(*
this, _M_zip.end());
5258 return _Sentinel<true>(_M_zip.end());
5263 {
return _M_zip.size(); }
5267 {
return _M_zip.size(); }
5270 template<
class _Fp,
class... _Rs>
5271 zip_transform_view(_Fp, _Rs&&...)
5272 -> zip_transform_view<_Fp, views::all_t<_Rs>...>;
5274 template<move_constructible _Fp,
input_range... _Vs>
5275 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0) && is_object_v<_Fp>
5276 && regular_invocable<_Fp&, range_reference_t<_Vs>...>
5277 && std::__detail::__can_reference<invoke_result_t<_Fp&, range_reference_t<_Vs>...>>
5278 template<bool _Const>
5279 class zip_transform_view<_Fp, _Vs...>::_Iterator : public __iter_cat<_Const>
5281 using _Parent = __detail::__maybe_const_t<_Const, zip_transform_view>;
5282 using _Fun_handle = __detail::__func_handle_t<
5283 __detail::__maybe_const_t<_Const, _Fp>,
5284 iterator_t<__detail::__maybe_const_t<_Const, _Vs>>...>;
5286 [[no_unique_address]] _Fun_handle _M_fun;
5287 __ziperator<_Const> _M_inner;
5290 _Iterator(_Fun_handle __fun, __ziperator<_Const> __inner)
5291 : _M_fun(__fun), _M_inner(std::move(__inner))
5295 _Iterator(_Parent& __parent, __ziperator<_Const> __inner)
5296 : _M_fun(*__parent._M_fun), _M_inner(std::move(__inner))
5299 friend class zip_transform_view;
5303 using iterator_concept =
typename __ziperator<_Const>::iterator_concept;
5305 = remove_cvref_t<invoke_result_t<__detail::__maybe_const_t<_Const, _Fp>&,
5306 range_reference_t<__detail::__maybe_const_t<_Const, _Vs>>...>>;
5307 using difference_type = range_difference_t<_Base<_Const>>;
5309 _Iterator() =
default;
5312 _Iterator(_Iterator<!_Const> __i)
5313 requires _Const && convertible_to<__ziperator<false>, __ziperator<_Const>>
5314 : _M_fun(__i._M_fun), _M_inner(
std::move(__i._M_inner))
5317 constexpr decltype(
auto)
5320 return std::apply([&](
const auto&... __iters) ->
decltype(
auto) {
5321 return _M_fun._M_call_deref(__iters...);
5322 }, _M_inner._M_current);
5325 constexpr _Iterator&
5344 constexpr _Iterator&
5359 constexpr _Iterator&
5366 constexpr _Iterator&
5373 constexpr decltype(
auto)
5376 return std::apply([&]<
typename... _Is>(
const _Is&... __iters) ->
decltype(
auto) {
5377 return _M_fun._M_call_subscript(__n, __iters...);
5378 }, _M_inner._M_current);
5381 friend constexpr bool
5382 operator==(
const _Iterator& __x,
const _Iterator& __y)
5383 requires equality_comparable<__ziperator<_Const>>
5384 {
return __x._M_inner == __y._M_inner; }
5386 friend constexpr auto
5387 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
5389 {
return __x._M_inner <=> __y._M_inner; }
5391 friend constexpr _Iterator
5392 operator+(
const _Iterator& __i, difference_type __n)
5394 {
return _Iterator(__i._M_fun, __i._M_inner + __n); }
5396 friend constexpr _Iterator
5397 operator+(difference_type __n,
const _Iterator& __i)
5399 {
return _Iterator(__i._M_fun, __i._M_inner + __n); }
5401 friend constexpr _Iterator
5402 operator-(
const _Iterator& __i, difference_type __n)
5404 {
return _Iterator(__i._M_fun, __i._M_inner - __n); }
5406 friend constexpr difference_type
5407 operator-(
const _Iterator& __x,
const _Iterator& __y)
5408 requires sized_sentinel_for<__ziperator<_Const>, __ziperator<_Const>>
5409 {
return __x._M_inner - __y._M_inner; }
5412 template<move_constructible _Fp,
input_range... _Vs>
5413 requires (
view<_Vs> && ...) && (
sizeof...(_Vs) > 0) && is_object_v<_Fp>
5414 && regular_invocable<_Fp&, range_reference_t<_Vs>...>
5415 && std::__detail::__can_reference<invoke_result_t<_Fp&, range_reference_t<_Vs>...>>
5416 template<bool _Const>
5417 class zip_transform_view<_Fp, _Vs...>::_Sentinel
5419 __zentinel<_Const> _M_inner;
5422 _Sentinel(__zentinel<_Const> __inner)
5426 friend class zip_transform_view;
5429 _Sentinel() =
default;
5432 _Sentinel(_Sentinel<!_Const> __i)
5433 requires _Const && convertible_to<__zentinel<false>, __zentinel<_Const>>
5437 template<
bool _OtherConst>
5438 requires sentinel_for<__zentinel<_Const>, __ziperator<_OtherConst>>
5439 friend constexpr bool
5440 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5441 {
return __x._M_inner == __y._M_inner; }
5443 template<
bool _OtherConst>
5444 requires sized_sentinel_for<__zentinel<_Const>, __ziperator<_OtherConst>>
5445 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _InnerView>>
5446 operator-(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5447 {
return __x._M_inner - __y._M_inner; }
5449 template<
bool _OtherConst>
5450 requires sized_sentinel_for<__zentinel<_Const>, __ziperator<_OtherConst>>
5451 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _InnerView>>
5452 operator-(
const _Sentinel& __x,
const _Iterator<_OtherConst>& __y)
5453 {
return __x._M_inner - __y._M_inner; }
5460 template<
typename _Fp,
typename... _Ts>
5461 concept __can_zip_transform_view
5465 struct _ZipTransform
5467 template<
typename _Fp>
5468 requires move_constructible<decay_t<_Fp>> && regular_invocable<decay_t<_Fp>&>
5469 && is_object_v<decay_t<invoke_result_t<decay_t<_Fp>&>>>
5471 operator() [[nodiscard]] (_Fp&&)
const
5473 return views::empty<decay_t<invoke_result_t<decay_t<_Fp>&>>>;
5476 template<
typename _Fp,
typename... _Ts>
5477 requires (
sizeof...(_Ts) != 0) && __detail::__can_zip_transform_view<_Fp, _Ts...>
5479 operator() [[nodiscard]] (_Fp&& __f, _Ts&&... __ts)
const
5485 inline constexpr _ZipTransform zip_transform;
5488 template<forward_range _Vp,
size_t _Nm>
5490 class adjacent_view : public
view_interface<adjacent_view<_Vp, _Nm>>
5492 _Vp _M_base = _Vp();
5494 template<
bool>
class _Iterator;
5495 template<
bool>
class _Sentinel;
5497 struct __as_sentinel
5501 adjacent_view()
requires default_initializable<_Vp> = default;
5504 adjacent_view(_Vp __base)
5505 : _M_base(std::move(__base))
5511 base() const & requires copy_constructible<_Vp>
5519 begin()
requires (!__detail::__simple_view<_Vp>)
5520 {
return _Iterator<false>(ranges::begin(_M_base), ranges::end(_M_base)); }
5524 {
return _Iterator<true>(ranges::begin(_M_base), ranges::end(_M_base)); }
5527 end()
requires (!__detail::__simple_view<_Vp>)
5530 return _Iterator<false>(__as_sentinel{}, ranges::begin(_M_base), ranges::end(_M_base));
5532 return _Sentinel<false>(ranges::end(_M_base));
5536 end() const requires
range<const _Vp>
5539 return _Iterator<true>(__as_sentinel{}, ranges::begin(_M_base), ranges::end(_M_base));
5541 return _Sentinel<true>(ranges::end(_M_base));
5547 using _ST =
decltype(ranges::size(_M_base));
5549 auto __sz =
static_cast<_CT
>(ranges::size(_M_base));
5551 return static_cast<_ST
>(__sz);
5557 using _ST =
decltype(ranges::size(_M_base));
5559 auto __sz =
static_cast<_CT
>(ranges::size(_M_base));
5561 return static_cast<_ST
>(__sz);
5565 template<
typename _Vp,
size_t _Nm>
5566 inline constexpr bool enable_borrowed_range<adjacent_view<_Vp, _Nm>>
5567 = enable_borrowed_range<_Vp>;
5572 template<
typename _Tp,
size_t _Nm>
5573 using __repeated_tuple =
typename __make_tuple<array<_Tp, _Nm>>::__type;
5577 template<
typename _Fp,
size_t _Nm>
5580 template<
typename... _Ts>
5581 static invoke_result_t<_Fp, _Ts...>
5582 __tuple_apply(
const tuple<_Ts...>&);
5584 template<
typename _Tp>
5585 decltype(__tuple_apply(
std::declval<__repeated_tuple<_Tp, _Nm>>()))
5590 template<forward_range _Vp,
size_t _Nm>
5592 template<bool _Const>
5593 class adjacent_view<_Vp, _Nm>::_Iterator
5595#ifdef _GLIBCXX_CLANG
5598 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
5599 array<iterator_t<_Base>, _Nm> _M_current = array<iterator_t<_Base>, _Nm>();
5602 _Iterator(iterator_t<_Base> __first, sentinel_t<_Base> __last)
5604 for (
auto& __i : _M_current)
5607 ranges::advance(__first, 1, __last);
5612 _Iterator(__as_sentinel, iterator_t<_Base> __first, iterator_t<_Base> __last)
5615 for (
auto& __it : _M_current)
5618 for (
size_t __i = 0; __i < _Nm; ++__i)
5620 _M_current[_Nm - 1 - __i] = __last;
5621 ranges::advance(__last, -1, __first);
5629 return random_access_iterator_tag{};
5631 return bidirectional_iterator_tag{};
5633 return forward_iterator_tag{};
5636 friend class adjacent_view;
5638#ifndef _GLIBCXX_CLANG
5639 template<forward_range _Wp, move_constructible _Fp,
size_t _Mm>
5640 requires view<_Wp> && (_Mm > 0) && is_object_v<_Fp>
5641 && regular_invocable<__detail::__unarize<_Fp&, _Mm>, range_reference_t<_Wp>>
5642 && std::__detail::__can_reference<invoke_result_t<__detail::__unarize<_Fp&, _Mm>,
5643 range_reference_t<_Wp>>>
5644 friend class adjacent_transform_view;
5648 using iterator_category = input_iterator_tag;
5649 using iterator_concept =
decltype(_S_iter_concept());
5650 using value_type = __detail::__repeated_tuple<range_value_t<_Base>, _Nm>;
5651 using difference_type = range_difference_t<_Base>;
5653 _Iterator() =
default;
5656 _Iterator(_Iterator<!_Const> __i)
5657 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
5659 for (
size_t __j = 0; __j < _Nm; ++__j)
5660 _M_current[__j] =
std::move(__i._M_current[__j]);
5666 auto __f = [](
auto& __i) ->
decltype(
auto) {
return *__i; };
5667 return __detail::__tuple_transform(__f, _M_current);
5670 constexpr _Iterator&
5673 for (
auto& __i : _M_current)
5686 constexpr _Iterator&
5689 for (
auto& __i : _M_current)
5702 constexpr _Iterator&
5703 operator+=(difference_type __x)
5706 for (
auto& __i : _M_current)
5711 constexpr _Iterator&
5712 operator-=(difference_type __x)
5715 for (
auto& __i : _M_current)
5721 operator[](difference_type __n)
const
5724 auto __f = [&](
auto& __i) ->
decltype(
auto) {
return __i[__n]; };
5725 return __detail::__tuple_transform(__f, _M_current);
5728 friend constexpr bool
5729 operator==(
const _Iterator& __x,
const _Iterator& __y)
5730 {
return __x._M_current.back() == __y._M_current.back(); }
5732 friend constexpr bool
5733 operator<(
const _Iterator& __x,
const _Iterator& __y)
5735 {
return __x._M_current.back() < __y._M_current.back(); }
5737 friend constexpr bool
5738 operator>(
const _Iterator& __x,
const _Iterator& __y)
5740 {
return __y < __x; }
5742 friend constexpr bool
5743 operator<=(
const _Iterator& __x,
const _Iterator& __y)
5745 {
return !(__y < __x); }
5747 friend constexpr bool
5748 operator>=(
const _Iterator& __x,
const _Iterator& __y)
5750 {
return !(__x < __y); }
5752 friend constexpr auto
5753 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
5755 && three_way_comparable<iterator_t<_Base>>
5756 {
return __x._M_current.back() <=> __y._M_current.back(); }
5758 friend constexpr _Iterator
5759 operator+(
const _Iterator& __i, difference_type __n)
5767 friend constexpr _Iterator
5768 operator+(difference_type __n,
const _Iterator& __i)
5776 friend constexpr _Iterator
5777 operator-(
const _Iterator& __i, difference_type __n)
5785 friend constexpr difference_type
5786 operator-(
const _Iterator& __x,
const _Iterator& __y)
5787 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
5788 {
return __x._M_current.back() - __y._M_current.back(); }
5790 friend constexpr auto
5791 iter_move(
const _Iterator& __i)
5792 {
return __detail::__tuple_transform(ranges::iter_move, __i._M_current); }
5794 friend constexpr void
5795 iter_swap(
const _Iterator& __l,
const _Iterator& __r)
5796 requires indirectly_swappable<iterator_t<_Base>>
5798 for (
size_t __i = 0; __i < _Nm; __i++)
5799 ranges::iter_swap(__l._M_current[__i], __r._M_current[__i]);
5803 template<forward_range _Vp,
size_t _Nm>
5805 template<bool _Const>
5806 class adjacent_view<_Vp, _Nm>::_Sentinel
5808 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
5810 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
5813 _Sentinel(sentinel_t<_Base> __end)
5817 friend class adjacent_view;
5820 _Sentinel() =
default;
5823 _Sentinel(_Sentinel<!_Const> __i)
5824 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
5828 template<
bool _OtherConst>
5829 requires sentinel_for<sentinel_t<_Base>,
5830 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vp>>>
5831 friend constexpr bool
5832 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5833 {
return __x._M_current.back() == __y._M_end; }
5835 template<
bool _OtherConst>
5836 requires sized_sentinel_for<sentinel_t<_Base>,
5837 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vp>>>
5838 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _Vp>>
5839 operator-(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
5840 {
return __x._M_current.back() - __y._M_end; }
5842 template<
bool _OtherConst>
5843 requires sized_sentinel_for<sentinel_t<_Base>,
5844 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vp>>>
5845 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _Vp>>
5846 operator-(
const _Sentinel& __y,
const _Iterator<_OtherConst>& __x)
5847 {
return __y._M_end - __x._M_current.back(); }
5854 template<
size_t _Nm,
typename _Range>
5855 concept __can_adjacent_view
5859 template<
size_t _Nm>
5860 struct _Adjacent : __adaptor::_RangeAdaptorClosure<_Adjacent<_Nm>>
5864 template<viewable_range _Range>
5865 requires ((_Nm == 0) && forward_range<_Range>)
5866 || __detail::__can_adjacent_view<_Nm, _Range>
5868 operator() [[nodiscard]] (_Range&& __r)
const
5870 if constexpr (_Nm == 0)
5871 return views::empty<tuple<>>;
5877 template<
size_t _Nm>
5878 inline constexpr _Adjacent<_Nm> adjacent;
5880 inline constexpr auto pairwise = adjacent<2>;
5883 template<forward_range _Vp, move_constructible _Fp,
size_t _Nm>
5884 requires view<_Vp> && (_Nm > 0) && is_object_v<_Fp>
5885 && regular_invocable<__detail::__unarize<_Fp&, _Nm>, range_reference_t<_Vp>>
5886 && std::__detail::__can_reference<invoke_result_t<__detail::__unarize<_Fp&, _Nm>,
5887 range_reference_t<_Vp>>>
5888 class adjacent_transform_view : public
view_interface<adjacent_transform_view<_Vp, _Fp, _Nm>>
5890 [[no_unique_address]] __detail::__box<_Fp> _M_fun;
5891 adjacent_view<_Vp, _Nm> _M_inner;
5893 using _InnerView = adjacent_view<_Vp, _Nm>;
5895 template<
bool _Const>
5896 using _InnerIter = iterator_t<__detail::__maybe_const_t<_Const, _InnerView>>;
5898 template<
bool _Const>
5899 using _InnerSent = sentinel_t<__detail::__maybe_const_t<_Const, _InnerView>>;
5901 template<
bool>
class _Iterator;
5902 template<
bool>
class _Sentinel;
5905 adjacent_transform_view() =
default;
5908 adjacent_transform_view(_Vp __base, _Fp __fun)
5909 : _M_fun(std::move(__fun)), _M_inner(std::move(
__base))
5916 base() const & requires copy_constructible<_Vp>
5917 {
return _M_inner.base(); }
5925 {
return _Iterator<false>(*
this, _M_inner.begin()); }
5929 requires
range<const _InnerView>
5930 && regular_invocable<__detail::__unarize<const _Fp&, _Nm>,
5931 range_reference_t<const _Vp>>
5932 {
return _Iterator<true>(*
this, _M_inner.begin()); }
5938 return _Iterator<false>(*
this, _M_inner.end());
5940 return _Sentinel<false>(_M_inner.end());
5945 requires
range<const _InnerView>
5946 && regular_invocable<__detail::__unarize<const _Fp&, _Nm>,
5947 range_reference_t<const _Vp>>
5950 return _Iterator<true>(*
this, _M_inner.end());
5952 return _Sentinel<true>(_M_inner.end());
5957 {
return _M_inner.size(); }
5961 {
return _M_inner.size(); }
5964 template<forward_range _Vp, move_constructible _Fp,
size_t _Nm>
5965 requires view<_Vp> && (_Nm > 0) && is_object_v<_Fp>
5966 && regular_invocable<__detail::__unarize<_Fp&, _Nm>, range_reference_t<_Vp>>
5967 && std::__detail::__can_reference<invoke_result_t<__detail::__unarize<_Fp&, _Nm>,
5968 range_reference_t<_Vp>>>
5969 template<bool _Const>
5970 class adjacent_transform_view<_Vp, _Fp, _Nm>::_Iterator
5972 using _Parent = __detail::__maybe_const_t<_Const, adjacent_transform_view>;
5973 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
5975 return __detail::__func_handle_t<
5976 __detail::__maybe_const_t<_Const, _Fp>,
5977 iterator_t<__detail::__maybe_const_t<(_Ids, _Const), _Vp>>...>();
5980 [[no_unique_address]] _Fun_handle _M_fun;
5981 _InnerIter<_Const> _M_inner;
5984 _Iterator(_Fun_handle __fun, _InnerIter<_Const> __inner)
5985 : _M_fun(__fun), _M_inner(std::move(__inner))
5989 _Iterator(_Parent& __parent, _InnerIter<_Const> __inner)
5990 : _M_fun(*__parent._M_fun), _M_inner(std::move(__inner))
5996 using __detail::__maybe_const_t;
5997 using __detail::__unarize;
5998 using _Res = invoke_result_t<__unarize<__maybe_const_t<_Const, _Fp>&, _Nm>,
5999 range_reference_t<_Base>>;
6000 using _Cat =
typename iterator_traits<iterator_t<_Base>>::iterator_category;
6003 if constexpr (!is_reference_v<_Res>)
6004 return input_iterator_tag{};
6005 else if constexpr (derived_from<_Cat, random_access_iterator_tag>)
6006 return random_access_iterator_tag{};
6007 else if constexpr (derived_from<_Cat, bidirectional_iterator_tag>)
6008 return bidirectional_iterator_tag{};
6009 else if constexpr (derived_from<_Cat, forward_iterator_tag>)
6010 return forward_iterator_tag{};
6012 return input_iterator_tag{};
6015 friend class adjacent_transform_view;
6018 using iterator_category =
decltype(_S_iter_cat());
6019 using iterator_concept =
typename _InnerIter<_Const>::iterator_concept;
6021 = remove_cvref_t<invoke_result_t
6022 <__detail::__unarize<__detail::__maybe_const_t<_Const, _Fp>&, _Nm>,
6023 range_reference_t<_Base>>>;
6024 using difference_type = range_difference_t<_Base>;
6026 _Iterator() =
default;
6029 _Iterator(_Iterator<!_Const> __i)
6030 requires _Const && convertible_to<_InnerIter<false>, _InnerIter<_Const>>
6031 : _M_fun(__i._M_fun), _M_inner(
std::move(__i._M_inner))
6034 constexpr decltype(
auto)
6037 return std::apply([&](
const auto&... __iters) ->
decltype(
auto) {
6038 return _M_fun._M_call_deref(__iters...);
6039 }, _M_inner._M_current);
6042 constexpr _Iterator&
6057 constexpr _Iterator&
6072 constexpr _Iterator&
6079 constexpr _Iterator&
6086 constexpr decltype(
auto)
6089 return std::apply([&](
const auto&... __iters) ->
decltype(
auto) {
6090 return _M_fun._M_call_subscript(__n, __iters...);
6091 }, _M_inner._M_current);
6094 friend constexpr bool
6095 operator==(
const _Iterator& __x,
const _Iterator& __y)
6096 {
return __x._M_inner == __y._M_inner; }
6098 friend constexpr bool
6099 operator<(
const _Iterator& __x,
const _Iterator& __y)
6101 {
return __x._M_inner < __y._M_inner; }
6103 friend constexpr bool
6104 operator>(
const _Iterator& __x,
const _Iterator& __y)
6106 {
return __x._M_inner > __y._M_inner; }
6108 friend constexpr bool
6109 operator<=(
const _Iterator& __x,
const _Iterator& __y)
6111 {
return __x._M_inner <= __y._M_inner; }
6113 friend constexpr bool
6114 operator>=(
const _Iterator& __x,
const _Iterator& __y)
6116 {
return __x._M_inner >= __y._M_inner; }
6118 friend constexpr auto
6119 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
6121 three_way_comparable<_InnerIter<_Const>>
6122 {
return __x._M_inner <=> __y._M_inner; }
6124 friend constexpr _Iterator
6125 operator+(
const _Iterator& __i, difference_type __n)
6127 {
return _Iterator(__i._M_fun, __i._M_inner + __n); }
6129 friend constexpr _Iterator
6130 operator+(difference_type __n,
const _Iterator& __i)
6132 {
return _Iterator(__i._M_fun, __i._M_inner + __n); }
6134 friend constexpr _Iterator
6135 operator-(
const _Iterator& __i, difference_type __n)
6137 {
return _Iterator(__i._M_fun, __i._M_inner - __n); }
6139 friend constexpr difference_type
6140 operator-(
const _Iterator& __x,
const _Iterator& __y)
6141 requires sized_sentinel_for<_InnerIter<_Const>, _InnerIter<_Const>>
6142 {
return __x._M_inner - __y._M_inner; }
6145 template<forward_range _Vp, move_constructible _Fp,
size_t _Nm>
6146 requires view<_Vp> && (_Nm > 0) && is_object_v<_Fp>
6147 && regular_invocable<__detail::__unarize<_Fp&, _Nm>, range_reference_t<_Vp>>
6148 && std::__detail::__can_reference<invoke_result_t<__detail::__unarize<_Fp&, _Nm>,
6149 range_reference_t<_Vp>>>
6150 template<bool _Const>
6151 class adjacent_transform_view<_Vp, _Fp, _Nm>::_Sentinel
6153 _InnerSent<_Const> _M_inner;
6156 _Sentinel(_InnerSent<_Const> __inner)
6160 friend class adjacent_transform_view;
6163 _Sentinel() =
default;
6166 _Sentinel(_Sentinel<!_Const> __i)
6167 requires _Const && convertible_to<_InnerSent<false>, _InnerSent<_Const>>
6171 template<
bool _OtherConst>
6172 requires sentinel_for<_InnerSent<_Const>, _InnerIter<_OtherConst>>
6173 friend constexpr bool
6174 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
6175 {
return __x._M_inner == __y._M_inner; }
6177 template<
bool _OtherConst>
6178 requires sized_sentinel_for<_InnerSent<_Const>, _InnerIter<_OtherConst>>
6179 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _InnerView>>
6180 operator-(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
6181 {
return __x._M_inner - __y._M_inner; }
6183 template<
bool _OtherConst>
6184 requires sized_sentinel_for<_InnerSent<_Const>, _InnerIter<_OtherConst>>
6185 friend constexpr range_difference_t<__detail::__maybe_const_t<_OtherConst, _InnerView>>
6186 operator-(
const _Sentinel& __x,
const _Iterator<_OtherConst>& __y)
6187 {
return __x._M_inner - __y._M_inner; }
6194 template<
size_t _Nm,
typename _Range,
typename _Fp>
6195 concept __can_adjacent_transform_view
6196 =
requires { adjacent_transform_view<all_t<_Range>,
decay_t<_Fp>, _Nm>
6200 template<
size_t _Nm>
6201 struct _AdjacentTransform : __adaptor::_RangeAdaptor<_AdjacentTransform<_Nm>>
6205 template<viewable_range _Range,
typename _Fp>
6206 requires ((_Nm == 0) && forward_range<_Range>)
6207 || __detail::__can_adjacent_transform_view<_Nm, _Range, _Fp>
6209 operator() [[nodiscard]] (_Range&& __r, _Fp&& __f)
const
6211 if constexpr (_Nm == 0)
6214 return adjacent_transform_view<all_t<_Range>, decay_t<_Fp>, _Nm>
6218 using __adaptor::_RangeAdaptor<_AdjacentTransform>::operator();
6219 static constexpr int _S_arity = 2;
6220 static constexpr bool _S_has_simple_extra_args =
true;
6223 template<
size_t _Nm>
6224 inline constexpr _AdjacentTransform<_Nm> adjacent_transform;
6226 inline constexpr auto pairwise_transform = adjacent_transform<2>;
6230#ifdef __cpp_lib_ranges_chunk
6233 template<
typename _Tp>
6234 constexpr _Tp __div_ceil(_Tp __num, _Tp __denom)
6236 _Tp __r = __num / __denom;
6237 if (__num % __denom)
6248 range_difference_t<_Vp> _M_n;
6249 range_difference_t<_Vp> _M_remainder = 0;
6250 __detail::__non_propagating_cache<iterator_t<_Vp>> _M_current;
6257 chunk_view(_Vp __base, range_difference_t<_Vp> __n)
6258 : _M_base(std::move(
__base)), _M_n(__n)
6259 { __glibcxx_assert(__n >= 0); }
6262 base() const & requires copy_constructible<_Vp>
6269 constexpr _OuterIter
6272 _M_current = ranges::begin(_M_base);
6273 _M_remainder = _M_n;
6274 return _OuterIter(*
this);
6277 constexpr default_sentinel_t
6278 end() const noexcept
6282 size()
requires sized_range<_Vp>
6284 return __detail::__to_unsigned_like(__detail::__div_ceil
6285 (ranges::distance(_M_base), _M_n));
6289 size() const requires sized_range<const _Vp>
6291 return __detail::__to_unsigned_like(__detail::__div_ceil
6292 (ranges::distance(_M_base), _M_n));
6296 template<
typename _Range>
6297 chunk_view(_Range&&, range_difference_t<_Range>) -> chunk_view<views::all_t<_Range>>;
6301 class chunk_view<_Vp>::_OuterIter
6303 chunk_view* _M_parent;
6306 _OuterIter(chunk_view& __parent) noexcept
6313 using iterator_concept = input_iterator_tag;
6314 using difference_type = range_difference_t<_Vp>;
6318 _OuterIter(_OuterIter&&) =
default;
6319 _OuterIter& operator=(_OuterIter&&) =
default;
6321 constexpr value_type
6324 __glibcxx_assert(*
this != default_sentinel);
6325 return value_type(*_M_parent);
6328 constexpr _OuterIter&
6331 __glibcxx_assert(*
this != default_sentinel);
6332 ranges::advance(*_M_parent->_M_current, _M_parent->_M_remainder,
6333 ranges::end(_M_parent->_M_base));
6334 _M_parent->_M_remainder = _M_parent->_M_n;
6342 friend constexpr bool
6343 operator==(
const _OuterIter& __x, default_sentinel_t)
6345 return *__x._M_parent->_M_current == ranges::end(__x._M_parent->_M_base)
6346 && __x._M_parent->_M_remainder != 0;
6349 friend constexpr difference_type
6350 operator-(default_sentinel_t,
const _OuterIter& __x)
6351 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
6353 const auto __dist = ranges::end(__x._M_parent->_M_base) - *__x._M_parent->_M_current;
6355 if (__dist < __x._M_parent->_M_remainder)
6356 return __dist == 0 ? 0 : 1;
6358 return 1 + __detail::__div_ceil(__dist - __x._M_parent->_M_remainder,
6359 __x._M_parent->_M_n);
6362 friend constexpr difference_type
6363 operator-(
const _OuterIter& __x, default_sentinel_t __y)
6364 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
6365 {
return -(__y - __x); }
6370 struct chunk_view<_Vp>::_OuterIter::value_type :
view_interface<value_type>
6373 chunk_view* _M_parent;
6376 value_type(chunk_view& __parent) noexcept
6383 constexpr _InnerIter
6384 begin() const noexcept
6385 {
return _InnerIter(*_M_parent); }
6387 constexpr default_sentinel_t
6388 end() const noexcept
6393 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
6395 return __detail::__to_unsigned_like
6396 (ranges::min(_M_parent->_M_remainder,
6397 ranges::end(_M_parent->_M_base) - *_M_parent->_M_current));
6403 class chunk_view<_Vp>::_InnerIter
6405 chunk_view* _M_parent;
6408 _InnerIter(chunk_view& __parent) noexcept
6412 friend _OuterIter::value_type;
6415 using iterator_concept = input_iterator_tag;
6416 using difference_type = range_difference_t<_Vp>;
6417 using value_type = range_value_t<_Vp>;
6419 _InnerIter(_InnerIter&&) =
default;
6420 _InnerIter& operator=(_InnerIter&&) =
default;
6422 constexpr const iterator_t<_Vp>&
6424 {
return *_M_parent->_M_current; }
6426 constexpr range_reference_t<_Vp>
6429 __glibcxx_assert(*
this != default_sentinel);
6430 return **_M_parent->_M_current;
6433 constexpr _InnerIter&
6436 __glibcxx_assert(*
this != default_sentinel);
6437 ++*_M_parent->_M_current;
6438 if (*_M_parent->_M_current == ranges::end(_M_parent->_M_base))
6439 _M_parent->_M_remainder = 0;
6441 --_M_parent->_M_remainder;
6449 friend constexpr bool
6450 operator==(
const _InnerIter& __x, default_sentinel_t)
noexcept
6451 {
return __x._M_parent->_M_remainder == 0; }
6453 friend constexpr difference_type
6454 operator-(default_sentinel_t,
const _InnerIter& __x)
6455 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
6457 return ranges::min(__x._M_parent->_M_remainder,
6458 ranges::end(__x._M_parent->_M_base) - *__x._M_parent->_M_current);
6461 friend constexpr difference_type
6462 operator-(
const _InnerIter& __x, default_sentinel_t __y)
6463 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
6464 {
return -(__y - __x); }
6468 friend constexpr range_rvalue_reference_t<_Vp>
6469 iter_move(
const _InnerIter& __i)
6470 noexcept(
noexcept(ranges::iter_move(*__i._M_parent->_M_current)))
6471 {
return ranges::iter_move(*__i._M_parent->_M_current); }
6473 friend constexpr void
6474 iter_swap(
const _InnerIter& __x,
const _InnerIter& __y)
6475 noexcept(
noexcept(ranges::iter_swap(*__x._M_parent->_M_current,
6476 *__x._M_parent->_M_current)))
6477 requires indirectly_swappable<iterator_t<_Vp>>
6478 {
return ranges::iter_swap(*__x._M_parent->_M_current, *__y._M_parent->_M_current); }
6486 range_difference_t<_Vp> _M_n;
6487 template<
bool>
class _Iterator;
6491 chunk_view(_Vp __base, range_difference_t<_Vp> __n)
6492 : _M_base(std::move(
__base)), _M_n(__n)
6493 { __glibcxx_assert(__n > 0); }
6496 base() const & requires copy_constructible<_Vp>
6504 begin()
requires (!__detail::__simple_view<_Vp>)
6505 {
return _Iterator<false>(
this, ranges::begin(_M_base)); }
6508 begin() const requires forward_range<const _Vp>
6509 {
return _Iterator<true>(
this, ranges::begin(_M_base)); }
6512 end()
requires (!__detail::__simple_view<_Vp>)
6514 if constexpr (common_range<_Vp> && sized_range<_Vp>)
6516 auto __missing = (_M_n - ranges::distance(_M_base) % _M_n) % _M_n;
6517 return _Iterator<false>(
this, ranges::end(_M_base), __missing);
6519 else if constexpr (common_range<_Vp> && !bidirectional_range<_Vp>)
6520 return _Iterator<false>(
this, ranges::end(_M_base));
6526 end() const requires forward_range<const _Vp>
6528 if constexpr (common_range<const _Vp> && sized_range<const _Vp>)
6530 auto __missing = (_M_n - ranges::distance(_M_base) % _M_n) % _M_n;
6531 return _Iterator<true>(
this, ranges::end(_M_base), __missing);
6533 else if constexpr (common_range<const _Vp> && !bidirectional_range<const _Vp>)
6534 return _Iterator<true>(
this, ranges::end(_M_base));
6540 size()
requires sized_range<_Vp>
6542 return __detail::__to_unsigned_like(__detail::__div_ceil
6543 (ranges::distance(_M_base), _M_n));
6547 size() const requires sized_range<const _Vp>
6549 return __detail::__to_unsigned_like(__detail::__div_ceil
6550 (ranges::distance(_M_base), _M_n));
6554 template<
typename _Vp>
6555 inline constexpr bool enable_borrowed_range<chunk_view<_Vp>>
6560 template<
bool _Const>
6561 class chunk_view<_Vp>::_Iterator
6563 using _Parent = __detail::__maybe_const_t<_Const, chunk_view>;
6564 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
6566 iterator_t<_Base> _M_current = iterator_t<_Base>();
6567 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
6568 range_difference_t<_Base> _M_n = 0;
6569 range_difference_t<_Base> _M_missing = 0;
6572 _Iterator(_Parent* __parent, iterator_t<_Base> __current,
6573 range_difference_t<_Base> __missing = 0)
6574 : _M_current(__current), _M_end(ranges::end(__parent->_M_base)),
6575 _M_n(__parent->_M_n), _M_missing(__missing)
6581 if constexpr (random_access_range<_Base>)
6582 return random_access_iterator_tag{};
6583 else if constexpr (bidirectional_range<_Base>)
6584 return bidirectional_iterator_tag{};
6586 return forward_iterator_tag{};
6592 using iterator_category = input_iterator_tag;
6593 using iterator_concept =
decltype(_S_iter_cat());
6594 using value_type =
decltype(views::take(subrange(_M_current, _M_end), _M_n));
6595 using difference_type = range_difference_t<_Base>;
6597 _Iterator() =
default;
6599 constexpr _Iterator(_Iterator<!_Const> __i)
6601 && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
6602 && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
6604 _M_n(__i._M_n), _M_missing(__i._M_missing)
6607 constexpr iterator_t<_Base>
6609 {
return _M_current; }
6611 constexpr value_type
6614 __glibcxx_assert(_M_current != _M_end);
6615 return views::take(subrange(_M_current, _M_end), _M_n);
6618 constexpr _Iterator&
6621 __glibcxx_assert(_M_current != _M_end);
6622 _M_missing = ranges::advance(_M_current, _M_n, _M_end);
6634 constexpr _Iterator&
6635 operator--()
requires bidirectional_range<_Base>
6637 ranges::advance(_M_current, _M_missing - _M_n);
6643 operator--(
int)
requires bidirectional_range<_Base>
6650 constexpr _Iterator&
6651 operator+=(difference_type __x)
6652 requires random_access_range<_Base>
6656 __glibcxx_assert(ranges::distance(_M_current, _M_end) > _M_n * (__x - 1));
6657 _M_missing = ranges::advance(_M_current, _M_n * __x, _M_end);
6661 ranges::advance(_M_current, _M_n * __x + _M_missing);
6667 constexpr _Iterator&
6668 operator-=(difference_type __x)
6669 requires random_access_range<_Base>
6670 {
return *
this += -__x; }
6672 constexpr value_type
6673 operator[](difference_type __n)
const
6674 requires random_access_range<_Base>
6675 {
return *(*
this + __n); }
6677 friend constexpr bool
6678 operator==(
const _Iterator& __x,
const _Iterator& __y)
6679 {
return __x._M_current == __y._M_current; }
6681 friend constexpr bool
6682 operator==(
const _Iterator& __x, default_sentinel_t)
6683 {
return __x._M_current == __x._M_end; }
6685 friend constexpr bool
6686 operator<(
const _Iterator& __x,
const _Iterator& __y)
6687 requires random_access_range<_Base>
6688 {
return __x._M_current > __y._M_current; }
6690 friend constexpr bool
6691 operator>(
const _Iterator& __x,
const _Iterator& __y)
6692 requires random_access_range<_Base>
6693 {
return __y < __x; }
6695 friend constexpr bool
6696 operator<=(
const _Iterator& __x,
const _Iterator& __y)
6697 requires random_access_range<_Base>
6698 {
return !(__y < __x); }
6700 friend constexpr bool
6701 operator>=(
const _Iterator& __x,
const _Iterator& __y)
6702 requires random_access_range<_Base>
6703 {
return !(__x < __y); }
6705 friend constexpr auto
6706 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
6707 requires random_access_range<_Base>
6708 && three_way_comparable<iterator_t<_Base>>
6709 {
return __x._M_current <=> __y._M_current; }
6711 friend constexpr _Iterator
6712 operator+(
const _Iterator& __i, difference_type __n)
6713 requires random_access_range<_Base>
6720 friend constexpr _Iterator
6721 operator+(difference_type __n,
const _Iterator& __i)
6722 requires random_access_range<_Base>
6729 friend constexpr _Iterator
6730 operator-(
const _Iterator& __i, difference_type __n)
6731 requires random_access_range<_Base>
6738 friend constexpr difference_type
6739 operator-(
const _Iterator& __x,
const _Iterator& __y)
6740 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
6742 return (__x._M_current - __y._M_current
6743 + __x._M_missing - __y._M_missing) / __x._M_n;
6746 friend constexpr difference_type
6747 operator-(default_sentinel_t,
const _Iterator& __x)
6748 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
6749 {
return __detail::__div_ceil(__x._M_end - __x._M_current, __x._M_n); }
6751 friend constexpr difference_type
6752 operator-(
const _Iterator& __x, default_sentinel_t __y)
6753 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
6754 {
return -(__y - __x); }
6761 template<
typename _Range,
typename _Dp>
6762 concept __can_chunk_view
6766 struct _Chunk : __adaptor::_RangeAdaptor<_Chunk>
6768 template<viewable_range _Range,
typename _Dp = range_difference_t<_Range>>
6769 requires __detail::__can_chunk_view<_Range, _Dp>
6771 operator() [[nodiscard]] (_Range&& __r, type_identity_t<_Dp> __n)
const
6774 using __adaptor::_RangeAdaptor<_Chunk>::operator();
6775 static constexpr int _S_arity = 2;
6776 static constexpr bool _S_has_simple_extra_args =
true;
6779 inline constexpr _Chunk chunk;
6783#ifdef __cpp_lib_ranges_slide
6786 template<
typename _Vp>
6787 concept __slide_caches_nothing = random_access_range<_Vp> && sized_range<_Vp>;
6789 template<
typename _Vp>
6790 concept __slide_caches_last
6791 = !__slide_caches_nothing<_Vp> && bidirectional_range<_Vp> && common_range<_Vp>;
6793 template<
typename _Vp>
6794 concept __slide_caches_first
6795 = !__slide_caches_nothing<_Vp> && !__slide_caches_last<_Vp>;
6798 template<forward_range _Vp>
6803 range_difference_t<_Vp> _M_n;
6804 [[no_unique_address]]
6805 __detail::__maybe_present_t<__detail::__slide_caches_first<_Vp>,
6806 __detail::_CachedPosition<_Vp>, 0> _M_cached_begin;
6807 [[no_unique_address]]
6808 __detail::__maybe_present_t<__detail::__slide_caches_last<_Vp>,
6809 __detail::_CachedPosition<_Vp>, 1> _M_cached_end;
6811 template<
bool>
class _Iterator;
6816 slide_view(_Vp __base, range_difference_t<_Vp> __n)
6817 : _M_base(std::move(
__base)), _M_n(__n)
6818 { __glibcxx_assert(__n > 0); }
6823 base() const & requires copy_constructible<_Vp>
6831 begin()
requires (!(__detail::__simple_view<_Vp>
6832 && __detail::__slide_caches_nothing<const _Vp>))
6834 if constexpr (__detail::__slide_caches_first<_Vp>)
6836 iterator_t<_Vp> __it;
6837 if (_M_cached_begin._M_has_value())
6838 __it = _M_cached_begin._M_get(_M_base);
6841 __it = ranges::next(ranges::begin(_M_base), _M_n - 1, ranges::end(_M_base));
6842 _M_cached_begin._M_set(_M_base, __it);
6844 return _Iterator<false>(ranges::begin(_M_base),
std::move(__it), _M_n);
6847 return _Iterator<false>(ranges::begin(_M_base), _M_n);
6851 begin() const requires __detail::__slide_caches_nothing<const _Vp>
6852 {
return _Iterator<true>(ranges::begin(_M_base), _M_n); }
6855 end()
requires (!(__detail::__simple_view<_Vp>
6856 && __detail::__slide_caches_nothing<const _Vp>))
6858 if constexpr (__detail::__slide_caches_nothing<_Vp>)
6859 return _Iterator<false>(ranges::begin(_M_base) + range_difference_t<_Vp>(size()),
6861 else if constexpr (__detail::__slide_caches_last<_Vp>)
6863 iterator_t<_Vp> __it;
6864 if (_M_cached_end._M_has_value())
6865 __it = _M_cached_end._M_get(_M_base);
6868 __it = ranges::prev(ranges::end(_M_base), _M_n - 1, ranges::begin(_M_base));
6869 _M_cached_end._M_set(_M_base, __it);
6871 return _Iterator<false>(
std::move(__it), _M_n);
6873 else if constexpr (common_range<_Vp>)
6874 return _Iterator<false>(ranges::end(_M_base), ranges::end(_M_base), _M_n);
6876 return _Sentinel(ranges::end(_M_base));
6880 end() const requires __detail::__slide_caches_nothing<const _Vp>
6881 {
return begin() + range_difference_t<const _Vp>(size()); }
6884 size()
requires sized_range<_Vp>
6886 auto __sz = ranges::distance(_M_base) - _M_n + 1;
6889 return __detail::__to_unsigned_like(__sz);
6893 size() const requires sized_range<const _Vp>
6895 auto __sz = ranges::distance(_M_base) - _M_n + 1;
6898 return __detail::__to_unsigned_like(__sz);
6902 template<
typename _Range>
6903 slide_view(_Range&&, range_difference_t<_Range>) -> slide_view<views::all_t<_Range>>;
6905 template<
typename _Vp>
6906 inline constexpr bool enable_borrowed_range<slide_view<_Vp>>
6907 = enable_borrowed_range<_Vp>;
6909 template<forward_range _Vp>
6911 template<
bool _Const>
6912 class slide_view<_Vp>::_Iterator
6914 using _Base = __detail::__maybe_const_t<_Const, _Vp>;
6915 static constexpr bool _S_last_elt_present
6916 = __detail::__slide_caches_first<_Base>;
6918 iterator_t<_Base> _M_current = iterator_t<_Base>();
6919 [[no_unique_address]]
6920 __detail::__maybe_present_t<_S_last_elt_present, iterator_t<_Base>>
6921 _M_last_elt =
decltype(_M_last_elt)();
6922 range_difference_t<_Base> _M_n = 0;
6925 _Iterator(iterator_t<_Base> __current, range_difference_t<_Base> __n)
6926 requires (!_S_last_elt_present)
6927 : _M_current(__current), _M_n(__n)
6931 _Iterator(iterator_t<_Base> __current, iterator_t<_Base> __last_elt,
6932 range_difference_t<_Base> __n)
6933 requires _S_last_elt_present
6934 : _M_current(__current), _M_last_elt(__last_elt), _M_n(__n)
6940 if constexpr (random_access_range<_Base>)
6941 return random_access_iterator_tag{};
6942 else if constexpr (bidirectional_range<_Base>)
6943 return bidirectional_iterator_tag{};
6945 return forward_iterator_tag{};
6949 friend slide_view::_Sentinel;
6952 using iterator_category = input_iterator_tag;
6953 using iterator_concept =
decltype(_S_iter_concept());
6954 using value_type =
decltype(views::counted(_M_current, _M_n));
6955 using difference_type = range_difference_t<_Base>;
6957 _Iterator() =
default;
6960 _Iterator(_Iterator<!_Const> __i)
6961 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
6962 : _M_current(
std::move(__i._M_current)), _M_n(__i._M_n)
6967 {
return views::counted(_M_current, _M_n); }
6969 constexpr _Iterator&
6973 if constexpr (_S_last_elt_present)
6986 constexpr _Iterator&
6987 operator--()
requires bidirectional_range<_Base>
6990 if constexpr (_S_last_elt_present)
6996 operator--(
int)
requires bidirectional_range<_Base>
7003 constexpr _Iterator&
7004 operator+=(difference_type __x)
7005 requires random_access_range<_Base>
7008 if constexpr (_S_last_elt_present)
7013 constexpr _Iterator&
7014 operator-=(difference_type __x)
7015 requires random_access_range<_Base>
7018 if constexpr (_S_last_elt_present)
7024 operator[](difference_type __n)
const
7025 requires random_access_range<_Base>
7026 {
return views::counted(_M_current + __n, _M_n); }
7028 friend constexpr bool
7029 operator==(
const _Iterator& __x,
const _Iterator& __y)
7031 if constexpr (_S_last_elt_present)
7032 return __x._M_last_elt == __y._M_last_elt;
7034 return __x._M_current == __y._M_current;
7037 friend constexpr bool
7038 operator<(
const _Iterator& __x,
const _Iterator& __y)
7039 requires random_access_range<_Base>
7040 {
return __x._M_current < __y._M_current; }
7042 friend constexpr bool
7043 operator>(
const _Iterator& __x,
const _Iterator& __y)
7044 requires random_access_range<_Base>
7045 {
return __y < __x; }
7047 friend constexpr bool
7048 operator<=(
const _Iterator& __x,
const _Iterator& __y)
7049 requires random_access_range<_Base>
7050 {
return !(__y < __x); }
7052 friend constexpr bool
7053 operator>=(
const _Iterator& __x,
const _Iterator& __y)
7054 requires random_access_range<_Base>
7055 {
return !(__x < __y); }
7057 friend constexpr auto
7058 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
7059 requires random_access_range<_Base>
7060 && three_way_comparable<iterator_t<_Base>>
7061 {
return __x._M_current <=> __y._M_current; }
7063 friend constexpr _Iterator
7064 operator+(
const _Iterator& __i, difference_type __n)
7065 requires random_access_range<_Base>
7072 friend constexpr _Iterator
7073 operator+(difference_type __n,
const _Iterator& __i)
7074 requires random_access_range<_Base>
7081 friend constexpr _Iterator
7082 operator-(
const _Iterator& __i, difference_type __n)
7083 requires random_access_range<_Base>
7090 friend constexpr difference_type
7091 operator-(
const _Iterator& __x,
const _Iterator& __y)
7092 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
7094 if constexpr (_S_last_elt_present)
7095 return __x._M_last_elt - __y._M_last_elt;
7097 return __x._M_current - __y._M_current;
7101 template<forward_range _Vp>
7103 class slide_view<_Vp>::_Sentinel
7105 sentinel_t<_Vp> _M_end = sentinel_t<_Vp>();
7108 _Sentinel(sentinel_t<_Vp> __end)
7115 _Sentinel() =
default;
7117 friend constexpr bool
7118 operator==(
const _Iterator<false>& __x,
const _Sentinel& __y)
7119 {
return __x._M_last_elt == __y._M_end; }
7121 friend constexpr range_difference_t<_Vp>
7122 operator-(
const _Iterator<false>& __x,
const _Sentinel& __y)
7123 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
7124 {
return __x._M_last_elt - __y._M_end; }
7126 friend constexpr range_difference_t<_Vp>
7127 operator-(
const _Sentinel& __y,
const _Iterator<false>& __x)
7128 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
7129 {
return __y._M_end -__x._M_last_elt; }
7136 template<
typename _Range,
typename _Dp>
7137 concept __can_slide_view
7141 struct _Slide : __adaptor::_RangeAdaptor<_Slide>
7143 template<viewable_range _Range,
typename _Dp = range_difference_t<_Range>>
7144 requires __detail::__can_slide_view<_Range, _Dp>
7146 operator() [[nodiscard]] (_Range&& __r, type_identity_t<_Dp> __n)
const
7149 using __adaptor::_RangeAdaptor<_Slide>::operator();
7150 static constexpr int _S_arity = 2;
7151 static constexpr bool _S_has_simple_extra_args =
true;
7154 inline constexpr _Slide slide;
7158#ifdef __cpp_lib_ranges_chunk_by
7160 indirect_binary_predicate<iterator_t<_Vp>, iterator_t<_Vp>> _Pred>
7161 requires view<_Vp> && is_object_v<_Pred>
7162 class chunk_by_view :
public view_interface<chunk_by_view<_Vp, _Pred>>
7164 _Vp _M_base = _Vp();
7165 __detail::__box<_Pred> _M_pred;
7166 __detail::_CachedPosition<_Vp> _M_cached_begin;
7168 constexpr iterator_t<_Vp>
7169 _M_find_next(iterator_t<_Vp> __current)
7171 __glibcxx_assert(_M_pred.has_value());
7172 auto __pred = [
this]<
typename _Tp,
typename _Up>(_Tp&& __x, _Up&& __y) {
7175 auto __it = ranges::adjacent_find(__current, ranges::end(_M_base), __pred);
7176 return ranges::next(__it, 1, ranges::end(_M_base));
7179 constexpr iterator_t<_Vp>
7180 _M_find_prev(iterator_t<_Vp> __current)
requires bidirectional_range<_Vp>
7182 __glibcxx_assert(_M_pred.has_value());
7183 auto __pred = [
this]<
typename _Tp,
typename _Up>(_Tp&& __x, _Up&& __y) {
7188 __glibcxx_assert(__rbegin != __rend);
7189 auto __it = ranges::adjacent_find(__rbegin, __rend, __pred).base();
7190 return ranges::prev(__it, 1, ranges::begin(_M_base));
7196 chunk_by_view()
requires (default_initializable<_Vp>
7197 && default_initializable<_Pred>)
7201 chunk_by_view(_Vp __base, _Pred __pred)
7202 : _M_base(std::move(
__base)), _M_pred(std::move(__pred))
7206 base() const & requires copy_constructible<_Vp>
7213 constexpr const _Pred&
7215 {
return *_M_pred; }
7220 __glibcxx_assert(_M_pred.has_value());
7221 iterator_t<_Vp> __it;
7222 if (_M_cached_begin._M_has_value())
7223 __it = _M_cached_begin._M_get(_M_base);
7226 __it = _M_find_next(ranges::begin(_M_base));
7227 _M_cached_begin._M_set(_M_base, __it);
7229 return _Iterator(*
this, ranges::begin(_M_base), __it);
7235 if constexpr (common_range<_Vp>)
7236 return _Iterator(*
this, ranges::end(_M_base), ranges::end(_M_base));
7242 template<
typename _Range,
typename _Pred>
7243 chunk_by_view(_Range&&, _Pred) -> chunk_by_view<views::all_t<_Range>, _Pred>;
7246 indirect_binary_predicate<iterator_t<_Vp>, iterator_t<_Vp>> _Pred>
7247 requires view<_Vp> && is_object_v<_Pred>
7248 class chunk_by_view<_Vp, _Pred>::_Iterator
7250 chunk_by_view* _M_parent =
nullptr;
7251 iterator_t<_Vp> _M_current = iterator_t<_Vp>();
7252 iterator_t<_Vp> _M_next = iterator_t<_Vp>();
7255 _Iterator(chunk_by_view& __parent, iterator_t<_Vp> __current, iterator_t<_Vp> __next)
7256 : _M_parent(std::
__addressof(__parent)), _M_current(__current), _M_next(__next)
7262 if constexpr (bidirectional_range<_Vp>)
7263 return bidirectional_iterator_tag{};
7265 return forward_iterator_tag{};
7268 friend chunk_by_view;
7271 using value_type = subrange<iterator_t<_Vp>>;
7272 using difference_type = range_difference_t<_Vp>;
7273 using iterator_category = input_iterator_tag;
7274 using iterator_concept =
decltype(_S_iter_concept());
7276 _Iterator() =
default;
7278 constexpr value_type
7281 __glibcxx_assert(_M_current != _M_next);
7282 return ranges::subrange(_M_current, _M_next);
7285 constexpr _Iterator&
7288 __glibcxx_assert(_M_current != _M_next);
7289 _M_current = _M_next;
7290 _M_next = _M_parent->_M_find_next(_M_current);
7302 constexpr _Iterator&
7303 operator--()
requires bidirectional_range<_Vp>
7305 _M_next = _M_current;
7306 _M_current = _M_parent->_M_find_prev(_M_next);
7311 operator--(
int)
requires bidirectional_range<_Vp>
7318 friend constexpr bool
7319 operator==(
const _Iterator& __x,
const _Iterator& __y)
7320 {
return __x._M_current == __y._M_current; }
7322 friend constexpr bool
7323 operator==(
const _Iterator& __x, default_sentinel_t)
7324 {
return __x._M_current == __x._M_next; }
7331 template<
typename _Range,
typename _Pred>
7332 concept __can_chunk_by_view
7336 struct _ChunkBy : __adaptor::_RangeAdaptor<_ChunkBy>
7338 template<viewable_range _Range,
typename _Pred>
7339 requires __detail::__can_chunk_by_view<_Range, _Pred>
7341 operator() [[nodiscard]] (_Range&& __r, _Pred&& __pred)
const
7344 using __adaptor::_RangeAdaptor<_ChunkBy>::operator();
7345 static constexpr int _S_arity = 2;
7346 static constexpr bool _S_has_simple_extra_args =
true;
7349 inline constexpr _ChunkBy chunk_by;
7353#ifdef __cpp_lib_ranges_join_with
7356 template<
typename _Range,
typename _Pattern>
7357 concept __compatible_joinable_ranges
7358 = common_with<range_value_t<_Range>, range_value_t<_Pattern>>
7359 && common_reference_with<range_reference_t<_Range>,
7360 range_reference_t<_Pattern>>
7361 && common_reference_with<range_rvalue_reference_t<_Range>,
7362 range_rvalue_reference_t<_Pattern>>;
7364 template<
typename _Range>
7365 concept __bidirectional_common = bidirectional_range<_Range> && common_range<_Range>;
7368 template<input_range _Vp, forward_range _Pattern>
7371 && __detail::__compatible_joinable_ranges<range_reference_t<_Vp>, _Pattern>
7372 class join_with_view :
public view_interface<join_with_view<_Vp, _Pattern>>
7374 using _InnerRange = range_reference_t<_Vp>;
7376 _Vp _M_base = _Vp();
7377 [[no_unique_address]]
7378 __detail::__maybe_present_t<!forward_range<_Vp>,
7379 __detail::__non_propagating_cache<iterator_t<_Vp>>> _M_outer_it;
7380 __detail::__non_propagating_cache<remove_cv_t<_InnerRange>> _M_inner;
7381 _Pattern _M_pattern = _Pattern();
7383 template<
bool _Const>
using _Base = __detail::__maybe_const_t<_Const, _Vp>;
7384 template<
bool _Const>
using _InnerBase = range_reference_t<_Base<_Const>>;
7385 template<
bool _Const>
using _PatternBase = __detail::__maybe_const_t<_Const, _Pattern>;
7387 template<
bool _Const>
using _OuterIter = iterator_t<_Base<_Const>>;
7388 template<
bool _Const>
using _InnerIter = iterator_t<_InnerBase<_Const>>;
7389 template<
bool _Const>
using _PatternIter = iterator_t<_PatternBase<_Const>>;
7391 template<
bool _Const>
7392 static constexpr bool _S_ref_is_glvalue = is_reference_v<_InnerBase<_Const>>;
7394 template<
bool _Const>
7398 template<
bool _Const>
7399 requires _S_ref_is_glvalue<_Const>
7400 && forward_range<_Base<_Const>>
7401 && forward_range<_InnerBase<_Const>>
7402 struct __iter_cat<_Const>
7408 using _OuterIter = join_with_view::_OuterIter<_Const>;
7409 using _InnerIter = join_with_view::_InnerIter<_Const>;
7410 using _PatternIter = join_with_view::_PatternIter<_Const>;
7411 using _OuterCat =
typename iterator_traits<_OuterIter>::iterator_category;
7412 using _InnerCat =
typename iterator_traits<_InnerIter>::iterator_category;
7413 using _PatternCat =
typename iterator_traits<_PatternIter>::iterator_category;
7416 if constexpr (!is_reference_v<common_reference_t<iter_reference_t<_InnerIter>,
7417 iter_reference_t<_PatternIter>>>)
7418 return input_iterator_tag{};
7419 else if constexpr (derived_from<_OuterCat, bidirectional_iterator_tag>
7420 && derived_from<_InnerCat, bidirectional_iterator_tag>
7421 && derived_from<_PatternCat, bidirectional_iterator_tag>
7422 && common_range<_InnerBase<_Const>>
7423 && common_range<_PatternBase<_Const>>)
7424 return bidirectional_iterator_tag{};
7425 else if constexpr (derived_from<_OuterCat, forward_iterator_tag>
7426 && derived_from<_InnerCat, forward_iterator_tag>
7427 && derived_from<_PatternCat, forward_iterator_tag>)
7428 return forward_iterator_tag{};
7430 return input_iterator_tag{};
7433 using iterator_category =
decltype(_S_iter_cat());
7436 template<
bool>
class _Iterator;
7437 template<
bool>
class _Sentinel;
7440 join_with_view()
requires (default_initializable<_Vp>
7441 && default_initializable<_Pattern>)
7445 join_with_view(_Vp __base, _Pattern __pattern)
7446 : _M_base(std::move(
__base)), _M_pattern(std::move(__pattern))
7449 template<input_range _Range>
7450 requires constructible_from<_Vp, views::all_t<_Range>>
7451 && constructible_from<_Pattern, single_view<range_value_t<_InnerRange>>>
7453 join_with_view(_Range&& __r, range_value_t<_InnerRange> __e)
7454 : _M_base(views::all(std::
forward<_Range>(__r))),
7455 _M_pattern(views::single(std::move(__e)))
7459 base() const& requires copy_constructible<_Vp>
7469 if constexpr (forward_range<_Vp>)
7471 constexpr bool __use_const = is_reference_v<_InnerRange>
7472 && __detail::__simple_view<_Vp> && __detail::__simple_view<_Pattern>;
7473 return _Iterator<__use_const>{*
this, ranges::begin(_M_base)};
7477 _M_outer_it = ranges::begin(_M_base);
7478 return _Iterator<false>{*
this};
7484 requires forward_range<const _Vp>
7485 && forward_range<const _Pattern>
7486 && is_reference_v<range_reference_t<const _Vp>>
7487 && input_range<range_reference_t<const _Vp>>
7488 {
return _Iterator<true>{*
this, ranges::begin(_M_base)}; }
7493 constexpr bool __use_const
7494 = __detail::__simple_view<_Vp> && __detail::__simple_view<_Pattern>;
7495 if constexpr (is_reference_v<_InnerRange>
7496 && forward_range<_Vp> && common_range<_Vp>
7497 && forward_range<_InnerRange> && common_range<_InnerRange>)
7498 return _Iterator<__use_const>{*
this, ranges::end(_M_base)};
7500 return _Sentinel<__use_const>{*
this};
7505 requires forward_range<const _Vp>
7506 && forward_range<const _Pattern>
7507 && is_reference_v<range_reference_t<const _Vp>>
7508 && input_range<range_reference_t<const _Vp>>
7510 using _InnerConstRange = range_reference_t<const _Vp>;
7511 if constexpr (forward_range<_InnerConstRange>
7512 && common_range<const _Vp>
7513 && common_range<_InnerConstRange>)
7514 return _Iterator<true>{*
this, ranges::end(_M_base)};
7516 return _Sentinel<true>{*
this};
7520 template<
typename _Range,
typename _Pattern>
7521 join_with_view(_Range&&, _Pattern&&)
7522 -> join_with_view<views::all_t<_Range>, views::all_t<_Pattern>>;
7524 template<input_range _Range>
7525 join_with_view(_Range&&, range_value_t<range_reference_t<_Range>>)
7526 -> join_with_view<views::all_t<_Range>,
7529 template<input_range _Vp, forward_range _Pattern>
7532 && __detail::__compatible_joinable_ranges<range_reference_t<_Vp>, _Pattern>
7533 template<
bool _Const>
7534 class join_with_view<_Vp, _Pattern>::_Iterator :
public __iter_cat<_Const>
7536 using _Parent = __detail::__maybe_const_t<_Const, join_with_view>;
7537 using _Base = join_with_view::_Base<_Const>;
7538 using _InnerBase = join_with_view::_InnerBase<_Const>;
7539 using _PatternBase = join_with_view::_PatternBase<_Const>;
7541 using _OuterIter = join_with_view::_OuterIter<_Const>;
7542 using _InnerIter = join_with_view::_InnerIter<_Const>;
7543 using _PatternIter = join_with_view::_PatternIter<_Const>;
7545 static constexpr bool _S_ref_is_glvalue = join_with_view::_S_ref_is_glvalue<_Const>;
7547 _Parent* _M_parent =
nullptr;
7548 [[no_unique_address]]
7549 __detail::__maybe_present_t<forward_range<_Base>, _OuterIter> _M_outer_it
7550 =
decltype(_M_outer_it)();
7551 variant<_PatternIter, _InnerIter> _M_inner_it;
7553 constexpr _OuterIter&
7556 if constexpr (forward_range<_Base>)
7559 return *_M_parent->_M_outer_it;
7562 constexpr const _OuterIter&
7563 _M_get_outer()
const
7565 if constexpr (forward_range<_Base>)
7568 return *_M_parent->_M_outer_it;
7572 _Iterator(_Parent& __parent, _OuterIter __outer)
7573 requires forward_range<_Base>
7576 if (_M_get_outer() != ranges::end(_M_parent->_M_base))
7578 auto&& __inner = _M_update_inner();
7579 _M_inner_it.template emplace<1>(ranges::begin(__inner));
7585 _Iterator(_Parent& __parent)
7586 requires (!forward_range<_Base>)
7589 if (_M_get_outer() != ranges::end(_M_parent->_M_base))
7591 auto&& __inner = _M_update_inner();
7592 _M_inner_it.template emplace<1>(ranges::begin(__inner));
7600 _OuterIter& __outer = _M_get_outer();
7601 if constexpr (_S_ref_is_glvalue)
7602 return __detail::__as_lvalue(*__outer);
7604 return _M_parent->_M_inner._M_emplace_deref(__outer);
7610 if constexpr (_S_ref_is_glvalue)
7611 return __detail::__as_lvalue(*_M_get_outer());
7613 return *_M_parent->_M_inner;
7621 if (_M_inner_it.index() == 0)
7623 if (std::get<0>(_M_inner_it) != ranges::end(_M_parent->_M_pattern))
7626 auto&& __inner = _M_update_inner();
7627 _M_inner_it.template emplace<1>(ranges::begin(__inner));
7631 auto&& __inner = _M_get_inner();
7632 if (std::get<1>(_M_inner_it) != ranges::end(__inner))
7635 if (++_M_get_outer() == ranges::end(_M_parent->_M_base))
7637 if constexpr (_S_ref_is_glvalue)
7638 _M_inner_it.template emplace<0>();
7642 _M_inner_it.template emplace<0>(ranges::begin(_M_parent->_M_pattern));
7650 if constexpr (_S_ref_is_glvalue
7651 && bidirectional_range<_Base>
7652 && __detail::__bidirectional_common<_InnerBase>
7653 && __detail::__bidirectional_common<_PatternBase>)
7654 return bidirectional_iterator_tag{};
7655 else if constexpr (_S_ref_is_glvalue
7656 && forward_range<_Base>
7657 && forward_range<_InnerBase>)
7658 return forward_iterator_tag{};
7660 return input_iterator_tag{};
7663 friend join_with_view;
7666 using iterator_concept =
decltype(_S_iter_concept());
7668 using value_type = common_type_t<iter_value_t<_InnerIter>,
7669 iter_value_t<_PatternIter>>;
7670 using difference_type = common_type_t<iter_difference_t<_OuterIter>,
7671 iter_difference_t<_InnerIter>,
7672 iter_difference_t<_PatternIter>>;
7674 _Iterator() =
default;
7677 _Iterator(_Iterator<!_Const> __i)
7679 && convertible_to<iterator_t<_Vp>, _OuterIter>
7680 && convertible_to<iterator_t<_InnerRange>, _InnerIter>
7681 && convertible_to<iterator_t<_Pattern>, _PatternIter>
7682 : _M_parent(__i._M_parent),
7685 if (__i._M_inner_it.index() == 0)
7686 _M_inner_it.template emplace<0>(std::get<0>(
std::move(__i._M_inner_it)));
7688 _M_inner_it.template emplace<1>(std::get<1>(
std::move(__i._M_inner_it)));
7691 constexpr common_reference_t<iter_reference_t<_InnerIter>,
7692 iter_reference_t<_PatternIter>>
7695 if (_M_inner_it.index() == 0)
7696 return *std::get<0>(_M_inner_it);
7698 return *std::get<1>(_M_inner_it);
7701 constexpr _Iterator&
7704 if (_M_inner_it.index() == 0)
7705 ++std::get<0>(_M_inner_it);
7707 ++std::get<1>(_M_inner_it);
7718 requires _S_ref_is_glvalue
7719 && forward_iterator<_OuterIter> && forward_iterator<_InnerIter>
7721 _Iterator __tmp = *
this;
7726 constexpr _Iterator&
7728 requires _S_ref_is_glvalue
7729 && bidirectional_range<_Base>
7730 && __detail::__bidirectional_common<_InnerBase>
7731 && __detail::__bidirectional_common<_PatternBase>
7733 if (_M_outer_it == ranges::end(_M_parent->_M_base))
7735 auto&& __inner = *--_M_outer_it;
7736 _M_inner_it.template emplace<1>(ranges::end(__inner));
7741 if (_M_inner_it.index() == 0)
7743 auto& __it = std::get<0>(_M_inner_it);
7744 if (__it == ranges::begin(_M_parent->_M_pattern))
7746 auto&& __inner = *--_M_outer_it;
7747 _M_inner_it.template emplace<1>(ranges::end(__inner));
7754 auto& __it = std::get<1>(_M_inner_it);
7755 auto&& __inner = *_M_outer_it;
7756 if (__it == ranges::begin(__inner))
7757 _M_inner_it.template emplace<0>(ranges::end(_M_parent->_M_pattern));
7763 if (_M_inner_it.index() == 0)
7764 --std::get<0>(_M_inner_it);
7766 --std::get<1>(_M_inner_it);
7772 requires _S_ref_is_glvalue && bidirectional_range<_Base>
7773 && __detail::__bidirectional_common<_InnerBase>
7774 && __detail::__bidirectional_common<_PatternBase>
7776 _Iterator __tmp = *
this;
7781 friend constexpr bool
7782 operator==(
const _Iterator& __x,
const _Iterator& __y)
7783 requires _S_ref_is_glvalue
7784 && forward_range<_Base> && equality_comparable<_InnerIter>
7785 {
return __x._M_outer_it == __y._M_outer_it && __x._M_inner_it ==__y._M_inner_it; }
7787 friend constexpr common_reference_t<iter_rvalue_reference_t<_InnerIter>,
7788 iter_rvalue_reference_t<_PatternIter>>
7789 iter_move(
const _Iterator& __x)
7791 if (__x._M_inner_it.index() == 0)
7792 return ranges::iter_move(std::get<0>(__x._M_inner_it));
7794 return ranges::iter_move(std::get<1>(__x._M_inner_it));
7797 friend constexpr void
7798 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
7799 requires indirectly_swappable<_InnerIter, _PatternIter>
7801 if (__x._M_inner_it.index() == 0)
7803 if (__y._M_inner_it.index() == 0)
7804 ranges::iter_swap(std::get<0>(__x._M_inner_it), std::get<0>(__y._M_inner_it));
7806 ranges::iter_swap(std::get<0>(__x._M_inner_it), std::get<1>(__y._M_inner_it));
7810 if (__y._M_inner_it.index() == 0)
7811 ranges::iter_swap(std::get<1>(__x._M_inner_it), std::get<0>(__y._M_inner_it));
7813 ranges::iter_swap(std::get<1>(__x._M_inner_it), std::get<1>(__y._M_inner_it));
7818 template<input_range _Vp, forward_range _Pattern>
7821 && __detail::__compatible_joinable_ranges<range_reference_t<_Vp>, _Pattern>
7822 template<
bool _Const>
7823 class join_with_view<_Vp, _Pattern>::_Sentinel
7825 using _Parent = __detail::__maybe_const_t<_Const, join_with_view>;
7826 using _Base = join_with_view::_Base<_Const>;
7828 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
7831 _Sentinel(_Parent& __parent)
7832 : _M_end(ranges::end(__parent._M_base))
7835 friend join_with_view;
7838 _Sentinel() =
default;
7841 _Sentinel(_Sentinel<!_Const> __s)
7842 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
7846 template<
bool _OtherConst>
7847 requires sentinel_for<sentinel_t<_Base>,
7848 iterator_t<__detail::__maybe_const_t<_OtherConst, _Vp>>>
7849 friend constexpr bool
7850 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
7851 {
return __x._M_get_outer() == __y._M_end; }
7858 template<
typename _Range,
typename _Pattern>
7859 concept __can_join_with_view
7863 struct _JoinWith : __adaptor::_RangeAdaptor<_JoinWith>
7865 template<viewable_range _Range,
typename _Pattern>
7866 requires __detail::__can_join_with_view<_Range, _Pattern>
7868 operator() [[nodiscard]] (_Range&& __r, _Pattern&& __f)
const
7873 using _RangeAdaptor<_JoinWith>::operator();
7874 static constexpr int _S_arity = 2;
7875 template<
typename _Pattern>
7876 static constexpr bool _S_has_simple_extra_args
7877 = _LazySplit::_S_has_simple_extra_args<_Pattern>;
7880 inline constexpr _JoinWith join_with;
7884#ifdef __cpp_lib_ranges_repeat
7885 template<move_constructible _Tp, semiregular _Bound = unreachable_sentinel_t>
7886 requires is_object_v<_Tp> && same_as<_Tp, remove_cv_t<_Tp>>
7887 && (__detail::__is_integer_like<_Bound> || same_as<_Bound, unreachable_sentinel_t>)
7888 class repeat_view : public
view_interface<repeat_view<_Tp, _Bound>>
7890 [[no_unique_address]] __detail::__box<_Tp> _M_value;
7891 [[no_unique_address]] _Bound _M_bound = _Bound();
7895 template<
typename _Range>
7896 friend constexpr auto
7897 views::__detail::__take_of_repeat_view(_Range&&, range_difference_t<_Range>);
7899 template<
typename _Range>
7900 friend constexpr auto
7901 views::__detail::__drop_of_repeat_view(_Range&&, range_difference_t<_Range>);
7904 repeat_view()
requires default_initializable<_Tp> = default;
7907 repeat_view(const _Tp& __value, _Bound __bound = _Bound())
7908 requires copy_constructible<_Tp>
7909 : _M_value(__value), _M_bound(__bound)
7911 if constexpr (!same_as<_Bound, unreachable_sentinel_t>)
7912 __glibcxx_assert(__bound >= 0);
7916 repeat_view(_Tp&& __value, _Bound __bound = _Bound())
7917 : _M_value(std::move(__value)), _M_bound(__bound)
7920 template<
typename... _Args,
typename... _BoundArgs>
7921 requires constructible_from<_Tp, _Args...>
7922 && constructible_from<_Bound, _BoundArgs...>
7924 repeat_view(piecewise_construct_t,
7925 tuple<_Args...> __args,
7926 tuple<_BoundArgs...> __bound_args = tuple<>{})
7927 : _M_value(std::make_from_tuple<_Tp>(std::move(__args))),
7928 _M_bound(std::make_from_tuple<_Bound>(std::move(__bound_args)))
7936 end() const requires (!same_as<_Bound, unreachable_sentinel_t>)
7939 constexpr unreachable_sentinel_t
7940 end() const noexcept
7941 {
return unreachable_sentinel; }
7944 size() const requires (!same_as<_Bound, unreachable_sentinel_t>)
7945 {
return __detail::__to_unsigned_like(_M_bound); }
7950 template<
typename _Tp,
typename _Bound = unreachable_sentinel_t>
7951 repeat_view(_Tp, _Bound = _Bound()) -> repeat_view<_Tp, _Bound>;
7953 template<move_constructible _Tp, semiregular _Bound>
7954 requires is_object_v<_Tp> && same_as<_Tp, remove_cv_t<_Tp>>
7955 && (__detail::__is_integer_like<_Bound> || same_as<_Bound, unreachable_sentinel_t>)
7956 class repeat_view<_Tp, _Bound>::_Iterator
7959 = __conditional_t<same_as<_Bound, unreachable_sentinel_t>, ptrdiff_t, _Bound>;
7961 const _Tp* _M_value =
nullptr;
7962 __index_type _M_current = __index_type();
7965 _Iterator(
const _Tp* __value, __index_type __bound = __index_type())
7966 : _M_value(__value), _M_current(__bound)
7968 if constexpr (!same_as<_Bound, unreachable_sentinel_t>)
7969 __glibcxx_assert(__bound >= 0);
7975 using iterator_concept = random_access_iterator_tag;
7976 using iterator_category = random_access_iterator_tag;
7977 using value_type = _Tp;
7978 using difference_type = __conditional_t<__detail::__is_signed_integer_like<__index_type>,
7980 __detail::__iota_diff_t<__index_type>>;
7982 _Iterator() =
default;
7984 constexpr const _Tp&
7986 {
return *_M_value; }
7988 constexpr _Iterator&
8003 constexpr _Iterator&
8006 if constexpr (!same_as<_Bound, unreachable_sentinel_t>)
8007 __glibcxx_assert(_M_current > 0);
8020 constexpr _Iterator&
8021 operator+=(difference_type __n)
8023 if constexpr (!same_as<_Bound, unreachable_sentinel_t>)
8024 __glibcxx_assert(_M_current + __n >= 0);
8029 constexpr _Iterator&
8030 operator-=(difference_type __n)
8032 if constexpr (!same_as<_Bound, unreachable_sentinel_t>)
8033 __glibcxx_assert(_M_current - __n >= 0);
8038 constexpr const _Tp&
8039 operator[](difference_type __n)
const noexcept
8040 {
return *(*
this + __n); }
8042 friend constexpr bool
8043 operator==(
const _Iterator& __x,
const _Iterator& __y)
8044 {
return __x._M_current == __y._M_current; }
8046 friend constexpr auto
8047 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
8048 {
return __x._M_current <=> __y._M_current; }
8050 friend constexpr _Iterator
8051 operator+(_Iterator __i, difference_type __n)
8057 friend constexpr _Iterator
8058 operator+(difference_type __n, _Iterator __i)
8059 {
return __i + __n; }
8061 friend constexpr _Iterator
8062 operator-(_Iterator __i, difference_type __n)
8068 friend constexpr difference_type
8069 operator-(
const _Iterator& __x,
const _Iterator& __y)
8071 return (
static_cast<difference_type
>(__x._M_current)
8072 -
static_cast<difference_type
>(__y._M_current));
8080 template<
typename _Tp,
typename _Bound>
8081 inline constexpr bool __is_repeat_view<repeat_view<_Tp, _Bound>> =
true;
8083 template<
typename _Tp>
8084 concept __can_repeat_view
8087 template<
typename _Tp,
typename _Bound>
8088 concept __can_bounded_repeat_view
8094 template<
typename _Tp>
8095 requires __detail::__can_repeat_view<_Tp>
8097 operator() [[nodiscard]] (_Tp&& __value)
const
8104 template<
typename _Tp,
typename _Bound>
8105 requires __detail::__can_bounded_repeat_view<_Tp, _Bound>
8107 operator() [[nodiscard]] (_Tp&& __value, _Bound __bound)
const
8111 inline constexpr _Repeat repeat;
8115 template<
typename _Range>
8117 __take_of_repeat_view(_Range&& __r, range_difference_t<_Range> __n)
8119 using _Tp = remove_cvref_t<_Range>;
8120 static_assert(__is_repeat_view<_Tp>);
8121 if constexpr (sized_range<_Tp>)
8123 std::min(ranges::distance(__r), __n));
8128 template<
typename _Range>
8130 __drop_of_repeat_view(_Range&& __r, range_difference_t<_Range> __n)
8132 using _Tp = remove_cvref_t<_Range>;
8133 static_assert(__is_repeat_view<_Tp>);
8134 if constexpr (sized_range<_Tp>)
8136 auto __sz = ranges::distance(__r);
8147#ifdef __cpp_lib_ranges_stride
8148 template<input_range _Vp>
8153 range_difference_t<_Vp> _M_stride;
8155 template<
bool _Const>
using _Base = __detail::__maybe_const_t<_Const, _Vp>;
8157 template<
bool _Const>
8161 template<
bool _Const>
8162 requires forward_range<_Base<_Const>>
8163 struct __iter_cat<_Const>
8169 using _Cat =
typename iterator_traits<iterator_t<_Base<_Const>>>::iterator_category;
8170 if constexpr (derived_from<_Cat, random_access_iterator_tag>)
8171 return random_access_iterator_tag{};
8176 using iterator_category =
decltype(_S_iter_cat());
8179 template<
bool>
class _Iterator;
8183 stride_view(_Vp __base, range_difference_t<_Vp> __stride)
8184 : _M_base(std::move(
__base)), _M_stride(__stride)
8185 { __glibcxx_assert(__stride > 0); }
8188 base() const& requires copy_constructible<_Vp>
8195 constexpr range_difference_t<_Vp>
8196 stride() const noexcept
8197 {
return _M_stride; }
8200 begin()
requires (!__detail::__simple_view<_Vp>)
8201 {
return _Iterator<false>(
this, ranges::begin(_M_base)); }
8204 begin() const requires range<const _Vp>
8205 {
return _Iterator<true>(
this, ranges::begin(_M_base)); }
8208 end()
requires (!__detail::__simple_view<_Vp>)
8210 if constexpr (common_range<_Vp> && sized_range<_Vp> && forward_range<_Vp>)
8212 auto __missing = (_M_stride - ranges::distance(_M_base) % _M_stride) % _M_stride;
8213 return _Iterator<false>(
this, ranges::end(_M_base), __missing);
8215 else if constexpr (common_range<_Vp> && !bidirectional_range<_Vp>)
8216 return _Iterator<false>(
this, ranges::end(_M_base));
8222 end() const requires range<const _Vp>
8224 if constexpr (common_range<const _Vp> && sized_range<const _Vp>
8225 && forward_range<const _Vp>)
8227 auto __missing = (_M_stride - ranges::distance(_M_base) % _M_stride) % _M_stride;
8228 return _Iterator<true>(
this, ranges::end(_M_base), __missing);
8230 else if constexpr (common_range<const _Vp> && !bidirectional_range<const _Vp>)
8231 return _Iterator<true>(
this, ranges::end(_M_base));
8237 size()
requires sized_range<_Vp>
8239 return __detail::__to_unsigned_like
8240 (__detail::__div_ceil(ranges::distance(_M_base), _M_stride));
8244 size() const requires sized_range<const _Vp>
8246 return __detail::__to_unsigned_like
8247 (__detail::__div_ceil(ranges::distance(_M_base), _M_stride));
8251 template<
typename _Range>
8252 stride_view(_Range&&, range_difference_t<_Range>) -> stride_view<views::all_t<_Range>>;
8254 template<
typename _Vp>
8255 inline constexpr bool enable_borrowed_range<stride_view<_Vp>>
8256 = enable_borrowed_range<_Vp>;
8258 template<input_range _Vp>
8260 template<
bool _Const>
8261 class stride_view<_Vp>::_Iterator :
public __iter_cat<_Const>
8263 using _Parent = __detail::__maybe_const_t<_Const, stride_view>;
8264 using _Base = stride_view::_Base<_Const>;
8266 iterator_t<_Base> _M_current = iterator_t<_Base>();
8267 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
8268 range_difference_t<_Base> _M_stride = 0;
8269 range_difference_t<_Base> _M_missing = 0;
8272 _Iterator(_Parent* __parent, iterator_t<_Base> __current,
8273 range_difference_t<_Base> __missing = 0)
8274 : _M_current(std::move(__current)), _M_end(ranges::end(__parent->_M_base)),
8275 _M_stride(__parent->_M_stride), _M_missing(__missing)
8281 if constexpr (random_access_range<_Base>)
8282 return random_access_iterator_tag{};
8283 else if constexpr (bidirectional_range<_Base>)
8284 return bidirectional_iterator_tag{};
8285 else if constexpr (forward_range<_Base>)
8286 return forward_iterator_tag{};
8288 return input_iterator_tag{};
8294 using difference_type = range_difference_t<_Base>;
8295 using value_type = range_value_t<_Base>;
8296 using iterator_concept =
decltype(_S_iter_concept());
8299 _Iterator()
requires default_initializable<iterator_t<_Base>> = default;
8302 _Iterator(_Iterator<!_Const> __other)
8304 && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
8305 && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
8306 : _M_current(std::move(__other._M_current)), _M_end(std::move(__other._M_end)),
8307 _M_stride(__other._M_stride), _M_missing(__other._M_missing)
8310 constexpr iterator_t<_Base>
8314 constexpr const iterator_t<_Base>&
8315 base() const & noexcept
8316 {
return _M_current; }
8318 constexpr decltype(
auto)
8320 {
return *_M_current; }
8322 constexpr _Iterator&
8325 __glibcxx_assert(_M_current != _M_end);
8326 _M_missing = ranges::advance(_M_current, _M_stride, _M_end);
8335 operator++(
int)
requires forward_range<_Base>
8342 constexpr _Iterator&
8343 operator--()
requires bidirectional_range<_Base>
8345 ranges::advance(_M_current, _M_missing - _M_stride);
8351 operator--(
int)
requires bidirectional_range<_Base>
8358 constexpr _Iterator&
8359 operator+=(difference_type __n)
requires random_access_range<_Base>
8363 __glibcxx_assert(ranges::distance(_M_current, _M_end) > _M_stride * (__n - 1));
8364 _M_missing = ranges::advance(_M_current, _M_stride * __n, _M_end);
8368 ranges::advance(_M_current, _M_stride * __n + _M_missing);
8374 constexpr _Iterator&
8375 operator-=(difference_type __n)
requires random_access_range<_Base>
8376 {
return *
this += -__n; }
8378 constexpr decltype(
auto)
operator[](difference_type __n)
const
8379 requires random_access_range<_Base>
8380 {
return *(*
this + __n); }
8382 friend constexpr bool
8383 operator==(
const _Iterator& __x, default_sentinel_t)
8384 {
return __x._M_current == __x._M_end; }
8386 friend constexpr bool
8387 operator==(
const _Iterator& __x,
const _Iterator& __y)
8388 requires equality_comparable<iterator_t<_Base>>
8389 {
return __x._M_current == __y._M_current; }
8391 friend constexpr bool
8392 operator<(
const _Iterator& __x,
const _Iterator& __y)
8393 requires random_access_range<_Base>
8394 {
return __x._M_current < __y._M_current; }
8396 friend constexpr bool
8397 operator>(
const _Iterator& __x,
const _Iterator& __y)
8398 requires random_access_range<_Base>
8399 {
return __y._M_current < __x._M_current; }
8401 friend constexpr bool
8402 operator<=(
const _Iterator& __x,
const _Iterator& __y)
8403 requires random_access_range<_Base>
8404 {
return !(__y._M_current < __x._M_current); }
8406 friend constexpr bool
8407 operator>=(
const _Iterator& __x,
const _Iterator& __y)
8408 requires random_access_range<_Base>
8409 {
return !(__x._M_current < __y._M_current); }
8411 friend constexpr auto
8412 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
8413 requires random_access_range<_Base> && three_way_comparable<iterator_t<_Base>>
8414 {
return __x._M_current <=> __y._M_current; }
8416 friend constexpr _Iterator
8417 operator+(
const _Iterator& __i, difference_type __n)
8418 requires random_access_range<_Base>
8425 friend constexpr _Iterator
8426 operator+(difference_type __n,
const _Iterator& __i)
8427 requires random_access_range<_Base>
8428 {
return __i + __n; }
8430 friend constexpr _Iterator
8431 operator-(
const _Iterator& __i, difference_type __n)
8432 requires random_access_range<_Base>
8439 friend constexpr difference_type
8440 operator-(
const _Iterator& __x,
const _Iterator& __y)
8441 requires sized_sentinel_for<iterator_t<_Base>, iterator_t<_Base>>
8443 auto __n = __x._M_current - __y._M_current;
8444 if constexpr (forward_range<_Base>)
8445 return (__n + __x._M_missing - __y._M_missing) / __x._M_stride;
8447 return -__detail::__div_ceil(-__n, __x._M_stride);
8449 return __detail::__div_ceil(__n, __x._M_stride);
8452 friend constexpr difference_type
8453 operator-(default_sentinel_t,
const _Iterator& __x)
8454 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
8455 {
return __detail::__div_ceil(__x._M_end - __x._M_current, __x._M_stride); }
8457 friend constexpr difference_type
8458 operator-(
const _Iterator& __x, default_sentinel_t __y)
8459 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
8460 {
return -(__y - __x); }
8462 friend constexpr range_rvalue_reference_t<_Base>
8463 iter_move(
const _Iterator& __i)
8464 noexcept(
noexcept(ranges::iter_move(__i._M_current)))
8465 {
return ranges::iter_move(__i._M_current); }
8467 friend constexpr void
8468 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
8469 noexcept(
noexcept(ranges::iter_swap(__x._M_current, __y._M_current)))
8470 requires indirectly_swappable<iterator_t<_Base>>
8471 { ranges::iter_swap(__x._M_current, __y._M_current); }
8478 template<
typename _Range,
typename _Dp>
8479 concept __can_stride_view
8483 struct _Stride : __adaptor::_RangeAdaptor<_Stride>
8485 template<viewable_range _Range,
typename _Dp = range_difference_t<_Range>>
8486 requires __detail::__can_stride_view<_Range, _Dp>
8488 operator() [[nodiscard]] (_Range&& __r, type_identity_t<_Dp> __n)
const
8491 using __adaptor::_RangeAdaptor<_Stride>::operator();
8492 static constexpr int _S_arity = 2;
8493 static constexpr bool _S_has_simple_extra_args =
true;
8496 inline constexpr _Stride stride;
8500#ifdef __cpp_lib_ranges_cartesian_product
8503 template<
bool _Const,
typename _First,
typename... _Vs>
8504 concept __cartesian_product_is_random_access
8505 = (random_access_range<__maybe_const_t<_Const, _First>>
8507 && (random_access_range<__maybe_const_t<_Const, _Vs>>
8508 && sized_range<__maybe_const_t<_Const, _Vs>>));
8510 template<
typename _Range>
8511 concept __cartesian_product_common_arg
8512 = common_range<_Range> || (sized_range<_Range> && random_access_range<_Range>);
8514 template<
bool _Const,
typename _First,
typename... _Vs>
8515 concept __cartesian_product_is_bidirectional
8516 = (bidirectional_range<__maybe_const_t<_Const, _First>>
8518 && (bidirectional_range<__maybe_const_t<_Const, _Vs>>
8519 && __cartesian_product_common_arg<__maybe_const_t<_Const, _Vs>>));
8521 template<
typename _First,
typename... _Vs>
8522 concept __cartesian_product_is_common = __cartesian_product_common_arg<_First>;
8524 template<
typename... _Vs>
8525 concept __cartesian_product_is_sized = (sized_range<_Vs> && ...);
8527 template<
bool _Const,
template<
typename>
class _FirstSent,
8528 typename _First,
typename... _Vs>
8529 concept __cartesian_is_sized_sentinel
8530 = (sized_sentinel_for<_FirstSent<__maybe_const_t<_Const, _First>>,
8531 iterator_t<__maybe_const_t<_Const, _First>>>
8533 && (sized_range<__maybe_const_t<_Const, _Vs>>
8534 && sized_sentinel_for<iterator_t<__maybe_const_t<_Const, _Vs>>,
8535 iterator_t<__maybe_const_t<_Const, _Vs>>>));
8537 template<__cartesian_product_common_arg _Range>
8539 __cartesian_common_arg_end(_Range& __r)
8541 if constexpr (common_range<_Range>)
8542 return ranges::end(__r);
8544 return ranges::begin(__r) + ranges::distance(__r);
8550 class cartesian_product_view : public
view_interface<cartesian_product_view<_First, _Vs...>>
8552 tuple<_First, _Vs...> _M_bases;
8554 template<
bool>
class _Iterator;
8557 _S_difference_type()
8563 range_difference_t<_First>,
8564 range_difference_t<_Vs>...>{};
8568 cartesian_product_view() =
default;
8571 cartesian_product_view(_First __first, _Vs... __rest)
8572 : _M_bases(std::move(__first), std::move(__rest)...)
8575 constexpr _Iterator<false>
8576 begin()
requires (!__detail::__simple_view<_First> || ... || !__detail::__simple_view<_Vs>)
8577 {
return _Iterator<false>(*
this, __detail::__tuple_transform(ranges::begin, _M_bases)); }
8579 constexpr _Iterator<true>
8580 begin() const requires (
range<const _First> && ... &&
range<const _Vs>)
8581 {
return _Iterator<true>(*
this, __detail::__tuple_transform(ranges::begin, _M_bases)); }
8583 constexpr _Iterator<false>
8584 end()
requires ((!__detail::__simple_view<_First> || ... || !__detail::__simple_view<_Vs>)
8585 && __detail::__cartesian_product_is_common<_First, _Vs...>)
8588 using _Ret = tuple<iterator_t<_First>, iterator_t<_Vs>...>;
8589 bool __empty_tail = (ranges::empty(std::get<1 + _Is>(_M_bases)) || ...);
8590 auto& __first = std::get<0>(_M_bases);
8591 return _Ret{(__empty_tail
8592 ? ranges::begin(__first)
8593 : __detail::__cartesian_common_arg_end(__first)),
8594 ranges::
begin(std::get<1 + _Is>(_M_bases))...};
8597 return _Iterator<false>{*
this,
std::move(__its)};
8600 constexpr _Iterator<true>
8601 end() const requires __detail::__cartesian_product_is_common<const _First, const _Vs...>
8604 using _Ret = tuple<iterator_t<const _First>, iterator_t<const _Vs>...>;
8605 bool __empty_tail = (ranges::empty(std::get<1 + _Is>(_M_bases)) || ...);
8606 auto& __first = std::get<0>(_M_bases);
8607 return _Ret{(__empty_tail
8608 ? ranges::begin(__first)
8609 : __detail::__cartesian_common_arg_end(__first)),
8610 ranges::
begin(std::get<1 + _Is>(_M_bases))...};
8613 return _Iterator<true>{*
this,
std::move(__its)};
8616 constexpr default_sentinel_t
8617 end() const noexcept
8621 size()
requires __detail::__cartesian_product_is_sized<_First, _Vs...>
8623 using _ST = __detail::__make_unsigned_like_t<
decltype(_S_difference_type())>;
8625 auto __size =
static_cast<_ST
>(1);
8626#ifdef _GLIBCXX_ASSERTIONS
8627 if constexpr (integral<_ST>)
8630 = (__builtin_mul_overflow(__size,
8631 static_cast<_ST
>(ranges::size(std::get<_Is>(_M_bases))),
8634 __glibcxx_assert(!__overflow);
8638 __size = (
static_cast<_ST
>(ranges::size(std::get<_Is>(_M_bases))) * ...);
8644 size() const requires __detail::__cartesian_product_is_sized<const _First, const _Vs...>
8646 using _ST = __detail::__make_unsigned_like_t<
decltype(_S_difference_type())>;
8648 auto __size =
static_cast<_ST
>(1);
8649#ifdef _GLIBCXX_ASSERTIONS
8650 if constexpr (integral<_ST>)
8653 = (__builtin_mul_overflow(__size,
8654 static_cast<_ST
>(ranges::size(std::get<_Is>(_M_bases))),
8657 __glibcxx_assert(!__overflow);
8661 __size = (
static_cast<_ST
>(ranges::size(std::get<_Is>(_M_bases))) * ...);
8667 template<
typename... _Vs>
8668 cartesian_product_view(_Vs&&...) -> cartesian_product_view<views::all_t<_Vs>...>;
8672 template<bool _Const>
8673 class cartesian_product_view<_First, _Vs...>::_Iterator
8675 using _Parent = __maybe_const_t<_Const, cartesian_product_view>;
8676 _Parent* _M_parent =
nullptr;
8677 tuple<iterator_t<__maybe_const_t<_Const, _First>>,
8678 iterator_t<__maybe_const_t<_Const, _Vs>>...> _M_current;
8681 _Iterator(_Parent& __parent,
decltype(_M_current) __current)
8683 _M_current(std::move(__current))
8689 if constexpr (__detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>)
8690 return random_access_iterator_tag{};
8691 else if constexpr (__detail::__cartesian_product_is_bidirectional<_Const, _First, _Vs...>)
8692 return bidirectional_iterator_tag{};
8694 return forward_iterator_tag{};
8696 return input_iterator_tag{};
8699 friend cartesian_product_view;
8702 using iterator_category = input_iterator_tag;
8703 using iterator_concept =
decltype(_S_iter_concept());
8705 = tuple<range_value_t<__maybe_const_t<_Const, _First>>,
8706 range_value_t<__maybe_const_t<_Const, _Vs>>...>;
8708 = tuple<range_reference_t<__maybe_const_t<_Const, _First>>,
8709 range_reference_t<__maybe_const_t<_Const, _Vs>>...>;
8710 using difference_type =
decltype(cartesian_product_view::_S_difference_type());
8712 _Iterator() =
default;
8715 _Iterator(_Iterator<!_Const> __i)
8717 && (convertible_to<iterator_t<_First>, iterator_t<const _First>>
8718 && ... && convertible_to<iterator_t<_Vs>, iterator_t<const _Vs>>)
8720 _M_current(std::move(__i._M_current))
8726 auto __f = [](
auto& __i) ->
decltype(
auto) {
8729 return __detail::__tuple_transform(__f, _M_current);
8732 constexpr _Iterator&
8751 constexpr _Iterator&
8753 requires __detail::__cartesian_product_is_bidirectional<_Const, _First, _Vs...>
8761 requires __detail::__cartesian_product_is_bidirectional<_Const, _First, _Vs...>
8768 constexpr _Iterator&
8769 operator+=(difference_type __x)
8770 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8776 constexpr _Iterator&
8777 operator-=(difference_type __x)
8778 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8779 {
return *
this += -__x; }
8782 operator[](difference_type __n)
const
8783 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8784 {
return *((*this) + __n); }
8786 friend constexpr bool
8787 operator==(
const _Iterator& __x,
const _Iterator& __y)
8788 requires equality_comparable<iterator_t<__maybe_const_t<_Const, _First>>>
8789 {
return __x._M_current == __y._M_current; }
8791 friend constexpr bool
8792 operator==(
const _Iterator& __x, default_sentinel_t)
8795 return ((std::get<_Is>(__x._M_current)
8796 == ranges::end(std::get<_Is>(__x._M_parent->_M_bases)))
8801 friend constexpr auto
8802 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
8803 requires __detail::__all_random_access<_Const, _First, _Vs...>
8804 {
return __x._M_current <=> __y._M_current; }
8806 friend constexpr _Iterator
8807 operator+(_Iterator __x, difference_type __y)
8808 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8809 {
return __x += __y; }
8811 friend constexpr _Iterator
8812 operator+(difference_type __x, _Iterator __y)
8813 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8814 {
return __y += __x; }
8816 friend constexpr _Iterator
8817 operator-(_Iterator __x, difference_type __y)
8818 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8819 {
return __x -= __y; }
8821 friend constexpr difference_type
8822 operator-(
const _Iterator& __x,
const _Iterator& __y)
8823 requires __detail::__cartesian_is_sized_sentinel<_Const, iterator_t, _First, _Vs...>
8824 {
return __x._M_distance_from(__y._M_current); }
8826 friend constexpr difference_type
8827 operator-(
const _Iterator& __i, default_sentinel_t)
8828 requires __detail::__cartesian_is_sized_sentinel<_Const, sentinel_t, _First, _Vs...>
8831 return tuple{ranges::end(std::get<0>(__i._M_parent->_M_bases)),
8832 ranges::begin(std::get<1 + _Is>(__i._M_parent->_M_bases))...};
8834 return __i._M_distance_from(__end_tuple);
8837 friend constexpr difference_type
8838 operator-(default_sentinel_t,
const _Iterator& __i)
8839 requires __detail::__cartesian_is_sized_sentinel<_Const, sentinel_t, _First, _Vs...>
8842 friend constexpr auto
8843 iter_move(
const _Iterator& __i)
8844 {
return __detail::__tuple_transform(ranges::iter_move, __i._M_current); }
8846 friend constexpr void
8847 iter_swap(
const _Iterator& __l,
const _Iterator& __r)
8848 requires (indirectly_swappable<iterator_t<__maybe_const_t<_Const, _First>>>
8850 && indirectly_swappable<iterator_t<__maybe_const_t<_Const, _Vs>>>)
8853 (ranges::iter_swap(std::get<_Is>(__l._M_current), std::get<_Is>(__r._M_current)), ...);
8858 template<
size_t _Nm =
sizeof...(_Vs)>
8862 auto& __it = std::get<_Nm>(_M_current);
8864 if constexpr (_Nm > 0)
8865 if (__it == ranges::end(std::get<_Nm>(_M_parent->_M_bases)))
8867 __it = ranges::begin(std::get<_Nm>(_M_parent->_M_bases));
8872 template<
size_t _Nm =
sizeof...(_Vs)>
8876 auto& __it = std::get<_Nm>(_M_current);
8877 if constexpr (_Nm > 0)
8878 if (__it == ranges::begin(std::get<_Nm>(_M_parent->_M_bases)))
8880 __it = __detail::__cartesian_common_arg_end(std::get<_Nm>(_M_parent->_M_bases));
8886 template<
size_t _Nm =
sizeof...(_Vs)>
8888 _M_advance(difference_type __x)
8889 requires __detail::__cartesian_product_is_random_access<_Const, _First, _Vs...>
8898 auto& __r = std::get<_Nm>(_M_parent->_M_bases);
8899 auto& __it = std::get<_Nm>(_M_current);
8900 if constexpr (_Nm == 0)
8902#ifdef _GLIBCXX_ASSERTIONS
8905 auto __size = ranges::ssize(__r);
8906 auto __begin = ranges::begin(__r);
8907 auto __offset = __it - __begin;
8908 __glibcxx_assert(__offset + __x >= 0 && __offset + __x <= __size);
8915 auto __size = ranges::ssize(__r);
8916 auto __begin = ranges::begin(__r);
8917 auto __offset = __it - __begin;
8919 __x = __offset / __size;
8923 __offset = __size + __offset;
8926 __it = __begin + __offset;
8927 _M_advance<_Nm - 1>(__x);
8932 template<
typename _Tuple>
8933 constexpr difference_type
8934 _M_distance_from(
const _Tuple& __t)
const
8937 auto __sum =
static_cast<difference_type
>(0);
8938#ifdef _GLIBCXX_ASSERTIONS
8939 if constexpr (integral<difference_type>)
8942 = (__builtin_add_overflow(__sum, _M_scaled_distance<_Is>(__t), &__sum)
8944 __glibcxx_assert(!__overflow);
8948 __sum = (_M_scaled_distance<_Is>(__t) + ...);
8953 template<
size_t _Nm,
typename _Tuple>
8954 constexpr difference_type
8955 _M_scaled_distance(
const _Tuple& __t)
const
8957 auto __dist =
static_cast<difference_type
>(std::get<_Nm>(_M_current)
8958 - std::get<_Nm>(__t));
8959#ifdef _GLIBCXX_ASSERTIONS
8960 if constexpr (integral<difference_type>)
8962 bool __overflow = __builtin_mul_overflow(__dist, _M_scaled_size<_Nm+1>(), &__dist);
8963 __glibcxx_assert(!__overflow);
8967 __dist *= _M_scaled_size<_Nm+1>();
8971 template<
size_t _Nm>
8972 constexpr difference_type
8973 _M_scaled_size()
const
8975 if constexpr (_Nm <=
sizeof...(_Vs))
8977 auto __size =
static_cast<difference_type
>(ranges::size
8978 (std::get<_Nm>(_M_parent->_M_bases)));
8979#ifdef _GLIBCXX_ASSERTIONS
8980 if constexpr (integral<difference_type>)
8982 bool __overflow = __builtin_mul_overflow(__size, _M_scaled_size<_Nm+1>(), &__size);
8983 __glibcxx_assert(!__overflow);
8987 __size *= _M_scaled_size<_Nm+1>();
8991 return static_cast<difference_type
>(1);
8999 template<
typename... _Ts>
9000 concept __can_cartesian_product_view
9004 struct _CartesianProduct
9006 template<
typename... _Ts>
9007 requires (
sizeof...(_Ts) == 0 || __detail::__can_cartesian_product_view<_Ts...>)
9009 operator() [[nodiscard]] (_Ts&&... __ts)
const
9011 if constexpr (
sizeof...(_Ts) == 0)
9012 return views::single(tuple{});
9018 inline constexpr _CartesianProduct cartesian_product;
9022#ifdef __cpp_lib_ranges_as_rvalue
9023 template<input_range _Vp>
9025 class as_rvalue_view :
public view_interface<as_rvalue_view<_Vp>>
9027 _Vp _M_base = _Vp();
9030 as_rvalue_view()
requires default_initializable<_Vp> = default;
9033 as_rvalue_view(_Vp __base)
9034 : _M_base(std::move(__base))
9038 base() const& requires copy_constructible<_Vp>
9046 begin()
requires (!__detail::__simple_view<_Vp>)
9047 {
return move_iterator(ranges::begin(_M_base)); }
9050 begin() const requires range<const _Vp>
9051 {
return move_iterator(ranges::begin(_M_base)); }
9054 end()
requires (!__detail::__simple_view<_Vp>)
9056 if constexpr (common_range<_Vp>)
9057 return move_iterator(ranges::end(_M_base));
9059 return move_sentinel(ranges::end(_M_base));
9063 end() const requires range<const _Vp>
9065 if constexpr (common_range<const _Vp>)
9066 return move_iterator(ranges::end(_M_base));
9068 return move_sentinel(ranges::end(_M_base));
9072 size()
requires sized_range<_Vp>
9073 {
return ranges::size(_M_base); }
9076 size() const requires sized_range<const _Vp>
9077 {
return ranges::size(_M_base); }
9080 template<
typename _Range>
9081 as_rvalue_view(_Range&&) -> as_rvalue_view<views::all_t<_Range>>;
9083 template<
typename _Tp>
9084 inline constexpr bool enable_borrowed_range<as_rvalue_view<_Tp>>
9085 = enable_borrowed_range<_Tp>;
9091 template<
typename _Tp>
9092 concept __can_as_rvalue_view =
requires { as_rvalue_view(
std::declval<_Tp>()); };
9095 struct _AsRvalue : __adaptor::_RangeAdaptorClosure<_AsRvalue>
9097 template<viewable_range _Range>
9098 requires __detail::__can_as_rvalue_view<_Range>
9100 operator() [[nodiscard]] (_Range&& __r)
const
9105 if constexpr (same_as<range_rvalue_reference_t<_Range>,
9106 range_reference_t<_Range>>)
9113 inline constexpr _AsRvalue as_rvalue;
9117#ifdef __cpp_lib_ranges_enumerate
9120 template<
typename _Range>
9121 concept __range_with_movable_reference = input_range<_Range>
9122 && move_constructible<range_reference_t<_Range>>
9123 && move_constructible<range_rvalue_reference_t<_Range>>;
9127 requires __detail::__range_with_movable_reference<_Vp>
9128 class enumerate_view :
public view_interface<enumerate_view<_Vp>>
9130 _Vp _M_base = _Vp();
9132 template<
bool _Const>
class _Iterator;
9133 template<
bool _Const>
class _Sentinel;
9136 enumerate_view()
requires default_initializable<_Vp> = default;
9139 enumerate_view(_Vp __base)
9140 : _M_base(std::move(__base))
9144 begin()
requires (!__detail::__simple_view<_Vp>)
9145 {
return _Iterator<false>(ranges::begin(_M_base), 0); }
9148 begin() const requires __detail::__range_with_movable_reference<const _Vp>
9149 {
return _Iterator<true>(ranges::begin(_M_base), 0); }
9152 end()
requires (!__detail::__simple_view<_Vp>)
9154 if constexpr (common_range<_Vp> && sized_range<_Vp>)
9155 return _Iterator<false>(ranges::end(_M_base), ranges::distance(_M_base));
9157 return _Sentinel<false>(ranges::end(_M_base));
9161 end() const requires __detail::__range_with_movable_reference<const _Vp>
9163 if constexpr (common_range<const _Vp> && sized_range<const _Vp>)
9164 return _Iterator<true>(ranges::end(_M_base), ranges::distance(_M_base));
9166 return _Sentinel<true>(ranges::end(_M_base));
9170 size()
requires sized_range<_Vp>
9171 {
return ranges::size(_M_base); }
9174 size() const requires sized_range<const _Vp>
9175 {
return ranges::size(_M_base); }
9178 base() const & requires copy_constructible<_Vp>
9186 template<
typename _Range>
9187 enumerate_view(_Range&&) -> enumerate_view<views::all_t<_Range>>;
9189 template<
typename _Tp>
9190 inline constexpr bool enable_borrowed_range<enumerate_view<_Tp>>
9191 = enable_borrowed_range<_Tp>;
9194 requires __detail::__range_with_movable_reference<_Vp>
9195 template<
bool _Const>
9196 class enumerate_view<_Vp>::_Iterator
9198 using _Base = __maybe_const_t<_Const, _Vp>;
9203 if constexpr (random_access_range<_Base>)
9204 return random_access_iterator_tag{};
9205 else if constexpr (bidirectional_range<_Base>)
9206 return bidirectional_iterator_tag{};
9207 else if constexpr (forward_range<_Base>)
9208 return forward_iterator_tag{};
9210 return input_iterator_tag{};
9213 friend enumerate_view;
9216 using iterator_category = input_iterator_tag;
9217 using iterator_concept =
decltype(_S_iter_concept());
9218 using difference_type = range_difference_t<_Base>;
9219 using value_type = tuple<difference_type, range_value_t<_Base>>;
9222 using __reference_type = tuple<difference_type, range_reference_t<_Base>>;
9224 iterator_t<_Base> _M_current = iterator_t<_Base>();
9225 difference_type _M_pos = 0;
9228 _Iterator(iterator_t<_Base> __current, difference_type __pos)
9229 : _M_current(std::move(__current)), _M_pos(__pos)
9233 _Iterator()
requires default_initializable<iterator_t<_Base>> = default;
9236 _Iterator(_Iterator<!_Const> __i)
9237 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
9238 : _M_current(std::move(__i._M_current)), _M_pos(__i._M_pos)
9241 constexpr const iterator_t<_Base> &
9242 base() const & noexcept
9243 {
return _M_current; }
9245 constexpr iterator_t<_Base>
9249 constexpr difference_type
9250 index() const noexcept
9255 {
return __reference_type(_M_pos, *_M_current); }
9257 constexpr _Iterator&
9270 operator++(
int)
requires forward_range<_Base>
9277 constexpr _Iterator&
9278 operator--()
requires bidirectional_range<_Base>
9286 operator--(
int)
requires bidirectional_range<_Base>
9293 constexpr _Iterator&
9294 operator+=(difference_type __n)
requires random_access_range<_Base>
9301 constexpr _Iterator&
9302 operator-=(difference_type __n)
requires random_access_range<_Base>
9310 operator[](difference_type __n)
const requires random_access_range<_Base>
9311 {
return __reference_type(_M_pos + __n, _M_current[__n]); }
9313 friend constexpr bool
9314 operator==(
const _Iterator& __x,
const _Iterator& __y)
noexcept
9315 {
return __x._M_pos == __y._M_pos; }
9317 friend constexpr strong_ordering
9318 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
noexcept
9319 {
return __x._M_pos <=> __y._M_pos; }
9321 friend constexpr _Iterator
9322 operator+(
const _Iterator& __x, difference_type __y)
9323 requires random_access_range<_Base>
9324 {
return (
auto(__x) += __y); }
9326 friend constexpr _Iterator
9327 operator+(difference_type __x,
const _Iterator& __y)
9328 requires random_access_range<_Base>
9329 {
return auto(__y) += __x; }
9331 friend constexpr _Iterator
9332 operator-(
const _Iterator& __x, difference_type __y)
9333 requires random_access_range<_Base>
9334 {
return auto(__x) -= __y; }
9336 friend constexpr difference_type
9337 operator-(
const _Iterator& __x,
const _Iterator& __y)
noexcept
9338 {
return __x._M_pos - __y._M_pos; }
9340 friend constexpr auto
9341 iter_move(
const _Iterator& __i)
9342 noexcept(
noexcept(ranges::iter_move(__i._M_current))
9343 && is_nothrow_move_constructible_v<range_rvalue_reference_t<_Base>>)
9345 return tuple<difference_type, range_rvalue_reference_t<_Base>>
9346 (__i._M_pos, ranges::iter_move(__i._M_current));
9351 requires __detail::__range_with_movable_reference<_Vp>
9352 template<
bool _Const>
9353 class enumerate_view<_Vp>::_Sentinel
9355 using _Base = __maybe_const_t<_Const, _Vp>;
9357 sentinel_t<_Base> _M_end = sentinel_t<_Base>();
9360 _Sentinel(sentinel_t<_Base> __end)
9361 : _M_end(std::move(__end))
9364 friend enumerate_view;
9367 _Sentinel() =
default;
9370 _Sentinel(_Sentinel<!_Const> __other)
9371 requires _Const && convertible_to<sentinel_t<_Vp>, sentinel_t<_Base>>
9375 constexpr sentinel_t<_Base>
9379 template<
bool _OtherConst>
9380 requires sentinel_for<sentinel_t<_Base>, iterator_t<__maybe_const_t<_OtherConst, _Vp>>>
9381 friend constexpr bool
9382 operator==(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
9383 {
return __x._M_current == __y._M_end; }
9385 template<
bool _OtherConst>
9386 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<__maybe_const_t<_OtherConst, _Vp>>>
9387 friend constexpr range_difference_t<__maybe_const_t<_OtherConst, _Vp>>
9388 operator-(
const _Iterator<_OtherConst>& __x,
const _Sentinel& __y)
9389 {
return __x._M_current - __y._M_end; }
9391 template<
bool _OtherConst>
9392 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<__maybe_const_t<_OtherConst, _Vp>>>
9393 friend constexpr range_difference_t<__maybe_const_t<_OtherConst, _Vp>>
9394 operator-(
const _Sentinel& __x,
const _Iterator<_OtherConst>& __y)
9395 {
return __x._M_end - __y._M_current; }
9402 template<
typename _Tp>
9403 concept __can_enumerate_view
9407 struct _Enumerate : __adaptor::_RangeAdaptorClosure<_Enumerate>
9409 template<viewable_range _Range>
9410 requires __detail::__can_enumerate_view<_Range>
9412 operator() [[nodiscard]] (_Range&& __r)
const
9416 inline constexpr _Enumerate enumerate;
9420#ifdef __cpp_lib_ranges_as_const
9425 _Vp _M_base = _Vp();
9428 as_const_view()
requires default_initializable<_Vp> = default;
9431 as_const_view(_Vp __base)
9432 noexcept(is_nothrow_move_constructible_v<_Vp>)
9433 : _M_base(std::move(__base))
9438 noexcept(is_nothrow_copy_constructible_v<_Vp>)
9439 requires copy_constructible<_Vp>
9444 noexcept(is_nothrow_move_constructible_v<_Vp>)
9448 begin()
requires (!__detail::__simple_view<_Vp>)
9449 {
return ranges::cbegin(_M_base); }
9452 begin() const requires range<const _Vp>
9453 {
return ranges::cbegin(_M_base); }
9456 end()
requires (!__detail::__simple_view<_Vp>)
9457 {
return ranges::cend(_M_base); }
9460 end() const requires range<const _Vp>
9461 {
return ranges::cend(_M_base); }
9464 size()
requires sized_range<_Vp>
9465 {
return ranges::size(_M_base); }
9468 size() const requires sized_range<const _Vp>
9469 {
return ranges::size(_M_base); }
9472 template<
typename _Range>
9473 as_const_view(_Range&&) -> as_const_view<views::all_t<_Range>>;
9475 template<
typename _Tp>
9476 inline constexpr bool enable_borrowed_range<as_const_view<_Tp>>
9477 = enable_borrowed_range<_Tp>;
9483 template<
typename _Tp>
9484 inline constexpr bool __is_constable_ref_view =
false;
9486 template<
typename _Range>
9487 inline constexpr bool __is_constable_ref_view<ref_view<_Range>>
9488 = constant_range<const _Range>;
9490 template<
typename _Range>
9494 struct _AsConst : __adaptor::_RangeAdaptorClosure<_AsConst>
9496 template<viewable_range _Range>
9498 operator()(_Range&& __r)
const
9500 requires __detail::__can_as_const_view<_Range>
9502 using _Tp = remove_cvref_t<_Range>;
9503 using element_type = remove_reference_t<range_reference_t<_Range>>;
9504 if constexpr (constant_range<views::all_t<_Range>>)
9506 else if constexpr (__detail::__is_empty_view<_Tp>)
9507 return views::empty<const element_type>;
9508#if __cpp_lib_optional >= 202506L && __cpp_lib_optional_range_support
9509 else if constexpr (__is_optional_ref_v<_Tp>)
9510 return optional<const typename _Tp::value_type&>(__r);
9512 else if constexpr (std::__detail::__is_span<_Tp>)
9514 else if constexpr (__detail::__is_constable_ref_view<_Tp>)
9516 else if constexpr (is_lvalue_reference_v<_Range>
9517 && constant_range<const _Tp>
9519 return ref_view(
static_cast<const _Tp&
>(__r));
9525 inline constexpr _AsConst as_const;
9530 namespace views = ranges::views;
9532#if __cpp_lib_ranges_to_container
9538 template<
typename _Container>
9539 constexpr bool __reservable_container
9540 = sized_range<_Container>
9541 &&
requires(_Container& __c, range_size_t<_Container> __n) {
9543 { __c.capacity() } -> same_as<
decltype(__n)>;
9544 { __c.max_size() } -> same_as<
decltype(__n)>;
9547 template<
typename _Cont,
typename _Range>
9548 constexpr bool __toable =
requires {
9549 requires (!input_range<_Cont>
9550 || convertible_to<range_reference_t<_Range>,
9551 range_value_t<_Cont>>);
9572 template<
typename _Cont, input_range _Rg,
typename... _Args>
9573 requires (!view<_Cont>)
9575 to [[nodiscard]] (_Rg&& __r, _Args&&... __args)
9577 static_assert(!is_const_v<_Cont> && !is_volatile_v<_Cont>);
9578 static_assert(is_class_v<_Cont> || is_union_v<_Cont>);
9580 if constexpr (__detail::__toable<_Cont, _Rg>)
9582 if constexpr (constructible_from<_Cont, _Rg, _Args...>)
9585 else if constexpr (constructible_from<_Cont, from_range_t, _Rg, _Args...>)
9588 else if constexpr (
requires {
requires common_range<_Rg>;
9589 typename __iter_category_t<iterator_t<_Rg>>;
9590 requires derived_from<__iter_category_t<iterator_t<_Rg>>,
9591 input_iterator_tag>;
9592 requires constructible_from<_Cont, iterator_t<_Rg>,
9593 sentinel_t<_Rg>, _Args...>;
9595 return _Cont(ranges::begin(__r), ranges::end(__r),
9599 static_assert(constructible_from<_Cont, _Args...>);
9601 if constexpr (sized_range<_Rg>
9602 && __detail::__reservable_container<_Cont>)
9603 __c.reserve(
static_cast<range_size_t<_Cont>
>(ranges::size(__r)));
9607 auto __it = ranges::begin(__r);
9608 const auto __sent = ranges::end(__r);
9609 while (__it != __sent)
9611 if constexpr (
requires { __c.emplace_back(*__it); })
9612 __c.emplace_back(*__it);
9613 else if constexpr (
requires { __c.push_back(*__it); })
9614 __c.push_back(*__it);
9615 else if constexpr (
requires { __c.emplace(__c.end(), *__it); })
9616 __c.emplace(__c.end(), *__it);
9618 __c.insert(__c.end(), *__it);
9626 static_assert(input_range<range_reference_t<_Rg>>);
9629 return ranges::to<_Cont>(ref_view(__r) | views::transform(
9630 []<
typename _Elt>(_Elt&& __elem) {
9631 using _ValT = range_value_t<_Cont>;
9640 template<
typename _Rg>
9643 using iterator_category = input_iterator_tag;
9644 using value_type = range_value_t<_Rg>;
9645 using difference_type = ptrdiff_t;
9646 using pointer = add_pointer_t<range_reference_t<_Rg>>;
9647 using reference = range_reference_t<_Rg>;
9649 pointer operator->()
const;
9650 _InputIter& operator++();
9651 _InputIter operator++(
int);
9652 bool operator==(
const _InputIter&)
const;
9655 template<
template<
typename...>
typename _Cont, input_range _Rg,
9660 template<
template<
typename...>
typename _Cont, input_range _Rg,
9666 template<
template<
typename...>
typename _Cont, input_range _Rg,
9676 template<
template<
typename...>
typename _Cont, input_range _Rg,
9679 to [[nodiscard]] (_Rg&& __r, _Args&&... __args)
9681 using __detail::_DeduceExpr1;
9682 using __detail::_DeduceExpr2;
9683 using __detail::_DeduceExpr3;
9684 if constexpr (
requires {
typename _DeduceExpr1<_Cont, _Rg, _Args...>; })
9685 return ranges::to<_DeduceExpr1<_Cont, _Rg, _Args...>>(
9687 else if constexpr (
requires {
typename _DeduceExpr2<_Cont, _Rg, _Args...>; })
9688 return ranges::to<_DeduceExpr2<_Cont, _Rg, _Args...>>(
9690 else if constexpr (
requires {
typename _DeduceExpr3<_Cont, _Rg, _Args...>; })
9691 return ranges::to<_DeduceExpr3<_Cont, _Rg, _Args...>>(
9694 static_assert(
false);
9700 template<
typename _Cont>
9703 template<
typename _Range,
typename... _Args>
9707 operator()(_Range&& __r, _Args&&... __args)
const
9731 template<
typename _Cont,
typename... _Args>
9732 requires (!view<_Cont>)
9734 to [[nodiscard]] (_Args&&... __args)
9736 static_assert(!is_const_v<_Cont> && !is_volatile_v<_Cont>);
9737 static_assert(is_class_v<_Cont> || is_union_v<_Cont>);
9739 using __detail::_To;
9740 using views::__adaptor::_Partial;
9747 template<
template<
typename...>
typename _Cont>
9750 template<
typename _Range,
typename... _Args>
9754 operator()(_Range&& __r, _Args&&... __args)
const
9780 template<
template<
typename...>
typename _Cont,
typename... _Args>
9782 to [[nodiscard]] (_Args&&... __args)
9784 using __detail::_To2;
9785 using views::__adaptor::_Partial;
9792#if __cpp_lib_ranges_concat
9797 template<
typename... _Rs>
9798 using __concat_reference_t = common_reference_t<range_reference_t<_Rs>...>;
9800 template<
typename... _Rs>
9801 using __concat_value_t = common_type_t<range_value_t<_Rs>...>;
9803 template<
typename... _Rs>
9804 using __concat_rvalue_reference_t
9805 = common_reference_t<range_rvalue_reference_t<_Rs>...>;
9807 template<
typename _Ref,
typename _RRef,
typename _It>
9808 concept __concat_indirectly_readable_impl =
requires(
const _It __it) {
9809 { *__it } -> convertible_to<_Ref>;
9810 { ranges::iter_move(__it) } -> convertible_to<_RRef>;
9813 template<
typename... _Rs>
9814 concept __concat_indirectly_readable
9815 = common_reference_with<__concat_reference_t<_Rs...>&&, __concat_value_t<_Rs...>&>
9816 && common_reference_with<__concat_reference_t<_Rs...>&&,
9817 __concat_rvalue_reference_t<_Rs...>&&>
9818 && common_reference_with<__concat_rvalue_reference_t<_Rs...>&&,
9819 __concat_value_t<_Rs...>
const&>
9820 && (__concat_indirectly_readable_impl<__concat_reference_t<_Rs...>,
9821 __concat_rvalue_reference_t<_Rs...>,
9825 template<
typename... _Rs>
9826 concept __concatable =
requires {
9827 typename __concat_reference_t<_Rs...>;
9828 typename __concat_value_t<_Rs...>;
9829 typename __concat_rvalue_reference_t<_Rs...>;
9830 } && __concat_indirectly_readable<_Rs...>;
9832 template<
bool _Const,
typename _Range,
typename... _Rs>
9833 struct __all_but_last_common
9835 static inline constexpr bool value
9836 =
requires {
requires (common_range<__maybe_const_t<_Const, _Range>>
9837 && __all_but_last_common<_Const, _Rs...>::value); };
9840 template<
bool _Const,
typename _Range>
9841 struct __all_but_last_common<_Const, _Range>
9842 {
static inline constexpr bool value =
true; };
9844 template<
bool _Const,
typename... _Rs>
9845 concept __concat_is_random_access = __all_random_access<_Const, _Rs...>
9846 && __all_but_last_common<_Const, _Rs...>::value;
9848 template<
bool _Const,
typename... _Rs>
9849 concept __concat_is_bidirectional = __all_bidirectional<_Const, _Rs...>
9850 && __all_but_last_common<_Const, _Rs...>::value;
9852 template<
typename _Range,
typename... _Rs>
9853 struct __all_but_first_sized
9854 {
static inline constexpr bool value = (sized_range<_Rs> && ...); };
9857 template<input_range... _Vs>
9858 requires (view<_Vs> && ...) && (
sizeof...(_Vs) > 0) && __detail::__concatable<_Vs...>
9859 class concat_view : public view_interface<concat_view<_Vs...>>
9861 tuple<_Vs...> _M_views;
9863 template<
bool _Const>
class _Iterator;
9866 constexpr concat_view() =
default;
9869 concat_view(_Vs... __views)
9870 : _M_views(std::move(__views)...)
9873 constexpr _Iterator<false>
9874 begin()
requires (!(__detail::__simple_view<_Vs> && ...))
9876 _Iterator<false> __it(
this, in_place_index<0>, ranges::begin(std::get<0>(_M_views)));
9877 __it.template _M_satisfy<0>();
9881 constexpr _Iterator<true>
9882 begin() const requires (range<const _Vs> && ...) && __detail::__concatable<const _Vs...>
9884 _Iterator<true> __it(
this, in_place_index<0>, ranges::begin(std::get<0>(_M_views)));
9885 __it.template _M_satisfy<0>();
9890 end()
requires (!(__detail::__simple_view<_Vs> && ...))
9892 constexpr auto __n =
sizeof...(_Vs);
9893 if constexpr (__detail::__all_forward<
false, _Vs...>
9894 && common_range<_Vs...[__n - 1]>)
9895 return _Iterator<false>(
this, in_place_index<__n - 1>,
9896 ranges::end(std::get<__n - 1>(_M_views)));
9902 end() const requires (range<const _Vs> && ...) && __detail::__concatable<const _Vs...>
9904 constexpr auto __n =
sizeof...(_Vs);
9905 if constexpr (__detail::__all_forward<
true, _Vs...>
9906 && common_range<
const _Vs...[__n - 1]>)
9907 return _Iterator<true>(
this, in_place_index<__n - 1>,
9908 ranges::end(std::get<__n - 1>(_M_views)));
9914 size()
requires (sized_range<_Vs>&&...)
9916 return std::apply([](
auto... __sizes) {
9917 using _CT = __detail::__make_unsigned_like_t<
common_type_t<
decltype(__sizes)...>>;
9918 return (_CT(__sizes) + ...);
9919 }, __detail::__tuple_transform(ranges::size, _M_views));
9923 size() const requires (sized_range<const _Vs>&&...)
9925 return std::apply([](
auto... __sizes) {
9926 using _CT = __detail::__make_unsigned_like_t<
common_type_t<
decltype(__sizes)...>>;
9927 return (_CT(__sizes) + ...);
9928 }, __detail::__tuple_transform(ranges::size, _M_views));
9932 template<
typename... _Rs>
9933 concat_view(_Rs&&...) -> concat_view<views::all_t<_Rs>...>;
9937 template<
bool _Const,
typename... _Vs>
9938 struct __concat_view_iter_cat
9941 template<
bool _Const,
typename... _Vs>
9942 requires __detail::__all_forward<_Const, _Vs...>
9943 struct __concat_view_iter_cat<_Const, _Vs...>
9948 if constexpr (!is_reference_v<__concat_reference_t<__maybe_const_t<_Const, _Vs>...>>)
9949 return input_iterator_tag{};
9951 return []<
typename... _Cats>(_Cats... __cats) {
9952 if constexpr ((derived_from<_Cats, random_access_iterator_tag> && ...)
9953 && __concat_is_random_access<_Const, _Vs...>)
9954 return random_access_iterator_tag{};
9955 else if constexpr ((derived_from<_Cats, bidirectional_iterator_tag> && ...)
9956 && __concat_is_bidirectional<_Const, _Vs...>)
9957 return bidirectional_iterator_tag{};
9958 else if constexpr ((derived_from<_Cats, forward_iterator_tag> && ...))
9959 return forward_iterator_tag{};
9961 return input_iterator_tag{};
9962 }(
typename iterator_traits<iterator_t<__maybe_const_t<_Const, _Vs>>>
9963 ::iterator_category{}...);
9968 template<input_range... _Vs>
9969 requires (view<_Vs> && ...) && (
sizeof...(_Vs) > 0) && __detail::__concatable<_Vs...>
9970 template<bool _Const>
9971 class concat_view<_Vs...>::_Iterator
9972 : public __detail::__concat_view_iter_cat<_Const, _Vs...>
9977 if constexpr (__detail::__concat_is_random_access<_Const, _Vs...>)
9978 return random_access_iterator_tag{};
9979 else if constexpr (__detail::__concat_is_bidirectional<_Const, _Vs...>)
9980 return bidirectional_iterator_tag{};
9981 else if constexpr (__detail::__all_forward<_Const, _Vs...>)
9982 return forward_iterator_tag{};
9984 return input_iterator_tag{};
9988 friend _Iterator<!_Const>;
9992 using iterator_concept =
decltype(_S_iter_concept());
9993 using value_type = __detail::__concat_value_t<__maybe_const_t<_Const, _Vs>...>;
9994 using difference_type = common_type_t<range_difference_t<__maybe_const_t<_Const, _Vs>>...>;
9997 using __base_iter = variant<iterator_t<__maybe_const_t<_Const, _Vs>>...>;
9999 __maybe_const_t<_Const, concat_view>* _M_parent =
nullptr;
10002 template<
size_t _Nm>
10006 if constexpr (_Nm < (
sizeof...(_Vs) - 1))
10008 if (std::get<_Nm>(_M_it) == ranges::end(std::get<_Nm>(_M_parent->_M_views)))
10010 _M_it.template emplace<_Nm + 1>(ranges::begin
10011 (std::get<_Nm + 1>(_M_parent->_M_views)));
10012 _M_satisfy<_Nm + 1>();
10017 template<
size_t _Nm>
10021 if constexpr (_Nm == 0)
10022 --std::get<0>(_M_it);
10025 if (std::get<_Nm>(_M_it) == ranges::begin(std::get<_Nm>(_M_parent->_M_views)))
10027 _M_it.template emplace<_Nm - 1>(ranges::end
10028 (std::get<_Nm - 1>(_M_parent->_M_views)));
10029 _M_prev<_Nm - 1>();
10032 --std::get<_Nm>(_M_it);
10036 template<
size_t _Nm>
10038 _M_advance_fwd(difference_type __offset, difference_type __steps)
10040 using _Dt = iter_difference_t<variant_alternative_t<_Nm, __base_iter>>;
10041 if constexpr (_Nm ==
sizeof...(_Vs) - 1)
10042 std::get<_Nm>(_M_it) +=
static_cast<_Dt
>(__steps);
10045 auto __n_size = ranges::distance(std::get<_Nm>(_M_parent->_M_views));
10046 if (__offset + __steps < __n_size)
10047 std::get<_Nm>(_M_it) +=
static_cast<_Dt
>(__steps);
10050 _M_it.template emplace<_Nm + 1>(ranges::begin
10051 (std::get<_Nm + 1>(_M_parent->_M_views)));
10052 _M_advance_fwd<_Nm + 1>(0, __offset + __steps - __n_size);
10057 template<
size_t _Nm>
10059 _M_advance_bwd(difference_type __offset, difference_type __steps)
10061 using _Dt = iter_difference_t<variant_alternative_t<_Nm, __base_iter>>;
10062 if constexpr (_Nm == 0)
10063 std::get<_Nm>(_M_it) -=
static_cast<_Dt
>(__steps);
10065 if (__offset >= __steps)
10066 std::get<_Nm>(_M_it) -=
static_cast<_Dt
>(__steps);
10069 auto __prev_size = ranges::distance(std::get<_Nm - 1>(_M_parent->_M_views));
10070 _M_it.template emplace<_Nm - 1>(ranges::end
10071 (std::get<_Nm - 1>(_M_parent->_M_views)));
10072 _M_advance_bwd<_Nm - 1>(__prev_size, __steps - __offset);
10080 template<
typename _Fp>
10081 static constexpr auto
10082 _S_invoke_with_runtime_index(_Fp&& __f,
size_t __index)
10084 return [&__f, __index]<
size_t _Idx>(
this auto&& __self) {
10085 if (_Idx == __index)
10086 return __f.template operator()<_Idx>();
10087 if constexpr (_Idx + 1 <
sizeof...(_Vs))
10088 return __self.template
operator()<_Idx + 1>();
10089 __builtin_unreachable();
10090 }.template operator()<0>();
10093 template<
typename _Fp>
10095 _M_invoke_with_runtime_index(_Fp&& __f)
10096 {
return _S_invoke_with_runtime_index(
std::forward<_Fp>(__f), _M_it.index()); }
10098 template<
typename... _Args>
10100 _Iterator(__maybe_const_t<_Const, concat_view>* __parent, _Args&&... __args)
10101 requires constructible_from<__base_iter, _Args&&...>
10106 _Iterator() =
default;
10109 _Iterator(_Iterator<!_Const> __it)
10110 requires _Const && (convertible_to<iterator_t<_Vs>, iterator_t<const _Vs>> && ...)
10111 : _M_parent(__it._M_parent),
10112 _M_it(_S_invoke_with_runtime_index([this, &__it]<size_t _Idx>() {
10113 return __base_iter(in_place_index<_Idx>,
10114 std::get<_Idx>(
std::move(__it._M_it)));
10115 }, __it._M_it.index()))
10118 constexpr decltype(
auto)
10121 __glibcxx_assert(!_M_it.valueless_by_exception());
10122 using reference = __detail::__concat_reference_t<__maybe_const_t<_Const, _Vs>...>;
10123 return std::visit([](
auto&& __it) -> reference {
return *__it; }, _M_it);
10126 constexpr _Iterator&
10129 _M_invoke_with_runtime_index([
this]<
size_t _Idx>() {
10130 ++std::get<_Idx>(_M_it);
10131 _M_satisfy<_Idx>();
10140 constexpr _Iterator
10142 requires __detail::__all_forward<_Const, _Vs...>
10144 auto __tmp = *
this;
10149 constexpr _Iterator&
10151 requires __detail::__concat_is_bidirectional<_Const, _Vs...>
10153 __glibcxx_assert(!_M_it.valueless_by_exception());
10154 _M_invoke_with_runtime_index([
this]<
size_t _Idx>() {
10160 constexpr _Iterator
10162 requires __detail::__concat_is_bidirectional<_Const, _Vs...>
10164 auto __tmp = *
this;
10169 constexpr _Iterator&
10170 operator+=(difference_type __n)
10171 requires __detail::__concat_is_random_access<_Const, _Vs...>
10173 __glibcxx_assert(!_M_it.valueless_by_exception());
10174 _M_invoke_with_runtime_index([
this, __n]<
size_t _Idx>() {
10175 auto __begin = ranges::begin(std::get<_Idx>(_M_parent->_M_views));
10177 _M_advance_fwd<_Idx>(std::get<_Idx>(_M_it) - __begin, __n);
10179 _M_advance_bwd<_Idx>(std::get<_Idx>(_M_it) - __begin, -__n);
10184 constexpr _Iterator&
10185 operator-=(difference_type __n)
10186 requires __detail::__concat_is_random_access<_Const, _Vs...>
10192 constexpr decltype(
auto)
10193 operator[](difference_type __n)
const
10194 requires __detail::__concat_is_random_access<_Const, _Vs...>
10195 {
return *((*this) + __n); }
10197 friend constexpr bool
10198 operator==(
const _Iterator& __x,
const _Iterator& __y)
10199 requires (equality_comparable<iterator_t<__maybe_const_t<_Const, _Vs>>> && ...)
10201 __glibcxx_assert(!__x._M_it.valueless_by_exception());
10202 __glibcxx_assert(!__y._M_it.valueless_by_exception());
10203 return __x._M_it == __y._M_it;
10206 friend constexpr bool
10207 operator==(
const _Iterator& __it, default_sentinel_t)
10209 __glibcxx_assert(!__it._M_it.valueless_by_exception());
10210 constexpr auto __last_idx =
sizeof...(_Vs) - 1;
10211 return (__it._M_it.index() == __last_idx
10212 && (std::get<__last_idx>(__it._M_it)
10213 == ranges::end(std::get<__last_idx>(__it._M_parent->_M_views))));
10216 friend constexpr bool
10217 operator<(
const _Iterator& __x,
const _Iterator& __y)
10218 requires __detail::__all_random_access<_Const, _Vs...>
10219 {
return __x._M_it < __y._M_it; }
10221 friend constexpr bool
10222 operator>(
const _Iterator& __x,
const _Iterator& __y)
10223 requires __detail::__all_random_access<_Const, _Vs...>
10224 {
return __x._M_it > __y._M_it; }
10226 friend constexpr bool
10227 operator<=(
const _Iterator& __x,
const _Iterator& __y)
10228 requires __detail::__all_random_access<_Const, _Vs...>
10229 {
return __x._M_it <= __y._M_it; }
10231 friend constexpr bool
10232 operator>=(
const _Iterator& __x,
const _Iterator& __y)
10233 requires __detail::__all_random_access<_Const, _Vs...>
10234 {
return __x._M_it >= __y._M_it; }
10236 friend constexpr auto
10237 operator<=>(
const _Iterator& __x,
const _Iterator& __y)
10238 requires __detail::__all_random_access<_Const, _Vs...>
10239 && (three_way_comparable<iterator_t<__maybe_const_t<_Const, _Vs>>> && ...)
10240 {
return __x._M_it <=> __y._M_it; }
10242 friend constexpr _Iterator
10243 operator+(
const _Iterator& __it, difference_type __n)
10244 requires __detail::__concat_is_random_access<_Const, _Vs...>
10245 {
return auto(__it) += __n; }
10247 friend constexpr _Iterator
10248 operator+(difference_type __n,
const _Iterator& __it)
10249 requires __detail::__concat_is_random_access<_Const, _Vs...>
10250 {
return __it + __n; }
10252 friend constexpr _Iterator
10253 operator-(
const _Iterator& __it, difference_type __n)
10254 requires __detail::__concat_is_random_access<_Const, _Vs...>
10255 {
return auto(__it) -= __n; }
10257 friend constexpr difference_type
10258 operator-(
const _Iterator& __x,
const _Iterator& __y)
10259 requires __detail::__concat_is_random_access<_Const, _Vs...>
10261 return _S_invoke_with_runtime_index([&]<
size_t _Ix>() -> difference_type {
10262 return _S_invoke_with_runtime_index([&]<
size_t _Iy>() -> difference_type {
10263 if constexpr (_Ix > _Iy)
10265 auto __dy = ranges::distance(std::get<_Iy>(__y._M_it),
10266 ranges::end(std::get<_Iy>(__y._M_parent
10268 auto __dx = ranges::distance(ranges::begin(std::get<_Ix>(__x._M_parent
10270 std::get<_Ix>(__x._M_it));
10271 difference_type __s = 0;
10272 [&]<
size_t _Idx = _Iy + 1>(
this auto&& __self) {
10273 if constexpr (_Idx < _Ix)
10275 __s += ranges::size(std::get<_Idx>(__x._M_parent->_M_views));
10276 __self.template operator()<_Idx + 1>();
10279 return __dy + __s + __dx;
10281 else if constexpr (_Ix < _Iy)
10282 return -(__y - __x);
10284 return std::get<_Ix>(__x._M_it) - std::get<_Iy>(__y._M_it);
10285 }, __y._M_it.index());
10286 }, __x._M_it.index());
10289 friend constexpr difference_type
10290 operator-(
const _Iterator& __x, default_sentinel_t)
10291 requires (sized_sentinel_for<sentinel_t<__maybe_const_t<_Const, _Vs>>,
10292 iterator_t<__maybe_const_t<_Const, _Vs>>> && ...)
10293 && __detail::__all_but_first_sized<__maybe_const_t<_Const, _Vs>...>::value
10295 return _S_invoke_with_runtime_index([&]<
size_t _Ix>() -> difference_type {
10296 auto __dx = ranges::distance(std::get<_Ix>(__x._M_it),
10297 ranges::end(std::get<_Ix>(__x._M_parent->_M_views)));
10298 difference_type __s = 0;
10299 [&]<
size_t _Idx = _Ix + 1>(
this auto&& __self) {
10300 if constexpr (_Idx <
sizeof...(_Vs))
10302 __s += ranges::size(std::get<_Idx>(__x._M_parent->_M_views));
10303 __self.template operator()<_Idx + 1>();
10306 return -(__dx + __s);
10307 }, __x._M_it.index());
10310 friend constexpr difference_type
10311 operator-(default_sentinel_t,
const _Iterator& __x)
10312 requires (sized_sentinel_for<sentinel_t<__maybe_const_t<_Const, _Vs>>,
10313 iterator_t<__maybe_const_t<_Const, _Vs>>> && ...)
10314 && __detail::__all_but_first_sized<__maybe_const_t<_Const, _Vs>...>::value
10317 friend constexpr decltype(
auto)
10318 iter_move(
const _Iterator& __it)
10320 using _Res = __detail::__concat_rvalue_reference_t<__maybe_const_t<_Const, _Vs>...>;
10321 return std::visit([](
const auto& __i) -> _Res {
10322 return ranges::iter_move(__i);
10326 friend constexpr void
10327 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
10328 requires swappable_with<iter_reference_t<_Iterator>, iter_reference_t<_Iterator>>
10329 && (... && indirectly_swappable<iterator_t<__maybe_const_t<_Const, _Vs>>>)
10331 std::visit([&]<
typename _Tp,
typename _Up>(
const _Tp& __it1,
const _Up& __it2) {
10332 if constexpr (is_same_v<_Tp, _Up>)
10333 ranges::iter_swap(__it1, __it2);
10335 ranges::swap(*__it1, *__it2);
10336 }, __x._M_it, __y._M_it);
10344 template<
typename... _Ts>
10350 template<
typename... _Ts>
10351 requires __detail::__can_concat_view<_Ts...>
10353 operator() [[nodiscard]] (_Ts&&... __ts)
const
10356 template<input_range _Range>
10358 operator() [[nodiscard]] (_Range&& __t)
const
10362 inline constexpr _Concat concat;
10368#if __cpp_lib_ranges_cache_latest
10371 template<input_range _Vp>
10373 class cache_latest_view :
public view_interface<cache_latest_view<_Vp>>
10375 _Vp _M_base = _Vp();
10377 using __cache_t = __conditional_t<is_reference_v<range_reference_t<_Vp>>,
10378 add_pointer_t<range_reference_t<_Vp>>,
10379 range_reference_t<_Vp>>;
10380 __detail::__non_propagating_cache<__cache_t> _M_cache;
10386 cache_latest_view()
requires default_initializable<_Vp> = default;
10389 cache_latest_view(_Vp __base)
10390 : _M_base(std::move(__base))
10394 base() const & requires copy_constructible<_Vp>
10395 {
return _M_base; }
10403 {
return _Iterator(*
this); }
10407 {
return _Sentinel(*
this); }
10410 size()
requires sized_range<_Vp>
10411 {
return ranges::size(_M_base); }
10414 size() const requires sized_range<const _Vp>
10415 {
return ranges::size(_M_base); }
10418 template<
typename _Range>
10419 cache_latest_view(_Range&&) -> cache_latest_view<views::all_t<_Range>>;
10421 template<input_range _Vp>
10423 class cache_latest_view<_Vp>::_Iterator
10425 cache_latest_view* _M_parent;
10426 iterator_t<_Vp> _M_current;
10429 _Iterator(cache_latest_view& __parent)
10431 _M_current(ranges::begin(__parent._M_base))
10434 friend class cache_latest_view;
10437 using difference_type = range_difference_t<_Vp>;
10438 using value_type = range_value_t<_Vp>;
10439 using iterator_concept = input_iterator_tag;
10441 _Iterator(_Iterator&&) =
default;
10444 operator=(_Iterator&&) =
default;
10446 constexpr iterator_t<_Vp>
10450 constexpr const iterator_t<_Vp>&
10451 base() const & noexcept
10452 {
return _M_current; }
10454 constexpr range_reference_t<_Vp>&
10457 if constexpr (is_reference_v<range_reference_t<_Vp>>)
10459 if (!_M_parent->_M_cache)
10460 _M_parent->_M_cache =
std::__addressof(__detail::__as_lvalue(*_M_current));
10461 return **_M_parent->_M_cache;
10465 if (!_M_parent->_M_cache)
10466 _M_parent->_M_cache._M_emplace_deref(_M_current);
10467 return *_M_parent->_M_cache;
10471 constexpr _Iterator&
10474 _M_parent->_M_cache._M_reset();
10483 friend constexpr range_rvalue_reference_t<_Vp>
10484 iter_move(
const _Iterator& __i)
10485 noexcept(
noexcept(ranges::iter_move(__i._M_current)))
10486 {
return ranges::iter_move(__i._M_current); }
10488 friend constexpr void
10489 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
10490 noexcept(
noexcept(ranges::iter_swap(__x._M_current, __y._M_current)))
10491 requires indirectly_swappable<iterator_t<_Vp>>
10492 { ranges::iter_swap(__x._M_current, __y._M_current); }
10495 template<input_range _Vp>
10497 class cache_latest_view<_Vp>::_Sentinel
10499 sentinel_t<_Vp> _M_end = sentinel_t<_Vp>();
10502 _Sentinel(cache_latest_view& __parent)
10503 : _M_end(ranges::end(__parent._M_base))
10506 friend class cache_latest_view;
10509 _Sentinel() =
default;
10511 constexpr sentinel_t<_Vp>
10515 friend constexpr bool
10516 operator==(
const _Iterator& __x,
const _Sentinel& __y)
10517 {
return __x._M_current == __y._M_end; }
10519 friend constexpr range_difference_t<_Vp>
10520 operator-(
const _Iterator& __x,
const _Sentinel& __y)
10521 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
10522 {
return __x._M_current - __y._M_end; }
10524 friend constexpr range_difference_t<_Vp>
10525 operator-(
const _Sentinel& __x,
const _Iterator& __y)
10526 requires sized_sentinel_for<sentinel_t<_Vp>, iterator_t<_Vp>>
10527 {
return __x._M_end - __y._M_current; }
10534 template<
typename _Tp>
10535 concept __can_cache_latest =
requires { cache_latest_view(
std::declval<_Tp>()); };
10538 struct _CacheLatest : __adaptor::_RangeAdaptorClosure<_CacheLatest>
10540 template<viewable_range _Range>
10541 requires __detail::__can_cache_latest<_Range>
10543 operator() [[nodiscard]] (_Range&& __r)
const
10546 static constexpr bool _S_has_simple_call_op =
true;
10549 inline constexpr _CacheLatest cache_latest;
10554#if __cpp_lib_ranges_as_input
10557 template<input_range _Vp>
10559 class as_input_view :
public view_interface<as_input_view<_Vp>>
10561 _Vp _M_base = _Vp();
10563 template<
bool _Const>
10567 as_input_view()
requires default_initializable<_Vp> = default;
10570 as_input_view(_Vp __base)
10571 : _M_base(std::move(__base))
10575 base() const & requires copy_constructible<_Vp>
10576 {
return _M_base; }
10583 begin()
requires (!__detail::__simple_view<_Vp>)
10584 {
return _Iterator<false>(ranges::begin(_M_base)); }
10587 begin() const requires range<const _Vp>
10588 {
return _Iterator<true>(ranges::begin(_M_base)); }
10591 end()
requires (!__detail::__simple_view<_Vp>)
10592 {
return ranges::end(_M_base); }
10595 end() const requires range<const _Vp>
10596 {
return ranges::end(_M_base); }
10599 size()
requires sized_range<_Vp>
10600 {
return ranges::size(_M_base); }
10603 size() const requires sized_range<const _Vp>
10604 {
return ranges::size(_M_base); }
10607 template<
typename _Range>
10608 as_input_view(_Range&&) -> as_input_view<views::all_t<_Range>>;
10610 template<input_range _Vp>
10612 template<
bool _Const>
10613 class as_input_view<_Vp>::_Iterator
10615 using _Base = __maybe_const_t<_Const, _Vp>;
10617 iterator_t<_Base> _M_current = iterator_t<_Base>();
10620 _Iterator(iterator_t<_Base> __current)
10621 : _M_current(std::move(__current))
10624 friend as_input_view;
10625 friend _Iterator<!_Const>;
10628 using difference_type = range_difference_t<_Base>;
10629 using value_type = range_value_t<_Base>;
10630 using iterator_concept = input_iterator_tag;
10632 _Iterator()
requires default_initializable<iterator_t<_Base>> = default;
10634 _Iterator(_Iterator&&) = default;
10635 _Iterator& operator=(_Iterator&&) = default;
10638 _Iterator(_Iterator<!_Const> __i)
10639 requires _Const && convertible_to<iterator_t<_Vp>, iterator_t<_Base>>
10640 : _M_current(std::move(__i._M_current))
10643 constexpr iterator_t<_Base>
10647 constexpr const iterator_t<_Base>&
10648 base() const & noexcept
10649 {
return _M_current; }
10651 constexpr decltype(
auto)
10653 {
return *_M_current; }
10655 constexpr _Iterator&
10666 friend constexpr bool
10667 operator==(
const _Iterator& __x,
const sentinel_t<_Base>& __y)
10668 {
return __x._M_current == __y; }
10670 friend constexpr difference_type
10671 operator-(
const sentinel_t<_Base>& __y,
const _Iterator& __x)
10672 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
10673 {
return __y - __x._M_current; }
10675 friend constexpr difference_type
10676 operator-(
const _Iterator& __x,
const sentinel_t<_Base>& __y)
10677 requires sized_sentinel_for<sentinel_t<_Base>, iterator_t<_Base>>
10678 {
return __x._M_current - __y; }
10680 friend constexpr range_rvalue_reference_t<_Base>
10681 iter_move(
const _Iterator& __i)
10682 noexcept(
noexcept(ranges::iter_move(__i._M_current)))
10683 {
return ranges::iter_move(__i._M_current); }
10685 friend constexpr void
10686 iter_swap(
const _Iterator& __x,
const _Iterator& __y)
10687 noexcept(
noexcept(ranges::iter_swap(__x._M_current, __y._M_current)))
10688 requires indirectly_swappable<iterator_t<_Base>>
10689 { ranges::iter_swap(__x._M_current, __y._M_current); }
10696 template<
typename _Tp>
10700 struct _AsInput : __adaptor::_RangeAdaptorClosure<_AsInput>
10702 template<viewable_range _Range>
10703 requires __detail::__can_as_input<_Range>
10705 operator() [[nodiscard]] (_Range&& __r)
const
10707 if constexpr (input_range<_Range>
10708 && !common_range<_Range>
10709 && !forward_range<_Range>)
10715 static constexpr bool _S_has_simple_call_op =
true;
10718 inline constexpr _AsInput as_input;
10723_GLIBCXX_END_NAMESPACE_VERSION
constexpr bool operator<=(const duration< _Rep1, _Period1 > &__lhs, const duration< _Rep2, _Period2 > &__rhs)
constexpr bool operator>=(const duration< _Rep1, _Period1 > &__lhs, const duration< _Rep2, _Period2 > &__rhs)
constexpr bool operator<(const duration< _Rep1, _Period1 > &__lhs, const duration< _Rep2, _Period2 > &__rhs)
constexpr bool operator>(const duration< _Rep1, _Period1 > &__lhs, const duration< _Rep2, _Period2 > &__rhs)
constexpr complex< _Tp > operator*(const complex< _Tp > &__x, const complex< _Tp > &__y)
Return new complex value x times y.
constexpr complex< _Tp > operator-(const complex< _Tp > &__x, const complex< _Tp > &__y)
Return new complex value x minus y.
constexpr complex< _Tp > operator+(const complex< _Tp > &__x, const complex< _Tp > &__y)
Return new complex value x plus y.
constexpr _Tp * to_address(_Tp *__ptr) noexcept
Obtain address referenced by a pointer to an object.
typename remove_reference< _Tp >::type remove_reference_t
Alias template for remove_reference.
typename common_type< _Tp... >::type common_type_t
Alias template for common_type.
typename make_unsigned< _Tp >::type make_unsigned_t
Alias template for make_unsigned.
typename decay< _Tp >::type decay_t
Alias template for decay.
auto declval() noexcept -> decltype(__declval< _Tp >(0))
constexpr _Tp * addressof(_Tp &__r) noexcept
Returns the actual address of the object or function referenced by r, even in the presence of an over...
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
constexpr __invoke_result< _Callable, _Args... >::type __invoke(_Callable &&__fn, _Args &&... __args) noexcept(__is_nothrow_invocable< _Callable, _Args... >::value)
Invoke a callable object.
constexpr _Tp * __addressof(_Tp &__r) noexcept
Same as C++11 std::addressof.
constexpr _Tp && forward(typename std::remove_reference< _Tp >::type &__t) noexcept
Forward an lvalue.
_Tp * end(valarray< _Tp > &__va) noexcept
Return an iterator pointing to one past the last element of the valarray.
_Tp * begin(valarray< _Tp > &__va) noexcept
Return an iterator pointing to the first element of the valarray.
constexpr const _Tp & min(const _Tp &, const _Tp &)
This does what you think it does.
constexpr reverse_iterator< _Iterator > make_reverse_iterator(_Iterator __i)
Generator function for reverse_iterator.
ISO C++ entities toplevel namespace is std.
make_integer_sequence< size_t, _Num > make_index_sequence
Alias template make_index_sequence.
integer_sequence< size_t, _Idx... > index_sequence
Alias template index_sequence.
constexpr auto empty(const _Container &__cont) noexcept(noexcept(__cont.empty())) -> decltype(__cont.empty())
Return whether a container is empty.
constexpr auto size(const _Container &__cont) noexcept(noexcept(__cont.size())) -> decltype(__cont.size())
Return the size of a container.
constexpr default_sentinel_t default_sentinel
A default sentinel value.
constexpr auto data(_Container &__cont) noexcept(noexcept(__cont.data())) -> decltype(__cont.data())
Return the data pointer of a container.
constexpr _Iterator __base(_Iterator __it)
A view that contains no elements.
A view that contains exactly one element.
constexpr reference_wrapper< _Tp > ref(_Tp &__t) noexcept
Denotes a reference should be taken to a variable.
constexpr reference_wrapper< const _Tp > cref(const _Tp &__t) noexcept
Denotes a const reference should be taken to a variable.
The ranges::view_interface class template.
[concept.constructible], concept constructible_from
[concept.defaultinitializable], concept default_initializable
[concept.moveconstructible], concept move_constructible
[concept.copyconstructible], concept copy_constructible
[range.range] The range concept.
[range.sized] The sized_range concept.
[range.view] The ranges::view concept.
A range for which ranges::begin returns an input iterator.
A range for which ranges::begin returns a forward iterator.
A range for which ranges::begin returns a bidirectional iterator.
A range for which ranges::begin returns a random access iterator.
A range for which ranges::begin returns a contiguous iterator.
A range for which ranges::begin and ranges::end return the same type.