2024-12-13 19:08:05 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
|
|
|
#include "Iterator/Utility.h"
|
|
|
|
#include "Iterator/Sentinel.h"
|
|
|
|
#include "Iterator/RandomAccessIterator.h"
|
|
|
|
#include "TypeTraits/TypeTraits.h"
|
2024-12-14 16:20:26 +08:00
|
|
|
#include "Memory/Address.h"
|
2024-12-13 19:08:05 +08:00
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
|
|
|
#if PLATFORM_COMPILER_GCC
|
|
|
|
# pragma GCC diagnostic push
|
|
|
|
# pragma GCC diagnostic ignored "-Wnon-template-friend"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A concept specifies a type is a contiguous iterator.
|
2024-12-14 14:02:26 +08:00
|
|
|
* Add the operator-> to the random access iterator and requires the operator* returns a true reference type.
|
2024-12-13 19:08:05 +08:00
|
|
|
*/
|
|
|
|
template <typename I>
|
|
|
|
concept CContiguousIterator = CRandomAccessIterator<I> && CLValueReference<TIteratorReference<I>>
|
|
|
|
&& CSameAs<TIteratorElement<I>, TRemoveCVRef<TIteratorReference<I>>>
|
|
|
|
&& CSameAs<TIteratorPointer<I>, TAddPointer<TIteratorReference<I>>>
|
|
|
|
&& requires(I& Iter)
|
|
|
|
{
|
|
|
|
{ ToAddress(Iter) } -> CSameAs<TAddPointer<TIteratorReference<I>>>;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** This is an example of a contiguous iterator, indicate the traits that define a contiguous iterator. */
|
|
|
|
template <CLValueReference T>
|
2024-12-14 14:02:26 +08:00
|
|
|
struct IContiguousIterator /* : IRandomAccessIterator<T> */
|
2024-12-13 19:08:05 +08:00
|
|
|
{
|
2024-12-14 14:02:26 +08:00
|
|
|
// ~Begin CRandomAccessIterator.
|
|
|
|
|
2024-12-16 19:34:47 +08:00
|
|
|
using FElementType = TRemoveCVRef<T>;
|
2024-12-13 19:08:05 +08:00
|
|
|
|
|
|
|
IContiguousIterator();
|
|
|
|
IContiguousIterator(const IContiguousIterator&);
|
2024-12-14 14:02:26 +08:00
|
|
|
IContiguousIterator(IContiguousIterator&&);
|
2024-12-13 19:08:05 +08:00
|
|
|
IContiguousIterator* operator=(const IContiguousIterator&);
|
2024-12-14 14:02:26 +08:00
|
|
|
IContiguousIterator* operator=(IContiguousIterator&&);
|
2024-12-13 19:08:05 +08:00
|
|
|
|
|
|
|
friend bool operator==(const IContiguousIterator&, const IContiguousIterator&);
|
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
friend strong_ordering operator<=>(const IContiguousIterator&, const IContiguousIterator&);
|
|
|
|
|
|
|
|
// ~End CRandomAccessIterator.
|
2024-12-13 19:08:05 +08:00
|
|
|
|
|
|
|
/**
|
2024-12-14 14:02:26 +08:00
|
|
|
* Dereference operator. See IForwardIterator.
|
2024-12-13 19:08:05 +08:00
|
|
|
* Specify, the return type must be a true reference type and refer to an element of a contiguous sequence, not a proxy class.
|
2024-12-14 14:02:26 +08:00
|
|
|
* Also satisfies CRandomAccessIterator.
|
2024-12-13 19:08:05 +08:00
|
|
|
*/
|
|
|
|
T operator*() const;
|
|
|
|
|
|
|
|
/** Indirection operator. Return the address of the element that the iterator is pointing to. */
|
|
|
|
TAddPointer<T> operator->() const;
|
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
// ~Begin CRandomAccessIterator.
|
|
|
|
|
|
|
|
T operator[](ptrdiff) const;
|
2024-12-13 19:08:05 +08:00
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
IContiguousIterator& operator++();
|
2024-12-13 19:08:05 +08:00
|
|
|
IContiguousIterator& operator--();
|
|
|
|
|
|
|
|
IContiguousIterator operator++(int);
|
|
|
|
IContiguousIterator operator--(int);
|
|
|
|
|
|
|
|
IContiguousIterator& operator+=(ptrdiff);
|
|
|
|
IContiguousIterator& operator-=(ptrdiff);
|
|
|
|
|
|
|
|
IContiguousIterator operator+(ptrdiff) const;
|
|
|
|
IContiguousIterator operator-(ptrdiff) const;
|
|
|
|
|
|
|
|
friend IContiguousIterator operator+(ptrdiff, const IContiguousIterator&);
|
|
|
|
|
|
|
|
friend ptrdiff operator-(const IContiguousIterator&, const IContiguousIterator&);
|
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
// ~End CRandomAccessIterator.
|
2024-12-13 19:08:05 +08:00
|
|
|
};
|
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
// Use IContiguousIterator<int> represents a contiguous iterator
|
2024-12-13 19:08:05 +08:00
|
|
|
static_assert(CContiguousIterator<IContiguousIterator<int&>>);
|
|
|
|
static_assert( COutputIterator<IContiguousIterator<int&>, int>);
|
|
|
|
|
2024-12-14 14:02:26 +08:00
|
|
|
// The int* is the most typical example of a contiguous iterator
|
2024-12-13 19:08:05 +08:00
|
|
|
static_assert(CContiguousIterator<int*>);
|
|
|
|
|
|
|
|
#if PLATFORM_COMPILER_GCC
|
|
|
|
# pragma GCC diagnostic pop
|
|
|
|
#endif
|
|
|
|
|
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|