2022-03-15 19:26:31 +08:00
# pragma once
# include "CoreTypes.h"
# include "Templates/Invoke.h"
# include "Templates/Utility.h"
2022-04-22 22:28:44 +08:00
# include "Templates/TypeHash.h"
2022-03-15 19:26:31 +08:00
# include "TypeTraits/TypeTraits.h"
2022-05-12 23:36:32 +08:00
# include "Memory/MemoryOperator.h"
2022-05-03 18:22:41 +08:00
# include "Miscellaneous/Compare.h"
2022-03-22 11:12:05 +08:00
# include "Miscellaneous/AssertionMacros.h"
2022-03-15 19:26:31 +08:00
NAMESPACE_REDCRAFT_BEGIN
NAMESPACE_MODULE_BEGIN ( Redcraft )
NAMESPACE_MODULE_BEGIN ( Utility )
2022-06-16 23:37:29 +08:00
template < typename . . . Types > requires ( true & & . . . & & CDestructible < Types > )
struct TVariant ;
2022-03-15 19:26:31 +08:00
NAMESPACE_PRIVATE_BEGIN
2022-06-16 23:37:29 +08:00
template < typename T > struct TIsTVariant : FFalse { } ;
template < typename . . . Types > struct TIsTVariant < TVariant < Types . . . > > : FTrue { } ;
template < typename TupleType >
struct TVariantNumImpl ;
template < typename . . . Types >
struct TVariantNumImpl < TVariant < Types . . . > > : TConstant < size_t , sizeof . . . ( Types ) > { } ;
template < typename . . . Types >
struct TVariantNumImpl < const TVariant < Types . . . > > : TConstant < size_t , sizeof . . . ( Types ) > { } ;
template < typename . . . Types >
struct TVariantNumImpl < volatile TVariant < Types . . . > > : TConstant < size_t , sizeof . . . ( Types ) > { } ;
template < typename . . . Types >
struct TVariantNumImpl < const volatile TVariant < Types . . . > > : TConstant < size_t , sizeof . . . ( Types ) > { } ;
template < typename T , typename TupleType >
struct TVariantIndexImpl ;
2022-03-15 19:26:31 +08:00
template < typename T , typename U , typename . . . Types >
2022-06-16 23:37:29 +08:00
struct TVariantIndexImpl < T , TVariant < U , Types . . . > > : TConstant < size_t , TVariantIndexImpl < T , TVariant < Types . . . > > : : Value + 1 >
{
static_assert ( sizeof . . . ( Types ) ! = 0 , " Non-existent types in variant " ) ;
} ;
template < typename T , typename . . . Types >
struct TVariantIndexImpl < T , TVariant < T , Types . . . > > : TConstant < size_t , 0 >
{
static_assert ( ( true & & . . . & & ! CSameAs < T , Types > ) , " Duplicate type in variant " ) ;
} ;
2022-03-15 19:26:31 +08:00
template < typename T >
2022-06-16 23:37:29 +08:00
struct TVariantIndexImpl < T , TVariant < > > : TConstant < size_t , INDEX_NONE > { } ;
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < typename T , typename . . . Types >
struct TVariantIndexImpl < T , const TVariant < Types . . . > > : TVariantIndexImpl < T , TVariant < Types . . . > > { } ;
template < typename T , typename . . . Types >
struct TVariantIndexImpl < T , volatile TVariant < Types . . . > > : TVariantIndexImpl < T , TVariant < Types . . . > > { } ;
template < typename T , typename . . . Types >
struct TVariantIndexImpl < T , const volatile TVariant < Types . . . > > : TVariantIndexImpl < T , TVariant < Types . . . > > { } ;
template < size_t I , typename TupleType >
struct TVariantAlternativeImpl ;
2022-03-15 19:26:31 +08:00
template < size_t I , typename T , typename . . . Types >
2022-06-16 23:37:29 +08:00
struct TVariantAlternativeImpl < I , TVariant < T , Types . . . > >
2022-03-15 19:26:31 +08:00
{
2022-06-16 23:37:29 +08:00
static_assert ( I < sizeof . . . ( Types ) + 1 , " Invalid index in variant " ) ;
using Type = TVariantAlternativeImpl < I - 1 , TVariant < Types . . . > > : : Type ;
2022-03-15 19:26:31 +08:00
} ;
template < typename T , typename . . . Types >
2022-06-16 23:37:29 +08:00
struct TVariantAlternativeImpl < 0 , TVariant < T , Types . . . > > { using Type = T ; } ;
template < size_t I , typename . . . Types >
struct TVariantAlternativeImpl < I , TVariant < Types . . . > > { } ;
2022-03-15 19:26:31 +08:00
2022-03-30 12:06:05 +08:00
template < >
2022-06-16 23:37:29 +08:00
struct TVariantAlternativeImpl < 0 , TVariant < > > { } ;
template < size_t I , typename . . . Types >
struct TVariantAlternativeImpl < I , const TVariant < Types . . . > > { using Type = TAddConst < typename TVariantAlternativeImpl < I , TVariant < Types . . . > > : : Type > ; } ;
template < size_t I , typename . . . Types >
struct TVariantAlternativeImpl < I , volatile TVariant < Types . . . > > { using Type = TAddVolatile < typename TVariantAlternativeImpl < I , TVariant < Types . . . > > : : Type > ; } ;
template < size_t I , typename . . . Types >
struct TVariantAlternativeImpl < I , const volatile TVariant < Types . . . > > { using Type = TAddCV < typename TVariantAlternativeImpl < I , TVariant < Types . . . > > : : Type > ; } ;
2022-03-30 12:06:05 +08:00
2022-03-15 19:26:31 +08:00
template < typename T , typename . . . Types >
struct TVariantSelectedType ;
template < typename T , typename U , typename . . . Types >
struct TVariantSelectedType < T , U , Types . . . >
{
2022-05-22 22:52:47 +08:00
using TypeAlternativeA = TConditional < CConstructibleFrom < U , T & & > , U , void > ;
2022-03-15 19:26:31 +08:00
using TypeAlternativeB = typename TVariantSelectedType < T , Types . . . > : : Type ;
2022-05-22 22:52:47 +08:00
using Type = TConditional < CSameAs < TRemoveCVRef < TypeAlternativeA > , void > , TypeAlternativeB ,
TConditional < CSameAs < TRemoveCVRef < TypeAlternativeB > , void > , TypeAlternativeA ,
TConditional < CSameAs < TRemoveCVRef < TypeAlternativeB > , TRemoveCVRef < T > > , TypeAlternativeB , TypeAlternativeA > > > ;
2022-03-15 19:26:31 +08:00
// 0 - Type not found
// 1 - Same type found
// 2 - Multiple types found
// 3 - The type found
2022-05-22 22:52:47 +08:00
static constexpr uint8 Flag = CSameAs < TRemoveCVRef < Type > , void > ? 0 :
CSameAs < TRemoveCVRef < TypeAlternativeA > , TRemoveCVRef < TypeAlternativeB > > ? 2 :
CSameAs < TRemoveCVRef < Type > , TRemoveCVRef < T > > ? 1 :
! CSameAs < TRemoveCVRef < TypeAlternativeA > , void > & & ! CSameAs < TypeAlternativeB , void > ? 2 : 3 ;
2022-03-15 19:26:31 +08:00
static constexpr bool Value = Flag & 1 ;
} ;
template < typename T >
struct TVariantSelectedType < T >
{
static constexpr uint8 Flag = 0 ;
using Type = void ;
} ;
NAMESPACE_PRIVATE_END
2022-06-16 23:37:29 +08:00
template < typename T >
concept CTVariant = NAMESPACE_PRIVATE : : TIsTVariant < T > : : Value ;
template < typename VariantType >
inline constexpr size_t TVariantNum = NAMESPACE_PRIVATE : : TVariantNumImpl < VariantType > : : Value ;
template < typename T , typename VariantType >
inline constexpr size_t TVariantIndex = NAMESPACE_PRIVATE : : TVariantIndexImpl < T , VariantType > : : Value ;
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < size_t I , typename VariantType >
using TVariantAlternative = typename NAMESPACE_PRIVATE : : TVariantAlternativeImpl < I , VariantType > : : Type ;
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < typename . . . Types > requires ( true & & . . . & & CDestructible < Types > )
struct TVariant
{
2022-04-22 22:28:44 +08:00
constexpr TVariant ( ) : TypeIndex ( 0xFF ) { } ;
2022-03-15 19:26:31 +08:00
constexpr TVariant ( FInvalid ) : TVariant ( ) { } ;
2022-05-16 22:42:17 +08:00
constexpr TVariant ( const TVariant & InValue ) requires ( true & & . . . & & CCopyConstructible < Types > )
2022-04-22 22:28:44 +08:00
: TypeIndex ( static_cast < uint8 > ( InValue . GetIndex ( ) ) )
2022-03-15 19:26:31 +08:00
{
2022-05-12 23:36:32 +08:00
if ( IsValid ( ) ) CopyConstructImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 22:42:17 +08:00
constexpr TVariant ( TVariant & & InValue ) requires ( true & & . . . & & CMoveConstructible < Types > )
2022-04-22 22:28:44 +08:00
: TypeIndex ( static_cast < uint8 > ( InValue . GetIndex ( ) ) )
2022-03-15 19:26:31 +08:00
{
2022-05-12 23:36:32 +08:00
if ( IsValid ( ) ) MoveConstructImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-06-16 23:37:29 +08:00
template < size_t I , typename . . . ArgTypes > requires ( I < sizeof . . . ( Types ) )
& & CConstructibleFrom < TVariantAlternative < I , TVariant < Types . . . > > , ArgTypes . . . >
2022-03-15 19:26:31 +08:00
constexpr explicit TVariant ( TInPlaceIndex < I > , ArgTypes & & . . . Args )
: TypeIndex ( I )
{
2022-06-16 23:37:29 +08:00
using SelectedType = TVariantAlternative < I , TVariant < Types . . . > > ;
2022-03-15 19:26:31 +08:00
new ( & Value ) SelectedType ( Forward < ArgTypes > ( Args ) . . . ) ;
}
2022-06-16 23:37:29 +08:00
template < typename T , typename . . . ArgTypes > requires CConstructibleFrom < T , ArgTypes . . . >
2022-03-15 19:26:31 +08:00
constexpr explicit TVariant ( TInPlaceType < T > , ArgTypes & & . . . Args )
2022-06-16 23:37:29 +08:00
: TVariant ( InPlaceIndex < TVariantIndex < T , TVariant < Types . . . > > > , Forward < ArgTypes > ( Args ) . . . )
2022-03-15 19:26:31 +08:00
{ }
2022-05-22 22:52:47 +08:00
template < typename T > requires NAMESPACE_PRIVATE : : TVariantSelectedType < TRemoveReference < T > , Types . . . > : : Value
& & ( ! CTInPlaceType < TRemoveCVRef < T > > ) & & ( ! CTInPlaceIndex < TRemoveCVRef < T > > )
& & ( ! CSameAs < TRemoveCVRef < T > , TVariant > )
constexpr TVariant ( T & & InValue ) : TVariant ( InPlaceType < typename NAMESPACE_PRIVATE : : TVariantSelectedType < TRemoveReference < T > , Types . . . > : : Type > , Forward < T > ( InValue ) )
2022-03-15 19:26:31 +08:00
{ }
constexpr ~ TVariant ( )
{
2022-05-16 22:42:17 +08:00
if constexpr ( ! ( true & & . . . & & CTriviallyDestructible < Types > ) ) Reset ( ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 22:42:17 +08:00
constexpr TVariant & operator = ( const TVariant & InValue ) requires ( true & & . . . & & ( CCopyConstructible < Types > & & CCopyAssignable < Types > ) )
2022-03-15 19:26:31 +08:00
{
if ( & InValue = = this ) return * this ;
if ( ! InValue . IsValid ( ) )
{
Reset ( ) ;
return * this ;
}
2022-05-12 23:36:32 +08:00
if ( GetIndex ( ) = = InValue . GetIndex ( ) ) CopyAssignImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-03-15 19:26:31 +08:00
else
{
Reset ( ) ;
2022-05-12 23:36:32 +08:00
CopyConstructImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-04-22 22:28:44 +08:00
TypeIndex = static_cast < uint8 > ( InValue . GetIndex ( ) ) ;
2022-03-15 19:26:31 +08:00
}
return * this ;
}
2022-05-16 22:42:17 +08:00
constexpr TVariant & operator = ( TVariant & & InValue ) requires ( true & & . . . & & ( CMoveConstructible < Types > & & CMoveAssignable < Types > ) )
2022-03-15 19:26:31 +08:00
{
if ( & InValue = = this ) return * this ;
if ( ! InValue . IsValid ( ) )
{
Reset ( ) ;
return * this ;
}
2022-05-12 23:36:32 +08:00
if ( GetIndex ( ) = = InValue . GetIndex ( ) ) MoveAssignImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-03-15 19:26:31 +08:00
else
{
Reset ( ) ;
2022-05-12 23:36:32 +08:00
MoveConstructImpl [ InValue . GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-04-22 22:28:44 +08:00
TypeIndex = static_cast < uint8 > ( InValue . GetIndex ( ) ) ;
2022-03-15 19:26:31 +08:00
}
return * this ;
}
2022-05-22 22:52:47 +08:00
template < typename T > requires NAMESPACE_PRIVATE : : TVariantSelectedType < TRemoveReference < T > , Types . . . > : : Value
2022-03-15 19:26:31 +08:00
constexpr TVariant & operator = ( T & & InValue )
{
2022-05-22 22:52:47 +08:00
using SelectedType = typename NAMESPACE_PRIVATE : : TVariantSelectedType < TRemoveReference < T > , Types . . . > : : Type ;
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
if ( GetIndex ( ) = = TVariantIndex < SelectedType , TVariant < Types . . . > > ) GetValue < SelectedType > ( ) = Forward < T > ( InValue ) ;
2022-03-15 19:26:31 +08:00
else
{
Reset ( ) ;
new ( & Value ) SelectedType ( Forward < T > ( InValue ) ) ;
2022-06-16 23:37:29 +08:00
TypeIndex = TVariantIndex < SelectedType , TVariant < Types . . . > > ;
2022-03-15 19:26:31 +08:00
}
return * this ;
}
2022-06-16 23:37:29 +08:00
template < size_t I , typename . . . ArgTypes > requires ( I < sizeof . . . ( Types ) )
& & CConstructibleFrom < TVariantAlternative < I , TVariant < Types . . . > > , ArgTypes . . . >
constexpr TVariantAlternative < I , TVariant < Types . . . > > & Emplace ( ArgTypes & & . . . Args )
2022-03-15 19:26:31 +08:00
{
Reset ( ) ;
2022-06-16 23:37:29 +08:00
using SelectedType = TVariantAlternative < I , TVariant < Types . . . > > ;
2022-03-15 19:26:31 +08:00
SelectedType * Result = new ( & Value ) SelectedType ( Forward < ArgTypes > ( Args ) . . . ) ;
TypeIndex = I ;
return * Result ;
}
2022-06-16 23:37:29 +08:00
template < typename T , typename . . . ArgTypes > requires CConstructibleFrom < T , ArgTypes . . . >
2022-03-15 19:26:31 +08:00
constexpr T & Emplace ( ArgTypes & & . . . Args )
{
2022-06-16 23:37:29 +08:00
return Emplace < TVariantIndex < T , TVariant < Types . . . > > > ( Forward < ArgTypes > ( Args ) . . . ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-12 23:36:32 +08:00
constexpr const type_info & GetTypeInfo ( ) const { return IsValid ( ) ? * TypeInfos [ GetIndex ( ) ] : typeid ( void ) ; }
2022-05-03 18:22:41 +08:00
2022-04-22 22:28:44 +08:00
constexpr size_t GetIndex ( ) const { return TypeIndex ! = 0xFF ? TypeIndex : INDEX_NONE ; }
constexpr bool IsValid ( ) const { return TypeIndex ! = 0xFF ; }
constexpr explicit operator bool ( ) const { return TypeIndex ! = 0xFF ; }
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < size_t I > constexpr bool HoldsAlternative ( ) const { return IsValid ( ) ? GetIndex ( ) = = I : false ; }
template < typename T > constexpr bool HoldsAlternative ( ) const { return IsValid ( ) ? GetIndex ( ) = = TVariantIndex < T , TVariant < Types . . . > > : false ; }
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) GetValue ( ) & { checkf ( HoldsAlternative < I > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return * reinterpret_cast < TVariantAlternative < I , TVariant < Types . . . > > * > ( & Value ) ; }
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) GetValue ( ) & & { checkf ( HoldsAlternative < I > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return MoveTemp ( * reinterpret_cast < TVariantAlternative < I , TVariant < Types . . . > > * > ( & Value ) ) ; }
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) GetValue ( ) const & { checkf ( HoldsAlternative < I > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return * reinterpret_cast < const TVariantAlternative < I , TVariant < Types . . . > > * > ( & Value ) ; }
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) GetValue ( ) const & & { checkf ( HoldsAlternative < I > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return MoveTemp ( * reinterpret_cast < const TVariantAlternative < I , TVariant < Types . . . > > * > ( & Value ) ) ; }
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < typename T > constexpr decltype ( auto ) GetValue ( ) & { checkf ( HoldsAlternative < T > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return * reinterpret_cast < T * > ( & Value ) ; }
template < typename T > constexpr decltype ( auto ) GetValue ( ) & & { checkf ( HoldsAlternative < T > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return MoveTemp ( * reinterpret_cast < T * > ( & Value ) ) ; }
template < typename T > constexpr decltype ( auto ) GetValue ( ) const & { checkf ( HoldsAlternative < T > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return * reinterpret_cast < const T * > ( & Value ) ; }
template < typename T > constexpr decltype ( auto ) GetValue ( ) const & & { checkf ( HoldsAlternative < T > ( ) , TEXT ( " It is an error to call GetValue() on an wrong TVariant. Please either check HoldsAlternative() or use Get(DefaultValue) instead. " ) ) ; return MoveTemp ( * reinterpret_cast < const T * > ( & Value ) ) ; }
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) Get ( TVariantAlternative < I , TVariant < Types . . . > > & DefaultValue ) & { return HoldsAlternative < I > ( ) ? GetValue < I > ( ) : DefaultValue ; }
template < size_t I > requires ( I < sizeof . . . ( Types ) ) constexpr decltype ( auto ) Get ( const TVariantAlternative < I , TVariant < Types . . . > > & DefaultValue ) const & { return HoldsAlternative < I > ( ) ? GetValue < I > ( ) : DefaultValue ; }
2022-03-15 19:26:31 +08:00
2022-06-16 23:37:29 +08:00
template < typename T > constexpr decltype ( auto ) Get ( T & DefaultValue ) & { return HoldsAlternative < T > ( ) ? GetValue < T > ( ) : DefaultValue ; }
template < typename T > constexpr decltype ( auto ) Get ( const T & DefaultValue ) const & { return HoldsAlternative < T > ( ) ? GetValue < T > ( ) : DefaultValue ; }
2022-03-15 19:26:31 +08:00
2022-05-16 23:09:04 +08:00
template < typename F > requires ( true & & . . . & & CInvocable < F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE decltype ( auto ) Visit ( F & & Func ) &
2022-03-15 19:26:31 +08:00
{
2022-04-26 23:04:50 +08:00
checkf ( IsValid ( ) , TEXT ( " It is an error to call Visit() on an wrong TVariant. Please either check IsValid(). " ) ) ;
2022-05-12 23:36:32 +08:00
2022-05-29 23:18:20 +08:00
using ReturnType = TCommonType < TInvokeResult < F , Types > . . . > ;
2022-05-12 23:36:32 +08:00
using FInvokeImpl = ReturnType ( * ) ( F & & , void * ) ;
static constexpr FInvokeImpl InvokeImpl [ ] = { [ ] ( F & & Func , void * This ) - > ReturnType { return InvokeResult < ReturnType > ( Forward < F > ( Func ) , * reinterpret_cast < Types * > ( This ) ) ; } . . . } ;
return InvokeImpl [ GetIndex ( ) ] ( Forward < F > ( Func ) , & Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 23:09:04 +08:00
template < typename F > requires ( true & & . . . & & CInvocable < F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE decltype ( auto ) Visit ( F & & Func ) & &
2022-03-15 19:26:31 +08:00
{
2022-04-26 23:04:50 +08:00
checkf ( IsValid ( ) , TEXT ( " It is an error to call Visit() on an wrong TVariant. Please either check IsValid(). " ) ) ;
2022-05-12 23:36:32 +08:00
2022-05-29 23:18:20 +08:00
using ReturnType = TCommonType < TInvokeResult < F , Types > . . . > ;
2022-05-12 23:36:32 +08:00
using FInvokeImpl = ReturnType ( * ) ( F & & , void * ) ;
static constexpr FInvokeImpl InvokeImpl [ ] = { [ ] ( F & & Func , void * This ) - > ReturnType { return InvokeResult < ReturnType > ( Forward < F > ( Func ) , MoveTemp ( * reinterpret_cast < Types * > ( This ) ) ) ; } . . . } ;
return InvokeImpl [ GetIndex ( ) ] ( Forward < F > ( Func ) , & Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 23:09:04 +08:00
template < typename F > requires ( true & & . . . & & CInvocable < F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE decltype ( auto ) Visit ( F & & Func ) const &
2022-03-15 19:26:31 +08:00
{
2022-04-26 23:04:50 +08:00
checkf ( IsValid ( ) , TEXT ( " It is an error to call Visit() on an wrong TVariant. Please either check IsValid(). " ) ) ;
2022-05-12 23:36:32 +08:00
2022-05-29 23:18:20 +08:00
using ReturnType = TCommonType < TInvokeResult < F , Types > . . . > ;
2022-05-12 23:36:32 +08:00
using FInvokeImpl = ReturnType ( * ) ( F & & , const void * ) ;
static constexpr FInvokeImpl InvokeImpl [ ] = { [ ] ( F & & Func , const void * This ) - > ReturnType { return InvokeResult < ReturnType > ( Forward < F > ( Func ) , * reinterpret_cast < const Types * > ( This ) ) ; } . . . } ;
return InvokeImpl [ GetIndex ( ) ] ( Forward < F > ( Func ) , & Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 23:09:04 +08:00
template < typename F > requires ( true & & . . . & & CInvocable < F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE decltype ( auto ) Visit ( F & & Func ) const & &
2022-03-15 19:26:31 +08:00
{
2022-04-26 23:04:50 +08:00
checkf ( IsValid ( ) , TEXT ( " It is an error to call Visit() on an wrong TVariant. Please either check IsValid(). " ) ) ;
2022-05-12 23:36:32 +08:00
2022-05-29 23:18:20 +08:00
using ReturnType = TCommonType < TInvokeResult < F , Types > . . . > ;
2022-05-12 23:36:32 +08:00
using FInvokeImpl = ReturnType ( * ) ( F & & , const void * ) ;
static constexpr FInvokeImpl InvokeImpl [ ] = { [ ] ( F & & Func , const void * This ) - > ReturnType { return InvokeResult < ReturnType > ( Forward < F > ( Func ) , MoveTemp ( * reinterpret_cast < const Types * > ( This ) ) ) ; } . . . } ;
return InvokeImpl [ GetIndex ( ) ] ( Forward < F > ( Func ) , & Value ) ;
2022-03-15 19:26:31 +08:00
}
2022-05-16 23:09:04 +08:00
template < typename R , typename F > requires ( true & & . . . & & CInvocableResult < R , F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE R Visit ( F & & Func ) & { return Visit ( Forward < F > ( Func ) ) ; }
2022-03-15 19:26:31 +08:00
2022-05-16 23:09:04 +08:00
template < typename R , typename F > requires ( true & & . . . & & CInvocableResult < R , F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE R Visit ( F & & Func ) & & { return MoveTemp ( * this ) . Visit ( Forward < F > ( Func ) ) ; }
2022-03-15 19:26:31 +08:00
2022-05-16 23:09:04 +08:00
template < typename R , typename F > requires ( true & & . . . & & CInvocableResult < R , F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE R Visit ( F & & Func ) const & { return Visit ( Forward < F > ( Func ) ) ; }
2022-03-15 19:26:31 +08:00
2022-05-16 23:09:04 +08:00
template < typename R , typename F > requires ( true & & . . . & & CInvocableResult < R , F , Types > )
2022-05-12 23:36:32 +08:00
FORCEINLINE R Visit ( F & & Func ) const & & { return MoveTemp ( * this ) . Visit ( Forward < F > ( Func ) ) ; }
2022-03-15 19:26:31 +08:00
constexpr void Reset ( )
{
2022-03-17 09:22:48 +08:00
if ( GetIndex ( ) = = INDEX_NONE ) return ;
2022-03-15 19:26:31 +08:00
2022-05-16 22:42:17 +08:00
if constexpr ( ! ( true & & . . . & & CTriviallyDestructible < Types > ) )
2022-03-23 17:49:30 +08:00
{
2022-05-12 23:36:32 +08:00
DestroyImpl [ GetIndex ( ) ] ( & Value ) ;
2022-03-23 17:49:30 +08:00
}
2022-03-15 19:26:31 +08:00
2022-04-22 22:28:44 +08:00
TypeIndex = static_cast < uint8 > ( INDEX_NONE ) ;
2022-03-15 19:26:31 +08:00
}
2022-04-22 22:28:44 +08:00
constexpr size_t GetTypeHash ( ) const requires ( true & & . . . & & CHashable < Types > )
{
if ( ! IsValid ( ) ) return 114514 ;
2022-05-12 23:36:32 +08:00
using NAMESPACE_REDCRAFT : : GetTypeHash ;
using FHashImpl = size_t ( * ) ( const void * ) ;
constexpr FHashImpl HashImpl [ ] = { [ ] ( const void * This ) - > size_t { return GetTypeHash ( * reinterpret_cast < const Types * > ( This ) ) ; } . . . } ;
return HashCombine ( GetTypeHash ( GetIndex ( ) ) , HashImpl [ GetIndex ( ) ] ( & Value ) ) ;
2022-04-22 22:28:44 +08:00
}
2022-05-20 23:35:36 +08:00
constexpr void Swap ( TVariant & InValue ) requires ( true & & . . . & & ( CMoveConstructible < Types > & & CSwappable < Types > ) )
2022-04-14 22:41:22 +08:00
{
if ( ! IsValid ( ) & & ! InValue . IsValid ( ) ) return ;
if ( IsValid ( ) & & ! InValue . IsValid ( ) )
{
InValue = MoveTemp ( * this ) ;
Reset ( ) ;
return ;
}
if ( InValue . IsValid ( ) & & ! IsValid ( ) )
{
* this = MoveTemp ( InValue ) ;
InValue . Reset ( ) ;
return ;
}
if ( GetIndex ( ) = = InValue . GetIndex ( ) )
{
2022-05-12 23:36:32 +08:00
using NAMESPACE_REDCRAFT : : Swap ;
using FSwapImpl = void ( * ) ( void * , void * ) ;
constexpr FSwapImpl SwapImpl [ ] = { [ ] ( void * A , void * B ) { Swap ( * reinterpret_cast < Types * > ( A ) , * reinterpret_cast < Types * > ( B ) ) ; } . . . } ;
SwapImpl [ GetIndex ( ) ] ( & Value , & InValue . Value ) ;
2022-04-14 22:41:22 +08:00
return ;
}
TVariant Temp = MoveTemp ( * this ) ;
* this = MoveTemp ( InValue ) ;
InValue = MoveTemp ( Temp ) ;
}
2022-03-15 19:26:31 +08:00
private :
2022-05-03 18:22:41 +08:00
2022-05-12 23:36:32 +08:00
static constexpr const type_info * TypeInfos [ ] = { & typeid ( Types ) . . . } ;
using FCopyConstructImpl = void ( * ) ( void * , const void * ) ;
using FMoveConstructImpl = void ( * ) ( void * , void * ) ;
using FCopyAssignImpl = void ( * ) ( void * , const void * ) ;
using FMoveAssignImpl = void ( * ) ( void * , void * ) ;
using FDestroyImpl = void ( * ) ( void * ) ;
static constexpr FCopyConstructImpl CopyConstructImpl [ ] = { [ ] ( void * A , const void * B ) { if constexpr ( requires ( Types * A , const Types * B ) { Memory : : CopyConstruct ( A , B ) ; } ) Memory : : CopyConstruct ( reinterpret_cast < Types * > ( A ) , reinterpret_cast < const Types * > ( B ) ) ; else checkf ( false , TEXT ( " The type '%s' is not copy constructible. " ) , typeid ( Types ) . name ( ) ) ; } . . . } ;
static constexpr FMoveConstructImpl MoveConstructImpl [ ] = { [ ] ( void * A , void * B ) { if constexpr ( requires ( Types * A , Types * B ) { Memory : : MoveConstruct ( A , B ) ; } ) Memory : : MoveConstruct ( reinterpret_cast < Types * > ( A ) , reinterpret_cast < Types * > ( B ) ) ; else checkf ( false , TEXT ( " The type '%s' is not move constructible. " ) , typeid ( Types ) . name ( ) ) ; } . . . } ;
static constexpr FCopyAssignImpl CopyAssignImpl [ ] = { [ ] ( void * A , const void * B ) { if constexpr ( requires ( Types * A , const Types * B ) { Memory : : CopyAssign ( A , B ) ; } ) Memory : : CopyAssign ( reinterpret_cast < Types * > ( A ) , reinterpret_cast < const Types * > ( B ) ) ; else checkf ( false , TEXT ( " The type '%s' is not copy assignable. " ) , typeid ( Types ) . name ( ) ) ; } . . . } ;
static constexpr FMoveAssignImpl MoveAssignImpl [ ] = { [ ] ( void * A , void * B ) { if constexpr ( requires ( Types * A , Types * B ) { Memory : : MoveAssign ( A , B ) ; } ) Memory : : MoveAssign ( reinterpret_cast < Types * > ( A ) , reinterpret_cast < Types * > ( B ) ) ; else checkf ( false , TEXT ( " The type '%s' is not move assignable. " ) , typeid ( Types ) . name ( ) ) ; } . . . } ;
static constexpr FDestroyImpl DestroyImpl [ ] = { [ ] ( void * A ) { if constexpr ( requires ( Types * A ) { Memory : : Destruct ( A ) ; } ) Memory : : Destruct ( reinterpret_cast < Types * > ( A ) ) ; else checkf ( false , TEXT ( " The type '%s' is not destructible. " ) , typeid ( Types ) . name ( ) ) ; } . . . } ;
2022-03-15 19:26:31 +08:00
2022-05-22 22:52:47 +08:00
TAlignedUnion < 1 , Types . . . > Value ;
2022-04-22 22:28:44 +08:00
uint8 TypeIndex ;
2022-03-15 19:26:31 +08:00
2022-03-31 12:07:15 +08:00
friend constexpr bool operator = = ( const TVariant & LHS , const TVariant & RHS ) requires ( true & & . . . & & CEqualityComparable < Types > )
2022-03-15 19:26:31 +08:00
{
if ( LHS . GetIndex ( ) ! = RHS . GetIndex ( ) ) return false ;
if ( LHS . IsValid ( ) = = false ) return true ;
2022-05-12 23:36:32 +08:00
using FCompareImpl = bool ( * ) ( const void * , const void * ) ;
constexpr FCompareImpl CompareImpl [ ] = { [ ] ( const void * LHS , const void * RHS ) - > bool { return * reinterpret_cast < const Types * > ( LHS ) = = * reinterpret_cast < const Types * > ( RHS ) ; } . . . } ;
return CompareImpl [ LHS . GetIndex ( ) ] ( & LHS . Value , & RHS . Value ) ;
2022-05-03 18:22:41 +08:00
}
friend constexpr partial_ordering operator < = > ( const TVariant & LHS , const TVariant & RHS ) requires ( true & & . . . & & CSynthThreeWayComparable < Types > )
{
if ( LHS . GetIndex ( ) ! = RHS . GetIndex ( ) ) return partial_ordering : : unordered ;
if ( LHS . IsValid ( ) = = false ) return partial_ordering : : equivalent ;
2022-05-12 23:36:32 +08:00
using FCompareImpl = partial_ordering ( * ) ( const void * , const void * ) ;
constexpr FCompareImpl CompareImpl [ ] = { [ ] ( const void * LHS , const void * RHS ) - > partial_ordering { return SynthThreeWayCompare ( * reinterpret_cast < const Types * > ( LHS ) , * reinterpret_cast < const Types * > ( RHS ) ) ; } . . . } ;
return CompareImpl [ LHS . GetIndex ( ) ] ( & LHS . Value , & RHS . Value ) ;
2022-03-15 19:26:31 +08:00
}
} ;
2022-05-16 23:09:04 +08:00
template < typename T , typename . . . Types > requires ( ! CSameAs < T , TVariant < Types . . . > > ) & & CEqualityComparable < T >
2022-03-15 19:26:31 +08:00
constexpr bool operator = = ( const TVariant < Types . . . > & LHS , const T & RHS )
{
return LHS . template HoldsAlternative < T > ( ) ? LHS . template GetValue < T > ( ) = = RHS : false ;
}
2022-03-22 11:12:05 +08:00
template < typename . . . Types >
constexpr bool operator = = ( const TVariant < Types . . . > & LHS , FInvalid )
{
return ! LHS . IsValid ( ) ;
}
2022-03-15 19:26:31 +08:00
NAMESPACE_MODULE_END ( Utility )
NAMESPACE_MODULE_END ( Redcraft )
NAMESPACE_REDCRAFT_END