struct _ZipTransform
{
+ template<typename _Fp>
+ requires move_constructible<decay_t<_Fp>> && regular_invocable<decay_t<_Fp>&>
+ && is_object_v<decay_t<invoke_result_t<decay_t<_Fp>&>>>
+ constexpr auto
+ operator() [[nodiscard]] (_Fp&& __f) const
+ {
+ return views::empty<decay_t<invoke_result_t<decay_t<_Fp>&>>>;
+ }
+
template<typename _Fp, typename... _Ts>
- requires (sizeof...(_Ts) == 0) || __detail::__can_zip_transform_view<_Fp, _Ts...>
+ requires (sizeof...(_Ts) != 0) && __detail::__can_zip_transform_view<_Fp, _Ts...>
constexpr auto
operator() [[nodiscard]] (_Fp&& __f, _Ts&&... __ts) const
{
- if constexpr (sizeof...(_Ts) == 0)
- return views::empty<decay_t<invoke_result_t<decay_t<_Fp>&>>>;
- else
- return zip_transform_view(std::forward<_Fp>(__f), std::forward<_Ts>(__ts)...);
+ return zip_transform_view(std::forward<_Fp>(__f), std::forward<_Ts>(__ts)...);
}
};
namespace ranges = std::ranges;
namespace views = std::views;
+template<typename T>
+concept can_zip_transform = requires (T t) {
+ views::zip_transform(std::forward<T>(t));
+};
+
+static_assert(!can_zip_transform<int>);
+
+struct NonMovable {
+ NonMovable(NonMovable&&) = delete;
+};
+
+static_assert(!can_zip_transform<NonMovable>);
+static_assert(!can_zip_transform<NonMovable&>);
+
+static_assert(!can_zip_transform<void(*)()>);
+static_assert(can_zip_transform<int(&(*)())[3]>);
+
constexpr bool
test01()
{
VERIFY( ranges::size(z3) == 3 );
VERIFY( ranges::equal(z3, (int[]){3, 6, 9}) );
+ auto z4 = views::zip_transform([] () { return 1; });
+ VERIFY( ranges::size(z4) == 0 );
+ static_assert( std::same_as<ranges::range_value_t<decltype(z4)>, int> );
+
return true;
}