2022-02-03 08:45:34 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
2023-01-02 14:03:40 +00:00
|
|
|
#include "TypeTraits/Swappable.h"
|
2022-02-03 08:45:34 +00:00
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
2023-01-03 00:33:30 +00:00
|
|
|
/** @return The pointer to the container element storage. */
|
|
|
|
template <typename T> requires (requires(T&& Container) { { Container.GetData() } -> CPointer; })
|
2022-12-13 14:02:39 +00:00
|
|
|
FORCEINLINE constexpr decltype(auto) GetData(T&& Container)
|
2022-02-03 08:45:34 +00:00
|
|
|
{
|
|
|
|
return Container.GetData();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetData algorithm for arrays. */
|
2022-12-13 14:02:39 +00:00
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr T* GetData( T(& Container)[N]) { return Container; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr T* GetData( T(&& Container)[N]) { return Container; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr const T* GetData(const T(& Container)[N]) { return Container; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr const T* GetData(const T(&& Container)[N]) { return Container; }
|
2022-02-03 08:45:34 +00:00
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetData algorithm for T::data(). */
|
2023-01-03 00:33:30 +00:00
|
|
|
template <typename T> requires (requires(T&& Container) { { Container.data() } -> CPointer; })
|
2022-12-13 14:02:39 +00:00
|
|
|
FORCEINLINE constexpr decltype(auto) GetData(T&& Container)
|
2022-02-03 08:45:34 +00:00
|
|
|
{
|
|
|
|
return Container.data();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetData algorithm for initializer_list. */
|
2022-11-23 15:27:45 +00:00
|
|
|
template <typename T>
|
2023-01-03 00:33:30 +00:00
|
|
|
FORCEINLINE constexpr const T* GetData(initializer_list<T> Container)
|
2022-11-23 15:27:45 +00:00
|
|
|
{
|
|
|
|
return Container.begin();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** @return The number of elements in the container. */
|
2023-01-03 00:33:30 +00:00
|
|
|
template <typename T> requires (requires(T&& Container) { { Container.Num() } -> CConvertibleTo<size_t>; })
|
2022-12-13 14:02:39 +00:00
|
|
|
FORCEINLINE constexpr decltype(auto) GetNum(T&& Container)
|
2022-02-03 08:45:34 +00:00
|
|
|
{
|
|
|
|
return Container.Num();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetNum algorithm for arrays. */
|
2022-12-13 14:02:39 +00:00
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr size_t GetNum( T(& Container)[N]) { return N; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr size_t GetNum( T(&& Container)[N]) { return N; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr size_t GetNum(const T(& Container)[N]) { return N; }
|
|
|
|
template <typename T, size_t N> FORCEINLINE constexpr size_t GetNum(const T(&& Container)[N]) { return N; }
|
2022-02-03 08:45:34 +00:00
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetNum algorithm for T::size(). */
|
2023-01-03 00:33:30 +00:00
|
|
|
template <typename T> requires (requires(T&& Container) { { Container.size() } -> CConvertibleTo<size_t>; })
|
2022-12-13 14:02:39 +00:00
|
|
|
FORCEINLINE constexpr decltype(auto) GetNum(T&& Container)
|
2022-02-03 08:45:34 +00:00
|
|
|
{
|
|
|
|
return Container.size();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the GetNum algorithm for initializer_list. */
|
2022-11-23 15:27:45 +00:00
|
|
|
template <typename T>
|
2023-01-03 00:33:30 +00:00
|
|
|
FORCEINLINE constexpr size_t GetNum(initializer_list<T> Container)
|
2022-11-23 15:27:45 +00:00
|
|
|
{
|
|
|
|
return Container.size();
|
|
|
|
}
|
|
|
|
|
2023-01-02 14:03:40 +00:00
|
|
|
/** Overloads the Swap algorithm for arrays. */
|
|
|
|
template <typename T, size_t N> requires (CSwappable<TRemoveAllExtents<T>>)
|
|
|
|
FORCEINLINE constexpr void Swap(T(&A)[N], T(&B)[N])
|
|
|
|
{
|
|
|
|
for (size_t Index = 0; Index < N; ++Index)
|
|
|
|
{
|
|
|
|
Swap(A[Index], B[Index]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 08:45:34 +00:00
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|