axmol/thirdparty/range-v3/include/range/v3/detail/variant.hpp

894 lines
31 KiB
C++

// Range v3 library
//
// Copyright Eric Niebler 2014-present
//
// Use, modification and distribution is subject to the
// Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt)
//
// Project home: https://github.com/ericniebler/range-v3
//
#ifndef RANGES_V3_DETAIL_VARIANT_HPP
#define RANGES_V3_DETAIL_VARIANT_HPP
#include <iterator>
#include <memory>
#include <new>
#include <stdexcept>
#include <tuple>
#include <type_traits>
#include <utility>
#include <meta/meta.hpp>
#include <concepts/concepts.hpp>
#include <range/v3/range_fwd.hpp>
#include <range/v3/functional/compose.hpp>
#include <range/v3/functional/identity.hpp>
#include <range/v3/functional/invoke.hpp>
#include <range/v3/iterator/concepts.hpp>
#include <range/v3/iterator/traits.hpp>
#include <range/v3/utility/get.hpp>
#include <range/v3/detail/prologue.hpp>
namespace ranges
{
template<std::size_t I>
struct emplaced_index_t;
template<std::size_t I>
struct emplaced_index_t : meta::size_t<I>
{};
#if RANGES_CXX_INLINE_VARIABLES < RANGES_CXX_INLINE_VARIABLES_17
namespace
{
template<std::size_t I>
constexpr auto & emplaced_index = static_const<emplaced_index_t<I>>::value;
}
#else // RANGES_CXX_INLINE_VARIABLES >= RANGES_CXX_INLINE_VARIABLES_17
template<std::size_t I>
inline constexpr emplaced_index_t<I> emplaced_index{};
#endif // RANGES_CXX_INLINE_VARIABLES
struct bad_variant_access : std::logic_error
{
explicit bad_variant_access(std::string const & what_arg)
: std::logic_error(what_arg)
{}
explicit bad_variant_access(char const * what_arg)
: std::logic_error(what_arg)
{}
};
template<typename T, std::size_t Index>
struct indexed_element
{
private:
std::add_pointer_t<T> t_;
public:
constexpr explicit indexed_element(T & t) noexcept
: t_(std::addressof(t))
{}
constexpr T & get() const noexcept
{
return *t_;
}
};
template<typename T, std::size_t Index>
struct indexed_element<T &&, Index>
{
private:
T * t_;
public:
constexpr explicit indexed_element(T && t) noexcept
: t_(std::addressof(t))
{}
constexpr T && get() const noexcept
{
return static_cast<T &&>(*t_);
}
};
template<std::size_t Index>
struct indexed_element<void, Index>
{
void get() const noexcept
{}
};
/// \cond
namespace detail
{
struct indexed_element_fn;
template(typename I, typename S, typename O)(
requires (!sized_sentinel_for<S, I>)) //
O uninitialized_copy(I first, S last, O out)
{
for(; first != last; ++first, ++out)
::new((void *)std::addressof(*out)) iter_value_t<O>(*first);
return out;
}
template(typename I, typename S, typename O)(
requires sized_sentinel_for<S, I>)
O uninitialized_copy(I first, S last, O out)
{
return std::uninitialized_copy_n(first, (last - first), out);
}
template<typename I, typename O>
O uninitialized_copy(I first, I last, O out)
{
return std::uninitialized_copy(first, last, out);
}
template<typename T, typename Index>
struct indexed_datum
{
private:
template<typename, typename>
friend struct indexed_datum;
T datum_;
public:
CPP_member
constexpr CPP_ctor(indexed_datum)()( //
noexcept(std::is_nothrow_default_constructible<T>::value) //
requires default_constructible<T>)
: datum_{}
{}
template(typename... Ts)(
requires constructible_from<T, Ts...> AND (sizeof...(Ts) != 0)) //
constexpr indexed_datum(Ts &&... ts) noexcept(
std::is_nothrow_constructible<T, Ts...>::value)
: datum_(static_cast<Ts &&>(ts)...)
{}
template(typename U)(
requires (!same_as<T, U>) AND convertible_to<U, T>)
constexpr indexed_datum(indexed_datum<U, Index> that) //
noexcept(std::is_nothrow_constructible<T, U>::value) //
: datum_(std::move(that.datum_))
{}
constexpr auto ref() noexcept
{
return indexed_element<T, Index::value>{datum_};
}
constexpr auto ref() const noexcept
{
return indexed_element<T const, Index::value>{datum_};
}
constexpr T & get() noexcept
{
return datum_;
}
constexpr T const & get() const noexcept
{
return datum_;
}
};
template<typename T, std::size_t N, typename Index>
struct indexed_datum<T[N], Index>;
template<typename T, typename Index>
struct indexed_datum<T &, Index>
{
private:
template<typename, typename>
friend struct indexed_datum;
T * t_;
public:
constexpr indexed_datum(T & t) noexcept
: t_(std::addressof(t))
{}
constexpr T & get() const noexcept
{
return *t_;
}
constexpr auto ref() const noexcept
{
return indexed_element<T &, Index::value>{*t_};
}
};
template<typename T, typename Index>
struct indexed_datum<T &&, Index>
{
private:
template<typename, typename>
friend struct indexed_datum;
T * t_;
public:
constexpr indexed_datum(T && t) noexcept
: t_(std::addressof(t))
{}
constexpr T && get() const noexcept
{
return static_cast<T &&>(*t_);
}
constexpr auto ref() const noexcept
{
return indexed_element<T &&, Index::value>{static_cast<T &&>(*t_)};
}
};
template<typename Index>
struct indexed_datum<void, Index>
{
void get() const noexcept
{}
constexpr indexed_element<void, Index::value> ref() const noexcept
{
return {};
}
};
template<std::size_t Index, typename... Ts>
using variant_datum_t =
detail::indexed_datum<meta::at_c<meta::list<Ts...>, Index>,
meta::size_t<Index>>;
using variant_nil = indexed_datum<void, meta::npos>;
template<typename Ts,
bool Trivial = meta::apply<
meta::quote<meta::and_>,
meta::transform<Ts, meta::quote<std::is_trivially_destructible>>>::
type::value>
struct variant_data_
{
using type = indexed_datum<void, meta::npos>;
};
template<typename T, typename... Ts>
struct variant_data_<meta::list<T, Ts...>, true>
{
struct type
{
using head_t = T;
using tail_t = meta::_t<variant_data_<meta::list<Ts...>>>;
union
{
head_t head;
tail_t tail;
};
type() noexcept
{}
template<typename... Args>
constexpr type(meta::size_t<0>, Args &&... args) noexcept(
std::is_nothrow_constructible<head_t, Args...>::value)
: head{((Args &&) args)...}
{}
template<std::size_t N, typename... Args>
constexpr type(meta::size_t<N>, Args &&... args) noexcept(
std::is_nothrow_constructible<tail_t, meta::size_t<N - 1>,
Args...>::value)
: tail{meta::size_t<N - 1>{}, ((Args &&) args)...}
{}
};
};
template<typename T, typename... Ts>
struct variant_data_<meta::list<T, Ts...>, false>
{
struct type
{
using head_t = T;
using tail_t = meta::_t<variant_data_<meta::list<Ts...>>>;
union
{
head_t head;
tail_t tail;
};
type() noexcept
{}
~type()
{}
template<typename... Args>
constexpr type(meta::size_t<0>, Args &&... args) noexcept(
std::is_nothrow_constructible<head_t, Args...>::value)
: head{((Args &&) args)...}
{}
template<std::size_t N, typename... Args>
constexpr type(meta::size_t<N>, Args &&... args) noexcept(
std::is_nothrow_constructible<tail_t, meta::size_t<N - 1>,
Args...>::value)
: tail{meta::size_t<N - 1>{}, ((Args &&) args)...}
{}
};
};
template<typename... Ts>
using variant_data = meta::_t<variant_data_<meta::transform<
meta::list<Ts...>, meta::as_list<meta::make_index_sequence<sizeof...(Ts)>>,
meta::quote<indexed_datum>>>>;
inline std::size_t variant_move_copy_(std::size_t, variant_nil, variant_nil)
{
return 0;
}
template<typename Data0, typename Data1>
std::size_t variant_move_copy_(std::size_t n, Data0 & self, Data1 && that)
{
using Head = typename Data0::head_t;
return 0 == n
? ((void)::new((void *)&self.head) Head(((Data1 &&) that).head), 0)
: variant_move_copy_(n - 1, self.tail, ((Data1 &&) that).tail) + 1;
}
constexpr bool variant_equal_(std::size_t, variant_nil, variant_nil)
{
return true;
}
template<typename Data0, typename Data1>
constexpr bool variant_equal_(std::size_t n, Data0 const & self,
Data1 const & that)
{
return n == 0 ? self.head.get() == that.head.get()
: variant_equal_(n - 1, self.tail, that.tail);
}
template<typename Fun, typename Proj = indexed_element_fn>
constexpr int variant_visit_(std::size_t, variant_nil, Fun, Proj = {})
{
return (RANGES_EXPECT(false), 0);
}
template<typename Data, typename Fun, typename Proj = indexed_element_fn>
constexpr int variant_visit_(std::size_t n, Data & self, Fun fun, Proj proj = {})
{
return 0 == n ? ((void)invoke(fun, invoke(proj, self.head)), 0)
: detail::variant_visit_(
n - 1, self.tail, detail::move(fun), detail::move(proj));
}
struct get_datum_fn
{
template<typename T>
decltype(auto) operator()(T && t) const noexcept
{
return t.get();
}
};
struct indexed_element_fn
{
template<typename T>
decltype(auto) operator()(T && t) const noexcept
{
return t.ref();
}
};
struct empty_variant_tag
{};
struct variant_core_access
{
template<typename... Ts>
static constexpr variant_data<Ts...> & data(variant<Ts...> & var) noexcept
{
return var.data_();
}
template<typename... Ts>
static constexpr variant_data<Ts...> const & data(
variant<Ts...> const & var) noexcept
{
return var.data_();
}
template<typename... Ts>
static constexpr variant_data<Ts...> && data(variant<Ts...> && var) noexcept
{
return detail::move(var.data_());
}
template<typename... Ts>
static variant<Ts...> make_empty(meta::id<variant<Ts...>> = {}) noexcept
{
return variant<Ts...>{empty_variant_tag{}};
}
};
struct delete_fn
{
template<typename T>
void operator()(T const & t) const noexcept
{
t.~T();
}
};
template<std::size_t N, typename... Ts>
struct construct_fn
{
std::tuple<Ts...> args_;
template<typename U, std::size_t... Is>
void construct_(U & u, meta::index_sequence<Is...>) noexcept(
std::is_nothrow_constructible<U, Ts...>::value)
{
::new((void *)std::addressof(u))
U(static_cast<Ts &&>(std::get<Is>(args_))...);
}
construct_fn(Ts &&... ts) noexcept(
std::is_nothrow_constructible<std::tuple<Ts...>, Ts...>::value)
: args_{static_cast<Ts &&>(ts)...}
{}
template<typename U, std::size_t M>
[[noreturn]] meta::if_c<N != M> operator()(
indexed_datum<U, meta::size_t<M>> &) noexcept
{
RANGES_EXPECT(false);
}
template<typename U>
meta::if_<std::is_object<U>> operator()(
indexed_datum<U, meta::size_t<N>> &
u) noexcept(std::is_nothrow_constructible<U, Ts...>::value)
{
this->construct_(u.get(), meta::make_index_sequence<sizeof...(Ts)>{});
}
template<typename U>
meta::if_<meta::not_<std::is_object<U>>> operator()(
indexed_datum<U, meta::size_t<N>> &
u) noexcept(std::is_nothrow_constructible<detail::decay_t<U>,
Ts...>::value)
{
this->construct_(u, meta::make_index_sequence<sizeof...(Ts)>{});
}
};
template<typename T, std::size_t N>
struct get_fn
{
T ** t_;
template<typename U, std::size_t M>
[[noreturn]] meta::if_c<M != N> operator()(indexed_element<U, M>) const
{
throw bad_variant_access("bad variant access");
}
template<typename U>
void operator()(indexed_element<U, N> t) const noexcept
{
*t_ = std::addressof(t.get());
}
template<typename U>
void operator()(indexed_element<U &&, N> t) const noexcept
{
U && u = t.get();
*t_ = std::addressof(u);
}
void operator()(indexed_element<void, N>) const noexcept
{}
};
template<typename Variant, std::size_t N>
struct emplace_fn
{
Variant * var_;
// clang-format off
template<typename...Ts>
auto CPP_auto_fun(operator())(Ts &&...ts) (const)
(
return var_->template emplace<N>(static_cast<Ts &&>(ts)...)
)
// clang-format on
};
template<typename Fun, typename Variant>
struct variant_visitor
{
Fun fun_;
Variant * var_;
// clang-format off
template<typename U, std::size_t N>
auto CPP_auto_fun(operator())(indexed_element<U, N> u)
(
return compose(emplace_fn<Variant, N>{var_}, fun_)(u)
)
// clang-format on
};
template<typename Variant, typename Fun>
variant_visitor<Fun, Variant> make_variant_visitor(
Variant & var,
Fun fun) noexcept(std::is_nothrow_move_constructible<Fun>::value)
{
return {detail::move(fun), &var};
}
template<typename To, typename From>
struct unique_visitor;
template<typename... To, typename... From>
struct unique_visitor<variant<To...>, variant<From...>>
{
variant<To...> * var_;
template<typename T, std::size_t N>
void operator()(indexed_element<T, N> t) const
{
using E = meta::at_c<meta::list<From...>, N>;
static_assert(RANGES_IS_SAME(T const, E const),
"Is indexed_element broken?");
using F = meta::find<meta::list<To...>, E>;
static constexpr std::size_t M = sizeof...(To) - F::size();
compose(emplace_fn<variant<To...>, M>{var_}, get_datum_fn{})(t);
}
};
template<typename T>
constexpr T & variant_deref_(T * t) noexcept
{
return *t;
}
inline void variant_deref_(void const volatile *) noexcept
{}
template<typename Variant>
struct variant_get
{
//////////////////////////////////////////////////////////////////////////////
// get
template<std::size_t N>
friend meta::_t<
std::add_lvalue_reference<meta::at_c<meta::as_list<Variant>, N>>>
get(Variant & var)
{
using elem_t = meta::_t<
std::remove_reference<meta::at_c<meta::as_list<Variant>, N>>>;
elem_t * elem = nullptr;
auto & data_var = detail::variant_core_access::data(var);
detail::variant_visit_(
var.index(), data_var, detail::get_fn<elem_t, N>{&elem});
return detail::variant_deref_(elem);
}
template<std::size_t N>
friend meta::_t<
std::add_lvalue_reference<meta::at_c<meta::as_list<Variant>, N> const>>
get(Variant const & var)
{
using elem_t = meta::_t<
std::remove_reference<meta::at_c<meta::as_list<Variant>, N> const>>;
elem_t * elem = nullptr;
auto & data_var = detail::variant_core_access::data(var);
detail::variant_visit_(
var.index(), data_var, detail::get_fn<elem_t, N>{&elem});
return detail::variant_deref_(elem);
}
template<std::size_t N>
friend meta::_t<
std::add_rvalue_reference<meta::at_c<meta::as_list<Variant>, N>>>
get(Variant && var)
{
using elem_t = meta::_t<
std::remove_reference<meta::at_c<meta::as_list<Variant>, N>>>;
elem_t * elem = nullptr;
auto & data_var = detail::variant_core_access::data(var);
detail::variant_visit_(
var.index(), data_var, detail::get_fn<elem_t, N>{&elem});
using res_t = meta::_t<
std::add_rvalue_reference<meta::at_c<meta::as_list<Variant>, N>>>;
return static_cast<res_t>(detail::variant_deref_(elem));
}
};
template<typename Variant,
bool Trivial = std::is_trivially_destructible<meta::apply<
meta::quote<variant_data>, meta::as_list<Variant>>>::value>
struct variant_base : variant_get<Variant>
{
~variant_base()
{
static_cast<Variant *>(this)->clear_();
}
};
template<typename... Ts>
struct variant_base<variant<Ts...>, true> : variant_get<variant<Ts...>>
{};
template<typename Fun, typename Types, typename Indices, typename = void>
struct variant_visit_results
{};
template<typename Fun, typename... Ts, std::size_t... Is>
struct variant_visit_results<
Fun, meta::list<Ts...>, meta::index_sequence<Is...>,
meta::void_<invoke_result_t<Fun &, indexed_element<Ts, Is>>...>>
{
using type = variant<invoke_result_t<Fun &, indexed_element<Ts, Is>>...>;
};
template<typename Fun, typename... Ts>
using variant_visit_results_t =
meta::_t<variant_visit_results<Fun, meta::list<Ts...>,
meta::make_index_sequence<sizeof...(Ts)>>>;
} // namespace detail
/// \endcond
/// \addtogroup group-utility
/// @{
template<typename... Ts>
struct variant
: private detail::variant_data<Ts...>
, private detail::variant_base<variant<Ts...>>
{
private:
friend detail::variant_core_access;
template<typename...>
friend struct variant;
friend detail::variant_base<variant, false>;
template<std::size_t Index>
using datum_t = detail::variant_datum_t<Index, Ts...>;
template<typename T>
using add_const_t = meta::if_<std::is_void<T>, void, T const>;
using unbox_fn = detail::get_datum_fn;
detail::variant_data<Ts...> & data_() & noexcept
{
return *this;
}
detail::variant_data<Ts...> const & data_() const & noexcept
{
return *this;
}
detail::variant_data<Ts...> && data_() && noexcept
{
return static_cast<detail::variant_data<Ts...> &&>(*this);
}
std::size_t index_;
void clear_() noexcept
{
if(valid())
{
detail::variant_visit_(index_, data_(), detail::delete_fn{}, identity{});
index_ = (std::size_t)-1;
}
}
template<typename That>
void assign_(That && that)
{
if(that.valid())
index_ = detail::variant_move_copy_(
that.index_, data_(), ((That &&) that).data_());
}
constexpr variant(detail::empty_variant_tag) noexcept
: detail::variant_data<Ts...>{}
, index_((std::size_t)-1)
{}
template(typename... Args)(
requires (sizeof...(Args) == sizeof...(Ts))) //
static constexpr bool all_convertible_to(int) noexcept
{
return and_v<convertible_to<Args, Ts>...>;
}
template<typename... Args>
static constexpr bool all_convertible_to(long) noexcept
{
return false;
}
public:
CPP_member
constexpr CPP_ctor(variant)()( //
noexcept(std::is_nothrow_default_constructible<datum_t<0>>::value) //
requires default_constructible<datum_t<0>>)
: variant{emplaced_index<0>}
{}
template(std::size_t N, typename... Args)(
requires constructible_from<datum_t<N>, Args...>)
constexpr variant(emplaced_index_t<N>, Args &&... args) noexcept(
std::is_nothrow_constructible<datum_t<N>, Args...>::value)
: detail::variant_data<Ts...>{meta::size_t<N>{}, static_cast<Args &&>(args)...}
, index_(N)
{}
template(std::size_t N, typename T, typename... Args)(
requires constructible_from<datum_t<N>, std::initializer_list<T> &,
Args...>)
constexpr variant(
emplaced_index_t<N>, std::initializer_list<T> il,
Args &&... args) noexcept(std::
is_nothrow_constructible<
datum_t<N>, std::initializer_list<T> &,
Args...>::value)
: detail::variant_data<Ts...>{meta::size_t<N>{},
il,
static_cast<Args &&>(args)...}
, index_(N)
{}
template(std::size_t N)(
requires constructible_from<datum_t<N>, meta::nil_>)
constexpr variant(emplaced_index_t<N>, meta::nil_)
noexcept(std::is_nothrow_constructible<datum_t<N>, meta::nil_>::value)
: detail::variant_data<Ts...>{meta::size_t<N>{}, meta::nil_{}}
, index_(N)
{}
variant(variant && that)
: detail::variant_data<Ts...>{}
, index_(detail::variant_move_copy_(that.index(), data_(),
std::move(that.data_())))
{}
variant(variant const & that)
: detail::variant_data<Ts...>{}
, index_(detail::variant_move_copy_(that.index(), data_(), that.data_()))
{}
template(typename... Args)(
requires (!same_as<variant<Args...>, variant>) AND
(all_convertible_to<Args...>(0))) //
variant(variant<Args...> that)
: detail::variant_data<Ts...>{}
, index_(detail::variant_move_copy_(that.index(), data_(),
std::move(that.data_())))
{}
variant & operator=(variant && that)
{
// TODO do a simple move assign when index()==that.index()
this->clear_();
this->assign_(detail::move(that));
return *this;
}
variant & operator=(variant const & that)
{
// TODO do a simple copy assign when index()==that.index()
this->clear_();
this->assign_(that);
return *this;
}
template(typename... Args)(
requires (!same_as<variant<Args...>, variant>) AND
(all_convertible_to<Args...>(0)))
variant & operator=(variant<Args...> that)
{
// TODO do a simple copy assign when index()==that.index() //
this->clear_();
this->assign_(that);
return *this;
}
static constexpr std::size_t size() noexcept
{
return sizeof...(Ts);
}
template(std::size_t N, typename... Args)(
requires constructible_from<datum_t<N>, Args...>)
void emplace(Args &&... args)
{
this->clear_();
detail::construct_fn<N, Args &&...> fn{static_cast<Args &&>(args)...};
detail::variant_visit_(N, data_(), std::ref(fn), identity{});
index_ = N;
}
constexpr bool valid() const noexcept
{
return index() != (std::size_t)-1;
}
constexpr std::size_t index() const noexcept
{
return index_;
}
template<typename Fun>
detail::variant_visit_results_t<composed<Fun, unbox_fn>, Ts...> visit(Fun fun)
{
detail::variant_visit_results_t<composed<Fun, unbox_fn>, Ts...> res{
detail::empty_variant_tag{}};
detail::variant_visit_(index_,
data_(),
detail::make_variant_visitor(
res, compose(detail::move(fun), unbox_fn{})));
return res;
}
template<typename Fun>
detail::variant_visit_results_t<composed<Fun, unbox_fn>, add_const_t<Ts>...>
visit(Fun fun) const
{
detail::variant_visit_results_t<composed<Fun, unbox_fn>, add_const_t<Ts>...>
res{detail::empty_variant_tag{}};
detail::variant_visit_(index_,
data_(),
detail::make_variant_visitor(
res, compose(detail::move(fun), unbox_fn{})));
return res;
}
template<typename Fun>
detail::variant_visit_results_t<Fun, Ts...> visit_i(Fun fun)
{
detail::variant_visit_results_t<Fun, Ts...> res{detail::empty_variant_tag{}};
detail::variant_visit_(
index_, data_(), detail::make_variant_visitor(res, detail::move(fun)));
return res;
}
template<typename Fun>
detail::variant_visit_results_t<Fun, add_const_t<Ts>...> visit_i(Fun fun) const
{
detail::variant_visit_results_t<Fun, add_const_t<Ts>...> res{
detail::empty_variant_tag{}};
detail::variant_visit_(
index_, data_(), detail::make_variant_visitor(res, detail::move(fun)));
return res;
}
};
template(typename... Ts, typename... Us)(
requires and_v<equality_comparable_with<Ts, Us>...>)
bool operator==(variant<Ts...> const & lhs, variant<Us...> const & rhs)
{
return (!lhs.valid() && !rhs.valid()) ||
(lhs.index() == rhs.index() &&
detail::variant_equal_(lhs.index(),
detail::variant_core_access::data(lhs),
detail::variant_core_access::data(rhs)));
}
template(typename... Ts, typename... Us)(
requires and_v<equality_comparable_with<Ts, Us>...>)
bool operator!=(variant<Ts...> const & lhs, variant<Us...> const & rhs)
{
return !(lhs == rhs);
}
//////////////////////////////////////////////////////////////////////////////////////
// emplace
template(std::size_t N, typename... Ts, typename... Args)(
requires constructible_from<detail::variant_datum_t<N, Ts...>, Args...>)
void emplace(variant<Ts...> & var, Args &&... args)
{
var.template emplace<N>(static_cast<Args &&>(args)...);
}
//////////////////////////////////////////////////////////////////////////////////////
// variant_unique
template<typename Var>
struct variant_unique
{};
template<typename... Ts>
struct variant_unique<variant<Ts...>>
{
using type = meta::apply<meta::quote<variant>, meta::unique<meta::list<Ts...>>>;
};
template<typename Var>
using variant_unique_t = meta::_t<variant_unique<Var>>;
//////////////////////////////////////////////////////////////////////////////////////
// unique_variant
template<typename... Ts>
variant_unique_t<variant<Ts...>> unique_variant(variant<Ts...> const & var)
{
using From = variant<Ts...>;
using To = variant_unique_t<From>;
auto res = detail::variant_core_access::make_empty(meta::id<To>{});
var.visit_i(detail::unique_visitor<To, From>{&res});
RANGES_EXPECT(res.valid());
return res;
}
/// @}
} // namespace ranges
RANGES_DIAGNOSTIC_PUSH
RANGES_DIAGNOSTIC_IGNORE_MISMATCHED_TAGS
namespace std
{
template<typename... Ts>
struct tuple_size<::ranges::variant<Ts...>> : tuple_size<tuple<Ts...>>
{};
template<size_t I, typename... Ts>
struct tuple_element<I, ::ranges::variant<Ts...>> : tuple_element<I, tuple<Ts...>>
{};
} // namespace std
RANGES_DIAGNOSTIC_POP
#include <range/v3/detail/epilogue.hpp>
#endif