2023-02-08 23:30:46 +08:00
# pragma once
# include "CoreTypes.h"
2024-12-18 16:58:43 +08:00
# include "TypeTraits/TypeTraits.h"
2023-02-08 23:30:46 +08:00
# include "Templates/Utility.h"
# include "Templates/TypeHash.h"
2024-12-20 18:09:27 +08:00
# include "Memory/Allocators.h"
2023-02-08 23:30:46 +08:00
# include "Memory/MemoryOperator.h"
2024-12-20 17:31:52 +08:00
# include "Iterators/Utility.h"
# include "Iterators/BasicIterator.h"
# include "Iterators/Sentinel.h"
# include "Iterators/ReverseIterator.h"
# include "Ranges/Utility.h"
# include "Ranges/Factory.h"
2024-12-18 16:58:43 +08:00
# include "Miscellaneous/Compare.h"
2023-02-08 23:30:46 +08:00
# include "Miscellaneous/AssertionMacros.h"
NAMESPACE_REDCRAFT_BEGIN
NAMESPACE_MODULE_BEGIN ( Redcraft )
NAMESPACE_MODULE_BEGIN ( Utility )
/** Dynamic array. The elements are stored contiguously, which means that elements can be accessed not only through iterators, but also using offsets to regular pointers to elements. */
2023-03-22 22:25:12 +08:00
template < CAllocatableObject T , CAllocator < T > Allocator = FHeapAllocator >
2024-11-02 20:49:48 +08:00
class TArray
2023-02-08 23:30:46 +08:00
{
2023-03-04 19:12:47 +08:00
private :
2024-10-30 23:14:52 +08:00
template < bool bConst , typename = TConditional < bConst , const T , T > >
class TIteratorImpl ;
2023-03-04 19:12:47 +08:00
2023-02-08 23:30:46 +08:00
public :
2024-12-16 19:34:47 +08:00
using FElementType = T ;
using FAllocatorType = Allocator ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
using FReference = T & ;
using FConstReference = const T & ;
2023-03-01 18:32:32 +08:00
2024-12-16 19:34:47 +08:00
using FIterator = TIteratorImpl < false > ;
using FConstIterator = TIteratorImpl < true > ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
using FReverseIterator = TReverseIterator < FIterator > ;
using FConstReverseIterator = TReverseIterator < FConstIterator > ;
2023-02-13 19:26:09 +08:00
2024-12-16 19:34:47 +08:00
static_assert ( CContiguousIterator < FIterator > ) ;
static_assert ( CContiguousIterator < FConstIterator > ) ;
2023-02-12 23:46:30 +08:00
2023-02-08 23:30:46 +08:00
/** Default constructor. Constructs an empty container with a default-constructed allocator. */
2023-02-22 23:34:51 +08:00
FORCEINLINE TArray ( ) : TArray ( 0 ) { }
2023-02-08 23:30:46 +08:00
/** Constructs the container with 'Count' default instances of T. */
2024-12-18 12:14:59 +08:00
explicit TArray ( size_t Count ) requires ( CDefaultConstructible < T > )
2023-02-12 23:46:30 +08:00
{
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : DefaultConstruct < FElementType > ( Impl . Pointer , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
2024-10-24 15:24:29 +08:00
2023-02-08 23:30:46 +08:00
/** Constructs the container with 'Count' copies of elements with 'InValue'. */
2024-12-18 12:14:59 +08:00
FORCEINLINE explicit TArray ( size_t Count , const FElementType & InValue ) requires ( CCopyConstructible < T > )
2024-12-20 18:09:27 +08:00
: TArray ( Ranges : : Repeat ( InValue , Count ) )
2024-10-29 21:19:44 +08:00
{ }
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
/** Constructs the container with the contents of the range ['First', 'Last'). */
2024-12-18 12:14:59 +08:00
template < CInputIterator I , CSentinelFor < I > S > requires ( CConstructibleFrom < T , TIteratorReference < I > > & & CMovable < T > )
explicit TArray ( I First , S Last )
2023-02-12 23:46:30 +08:00
{
if constexpr ( CForwardIterator < I > )
{
2024-12-17 21:49:37 +08:00
size_t Count = 0 ;
2023-02-12 23:46:30 +08:00
2024-12-17 21:49:37 +08:00
if constexpr ( CSizedSentinelFor < S , I > )
{
checkf ( First - Last < = 0 , TEXT ( " Illegal range iterator. Please check First <= Last. " ) ) ;
Count = Last - First ;
}
else for ( I Iter = First ; Iter ! = Last ; + + Iter ) + + Count ;
2023-02-12 23:46:30 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-12 23:46:30 +08:00
for ( size_t Index = 0 ; Index ! = Count ; + + Index )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Index ) FElementType ( * First + + ) ;
2023-02-12 23:46:30 +08:00
}
}
else
{
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = 0 ;
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-12 23:46:30 +08:00
while ( First ! = Last )
{
PushBack ( * First ) ;
+ + First ;
}
}
}
2024-12-17 21:49:37 +08:00
/** Constructs the container with the contents of the range. */
2024-12-18 12:14:59 +08:00
template < CInputRange R > requires ( ! CSameAs < TRemoveCVRef < R > , TArray > & & CConstructibleFrom < T , TRangeReference < R > > & & CMovable < T > )
2024-12-20 18:09:27 +08:00
FORCEINLINE explicit TArray ( R & & Range ) : TArray ( Ranges : : Begin ( Range ) , Ranges : : End ( Range ) ) { }
2024-12-17 21:49:37 +08:00
2023-02-08 23:30:46 +08:00
/** Copy constructor. Constructs the container with the copy of the contents of 'InValue'. */
2024-12-18 12:14:59 +08:00
TArray ( const TArray & InValue ) requires ( CCopyConstructible < T > )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : CopyConstruct < FElementType > ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
/** Move constructor. After the move, 'InValue' is guaranteed to be empty. */
2024-12-18 12:14:59 +08:00
TArray ( TArray & & InValue ) requires ( CMoveConstructible < T > )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
if ( InValue . Impl - > IsTransferable ( InValue . Impl . Pointer ) )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Impl . ArrayMax = InValue . Max ( ) ;
Impl . Pointer = InValue . Impl . Pointer ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
InValue . Impl . ArrayNum = 0 ;
InValue . Impl . ArrayMax = InValue . Impl - > CalculateSlackReserve ( InValue . Num ( ) ) ;
InValue . Impl . Pointer = InValue . Impl - > Allocate ( InValue . Max ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else
{
2023-03-02 22:51:45 +08:00
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
2024-10-28 15:59:45 +08:00
InValue . Reset ( ) ;
2023-02-08 23:30:46 +08:00
}
/** Constructs the container with the contents of the initializer list. */
2024-12-20 18:09:27 +08:00
FORCEINLINE TArray ( initializer_list < FElementType > IL ) requires ( CCopyConstructible < T > ) : TArray ( Ranges : : Begin ( IL ) , Ranges : : End ( IL ) ) { }
2023-02-08 23:30:46 +08:00
/** Destructs the array. The destructors of the elements are called and the used storage is deallocated. */
2023-02-22 23:34:51 +08:00
~ TArray ( )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2023-02-08 23:30:46 +08:00
}
/** Copy assignment operator. Replaces the contents with a copy of the contents of 'InValue'. */
2024-12-18 12:14:59 +08:00
TArray & operator = ( const TArray & InValue ) requires ( CCopyable < T > )
2023-02-08 23:30:46 +08:00
{
if ( & InValue = = this ) UNLIKELY return * this ;
size_t NumToAllocate = InValue . Num ( ) ;
2023-03-02 22:51:45 +08:00
NumToAllocate = NumToAllocate > Max ( ) ? Impl - > CalculateSlackGrow ( InValue . Num ( ) , Max ( ) ) : NumToAllocate ;
NumToAllocate = NumToAllocate < Max ( ) ? Impl - > CalculateSlackShrink ( InValue . Num ( ) , Max ( ) ) : NumToAllocate ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : CopyConstruct < FElementType > ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
return * this ;
}
if ( InValue . Num ( ) < = Num ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : CopyAssign ( Impl . Pointer , InValue . Impl . Pointer , InValue . Num ( ) ) ;
Memory : : Destruct ( Impl . Pointer + InValue . Num ( ) , Num ( ) - InValue . Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else if ( InValue . Num ( ) < = Max ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : CopyAssign ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2024-12-16 19:34:47 +08:00
Memory : : CopyConstruct < FElementType > ( Impl . Pointer + Num ( ) , InValue . Impl . Pointer + Num ( ) , InValue . Num ( ) - Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else check_no_entry ( ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
2023-02-08 23:30:46 +08:00
return * this ;
}
/** Move assignment operator. After the move, 'InValue' is guaranteed to be empty. */
2024-12-18 12:14:59 +08:00
TArray & operator = ( TArray & & InValue ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
if ( & InValue = = this ) UNLIKELY return * this ;
2023-03-02 22:51:45 +08:00
if ( InValue . Impl - > IsTransferable ( InValue . Impl . Pointer ) )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2023-02-08 23:30:46 +08:00
2024-11-02 23:07:05 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
Impl . ArrayMax = InValue . Max ( ) ;
Impl . Pointer = InValue . Impl . Pointer ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
InValue . Impl . ArrayNum = 0 ;
InValue . Impl . ArrayMax = InValue . Impl - > CalculateSlackReserve ( InValue . Num ( ) ) ;
InValue . Impl . Pointer = InValue . Impl - > Allocate ( InValue . Max ( ) ) ;
2023-02-08 23:30:46 +08:00
return * this ;
}
size_t NumToAllocate = InValue . Num ( ) ;
2023-03-02 22:51:45 +08:00
NumToAllocate = NumToAllocate > Max ( ) ? Impl - > CalculateSlackGrow ( InValue . Num ( ) , Max ( ) ) : NumToAllocate ;
NumToAllocate = NumToAllocate < Max ( ) ? Impl - > CalculateSlackShrink ( InValue . Num ( ) , Max ( ) ) : NumToAllocate ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2024-10-24 15:24:29 +08:00
2023-02-08 23:30:46 +08:00
InValue . Reset ( ) ;
return * this ;
}
if ( InValue . Num ( ) < = Num ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : MoveAssign ( Impl . Pointer , InValue . Impl . Pointer , InValue . Num ( ) ) ;
Memory : : Destruct ( Impl . Pointer + InValue . Num ( ) , Num ( ) - InValue . Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else if ( InValue . Num ( ) < = Max ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : MoveAssign ( Impl . Pointer , InValue . Impl . Pointer , Num ( ) ) ;
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + Num ( ) , InValue . Impl . Pointer + Num ( ) , InValue . Num ( ) - Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else check_no_entry ( ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = InValue . Num ( ) ;
2023-02-08 23:30:46 +08:00
InValue . Reset ( ) ;
return * this ;
}
/** Replaces the contents with those identified by initializer list. */
2024-12-18 12:14:59 +08:00
TArray & operator = ( initializer_list < FElementType > IL ) requires ( CCopyable < T > )
2023-02-08 23:30:46 +08:00
{
2024-12-20 18:09:27 +08:00
size_t NumToAllocate = Ranges : : Num ( IL ) ;
2023-02-08 23:30:46 +08:00
2024-12-20 18:09:27 +08:00
NumToAllocate = NumToAllocate > Max ( ) ? Impl - > CalculateSlackGrow ( Ranges : : Num ( IL ) , Max ( ) ) : NumToAllocate ;
NumToAllocate = NumToAllocate < Max ( ) ? Impl - > CalculateSlackShrink ( Ranges : : Num ( IL ) , Max ( ) ) : NumToAllocate ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2024-10-24 15:24:29 +08:00
2024-12-20 18:09:27 +08:00
Impl . ArrayNum = Ranges : : Num ( IL ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-20 18:09:27 +08:00
Memory : : CopyConstruct < FElementType > ( Impl . Pointer , Ranges : : GetData ( IL ) , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
return * this ;
}
2024-12-20 18:09:27 +08:00
if ( Ranges : : Num ( IL ) < = Num ( ) )
2023-02-08 23:30:46 +08:00
{
2024-12-20 18:09:27 +08:00
Memory : : CopyAssign ( Impl . Pointer , Ranges : : GetData ( IL ) , Ranges : : Num ( IL ) ) ;
Memory : : Destruct ( Impl . Pointer + Ranges : : Num ( IL ) , Num ( ) - Ranges : : Num ( IL ) ) ;
2023-02-08 23:30:46 +08:00
}
2024-12-20 18:09:27 +08:00
else if ( Ranges : : Num ( IL ) < = Max ( ) )
2023-02-08 23:30:46 +08:00
{
2024-12-20 18:09:27 +08:00
Memory : : CopyAssign ( Impl . Pointer , Ranges : : GetData ( IL ) , Num ( ) ) ;
Memory : : CopyConstruct < FElementType > ( Impl . Pointer + Num ( ) , Ranges : : GetData ( IL ) + Num ( ) , Ranges : : Num ( IL ) - Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else check_no_entry ( ) ;
2024-12-20 18:09:27 +08:00
Impl . ArrayNum = Ranges : : Num ( IL ) ;
2023-02-08 23:30:46 +08:00
return * this ;
}
/** Compares the contents of two arrays. */
2024-12-18 12:14:59 +08:00
NODISCARD friend bool operator = = ( const TArray & LHS , const TArray & RHS ) requires ( CWeaklyEqualityComparable < T > )
2023-02-08 23:30:46 +08:00
{
if ( LHS . Num ( ) ! = RHS . Num ( ) ) return false ;
2023-04-01 19:28:03 +08:00
for ( size_t Index = 0 ; Index < LHS . Num ( ) ; + + Index )
2023-02-08 23:30:46 +08:00
{
2023-04-01 19:28:03 +08:00
if ( LHS [ Index ] ! = RHS [ Index ] ) return false ;
2023-02-08 23:30:46 +08:00
}
return true ;
}
2023-03-13 21:53:47 +08:00
/** Compares the contents of 'LHS' and 'RHS' lexicographically. */
2024-12-18 12:14:59 +08:00
NODISCARD friend auto operator < = > ( const TArray & LHS , const TArray & RHS ) requires ( CSynthThreeWayComparable < T > )
2023-02-08 23:30:46 +08:00
{
2023-04-01 19:28:03 +08:00
const size_t NumToCompare = LHS . Num ( ) < RHS . Num ( ) ? LHS . Num ( ) : RHS . Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-04-01 19:28:03 +08:00
for ( size_t Index = 0 ; Index < NumToCompare ; + + Index )
2023-02-08 23:30:46 +08:00
{
2023-04-01 19:28:03 +08:00
if ( const auto Result = SynthThreeWayCompare ( LHS [ Index ] , RHS [ Index ] ) ; Result ! = 0 ) return Result ;
2023-02-08 23:30:46 +08:00
}
2023-03-13 21:53:47 +08:00
return LHS . Num ( ) < = > RHS . Num ( ) ;
2023-02-08 23:30:46 +08:00
}
2024-10-24 15:24:29 +08:00
2023-02-08 23:30:46 +08:00
/** Inserts 'InValue' before 'Iter' in the container. */
2024-12-18 12:14:59 +08:00
FIterator Insert ( FConstIterator Iter , const FElementType & InValue ) requires ( CCopyable < T > )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
const size_t InsertIndex = Iter - Begin ( ) ;
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Num ( ) + 1 > Max ( ) ? Impl - > CalculateSlackGrow ( Num ( ) + 1 , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate > = Num ( ) + 1 ) ;
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , InsertIndex ) ;
new ( Impl . Pointer + InsertIndex ) FElementType ( InValue ) ;
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + InsertIndex + 1 , OldAllocation + InsertIndex , NumToDestruct - InsertIndex ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
if ( InsertIndex ! = Num ( ) )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Num ( ) ) FElementType ( MoveTemp ( Impl . Pointer [ Num ( ) - 1 ] ) ) ;
2023-02-08 23:30:46 +08:00
for ( size_t Index = Num ( ) - 1 ; Index ! = InsertIndex ; - - Index )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ Index ] = MoveTemp ( Impl . Pointer [ Index - 1 ] ) ;
2023-02-08 23:30:46 +08:00
}
2023-03-02 22:51:45 +08:00
Impl . Pointer [ InsertIndex ] = InValue ;
2023-02-08 23:30:46 +08:00
}
2024-12-16 19:34:47 +08:00
else new ( Impl . Pointer + Num ( ) ) FElementType ( InValue ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
/** Inserts 'InValue' before 'Iter' in the container. */
2024-12-18 12:14:59 +08:00
FIterator Insert ( FConstIterator Iter , FElementType & & InValue ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
const size_t InsertIndex = Iter - Begin ( ) ;
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Num ( ) + 1 > Max ( ) ? Impl - > CalculateSlackGrow ( Num ( ) + 1 , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate > = Num ( ) + 1 ) ;
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , InsertIndex ) ;
new ( Impl . Pointer + InsertIndex ) FElementType ( MoveTemp ( InValue ) ) ;
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + InsertIndex + 1 , OldAllocation + InsertIndex , NumToDestruct - InsertIndex ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
if ( InsertIndex ! = Num ( ) )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Num ( ) ) FElementType ( MoveTemp ( Impl . Pointer [ Num ( ) - 1 ] ) ) ;
2023-02-08 23:30:46 +08:00
for ( size_t Index = Num ( ) - 1 ; Index ! = InsertIndex ; - - Index )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ Index ] = MoveTemp ( Impl . Pointer [ Index - 1 ] ) ;
2023-02-08 23:30:46 +08:00
}
2023-03-02 22:51:45 +08:00
Impl . Pointer [ InsertIndex ] = MoveTemp ( InValue ) ;
2023-02-08 23:30:46 +08:00
}
2024-12-16 19:34:47 +08:00
else new ( Impl . Pointer + Num ( ) ) FElementType ( MoveTemp ( InValue ) ) ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
/** Inserts 'Count' copies of the 'InValue' before 'Iter' in the container. */
2024-12-18 12:14:59 +08:00
FIterator Insert ( FConstIterator Iter , size_t Count , const FElementType & InValue ) requires ( CCopyable < T > )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
2024-12-20 18:09:27 +08:00
return Insert ( Iter , Ranges : : Repeat ( InValue , Count ) ) ;
2023-02-08 23:30:46 +08:00
}
2024-10-24 15:24:29 +08:00
2023-02-12 23:46:30 +08:00
/** Inserts elements from range ['First', 'Last') before 'Iter'. */
2024-12-18 12:14:59 +08:00
template < CInputIterator I , CSentinelFor < I > S > requires ( CConstructibleFrom < T , TIteratorReference < I > > & & CAssignableFrom < T & , TIteratorReference < I > > & & CMovable < T > )
2024-12-16 19:34:47 +08:00
FIterator Insert ( FConstIterator Iter , I First , S Last )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
2023-02-12 23:46:30 +08:00
if constexpr ( CForwardIterator < I > )
{
const size_t InsertIndex = Iter - Begin ( ) ;
2024-12-17 21:49:37 +08:00
size_t Count = 0 ;
if constexpr ( CSizedSentinelFor < S , I > )
{
checkf ( First - Last < = 0 , TEXT ( " Illegal range iterator. Please check First <= Last. " ) ) ;
Count = Last - First ;
}
else for ( I Jter = First ; Jter ! = Last ; + + Jter ) + + Count ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
if ( Count = = 0 ) return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Num ( ) + Count > Max ( ) ? Impl - > CalculateSlackGrow ( Num ( ) + Count , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
check ( NumToAllocate > = Num ( ) + Count ) ;
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + Count ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , InsertIndex ) ;
2023-02-12 23:46:30 +08:00
for ( size_t Index = InsertIndex ; Index ! = InsertIndex + Count ; + + Index )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Index ) FElementType ( * First + + ) ;
2023-02-12 23:46:30 +08:00
}
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + InsertIndex + Count , OldAllocation + InsertIndex , NumToDestruct - InsertIndex ) ;
2023-02-12 23:46:30 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-12 23:46:30 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
2024-10-29 21:19:44 +08:00
/*
* NO ( XA ) - No Operation
* IA ( AB ) - Insert Assignment
* IC ( BC ) - Insert Construction
* MA ( CD ) - Move Assignment
* MC ( DO ) - Move Construction
*
* IR ( AC ) - Insert Range
* UI ( UO ) - Uninitialized
*
* | X | - - - - - - - - - - - - - - - - - - - | | - UI - | O |
* | X | - - - - | A | - IR - | C | - - - - - - - - - - - | O |
* | X | - NO - | A | - IA - | BC | - MA - | D | - MC - | O |
*
* | X | - - - - - - - - - - - - - - - - - | | - UI - | O |
* | X | - - - - - - - - - - | A | - IR - | CD | - - - - | O |
* | X | - - - - NO - - - - | A | - IA - | BCD | - MC - | O |
*
* | X | - - - - - - - - - - - | | - - - - - UI - - - - - | O |
* | X | - - - - | A | - - - - IR - - - - - | C | - - - - | O |
* | X | - NO - | A | - IA - | B | - IC - | CD | - MC - | O |
*
* | X | - - - - - - - - - - - - - - - - | | - UI - | O |
* | X | - - - - - - - - - - - - - - - - | A | - IR - | C O |
* | X | - - - - - - - NO - - - - - - - | AB | - IC - | CDO |
*
* | X | - - - - - - - - - - - | | - - - - UI - - - - | O |
* | X | - - - - - - - - - - - - - - - - | A | - IR - | C O |
* | X | - - - - - - - NO - - - - - - - | AB | - IC - | CDO |
*/
2023-02-12 23:46:30 +08:00
const size_t IndexA = InsertIndex ;
const size_t IndexC = InsertIndex + Count ;
const size_t IndexB = Num ( ) > IndexA ? ( Num ( ) < IndexC ? Num ( ) : IndexC ) : IndexA ;
const size_t IndexD = Num ( ) > IndexC ? Num ( ) : IndexC ;
const size_t IndexO = Num ( ) + Count ;
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
for ( size_t TargetIndex = IndexO - 1 ; TargetIndex ! = IndexD - 1 ; - - TargetIndex )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + TargetIndex ) FElementType ( MoveTemp ( Impl . Pointer [ TargetIndex - Count ] ) ) ;
2023-02-12 23:46:30 +08:00
}
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
for ( size_t TargetIndex = IndexD - 1 ; TargetIndex ! = IndexC - 1 ; - - TargetIndex )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ TargetIndex ] = MoveTemp ( Impl . Pointer [ TargetIndex - Count ] ) ;
2023-02-12 23:46:30 +08:00
}
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
for ( size_t TargetIndex = IndexA ; TargetIndex ! = IndexB ; + + TargetIndex )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ TargetIndex ] = * First + + ;
2023-02-12 23:46:30 +08:00
}
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
for ( size_t TargetIndex = IndexB ; TargetIndex ! = IndexC ; + + TargetIndex )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + TargetIndex ) FElementType ( * First + + ) ;
2023-02-12 23:46:30 +08:00
}
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
check ( First = = Last ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + Count ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-12 23:46:30 +08:00
}
else
2023-02-08 23:30:46 +08:00
{
2023-02-12 23:46:30 +08:00
TArray Temp ( MoveTemp ( First ) , MoveTemp ( Last ) ) ;
2024-10-24 15:24:29 +08:00
return Insert ( Iter , MakeMoveIterator ( Temp . Begin ( ) ) , MakeMoveSentinel ( Temp . End ( ) ) ) ;
2023-02-08 23:30:46 +08:00
}
2023-02-12 23:46:30 +08:00
}
2023-02-08 23:30:46 +08:00
2024-12-17 21:49:37 +08:00
/** Inserts elements from range before 'Iter'. */
2024-12-18 12:14:59 +08:00
template < CInputRange R > requires ( CConstructibleFrom < T , TRangeReference < R > > & & CAssignableFrom < T & , TRangeReference < R > > & & CMovable < T > )
2024-12-17 21:49:37 +08:00
FORCEINLINE FIterator Insert ( FConstIterator Iter , R & & Range )
{
2024-12-20 18:09:27 +08:00
return Insert ( Iter , Ranges : : Begin ( Range ) , Ranges : : End ( Range ) ) ;
2024-12-17 21:49:37 +08:00
}
2023-02-12 23:46:30 +08:00
/** Inserts elements from initializer list before 'Iter' in the container. */
2024-12-18 12:14:59 +08:00
FORCEINLINE FIterator Insert ( FConstIterator Iter , initializer_list < FElementType > IL ) requires ( CCopyable < T > )
2023-02-12 23:46:30 +08:00
{
2024-12-20 18:09:27 +08:00
return Insert ( Iter , Ranges : : Begin ( IL ) , Ranges : : End ( IL ) ) ;
2023-02-08 23:30:46 +08:00
}
/** Inserts a new element into the container directly before 'Iter'. */
2024-12-18 12:14:59 +08:00
template < typename . . . Ts > requires ( CConstructibleFrom < T , Ts . . . > & & CMovable < T > )
2024-12-16 19:34:47 +08:00
FIterator Emplace ( FConstIterator Iter , Ts & & . . . Args )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
const size_t InsertIndex = Iter - Begin ( ) ;
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Num ( ) + 1 > Max ( ) ? Impl - > CalculateSlackGrow ( Num ( ) + 1 , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate > = Num ( ) + 1 ) ;
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , InsertIndex ) ;
new ( Impl . Pointer + InsertIndex ) FElementType ( Forward < Ts > ( Args ) . . . ) ;
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + InsertIndex + 1 , OldAllocation + InsertIndex , NumToDestruct - InsertIndex ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
if ( InsertIndex ! = Num ( ) )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Num ( ) ) FElementType ( MoveTemp ( Impl . Pointer [ Num ( ) - 1 ] ) ) ;
2023-02-08 23:30:46 +08:00
for ( size_t Index = Num ( ) - 1 ; Index ! = InsertIndex ; - - Index )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ Index ] = MoveTemp ( Impl . Pointer [ Index - 1 ] ) ;
2023-02-08 23:30:46 +08:00
}
2024-12-16 19:34:47 +08:00
Impl . Pointer [ InsertIndex ] = FElementType ( Forward < Ts > ( Args ) . . . ) ;
2023-02-08 23:30:46 +08:00
}
2024-12-16 19:34:47 +08:00
else new ( Impl . Pointer + Num ( ) ) FElementType ( Forward < Ts > ( Args ) . . . ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + InsertIndex ) ;
2023-02-08 23:30:46 +08:00
}
/** Removes the element at 'Iter' in the container. Without changing the order of elements. */
2024-12-18 12:14:59 +08:00
FORCEINLINE FIterator StableErase ( FConstIterator Iter , bool bAllowShrinking = true ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) & & Iter ! = End ( ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
return StableErase ( Iter , Iter + 1 , bAllowShrinking ) ;
}
2023-02-12 23:46:30 +08:00
/** Removes the elements in the range ['First', 'Last') in the container. Without changing the order of elements. */
2024-12-18 12:14:59 +08:00
FIterator StableErase ( FConstIterator First , FConstIterator Last , bool bAllowShrinking = true ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
2023-02-12 23:46:30 +08:00
checkf ( IsValidIterator ( First ) & & IsValidIterator ( Last ) & & First < = Last , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
const size_t EraseIndex = First - Begin ( ) ;
const size_t EraseCount = Last - First ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
if ( EraseCount = = 0 ) return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = bAllowShrinking ? Impl - > CalculateSlackShrink ( Num ( ) - EraseCount , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) - EraseCount ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , EraseIndex ) ;
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + EraseIndex , OldAllocation + EraseIndex + EraseCount , NumToDestruct - EraseIndex - EraseCount ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
}
for ( size_t Index = EraseIndex + EraseCount ; Index ! = Num ( ) ; + + Index )
{
2023-03-02 22:51:45 +08:00
Impl . Pointer [ Index - EraseCount ] = MoveTemp ( Impl . Pointer [ Index ] ) ;
2023-02-08 23:30:46 +08:00
}
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer + Num ( ) - EraseCount , EraseCount ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) - EraseCount ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
}
/** Removes the element at 'Iter' in the container. But it may change the order of elements. */
2024-12-18 12:14:59 +08:00
FORCEINLINE FIterator Erase ( FConstIterator Iter , bool bAllowShrinking = true ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
checkf ( IsValidIterator ( Iter ) & & Iter ! = End ( ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
return Erase ( Iter , Iter + 1 , bAllowShrinking ) ;
}
2023-02-12 23:46:30 +08:00
/** Removes the elements in the range ['First', 'Last') in the container. But it may change the order of elements. */
2024-12-18 12:14:59 +08:00
FIterator Erase ( FConstIterator First , FConstIterator Last , bool bAllowShrinking = true ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
2023-02-12 23:46:30 +08:00
checkf ( IsValidIterator ( First ) & & IsValidIterator ( Last ) & & First < = Last , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
2024-10-24 15:24:29 +08:00
2023-02-12 23:46:30 +08:00
const size_t EraseIndex = First - Begin ( ) ;
const size_t EraseCount = Last - First ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
if ( EraseCount = = 0 ) return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = bAllowShrinking ? Impl - > CalculateSlackShrink ( Num ( ) - EraseCount , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) - EraseCount ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , EraseIndex ) ;
Memory : : MoveConstruct < FElementType > ( Impl . Pointer + EraseIndex , OldAllocation + EraseIndex + EraseCount , NumToDestruct - EraseIndex - EraseCount ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
}
for ( size_t Index = 0 ; Index ! = EraseCount ; + + Index )
{
if ( EraseIndex + Index > = Num ( ) - EraseCount ) break ;
2023-03-02 22:51:45 +08:00
Impl . Pointer [ EraseIndex + Index ] = MoveTemp ( Impl . Pointer [ Num ( ) - Index - 1 ] ) ;
2023-02-08 23:30:46 +08:00
}
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer + Num ( ) - EraseCount , EraseCount ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) - EraseCount ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
return FIterator ( this , Impl . Pointer + EraseIndex ) ;
2023-02-08 23:30:46 +08:00
}
/** Appends the given element value to the end of the container. */
2024-12-18 12:14:59 +08:00
FORCEINLINE void PushBack ( const FElementType & InValue ) requires ( CCopyable < T > )
2023-02-08 23:30:46 +08:00
{
EmplaceBack ( InValue ) ;
}
/** Appends the given element value to the end of the container. */
2024-12-18 12:14:59 +08:00
FORCEINLINE void PushBack ( FElementType & & InValue ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
EmplaceBack ( MoveTemp ( InValue ) ) ;
}
/** Appends a new element to the end of the container. */
2024-12-18 12:14:59 +08:00
template < typename . . . Ts > requires ( CConstructibleFrom < T , Ts . . . > & & CMovable < T > )
2024-12-16 19:34:47 +08:00
FElementType & EmplaceBack ( Ts & & . . . Args )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Num ( ) + 1 > Max ( ) ? Impl - > CalculateSlackGrow ( Num ( ) + 1 , Max ( ) ) : Max ( ) ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate > = Num ( ) + 1 ) ;
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , Num ( ) - 1 ) ;
new ( Impl . Pointer + Num ( ) - 1 ) FElementType ( Forward < Ts > ( Args ) . . . ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
return Impl . Pointer [ Num ( ) - 1 ] ;
2023-02-08 23:30:46 +08:00
}
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Num ( ) ) FElementType ( Forward < Ts > ( Args ) . . . ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Num ( ) + 1 ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
return Impl . Pointer [ Num ( ) - 1 ] ;
2023-02-08 23:30:46 +08:00
}
/** Removes the last element of the container. The array cannot be empty. */
2024-12-18 12:14:59 +08:00
FORCEINLINE void PopBack ( bool bAllowShrinking = true ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
Erase ( End ( ) - 1 , bAllowShrinking ) ;
}
/** Resizes the container to contain 'Count' elements. Additional default elements are appended. */
2024-12-18 12:14:59 +08:00
void SetNum ( size_t Count , bool bAllowShrinking = true ) requires ( CDefaultConstructible < T > & & CMovable < T > )
2023-02-08 23:30:46 +08:00
{
size_t NumToAllocate = Count ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
NumToAllocate = NumToAllocate > Max ( ) ? Impl - > CalculateSlackGrow ( Count , Max ( ) ) : NumToAllocate ;
NumToAllocate = NumToAllocate < Max ( ) ? ( bAllowShrinking ? Impl - > CalculateSlackShrink ( Count , Max ( ) ) : Max ( ) ) : NumToAllocate ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
if ( NumToDestruct < = Num ( ) )
{
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , NumToDestruct ) ;
Memory : : DefaultConstruct < FElementType > ( Impl . Pointer + NumToDestruct , Num ( ) - NumToDestruct ) ;
2023-02-08 23:30:46 +08:00
}
else
{
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
return ;
}
2024-10-24 15:24:29 +08:00
2023-02-08 23:30:46 +08:00
if ( Count < = Num ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer + Count , Num ( ) - Count ) ;
2023-02-08 23:30:46 +08:00
}
else if ( Count < = Max ( ) )
{
2024-12-16 19:34:47 +08:00
Memory : : DefaultConstruct < FElementType > ( Impl . Pointer + Num ( ) , Count - Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
else check_no_entry ( ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
2023-02-08 23:30:46 +08:00
}
/** Resizes the container to contain 'Count' elements. Additional copies of 'InValue' are appended. */
2024-12-18 12:14:59 +08:00
void SetNum ( size_t Count , const FElementType & InValue , bool bAllowShrinking = true ) requires ( CCopyConstructible < T > & & CMovable < T > )
2023-02-08 23:30:46 +08:00
{
size_t NumToAllocate = Count ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
NumToAllocate = NumToAllocate > Max ( ) ? Impl - > CalculateSlackGrow ( Count , Max ( ) ) : NumToAllocate ;
NumToAllocate = NumToAllocate < Max ( ) ? ( bAllowShrinking ? Impl - > CalculateSlackShrink ( Count , Max ( ) ) : Max ( ) ) : NumToAllocate ;
2023-02-08 23:30:46 +08:00
if ( NumToAllocate ! = Max ( ) )
{
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
const size_t NumToDestruct = Num ( ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
if ( NumToDestruct < = Num ( ) )
{
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , NumToDestruct ) ;
2023-02-08 23:30:46 +08:00
for ( size_t Index = NumToDestruct ; Index ! = Num ( ) ; + + Index )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Index ) FElementType ( InValue ) ;
2023-02-08 23:30:46 +08:00
}
}
else
{
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
}
Memory : : Destruct ( OldAllocation , NumToDestruct ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
return ;
}
2024-10-24 15:24:29 +08:00
2023-02-08 23:30:46 +08:00
if ( Count < = Num ( ) )
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer + Count , Num ( ) - Count ) ;
2023-02-08 23:30:46 +08:00
}
else if ( Count < = Max ( ) )
{
for ( size_t Index = Num ( ) ; Index ! = Count ; + + Index )
{
2024-12-16 19:34:47 +08:00
new ( Impl . Pointer + Index ) FElementType ( InValue ) ;
2023-02-08 23:30:46 +08:00
}
}
else check_no_entry ( ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = Count ;
2023-02-08 23:30:46 +08:00
}
/** Increase the max capacity of the array to a value that's greater or equal to 'Count'. */
2024-12-18 12:14:59 +08:00
void Reserve ( size_t Count ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
if ( Count < = Max ( ) ) return ;
2024-12-16 19:34:47 +08:00
const size_t NumToAllocate = Impl - > CalculateSlackReserve ( Count ) ;
FElementType * OldAllocation = Impl . Pointer ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate > Max ( ) ) ;
2023-03-02 22:51:45 +08:00
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , Num ( ) ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
}
/** Requests the removal of unused capacity. */
2023-02-22 23:34:51 +08:00
void Shrink ( )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
size_t NumToAllocate = Impl - > CalculateSlackReserve ( Num ( ) ) ;
2023-02-08 23:30:46 +08:00
check ( NumToAllocate < = Max ( ) ) ;
if ( NumToAllocate = = Max ( ) ) return ;
2024-12-16 19:34:47 +08:00
FElementType * OldAllocation = Impl . Pointer ;
2024-10-24 15:24:29 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayMax = NumToAllocate ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
2024-12-16 19:34:47 +08:00
Memory : : MoveConstruct < FElementType > ( Impl . Pointer , OldAllocation , Num ( ) ) ;
2023-02-08 23:30:46 +08:00
Memory : : Destruct ( OldAllocation , Num ( ) ) ;
2023-03-02 22:51:45 +08:00
Impl - > Deallocate ( OldAllocation ) ;
2023-02-08 23:30:46 +08:00
}
2023-02-27 23:25:40 +08:00
2023-02-08 23:30:46 +08:00
/** @return The pointer to the underlying element storage. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE FElementType * GetData ( ) { return Impl . Pointer ; }
NODISCARD FORCEINLINE const FElementType * GetData ( ) const { return Impl . Pointer ; }
2023-02-08 23:30:46 +08:00
/** @return The iterator to the first or end element. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE FIterator Begin ( ) { return FIterator ( this , Impl . Pointer ) ; }
NODISCARD FORCEINLINE FConstIterator Begin ( ) const { return FConstIterator ( this , Impl . Pointer ) ; }
NODISCARD FORCEINLINE FIterator End ( ) { return FIterator ( this , Impl . Pointer + Num ( ) ) ; }
NODISCARD FORCEINLINE FConstIterator End ( ) const { return FConstIterator ( this , Impl . Pointer + Num ( ) ) ; }
2023-02-27 23:25:40 +08:00
2023-02-13 19:26:09 +08:00
/** @return The reverse iterator to the first or end element. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE FReverseIterator RBegin ( ) { return FReverseIterator ( End ( ) ) ; }
NODISCARD FORCEINLINE FConstReverseIterator RBegin ( ) const { return FConstReverseIterator ( End ( ) ) ; }
NODISCARD FORCEINLINE FReverseIterator REnd ( ) { return FReverseIterator ( Begin ( ) ) ; }
NODISCARD FORCEINLINE FConstReverseIterator REnd ( ) const { return FConstReverseIterator ( Begin ( ) ) ; }
2023-02-08 23:30:46 +08:00
/** @return The number of elements in the container. */
2023-03-02 22:51:45 +08:00
NODISCARD FORCEINLINE size_t Num ( ) const { return Impl . ArrayNum ; }
2023-02-08 23:30:46 +08:00
/** @return The number of elements that can be held in currently allocated storage. */
2023-03-02 22:51:45 +08:00
NODISCARD FORCEINLINE size_t Max ( ) const { return Impl . ArrayMax ; }
2023-02-08 23:30:46 +08:00
/** @return true if the container is empty, false otherwise. */
2023-02-22 23:34:51 +08:00
NODISCARD FORCEINLINE bool IsEmpty ( ) const { return Num ( ) = = 0 ; }
2023-02-08 23:30:46 +08:00
/** @return true if the iterator is valid, false otherwise. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE bool IsValidIterator ( FConstIterator Iter ) const { return Begin ( ) < = Iter & & Iter < = End ( ) ; }
2023-02-08 23:30:46 +08:00
/** @return The reference to the requested element. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE FElementType & operator [ ] ( size_t Index ) { checkf ( Index < Num ( ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ; return Impl . Pointer [ Index ] ; }
NODISCARD FORCEINLINE const FElementType & operator [ ] ( size_t Index ) const { checkf ( Index < Num ( ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ; return Impl . Pointer [ Index ] ; }
2023-02-08 23:30:46 +08:00
/** @return The reference to the first or last element. */
2024-12-16 19:34:47 +08:00
NODISCARD FORCEINLINE FElementType & Front ( ) { return * Begin ( ) ; }
NODISCARD FORCEINLINE const FElementType & Front ( ) const { return * Begin ( ) ; }
NODISCARD FORCEINLINE FElementType & Back ( ) { return * ( End ( ) - 1 ) ; }
NODISCARD FORCEINLINE const FElementType & Back ( ) const { return * ( End ( ) - 1 ) ; }
2023-02-08 23:30:46 +08:00
/** Erases all elements from the container. After this call, Num() returns zero. */
2023-02-22 23:34:51 +08:00
void Reset ( bool bAllowShrinking = true )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
const size_t NumToAllocate = Impl - > CalculateSlackReserve ( 0 ) ;
2023-02-08 23:30:46 +08:00
2023-02-12 23:46:30 +08:00
if ( bAllowShrinking & & NumToAllocate ! = Max ( ) )
2023-02-08 23:30:46 +08:00
{
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl - > Deallocate ( Impl . Pointer ) ;
2023-02-08 23:30:46 +08:00
2023-03-02 22:51:45 +08:00
Impl . ArrayNum = 0 ;
Impl . ArrayMax = Impl - > CalculateSlackReserve ( Num ( ) ) ;
Impl . Pointer = Impl - > Allocate ( Max ( ) ) ;
2023-02-08 23:30:46 +08:00
return ;
}
2023-03-02 22:51:45 +08:00
Memory : : Destruct ( Impl . Pointer , Num ( ) ) ;
Impl . ArrayNum = 0 ;
2023-02-08 23:30:46 +08:00
}
/** Overloads the GetTypeHash algorithm for TArray. */
2024-12-18 12:14:59 +08:00
NODISCARD friend FORCEINLINE size_t GetTypeHash ( const TArray & A ) requires ( CHashable < T > )
2023-02-08 23:30:46 +08:00
{
size_t Result = 0 ;
2024-12-16 19:34:47 +08:00
for ( FConstIterator Iter = A . Begin ( ) ; Iter ! = A . End ( ) ; + + Iter )
2023-02-08 23:30:46 +08:00
{
2023-02-24 19:09:22 +08:00
Result = HashCombine ( Result , GetTypeHash ( * Iter ) ) ;
2023-02-08 23:30:46 +08:00
}
return Result ;
}
/** Overloads the Swap algorithm for TArray. */
2024-12-18 12:14:59 +08:00
friend void Swap ( TArray & A , TArray & B ) requires ( CMovable < T > )
2023-02-08 23:30:46 +08:00
{
const bool bIsTransferable =
2023-03-02 22:51:45 +08:00
A . Impl - > IsTransferable ( A . Impl . Pointer ) & &
B . Impl - > IsTransferable ( B . Impl . Pointer ) ;
2023-02-08 23:30:46 +08:00
if ( bIsTransferable )
{
2023-03-02 22:51:45 +08:00
Swap ( A . Impl . ArrayNum , B . Impl . ArrayNum ) ;
Swap ( A . Impl . ArrayMax , B . Impl . ArrayMax ) ;
Swap ( A . Impl . Pointer , B . Impl . Pointer ) ;
2023-02-08 23:30:46 +08:00
return ;
}
TArray Temp = MoveTemp ( A ) ;
A = MoveTemp ( B ) ;
B = MoveTemp ( Temp ) ;
}
2023-02-15 23:41:05 +08:00
ENABLE_RANGE_BASED_FOR_LOOP_SUPPORT
2023-02-08 23:30:46 +08:00
private :
2024-12-16 19:34:47 +08:00
ALLOCATOR_WRAPPER_BEGIN ( FAllocatorType , FElementType , Impl )
2023-02-27 23:25:40 +08:00
{
size_t ArrayNum ;
size_t ArrayMax ;
2024-12-16 19:34:47 +08:00
FElementType * Pointer ;
2023-03-02 22:51:45 +08:00
}
2024-12-16 19:34:47 +08:00
ALLOCATOR_WRAPPER_END ( FAllocatorType , FElementType , Impl )
2023-02-08 23:30:46 +08:00
2023-03-04 19:12:47 +08:00
private :
2024-10-30 23:14:52 +08:00
template < bool bConst , typename U >
2024-11-02 20:52:10 +08:00
class TIteratorImpl final
2023-03-04 19:12:47 +08:00
{
public :
2024-12-18 12:14:59 +08:00
using FElementType = T ;
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( ) = default ;
2023-03-04 19:12:47 +08:00
# if DO_CHECK
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( const TIteratorImpl < false > & InValue ) requires ( bConst )
2023-03-04 19:12:47 +08:00
: Owner ( InValue . Owner ) , Pointer ( InValue . Pointer )
{ }
# else
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( const TIteratorImpl < false > & InValue ) requires ( bConst )
2023-03-04 19:12:47 +08:00
: Pointer ( InValue . Pointer )
{ }
# endif
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( const TIteratorImpl & ) = default ;
FORCEINLINE TIteratorImpl ( TIteratorImpl & & ) = default ;
FORCEINLINE TIteratorImpl & operator = ( const TIteratorImpl & ) = default ;
FORCEINLINE TIteratorImpl & operator = ( TIteratorImpl & & ) = default ;
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD friend FORCEINLINE bool operator = = ( const TIteratorImpl & LHS , const TIteratorImpl & RHS ) { return LHS . Pointer = = RHS . Pointer ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD friend FORCEINLINE strong_ordering operator < = > ( const TIteratorImpl & LHS , const TIteratorImpl & RHS ) { return LHS . Pointer < = > RHS . Pointer ; }
2023-03-04 19:12:47 +08:00
2024-11-01 19:51:44 +08:00
NODISCARD FORCEINLINE U & operator * ( ) const { CheckThis ( true ) ; return * Pointer ; }
NODISCARD FORCEINLINE U * operator - > ( ) const { CheckThis ( false ) ; return Pointer ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD FORCEINLINE U & operator [ ] ( ptrdiff Index ) const { TIteratorImpl Temp = * this + Index ; return * Temp ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl & operator + + ( ) { + + Pointer ; CheckThis ( ) ; return * this ; }
FORCEINLINE TIteratorImpl & operator - - ( ) { - - Pointer ; CheckThis ( ) ; return * this ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl operator + + ( int ) { TIteratorImpl Temp = * this ; + + * this ; return Temp ; }
FORCEINLINE TIteratorImpl operator - - ( int ) { TIteratorImpl Temp = * this ; - - * this ; return Temp ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl & operator + = ( ptrdiff Offset ) { Pointer + = Offset ; CheckThis ( ) ; return * this ; }
FORCEINLINE TIteratorImpl & operator - = ( ptrdiff Offset ) { Pointer - = Offset ; CheckThis ( ) ; return * this ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD friend FORCEINLINE TIteratorImpl operator + ( TIteratorImpl Iter , ptrdiff Offset ) { TIteratorImpl Temp = Iter ; Temp + = Offset ; return Temp ; }
NODISCARD friend FORCEINLINE TIteratorImpl operator + ( ptrdiff Offset , TIteratorImpl Iter ) { TIteratorImpl Temp = Iter ; Temp + = Offset ; return Temp ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD FORCEINLINE TIteratorImpl operator - ( ptrdiff Offset ) const { TIteratorImpl Temp = * this ; Temp - = Offset ; return Temp ; }
2023-03-04 19:12:47 +08:00
2024-10-30 23:14:52 +08:00
NODISCARD friend FORCEINLINE ptrdiff operator - ( const TIteratorImpl & LHS , const TIteratorImpl & RHS ) { LHS . CheckThis ( ) ; RHS . CheckThis ( ) ; return LHS . Pointer - RHS . Pointer ; }
2023-03-04 19:12:47 +08:00
private :
# if DO_CHECK
const TArray * Owner = nullptr ;
# endif
2024-10-30 23:14:52 +08:00
U * Pointer = nullptr ;
2023-03-04 19:12:47 +08:00
# if DO_CHECK
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( const TArray * InContainer , U * InPointer )
2023-03-04 19:12:47 +08:00
: Owner ( InContainer ) , Pointer ( InPointer )
{ }
# else
2024-10-30 23:14:52 +08:00
FORCEINLINE TIteratorImpl ( const TArray * InContainer , U * InPointer )
2023-03-04 19:12:47 +08:00
: Pointer ( InPointer )
{ }
# endif
FORCEINLINE void CheckThis ( bool bExceptEnd = false ) const
{
checkf ( Owner & & Owner - > IsValidIterator ( * this ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
checkf ( ! ( bExceptEnd & & Owner - > End ( ) = = * this ) , TEXT ( " Read access violation. Please check IsValidIterator(). " ) ) ;
}
2024-10-30 23:14:52 +08:00
template < bool , typename > friend class TIteratorImpl ;
2023-03-04 19:12:47 +08:00
friend TArray ;
} ;
2023-02-08 23:30:46 +08:00
} ;
2023-02-26 19:01:32 +08:00
template < typename I , typename S >
2024-12-17 21:49:37 +08:00
TArray ( I , S ) - > TArray < TIteratorElement < I > > ;
template < typename R >
TArray ( R ) - > TArray < TRangeElement < R > > ;
2023-02-26 19:01:32 +08:00
template < typename T >
TArray ( initializer_list < T > ) - > TArray < T > ;
2023-02-08 23:30:46 +08:00
NAMESPACE_MODULE_END ( Utility )
NAMESPACE_MODULE_END ( Redcraft )
NAMESPACE_REDCRAFT_END