2024-12-20 17:35:31 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
|
|
|
#include "TypeTraits/TypeTraits.h"
|
|
|
|
#include "Iterators/Utility.h"
|
2024-12-20 18:09:27 +08:00
|
|
|
#include "Iterators/Sentinel.h"
|
2024-12-20 17:35:31 +08:00
|
|
|
#include "Iterators/BasicIterator.h"
|
|
|
|
#include "Ranges/Utility.h"
|
|
|
|
#include "Miscellaneous/AssertionMacros.h"
|
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
2024-12-20 18:09:27 +08:00
|
|
|
NAMESPACE_BEGIN(Algorithms)
|
2024-12-20 17:35:31 +08:00
|
|
|
|
|
|
|
/** Increments given iterator 'Iter' by 'N' elements. */
|
|
|
|
template <CInputIterator I>
|
|
|
|
FORCEINLINE constexpr void Advance(I& Iter, ptrdiff N)
|
|
|
|
{
|
|
|
|
if constexpr (CRandomAccessIterator<I>)
|
|
|
|
{
|
|
|
|
Iter += N;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if constexpr (CBidirectionalIterator<I>)
|
|
|
|
{
|
|
|
|
for (; N > 0; --N) ++Iter;
|
|
|
|
for (; N < 0; ++N) --Iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
checkf(N >= 0, TEXT("The iterator must satisfy the CBidirectionalIterator in order to be decremented."));
|
|
|
|
for (; N > 0; --N) ++Iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The number of hops from 'First' to 'Last'. */
|
|
|
|
template <CInputIterator I, CSentinelFor<I> S>
|
|
|
|
NODISCARD FORCEINLINE constexpr ptrdiff Distance(I First, S Last)
|
|
|
|
{
|
|
|
|
if constexpr (CSizedSentinelFor<S, I>)
|
|
|
|
{
|
|
|
|
return Last - First;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ptrdiff Result = 0;
|
|
|
|
for (; First != Last; ++First) ++Result;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The size of the range. */
|
|
|
|
template <CRange R>
|
|
|
|
NODISCARD FORCEINLINE constexpr ptrdiff Distance(R&& Range)
|
|
|
|
{
|
|
|
|
if constexpr (CSizedRange<R>)
|
|
|
|
{
|
2024-12-20 18:09:27 +08:00
|
|
|
return static_cast<ptrdiff>(Ranges::Num(Range));
|
2024-12-20 17:35:31 +08:00
|
|
|
}
|
2024-12-20 18:09:27 +08:00
|
|
|
else return Algorithms::Distance(Ranges::Begin(Range), Ranges::End(Range));
|
2024-12-20 17:35:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The 'N'-th successor of iterator 'Iter'. */
|
|
|
|
template <CInputIterator I>
|
|
|
|
NODISCARD FORCEINLINE constexpr I Next(I Iter, ptrdiff N = 1)
|
|
|
|
{
|
2024-12-20 18:09:27 +08:00
|
|
|
Algorithms::Advance(Iter, N);
|
2024-12-20 17:35:31 +08:00
|
|
|
return Iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The 'N'-th predecessor of iterator 'Iter'. */
|
|
|
|
template <CBidirectionalIterator I>
|
|
|
|
NODISCARD FORCEINLINE constexpr I Prev(I Iter, ptrdiff N = 1)
|
|
|
|
{
|
2024-12-20 18:09:27 +08:00
|
|
|
Algorithms::Advance(Iter, -N);
|
2024-12-20 17:35:31 +08:00
|
|
|
return Iter;
|
|
|
|
}
|
|
|
|
|
2024-12-20 18:09:27 +08:00
|
|
|
NAMESPACE_END(Algorithms)
|
2024-12-20 17:35:31 +08:00
|
|
|
|
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|