diff --git a/Redcraft.Utility/Source/Private/Testing/TemplatesTesting.cpp b/Redcraft.Utility/Source/Private/Testing/TemplatesTesting.cpp index 98b256c..018f684 100644 --- a/Redcraft.Utility/Source/Private/Testing/TemplatesTesting.cpp +++ b/Redcraft.Utility/Source/Private/Testing/TemplatesTesting.cpp @@ -1155,6 +1155,87 @@ void TestFunction() always_check(FunctionDebug.Output[10] == 720); always_check(FunctionDebug.Output[11] == 5040); } + + { + TFunction Identity = TIdentity<>(); + TFunction NotIdentity = NotFn(Identity); + + always_check(Identity(true)); + always_check(NotIdentity(false)); + } + + { + always_check(TPlus ()(4, 2) == 6); + always_check(TMinus ()(4, 2) == 2); + always_check(TMultiplies()(4, 2) == 8); + always_check(TDivides ()(4, 2) == 2); + always_check(TModulus ()(4, 2) == 0); + always_check(TNegate ()(4 ) == -4); + + always_check(TEqualTo ()(4, 2) == false); + always_check(TNotEqualTo ()(4, 2) == true); + always_check(TGreater ()(4, 2) == true); + always_check(TLess ()(4, 2) == false); + always_check(TGreaterEqual()(4, 2) == true); + always_check(TLessEqual ()(4, 2) == false); + + always_check(TLogicalAnd()(4, 2) == true); + always_check(TLogicalOr ()(4, 2) == true); + always_check(TLogicalNot()(4 ) == false); + + always_check(TBitAnd()(4, 2) == 0); + always_check(TBitOr ()(4, 2) == 6); + always_check(TBitXor()(4, 2) == 6); + always_check(TBitNot()(4 ) == -5); + } + + { + TFunction TempA = TPlus <>(); + TFunction TempB = TMinus <>(); + TFunction TempC = TMultiplies<>(); + TFunction TempD = TDivides <>(); + TFunction TempE = TModulus <>(); + TFunction TempF = TNegate <>(); + + always_check(TempA(4, 2) == 6); + always_check(TempB(4, 2) == 2); + always_check(TempC(4, 2) == 8); + always_check(TempD(4, 2) == 2); + always_check(TempE(4, 2) == 0); + always_check(TempF(4 ) == -4); + + TFunction TempG = TEqualTo <>(); + TFunction TempH = TNotEqualTo <>(); + TFunction TempI = TGreater <>(); + TFunction TempJ = TLess <>(); + TFunction TempK = TGreaterEqual<>(); + TFunction TempL = TLessEqual <>(); + + always_check(TempG(4, 2) == false); + always_check(TempH(4, 2) == true); + always_check(TempI(4, 2) == true); + always_check(TempJ(4, 2) == false); + always_check(TempK(4, 2) == true); + always_check(TempL(4, 2) == false); + + TFunction TempM = TLogicalAnd<>(); + TFunction TempN = TLogicalOr <>(); + TFunction TempO = TLogicalNot<>(); + + always_check(TempM(4, 2) == true); + always_check(TempN(4, 2) == true); + always_check(TempO(4 ) == false); + + TFunction TempP = TBitAnd<>(); + TFunction TempQ = TBitOr <>(); + TFunction TempR = TBitXor<>(); + TFunction TempS = TBitNot<>(); + + always_check(TempP(4, 2) == 0); + always_check(TempQ(4, 2) == 6); + always_check(TempR(4, 2) == 6); + always_check(TempS(4 ) == -5); + } } NAMESPACE_UNNAMED_BEGIN diff --git a/Redcraft.Utility/Source/Public/Templates/Function.h b/Redcraft.Utility/Source/Public/Templates/Function.h index 1fcfeb3..40482dd 100644 --- a/Redcraft.Utility/Source/Public/Templates/Function.h +++ b/Redcraft.Utility/Source/Public/Templates/Function.h @@ -2,12 +2,17 @@ #include "CoreTypes.h" #include "Memory/Memory.h" +#include "Concepts/Same.h" #include "Templates/Any.h" #include "Templates/Tuple.h" #include "Templates/Invoke.h" +#include "Templates/Utility.h" #include "Templates/Container.h" +#include "Concepts/Comparable.h" +#include "Concepts/Convertible.h" #include "TypeTraits/TypeTraits.h" #include "Miscellaneous/TypeInfo.h" +#include "Concepts/BooleanTestable.h" #include "Miscellaneous/AssertionMacros.h" NAMESPACE_REDCRAFT_BEGIN @@ -454,6 +459,203 @@ static_assert(sizeof(TFunctionRef) == 16, "The byte size of TFunction static_assert(sizeof(TFunction) == 64, "The byte size of TFunction is unexpected"); static_assert(sizeof(TUniqueFunction) == 64, "The byte size of TUniqueFunction is unexpected"); +template +struct TIdentity +{ + using Type = T; + + constexpr T&& operator()(T&& InValue) const + { + return Forward(InValue); + } +}; + +template <> +struct TIdentity +{ + using Type = void; + + template + constexpr T&& operator()(T&& InValue) const + { + return Forward(InValue); + } +}; + +NAMESPACE_PRIVATE_BEGIN + +template +struct NotFunctionType +{ + F Func; + + NotFunctionType(const NotFunctionType&) = default; + NotFunctionType(NotFunctionType&&) = default; + + template + constexpr NotFunctionType(InF&& InFunc) : Func(Forward(InFunc)) { } + + template requires TIsInvocable::Value + constexpr auto operator()(Types&&... Args) & + -> decltype(!Invoke(Func, Forward(Args)...)) + { + return !Invoke(Func, Forward(Args)...); + } + + template requires TIsInvocable::Value + constexpr auto operator()(Types&&... Args) && + -> decltype(!Invoke(MoveTemp(Func), Forward(Args)...)) + { + return !Invoke(MoveTemp(Func), Forward(Args)...); + } + + template requires TIsInvocable::Value + constexpr auto operator()(Types&&... Args) const& + -> decltype(!Invoke(Func, Forward(Args)...)) + { + return !Invoke(Func, Forward(Args)...); + } + + template requires TIsInvocable::Value + constexpr auto operator()(Types&&... Args) const&& + -> decltype(!Invoke(MoveTemp(Func), Forward(Args)...)) + { + return !Invoke(MoveTemp(Func), Forward(Args)...); + } +}; + +NAMESPACE_PRIVATE_END + +template +constexpr NAMESPACE_PRIVATE::NotFunctionType::Type> NotFn(F&& Func) +{ + return NAMESPACE_PRIVATE::NotFunctionType::Type>(Forward(Func)); +} + +#define FUNCTOR_UNARY_OPERATOR_IMPL(Name, Operator, ReturnType, ConceptT, ConceptU) \ + template requires (CSameAs || ConceptT) \ + struct Name \ + { \ + constexpr ReturnType operator()(const T& InValue) const { \ + return Operator InValue; \ + } \ + }; \ + \ + template <> \ + struct Name \ + { \ + template requires ConceptU \ + constexpr auto operator()(U&& InValue) const \ + -> decltype(Operator Forward(InValue)) \ + { \ + return Operator Forward(InValue); \ + } \ + } + +#define FUNCTOR_BINARY_OPERATOR_IMPL(Name, Operator, ReturnType, ConceptT, ConceptTU) \ + template requires (CSameAs || ConceptT) \ + struct Name \ + { \ + constexpr ReturnType operator()(const T& LHS, const T& RHS) const \ + { \ + return LHS Operator RHS; \ + } \ + }; \ + \ + template <> \ + struct Name \ + { \ + template requires ConceptTU \ + constexpr auto operator()(T&& LHS, U&& RHS) const \ + -> decltype(Forward(LHS) Operator Forward(RHS)) \ + { \ + return Forward(LHS) Operator Forward(RHS); \ + } \ + } + +#define FUNCTOR_UNARY_OPERATOR_A_IMPL(Name, Operator) \ + FUNCTOR_UNARY_OPERATOR_IMPL \ + ( \ + Name, Operator, T, \ + (requires(const T& InValue) { { Operator InValue } -> CConvertibleTo; }), \ + (requires(U&& InValue) { Operator Forward(InValue); }) \ + ) + +#define FUNCTOR_BINARY_OPERATOR_A_IMPL(Name, Operator) \ + FUNCTOR_BINARY_OPERATOR_IMPL \ + ( \ + Name, Operator, T, \ + (requires(const T& LHS, const T& RHS) { { LHS Operator RHS } -> CConvertibleTo; }), \ + (requires(T&& LHS, U&& RHS) { Forward(LHS) Operator Forward(RHS); }) \ + ) + +#define FUNCTOR_UNARY_OPERATOR_B_IMPL(Name, Operator) \ + FUNCTOR_UNARY_OPERATOR_IMPL \ + ( \ + Name, Operator, bool, \ + (requires(const T& InValue) { { Operator InValue } -> CBooleanTestable; }), \ + (requires(U&& InValue) { { Operator Forward(InValue) } -> CBooleanTestable; }) \ + ) + +#define FUNCTOR_BINARY_OPERATOR_B_IMPL(Name, Operator) \ + FUNCTOR_BINARY_OPERATOR_IMPL \ + ( \ + Name, Operator, bool, \ + (requires(const T& LHS, const T& RHS) { { LHS Operator RHS } -> CBooleanTestable; }), \ + (requires(T&& LHS, U&& RHS) { { Forward(LHS) Operator Forward(RHS) } -> CBooleanTestable; }) \ + ) + +#define FUNCTOR_BINARY_OPERATOR_C_IMPL(Name, Operator) \ + FUNCTOR_BINARY_OPERATOR_IMPL \ + ( \ + Name, Operator, bool, \ + (CEqualityComparable), \ + (CEqualityComparableWith) \ + ) + +#define FUNCTOR_BINARY_OPERATOR_D_IMPL(Name, Operator) \ + FUNCTOR_BINARY_OPERATOR_IMPL \ + ( \ + Name, Operator, bool, \ + (CTotallyOrdered), \ + (CTotallyOrderedWith) \ + ) + +FUNCTOR_BINARY_OPERATOR_A_IMPL(TPlus, +); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TMinus, -); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TMultiplies, *); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TDivides, /); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TModulus, %); +FUNCTOR_UNARY_OPERATOR_A_IMPL (TNegate, -); + +FUNCTOR_BINARY_OPERATOR_C_IMPL(TEqualTo, ==); +FUNCTOR_BINARY_OPERATOR_C_IMPL(TNotEqualTo, !=); +FUNCTOR_BINARY_OPERATOR_D_IMPL(TGreater, > ); +FUNCTOR_BINARY_OPERATOR_D_IMPL(TLess, < ); +FUNCTOR_BINARY_OPERATOR_D_IMPL(TGreaterEqual, >=); +FUNCTOR_BINARY_OPERATOR_D_IMPL(TLessEqual, <=); + +FUNCTOR_BINARY_OPERATOR_B_IMPL(TLogicalAnd, &&); +FUNCTOR_BINARY_OPERATOR_B_IMPL(TLogicalOr, ||); +FUNCTOR_UNARY_OPERATOR_B_IMPL (TLogicalNot, ! ); + +FUNCTOR_BINARY_OPERATOR_A_IMPL(TBitAnd, &); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TBitOr, |); +FUNCTOR_BINARY_OPERATOR_A_IMPL(TBitXor, ^); +FUNCTOR_UNARY_OPERATOR_A_IMPL (TBitNot, ~); + +#undef FUNCTOR_BINARY_OPERATOR_D_IMPL +#undef FUNCTOR_BINARY_OPERATOR_C_IMPL + +#undef FUNCTOR_BINARY_OPERATOR_B_IMPL +#undef FUNCTOR_UNARY_OPERATOR_B_IMPL + +#undef FUNCTOR_BINARY_OPERATOR_A_IMPL +#undef FUNCTOR_UNARY_OPERATOR_A_IMPL + +#undef FUNCTOR_BINARY_OPERATOR_IMPL +#undef FUNCTOR_UNARY_OPERATOR_IMPL + NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END diff --git a/Redcraft.Utility/Source/Public/Templates/Invoke.h b/Redcraft.Utility/Source/Public/Templates/Invoke.h index 34e213f..844a9b4 100644 --- a/Redcraft.Utility/Source/Public/Templates/Invoke.h +++ b/Redcraft.Utility/Source/Public/Templates/Invoke.h @@ -1,6 +1,7 @@ #pragma once #include "CoreTypes.h" +#include "Templates/Utility.h" #include "TypeTraits/TypeTraits.h" NAMESPACE_REDCRAFT_BEGIN @@ -13,6 +14,7 @@ struct InvokeFunction { template static auto Invoke(F&& Object, Types&&... Args) + -> decltype(Forward(Object)(Forward(Args)...)) { return Forward(Object)(Forward(Args)...); } @@ -21,15 +23,15 @@ struct InvokeFunction struct InvokeMemberFunction { template - static auto Invoke(F&& Func, ObjectType&& Object, Types&&... Args) -> - decltype((Forward(Object)->*Func)(Forward(Args)...)) + static auto Invoke(F&& Func, ObjectType&& Object, Types&&... Args) + -> decltype((Forward(Object)->*Func)(Forward(Args)...)) { return (Forward(Object)->*Func)(Forward(Args)...); } template - static auto Invoke(F&& Func, ObjectType&& Object, Types&&... Args) -> - decltype((Forward(Object).*Func)(Forward(Args)...)) + static auto Invoke(F&& Func, ObjectType&& Object, Types&&... Args) + -> decltype((Forward(Object).*Func)(Forward(Args)...)) { return (Forward(Object).*Func)(Forward(Args)...); } @@ -38,15 +40,15 @@ struct InvokeMemberFunction struct InvokeMemberObject { template - static auto Invoke(F&& Func, ObjectType&& Object) -> - decltype(Forward(Object)->*Func) + static auto Invoke(F&& Func, ObjectType&& Object) + -> decltype(Forward(Object)->*Func) { return (Forward(Object)->*Func); } template - static auto Invoke(F&& Func, ObjectType&& Object) -> - decltype(Forward(Object).*Func) + static auto Invoke(F&& Func, ObjectType&& Object) + -> decltype(Forward(Object).*Func) { return (Forward(Object).*Func); }