2022-02-09 21:46:39 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
2024-11-01 15:11:45 +08:00
|
|
|
#include "Memory/Address.h"
|
2022-02-09 21:46:39 +08:00
|
|
|
#include "Templates/Invoke.h"
|
|
|
|
#include "Templates/Utility.h"
|
2022-05-03 22:20:08 +08:00
|
|
|
#include "Templates/Optional.h"
|
2023-01-19 19:34:17 +08:00
|
|
|
#include "Templates/TypeHash.h"
|
2022-02-09 21:46:39 +08:00
|
|
|
#include "TypeTraits/TypeTraits.h"
|
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/**
|
|
|
|
* TReferenceWrapper is a class template that wraps a reference object.
|
|
|
|
* It is frequently used as a mechanism to store references inside standard
|
|
|
|
* containers which cannot normally hold references.
|
|
|
|
*/
|
2022-05-15 23:10:02 +08:00
|
|
|
template <typename ReferencedType> requires (CObject<ReferencedType> || CFunction<ReferencedType>)
|
2022-12-30 19:11:01 +08:00
|
|
|
class TReferenceWrapper final
|
2022-02-09 21:46:39 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2024-12-16 19:34:47 +08:00
|
|
|
using FType = ReferencedType;
|
2022-02-09 21:46:39 +08:00
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/** Constructs a new reference wrapper. */
|
2023-01-06 19:14:35 +08:00
|
|
|
template <typename T = ReferencedType&> requires (CConvertibleTo<T&&, ReferencedType&> && !CSameAs<TReferenceWrapper, TRemoveCVRef<T>>)
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper(T&& Object)
|
2023-01-06 19:14:35 +08:00
|
|
|
: Pointer(AddressOf(static_cast<ReferencedType&>(Forward<T>(Object))))
|
|
|
|
{ }
|
2022-02-09 21:46:39 +08:00
|
|
|
|
2023-01-06 19:14:35 +08:00
|
|
|
/** Copies content of other into a new instance. */
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper(const TReferenceWrapper&) = default;
|
2024-11-01 15:11:45 +08:00
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/** Converting copy constructor. */
|
2022-11-16 22:03:54 +08:00
|
|
|
template <typename T = ReferencedType> requires (CConvertibleTo<T&, ReferencedType&>)
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper(const TReferenceWrapper<T>& InValue)
|
2022-05-03 22:20:08 +08:00
|
|
|
: Pointer(InValue.Pointer)
|
|
|
|
{ }
|
2022-02-09 21:46:39 +08:00
|
|
|
|
2024-11-01 23:20:55 +08:00
|
|
|
/** Assigns by copying the content of others. */
|
2023-01-06 19:14:35 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper& operator=(const TReferenceWrapper&) = default;
|
|
|
|
|
2024-11-01 23:20:55 +08:00
|
|
|
/** Assigns by copying the content of others. */
|
2023-01-06 19:14:35 +08:00
|
|
|
template <typename T = ReferencedType> requires (CConvertibleTo<T&, ReferencedType&>)
|
|
|
|
FORCEINLINE constexpr TReferenceWrapper& operator=(const TReferenceWrapper<T>& InValue)
|
|
|
|
{
|
|
|
|
Pointer = InValue.Pointer;
|
2024-11-01 23:20:55 +08:00
|
|
|
|
|
|
|
return *this;
|
2023-01-06 19:14:35 +08:00
|
|
|
}
|
2022-12-13 22:11:10 +08:00
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/** @return The stored reference. */
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr ReferencedType& Get() const { return *Pointer; }
|
2022-12-29 21:55:02 +08:00
|
|
|
FORCEINLINE constexpr operator ReferencedType&() const { return *Pointer; }
|
2022-02-09 21:46:39 +08:00
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/** Calls the Callable object, reference to which is stored. */
|
2022-11-16 19:13:37 +08:00
|
|
|
template <typename... Ts>
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TInvokeResult<ReferencedType&, Ts...> operator()(Ts&&... Args) const
|
2022-02-09 21:46:39 +08:00
|
|
|
{
|
2022-11-16 19:13:37 +08:00
|
|
|
return Invoke(Get(), Forward<Ts>(Args)...);
|
2022-02-09 21:46:39 +08:00
|
|
|
}
|
2022-05-03 22:20:08 +08:00
|
|
|
|
2022-12-29 21:55:02 +08:00
|
|
|
/** Overloads the GetTypeHash algorithm for TReferenceWrapper. */
|
|
|
|
NODISCARD friend FORCEINLINE constexpr size_t GetTypeHash(TReferenceWrapper A) requires (CHashable<ReferencedType>)
|
2022-05-03 22:20:08 +08:00
|
|
|
{
|
2022-12-19 18:00:52 +08:00
|
|
|
return GetTypeHash(A.Get());
|
2022-05-03 22:20:08 +08:00
|
|
|
}
|
2022-12-29 21:55:02 +08:00
|
|
|
|
2022-02-09 21:46:39 +08:00
|
|
|
private:
|
|
|
|
|
2022-05-03 22:20:08 +08:00
|
|
|
ReferencedType* Pointer;
|
|
|
|
|
2022-11-15 19:28:43 +08:00
|
|
|
template <typename T> requires (CObject<T> || CFunction<T>) friend class TReferenceWrapper;
|
2022-05-03 22:20:08 +08:00
|
|
|
|
2022-02-09 21:46:39 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
TReferenceWrapper(T&) -> TReferenceWrapper<T>;
|
|
|
|
|
2022-04-24 23:08:00 +08:00
|
|
|
template <typename T>
|
|
|
|
void Ref(const T&&) = delete;
|
|
|
|
|
|
|
|
template <typename T>
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper<T> Ref(T& InValue)
|
2022-04-24 23:08:00 +08:00
|
|
|
{
|
|
|
|
return TReferenceWrapper<T>(InValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
2022-12-13 22:02:39 +08:00
|
|
|
FORCEINLINE constexpr TReferenceWrapper<T> Ref(TReferenceWrapper<T> InValue)
|
2022-04-24 23:08:00 +08:00
|
|
|
{
|
|
|
|
return Ref(InValue.Get());
|
|
|
|
}
|
|
|
|
|
2022-05-20 23:35:36 +08:00
|
|
|
NAMESPACE_PRIVATE_BEGIN
|
|
|
|
|
2022-06-16 23:37:29 +08:00
|
|
|
template <typename T> struct TIsTReferenceWrapperImpl : FFalse { };
|
|
|
|
template <typename T> struct TIsTReferenceWrapperImpl<TReferenceWrapper<T>> : FTrue { };
|
|
|
|
|
2024-12-16 19:34:47 +08:00
|
|
|
template <typename T> struct TUnwrapReferenceImpl { using FType = T; };
|
|
|
|
template <typename T> struct TUnwrapReferenceImpl<TReferenceWrapper<T>> { using FType = T&; };
|
2022-06-16 23:37:29 +08:00
|
|
|
|
2024-12-16 19:34:47 +08:00
|
|
|
template <typename T> struct TUnwrapRefDecayImpl { using FType = typename TUnwrapReferenceImpl<TDecay<T>>::FType; };
|
2022-03-31 17:36:48 +08:00
|
|
|
|
2022-05-20 23:35:36 +08:00
|
|
|
NAMESPACE_PRIVATE_END
|
|
|
|
|
2022-06-16 23:37:29 +08:00
|
|
|
template <typename T>
|
2022-11-17 20:57:54 +08:00
|
|
|
concept CTReferenceWrapper = NAMESPACE_PRIVATE::TIsTReferenceWrapperImpl<TRemoveCV<T>>::Value;
|
2022-05-20 23:35:36 +08:00
|
|
|
|
2022-06-16 23:37:29 +08:00
|
|
|
template <typename T>
|
2024-12-16 19:34:47 +08:00
|
|
|
using TUnwrapReference = typename NAMESPACE_PRIVATE::TUnwrapReferenceImpl<T>::FType;
|
2022-03-26 19:33:28 +08:00
|
|
|
|
2022-06-16 23:37:29 +08:00
|
|
|
template <typename T>
|
2024-12-16 19:34:47 +08:00
|
|
|
using TUnwrapRefDecay = typename NAMESPACE_PRIVATE::TUnwrapRefDecayImpl<T>::FType;
|
2022-03-26 19:33:28 +08:00
|
|
|
|
2022-02-09 21:46:39 +08:00
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|