2022-02-04 21:11:05 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
2022-04-08 11:44:36 +08:00
|
|
|
#include "Templates/Utility.h"
|
2022-03-26 19:31:45 +08:00
|
|
|
#include "TypeTraits/TypeTraits.h"
|
2022-02-04 21:11:05 +08:00
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
|
|
|
NAMESPACE_PRIVATE_BEGIN
|
|
|
|
|
|
|
|
struct InvokeFunction
|
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename... Ts>
|
|
|
|
static auto Invoke(F&& Object, Ts&&... Args)
|
|
|
|
-> decltype(Forward<F>(Object)(Forward<Ts>(Args)...))
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
return Forward<F>(Object)(Forward<Ts>(Args)...);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct InvokeMemberFunction
|
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename ObjectType, typename... Ts>
|
|
|
|
static auto Invoke(F&& Func, ObjectType&& Object, Ts&&... Args)
|
|
|
|
-> decltype((Forward<ObjectType>(Object)->*Func)(Forward<Ts>(Args)...))
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
return (Forward<ObjectType>(Object)->*Func)(Forward<Ts>(Args)...);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename ObjectType, typename... Ts>
|
|
|
|
static auto Invoke(F&& Func, ObjectType&& Object, Ts&&... Args)
|
|
|
|
-> decltype((Forward<ObjectType>(Object).*Func)(Forward<Ts>(Args)...))
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
return (Forward<ObjectType>(Object).*Func)(Forward<Ts>(Args)...);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct InvokeMemberObject
|
|
|
|
{
|
|
|
|
template <typename F, typename ObjectType>
|
2022-04-08 11:44:36 +08:00
|
|
|
static auto Invoke(F&& Func, ObjectType&& Object)
|
|
|
|
-> decltype(Forward<ObjectType>(Object)->*Func)
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-04-07 14:52:41 +08:00
|
|
|
return (Forward<ObjectType>(Object)->*Func);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename F, typename ObjectType>
|
2022-04-08 11:44:36 +08:00
|
|
|
static auto Invoke(F&& Func, ObjectType&& Object)
|
|
|
|
-> decltype(Forward<ObjectType>(Object).*Func)
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-04-07 14:52:41 +08:00
|
|
|
return (Forward<ObjectType>(Object).*Func);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename F,
|
|
|
|
typename T,
|
2022-05-22 22:52:47 +08:00
|
|
|
typename Decayed = TDecay<F>,
|
2022-05-15 22:56:53 +08:00
|
|
|
bool IsMemberFunction = CMemberFunctionPointer<Decayed>,
|
|
|
|
bool IsMemberObject = CMemberObjectPointer<Decayed>>
|
2022-02-04 21:11:05 +08:00
|
|
|
struct InvokeMember;
|
|
|
|
|
|
|
|
template <typename F, typename T, typename Decayed>
|
|
|
|
struct InvokeMember<F, T, Decayed, true, false> : InvokeMemberFunction { };
|
|
|
|
|
|
|
|
template <typename F, typename T, typename Decayed>
|
|
|
|
struct InvokeMember<F, T, Decayed, false, true> : InvokeMemberObject { };
|
|
|
|
|
|
|
|
template <typename F, typename T, typename Decayed>
|
|
|
|
struct InvokeMember<F, T, Decayed, false, false> : InvokeFunction { };
|
|
|
|
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename... Ts>
|
2022-02-04 21:11:05 +08:00
|
|
|
struct InvokeImpl;
|
|
|
|
|
|
|
|
template <typename F>
|
|
|
|
struct InvokeImpl<F> : InvokeFunction { };
|
|
|
|
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename T, typename... Ts>
|
|
|
|
struct InvokeImpl<F, T, Ts...> : InvokeMember<F, T> { };
|
2022-02-04 21:11:05 +08:00
|
|
|
|
|
|
|
NAMESPACE_PRIVATE_END
|
|
|
|
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename F, typename... Ts> requires CInvocable<F, Ts...>
|
|
|
|
constexpr auto Invoke(F&& Func, Ts&&... Args)
|
|
|
|
-> decltype(NAMESPACE_PRIVATE::InvokeImpl<F, Ts...>::Invoke(Forward<F>(Func), Forward<Ts>(Args)...))
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
return NAMESPACE_PRIVATE::InvokeImpl<F, Ts...>::Invoke(Forward<F>(Func), Forward<Ts>(Args)...);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename R, typename F, typename... Ts> requires CInvocableResult<R, F, Ts...>
|
|
|
|
constexpr R InvokeResult(F&& Func, Ts&&... Args)
|
2022-02-04 21:11:05 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
if constexpr (CVoid<R>) Invoke(Forward<F>(Func), Forward<Ts>(Args)...);
|
|
|
|
else return Invoke(Forward<F>(Func), Forward<Ts>(Args)...);
|
2022-02-04 21:11:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|