#pragma once #include "CoreTypes.h" #include "Templates/Meta.h" #include "Templates/Invoke.h" #include "Templates/Utility.h" #include "Templates/TypeHash.h" #include "TypeTraits/TypeTraits.h" #include "Miscellaneous/Compare.h" #include "Templates/ReferenceWrapper.h" #include NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) #define RS_TUPLE_ELEMENT_STATIC_ALIAS 1 template class TTuple; NAMESPACE_PRIVATE_BEGIN template struct TIsTTuple : FFalse { }; template struct TIsTTuple> : FTrue { }; struct FForwardingConstructor { explicit FForwardingConstructor() = default; }; struct FOtherTupleConstructor { explicit FOtherTupleConstructor() = default; }; inline constexpr FForwardingConstructor ForwardingConstructor{ }; inline constexpr FOtherTupleConstructor OtherTupleConstructor{ }; template struct TTupleArityImpl; template struct TTupleArityImpl> : TConstant>> { }; template struct TTupleIndexImpl; template struct TTupleIndexImpl> : TConstant>> { }; template struct TTupleElementImpl; template struct TTupleElementImpl> { using FType = Meta::TType>; }; template struct TTupleConvertCopy : FTrue { }; template struct TTupleConvertCopy : TBoolConstant&, T> || CConstructibleFrom&> || CSameAs)> { }; template struct TTupleConvertMove : FTrue { }; template struct TTupleConvertMove : TBoolConstant&&, T> || CConstructibleFrom&&> || CSameAs)> { }; template struct TTupleBasicElement { private: using FValueType = T; FValueType Value; public: template requires (CConstructibleFrom) FORCEINLINE constexpr TTupleBasicElement(Type&& Arg) : Value(Forward(Arg)) { } FORCEINLINE constexpr TTupleBasicElement() = default; FORCEINLINE constexpr TTupleBasicElement(const TTupleBasicElement&) = default; FORCEINLINE constexpr TTupleBasicElement(TTupleBasicElement&&) = default; FORCEINLINE constexpr TTupleBasicElement& operator=(const TTupleBasicElement&) = default; FORCEINLINE constexpr TTupleBasicElement& operator=(TTupleBasicElement&&) = default; FORCEINLINE constexpr ~TTupleBasicElement() = default; FORCEINLINE constexpr T& GetValue() & { return static_cast< T& >(Value); } FORCEINLINE constexpr const T& GetValue() const & { return static_cast(Value); } FORCEINLINE constexpr volatile T& GetValue() volatile& { return static_cast< volatile T& >(Value); } FORCEINLINE constexpr const volatile T& GetValue() const volatile& { return static_cast(Value); } FORCEINLINE constexpr T&& GetValue() && { return static_cast< T&&>(Value); } FORCEINLINE constexpr const T&& GetValue() const && { return static_cast(Value); } FORCEINLINE constexpr volatile T&& GetValue() volatile&& { return static_cast< volatile T&&>(Value); } FORCEINLINE constexpr const volatile T&& GetValue() const volatile&& { return static_cast(Value); } }; #if RS_TUPLE_ELEMENT_STATIC_ALIAS #define DEFINE_TTupleBasicElement(Index, Name) \ template \ struct TTupleBasicElement \ { \ using F##Name##Type = T; \ F##Name##Type Name; \ \ template requires (CConstructibleFrom) \ FORCEINLINE constexpr TTupleBasicElement(Type&& Arg) \ : Name(Forward(Arg)) \ { } \ \ FORCEINLINE constexpr TTupleBasicElement() = default; \ FORCEINLINE constexpr TTupleBasicElement(const TTupleBasicElement&) = default; \ FORCEINLINE constexpr TTupleBasicElement(TTupleBasicElement&&) = default; \ FORCEINLINE constexpr TTupleBasicElement& operator=(const TTupleBasicElement&) = default; \ FORCEINLINE constexpr TTupleBasicElement& operator=(TTupleBasicElement&&) = default; \ FORCEINLINE constexpr ~TTupleBasicElement() = default; \ \ FORCEINLINE constexpr T& GetValue() & { return static_cast< T& >(Name); } \ FORCEINLINE constexpr const T& GetValue() const & { return static_cast(Name); } \ FORCEINLINE constexpr volatile T& GetValue() volatile& { return static_cast< volatile T& >(Name); } \ FORCEINLINE constexpr const volatile T& GetValue() const volatile& { return static_cast(Name); } \ FORCEINLINE constexpr T&& GetValue() && { return static_cast< T&&>(Name); } \ FORCEINLINE constexpr const T&& GetValue() const && { return static_cast(Name); } \ FORCEINLINE constexpr volatile T&& GetValue() volatile&& { return static_cast< volatile T&&>(Name); } \ FORCEINLINE constexpr const volatile T&& GetValue() const volatile&& { return static_cast(Name); } \ } DEFINE_TTupleBasicElement(0x0, First); DEFINE_TTupleBasicElement(0x1, Second); DEFINE_TTupleBasicElement(0x2, Third); DEFINE_TTupleBasicElement(0x3, Fourth); DEFINE_TTupleBasicElement(0x4, Fifth); DEFINE_TTupleBasicElement(0x5, Sixth); DEFINE_TTupleBasicElement(0x6, Seventh); DEFINE_TTupleBasicElement(0x7, Eighth); DEFINE_TTupleBasicElement(0x8, Ninth); DEFINE_TTupleBasicElement(0x9, Tenth); DEFINE_TTupleBasicElement(0xA, Eleventh); DEFINE_TTupleBasicElement(0xB, Twelfth); DEFINE_TTupleBasicElement(0xC, Thirteenth); DEFINE_TTupleBasicElement(0xD, Fourteenth); DEFINE_TTupleBasicElement(0xE, Fifteenth); DEFINE_TTupleBasicElement(0xF, Sixteenth); #undef DEFINE_TTupleBasicElement #endif template FORCEINLINE constexpr TTuple...> MakeTupleImpl(Ts&&... Args) { return TTuple...>(Forward(Args)...); } template class TTupleImpl; template class TTupleImpl, Ts...> : public TTupleBasicElement... { protected: FORCEINLINE constexpr TTupleImpl() = default; FORCEINLINE constexpr TTupleImpl(const TTupleImpl&) = default; FORCEINLINE constexpr TTupleImpl(TTupleImpl&&) = default; FORCEINLINE constexpr TTupleImpl& operator=(const TTupleImpl&) = default; FORCEINLINE constexpr TTupleImpl& operator=(TTupleImpl&&) = default; FORCEINLINE constexpr ~TTupleImpl() = default; template FORCEINLINE constexpr explicit TTupleImpl(FForwardingConstructor, Us&&... Args) : TTupleBasicElement(Forward(Args))... { } template FORCEINLINE constexpr explicit TTupleImpl(FOtherTupleConstructor, TupleType&& InValue) : TTupleBasicElement(Forward(InValue).template GetValue())... { } }; template class TTupleHelper; template class TTupleHelper> { public: template FORCEINLINE static constexpr void Assign(LHSTupleType& LHS, RHSTupleType&& RHS) { static_assert(sizeof...(Indices) == TTupleArityImpl>::Value && TTupleArityImpl>::Value == TTupleArityImpl>::Value, "Cannot assign tuple from different size"); ((LHS.template GetValue() = Forward(RHS).template GetValue()), ...); } template FORCEINLINE static constexpr auto Apply(F&& Func, TTupleType&& Arg) { return Invoke(Forward(Func), Forward(Arg).template GetValue()...); } template FORCEINLINE static constexpr auto Transform(F&& Func, TTupleType&& Arg) { return MakeTupleImpl(Invoke(Forward(Func), Forward(Arg).template GetValue())...); } template FORCEINLINE static constexpr T Construct(TTupleType&& Arg) { return T(Forward(Arg).template GetValue()...); } }; template struct TTupleThreeWay; template struct TTupleThreeWay> { template FORCEINLINE static constexpr R Do(const LHSTupleType& LHS, const RHSTupleType& RHS) { auto Result = SynthThreeWayCompare(LHS.template GetValue(), RHS.template GetValue()); if (Result != 0) return Result; return TTupleThreeWay>::Do(LHS, RHS); } }; template struct TTupleThreeWay> { template FORCEINLINE static constexpr R Do(const LHSTupleType& LHS, const RHSTupleType& RHS) { return R::equivalent; } }; template struct TTTupleWeaklyEqualityComparable; template struct TTTupleWeaklyEqualityComparable, TTypeSequence> : TBoolConstant && TTTupleWeaklyEqualityComparable, TTypeSequence>::Value> { }; template <> struct TTTupleWeaklyEqualityComparable, TTypeSequence<>> : FTrue { }; template concept CTTupleWeaklyEqualityComparable = TTTupleWeaklyEqualityComparable::Value; template struct TTTupleSynthThreeWayComparable; template struct TTTupleSynthThreeWayComparable, TTypeSequence> : TBoolConstant && TTTupleSynthThreeWayComparable, TTypeSequence>::Value> { }; template <> struct TTTupleSynthThreeWayComparable, TTypeSequence<>> : FTrue { }; template concept CTTupleSynthThreeWayComparable = TTTupleSynthThreeWayComparable::Value; template struct TTupleVisitElementByIndex; template struct TTupleVisitElementByIndex> { template FORCEINLINE static constexpr decltype(auto) Do(F&& Func, TTupleType&& Arg, size_t Index) { if (Index == I) return InvokeResult(Forward(Func), Forward(Arg).template GetValue()); return TTupleVisitElementByIndex>::Do(Forward(Func), Forward(Arg), Index); } }; template struct TTupleVisitElementByIndex> { template FORCEINLINE static constexpr decltype(auto) Do(F&& Func, TTupleType&& Arg, size_t) { checkf(false, "Read access violation. Please check Index."); return InvokeResult(Forward(Func), Forward(Arg).template GetValue<0>()); } }; NAMESPACE_PRIVATE_END template concept CTTuple = NAMESPACE_PRIVATE::TIsTTuple>::Value; template inline constexpr size_t TTupleArity = NAMESPACE_PRIVATE::TTupleArityImpl>::Value; template inline constexpr size_t TTupleIndex = NAMESPACE_PRIVATE::TTupleIndexImpl>::Value; template using TTupleElement = TCopyCV>::FType>; template class TTuple final : public NAMESPACE_PRIVATE::TTupleImpl, Ts...> { private: using FSuper = NAMESPACE_PRIVATE::TTupleImpl, Ts...>; using FHelper = NAMESPACE_PRIVATE::TTupleHelper>; public: /** Default constructor. Value-initializes all elements, if any. */ FORCEINLINE constexpr TTuple() = default; /** Converting constructor. Initializes each element of the tuple with the corresponding value in Forward(Args). */ template requires (sizeof...(Ts) >= 1 && sizeof...(Us) == sizeof...(Ts)) && (true && ... && CConstructibleFrom) FORCEINLINE constexpr explicit (!(true && ... && CConvertibleTo)) TTuple(Us&&... Args) : FSuper(NAMESPACE_PRIVATE::ForwardingConstructor, Forward(Args)...) { } /** Converting copy constructor. Initializes each element of the tuple with the corresponding element of other. */ template requires (sizeof...(Us) == sizeof...(Ts) && (true && ... && CConstructibleFrom) && NAMESPACE_PRIVATE::TTupleConvertCopy::Value) FORCEINLINE constexpr explicit (!(true && ... && CConvertibleTo)) TTuple(const TTuple& InValue) : FSuper(NAMESPACE_PRIVATE::OtherTupleConstructor, InValue) { } /** Converting move constructor. Initializes each element of the tuple with the corresponding element of other. */ template requires (sizeof...(Us) == sizeof...(Ts) && (true && ... && CConstructibleFrom) && NAMESPACE_PRIVATE::TTupleConvertMove::Value) FORCEINLINE constexpr explicit (!(true && ... && CConvertibleTo)) TTuple(TTuple&& InValue) : FSuper(NAMESPACE_PRIVATE::OtherTupleConstructor, MoveTemp(InValue)) { } /** Copies/moves content of other into a new instance. */ FORCEINLINE constexpr TTuple(const TTuple&) = default; FORCEINLINE constexpr TTuple(TTuple&&) = default; /** Converting copy assignment operator. Assigns each element of other to the corresponding element of this. */ template requires (sizeof...(Us) == sizeof...(Ts) && (true && ... && CAssignableFrom)) FORCEINLINE constexpr TTuple& operator=(const TTuple& InValue) { FHelper::Assign(*this, InValue); return *this; } /** Converting move assignment operator. Assigns each element of other to the corresponding element of this. */ template requires (sizeof...(Us) == sizeof...(Ts) && (true && ... && CAssignableFrom)) FORCEINLINE constexpr TTuple& operator=(TTuple&& InValue) { FHelper::Assign(*this, MoveTemp(InValue)); return *this; } /** Copy/move assignment operator. */ FORCEINLINE constexpr TTuple& operator=(const TTuple&) = default; FORCEINLINE constexpr TTuple& operator=(TTuple&&) = default; /** Compares every element of the tuple lhs with the corresponding element of the tuple rhs. */ template requires (sizeof...(Ts) == sizeof...(Us) && NAMESPACE_PRIVATE::CTTupleWeaklyEqualityComparable, TTypeSequence>) NODISCARD friend FORCEINLINE constexpr bool operator==(const TTuple& LHS, const TTuple& RHS) { if constexpr (sizeof...(Ts) != sizeof...(Us)) return false; return [&LHS, &RHS](TIndexSequence) -> bool { return (true && ... && (LHS.template GetValue() == RHS.template GetValue())); } (TMakeIndexSequence()); } /** Compares lhs and rhs lexicographically by synthesized three-way comparison. */ template requires (sizeof...(Ts) == sizeof...(Us) && NAMESPACE_PRIVATE::CTTupleSynthThreeWayComparable, TTypeSequence>) NODISCARD friend FORCEINLINE constexpr TCommonComparisonCategory...> operator<=>(const TTuple& LHS, const TTuple& RHS) { using FResult = TCommonComparisonCategory...>; return NAMESPACE_PRIVATE::TTupleThreeWay>::Do(LHS, RHS); } /** Extracts the Ith element from the tuple. I must be an integer value in [0, sizeof...(Ts)). */ template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() & { return static_cast< NAMESPACE_PRIVATE::TTupleBasicElement, I>& >(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const & { return static_cast, I>& >(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() volatile& { return static_cast< volatile NAMESPACE_PRIVATE::TTupleBasicElement, I>& >(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const volatile& { return static_cast, I>& >(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() && { return static_cast< NAMESPACE_PRIVATE::TTupleBasicElement, I>&&>(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const && { return static_cast, I>&&>(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() volatile&& { return static_cast< volatile NAMESPACE_PRIVATE::TTupleBasicElement, I>&&>(*this).GetValue(); } template requires (I < sizeof...(Ts)) NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const volatile&& { return static_cast, I>&&>(*this).GetValue(); } /** Extracts the element of the tuple whose type is T. Fails to compile unless the tuple has exactly one element of that type. */ template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() & { return static_cast< TTuple& >(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const & { return static_cast(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() volatile& { return static_cast< volatile TTuple& >(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const volatile& { return static_cast(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() && { return static_cast< TTuple&&>(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const && { return static_cast(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() volatile&& { return static_cast< volatile TTuple&&>(*this).GetValue>(); } template NODISCARD FORCEINLINE constexpr decltype(auto) GetValue() const volatile&& { return static_cast(*this).GetValue>(); } /** Invoke the callable object 'Func' with a tuple of arguments. */ template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) & { return FHelper::Apply(Forward(Func), static_cast< TTuple& >(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) const & { return FHelper::Apply(Forward(Func), static_cast(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) volatile& { return FHelper::Apply(Forward(Func), static_cast< volatile TTuple& >(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) const volatile& { return FHelper::Apply(Forward(Func), static_cast(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) && { return FHelper::Apply(Forward(Func), static_cast< TTuple&&>(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) const && { return FHelper::Apply(Forward(Func), static_cast(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) volatile&& { return FHelper::Apply(Forward(Func), static_cast< volatile TTuple&&>(*this)); } template requires (CInvocable) FORCEINLINE constexpr decltype(auto) Apply(F&& Func) const volatile&& { return FHelper::Apply(Forward(Func), static_cast(*this)); } /** Visits each element in a tuple in parallel and applies it as arguments to the function. */ template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) & { VisitTuple(Forward(Func), static_cast< TTuple& >(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) const & { VisitTuple(Forward(Func), static_cast(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) volatile& { VisitTuple(Forward(Func), static_cast< volatile TTuple& >(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) const volatile& { VisitTuple(Forward(Func), static_cast(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) && { VisitTuple(Forward(Func), static_cast< TTuple&&>(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) const && { VisitTuple(Forward(Func), static_cast(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) volatile&& { VisitTuple(Forward(Func), static_cast< volatile TTuple&&>(*this)); } template requires (true && ... && CInvocable) FORCEINLINE constexpr void Visit(F&& Func) const volatile&& { VisitTuple(Forward(Func), static_cast(*this)); } /** Visits specified element in a tuple and applies it as arguments to the function. */ template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) & { return static_cast< TTuple& >(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) const & { return static_cast(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) volatile& { return static_cast< volatile TTuple& >(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) const volatile& { return static_cast(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) && { return static_cast< TTuple&&>(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) const && { return static_cast(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) volatile&& { return static_cast< volatile TTuple&&>(*this).Visit...>>(Forward(Func), Index); } template requires ((sizeof...(Ts) >= 1 && CCommonReference...>) && ... && (CInvocable)) FORCEINLINE constexpr decltype(auto) Visit(F&& Func, size_t Index) const volatile&& { return static_cast(*this).Visit...>>(Forward(Func), Index); } /** Visits specified element in a tuple and applies it as arguments to the function. */ template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) & { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast< TTuple& >(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) const & { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) volatile& { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast< volatile TTuple& >(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) const volatile& { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) && { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast< TTuple&&>(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) const && { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) volatile&& { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast< volatile TTuple&&>(*this), Index); } template requires ((sizeof...(Ts) >= 1) && ... && CInvocableResult) FORCEINLINE constexpr Ret Visit(F&& Func, size_t Index) const volatile&& { return NAMESPACE_PRIVATE::TTupleVisitElementByIndex>::Do(Forward(Func), static_cast(*this), Index); } /** Transform a tuple into another tuple using the given function. */ template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) & { return FHelper::Transform(Forward(Func), static_cast< TTuple& >(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) const & { return FHelper::Transform(Forward(Func), static_cast(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) volatile& { return FHelper::Transform(Forward(Func), static_cast< volatile TTuple& >(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) const volatile& { return FHelper::Transform(Forward(Func), static_cast(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) && { return FHelper::Transform(Forward(Func), static_cast< TTuple&&>(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) const && { return FHelper::Transform(Forward(Func), static_cast(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) volatile&& { return FHelper::Transform(Forward(Func), static_cast< volatile TTuple&&>(*this)); } template requires (true && ... && (CInvocable && !CSameAs>)) NODISCARD FORCEINLINE constexpr decltype(auto) Transform(F&& Func) const volatile&& { return FHelper::Transform(Forward(Func), static_cast(*this)); } /** Constructs an object of type T with a tuple as an argument. */ template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() & { return FHelper::template Construct(static_cast< TTuple& >(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() const & { return FHelper::template Construct(static_cast(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() volatile& { return FHelper::template Construct(static_cast< volatile TTuple& >(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() const volatile& { return FHelper::template Construct(static_cast(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() && { return FHelper::template Construct(static_cast< TTuple&&>(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() const && { return FHelper::template Construct(static_cast(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() volatile&& { return FHelper::template Construct(static_cast< volatile TTuple&&>(*this)); } template requires (CConstructibleFrom) NODISCARD FORCEINLINE constexpr T Construct() const volatile&& { return FHelper::template Construct(static_cast(*this)); } /** @return The number of elements in the tuple. */ NODISCARD static FORCEINLINE constexpr size_t Num() { return sizeof...(Ts); } /** @return true if the tuple is empty, false otherwise. */ NODISCARD static FORCEINLINE constexpr bool IsEmpty() { return Num() == 0; } /** Overloads the GetTypeHash algorithm for TTuple. */ NODISCARD friend FORCEINLINE constexpr size_t GetTypeHash(const TTuple& A) requires (true && ... && CHashable) { return [&A](TIndexSequence) -> size_t { return HashCombine(GetTypeHash(A.template GetValue())...); } (TMakeIndexSequence()); } /** Overloads the Swap algorithm for TTuple. */ friend FORCEINLINE constexpr void Swap(TTuple& A, TTuple& B) requires (true && ... && (CMoveConstructible && CSwappable)) { [&A, &B](TIndexSequence) { ((Swap(A.template GetValue(), B.template GetValue())), ...); } (TMakeIndexSequence()); } }; template TTuple(Ts...) -> TTuple; template using TPair = TTuple; /** Creates a tuple object of the type defined by the argument types. */ template FORCEINLINE constexpr TTuple...> MakeTuple(Ts&&... Args) { return TTuple...>(Forward(Args)...); } /** * Creates a tuple of lvalue references or unpacks a tuple into individual objects. * * TTuple> SomeFunction(); * * FString Ret1; * float Ret2; * TArray Ret3; * * Tie(Ret1, Ret2, Ret3) = SomeFunction(); */ template FORCEINLINE constexpr TTuple Tie(Ts&... Args) { return TTuple(Args...); } /** Creates a tuple of forwarding references. */ template FORCEINLINE constexpr TTuple ForwardAsTuple(Ts&&... Args) { return TTuple(Forward(Args)...); } NAMESPACE_PRIVATE_BEGIN struct FTupleEndFlag { }; template struct TTupleCatResultImpl; template struct TTupleCatResultImpl, TTupleTypes...> { using FType = typename TTupleCatResultImpl::FType; }; template struct TTupleCatResultImpl { using FType = TTuple; }; template struct TTupleCatMake; template struct TTupleCatMake, TIndexSequence> { template struct FForwardType { using FType = TConditional, TRemoveReference&&, U>; }; template FORCEINLINE static constexpr TTuple Do(TTupleType&& InValue) { return TTuple ( static_cast(InValue).template GetValue())>::FType> ( Forward(InValue).template GetValue() )... ); } }; template struct TTupleCatForward; template struct TTupleCatForward, TIndexSequence> { template FORCEINLINE static constexpr decltype(auto) Do(ForwardType&& ForwardTuple, TTupleType&& InValue) { return ForwardAsTuple(Forward(ForwardTuple).template GetValue()..., Forward(InValue).template GetValue()...); } }; template struct TTupleCatImpl { template FORCEINLINE static constexpr decltype(auto) Do(ForwardType&& ForwardTuple, TTupleType&& InValue, OtherTTupleTypes&&... OtherValue) { return Do(TTupleCatForward< TMakeIndexSequence>>, TMakeIndexSequence>>> ::Do(Forward(ForwardTuple), Forward(InValue)), Forward(OtherValue)...); } template FORCEINLINE static constexpr decltype(auto) Do(ForwardType&& ForwardTuple) { return TTupleCatMake>>::Do(Forward(ForwardTuple)); } }; template struct TTupleVisitImpl; template struct TTupleVisitImpl> { template FORCEINLINE static constexpr void Do(F&& Func, TupleTypes&&... Tuples) { Invoke(Forward(Func), Forward(Tuples).template GetValue()...); TTupleVisitImpl>::Do(Forward(Func), Forward(Tuples)...); } }; template <> struct TTupleVisitImpl> { template FORCEINLINE static constexpr void Do(TupleTypes&&...) { } }; NAMESPACE_PRIVATE_END template requires (true && ... && CTTuple>) using TTupleCatResult = typename NAMESPACE_PRIVATE::TTupleCatResultImpl..., NAMESPACE_PRIVATE::FTupleEndFlag>::FType; /** Creates a tuple by concatenating any number of tuples. */ template requires (true && ... && CTTuple>) FORCEINLINE constexpr decltype(auto) TupleCat(TTupleTypes&&... Args) { using FResult = TTupleCatResult; if constexpr (sizeof...(Args) == 0) return FResult(); else return NAMESPACE_PRIVATE::TTupleCatImpl::Do(Forward(Args)...); } /** * Visits each element in the specified tuples in parallel and applies them as arguments to the function. * * @param Func - The function to apply. * @param Tuples - The tuples whose elements are to be applied to the function. * * void SomeFunction(const TTuple& TupleA, const TTuple& TupleB) * { * // Equivalent to: * // Func(TupleA.Get<0>(), TupleB.Get<0>()); * // Func(TupleA.Get<1>(), TupleB.Get<1>()); * // Func(TupleA.Get<2>(), TupleB.Get<2>()); * VisitTuple(Func, TupleA, TupleB); * } */ template requires (CTTuple> && (true && ... && CTTuple>)) FORCEINLINE constexpr void VisitTuple(F&& Func, FirstTupleType&& FirstTuple, TupleTypes&&... Tuples) { NAMESPACE_PRIVATE::TTupleVisitImpl>>> ::Do(Forward(Func), Forward(FirstTuple), Forward(Tuples)...); } template requires (requires { typename TTuple...>; }) struct TBasicCommonType, TTuple> { using FType = TTuple...>; }; template typename TQualifiers, template typename UQualifiers> requires (requires { typename TTuple, UQualifiers>...>; }) struct TBasicCommonReference, TTuple, TQualifiers, UQualifiers> { using FType = TTuple, UQualifiers>...>; }; NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END // ReSharper disable CppInconsistentNaming NAMESPACE_STD_BEGIN // Support structure binding, should not be directly used. template struct tuple_size> : integral_constant>> { }; template struct tuple_element> { using type = NAMESPACE_REDCRAFT::TTupleElement>; }; NAMESPACE_STD_END NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) // Support structure binding, should not be directly used. template FORCEINLINE constexpr decltype(auto) get( TTuple& InValue) { return static_cast< TTuple& >(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get(const TTuple& InValue) { return static_cast& >(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get( volatile TTuple& InValue) { return static_cast< volatile TTuple& >(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get(const volatile TTuple& InValue) { return static_cast& >(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get( TTuple&& InValue) { return static_cast< TTuple&&>(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get(const TTuple&& InValue) { return static_cast&&>(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get( volatile TTuple&& InValue) { return static_cast< volatile TTuple&&>(InValue).template GetValue(); } template FORCEINLINE constexpr decltype(auto) get(const volatile TTuple&& InValue) { return static_cast&&>(InValue).template GetValue(); } NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END // ReSharper restore CppInconsistentNaming