2024-10-27 11:29:45 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "CoreTypes.h"
|
|
|
|
#include "String/Char.h"
|
|
|
|
#include "Memory/Allocator.h"
|
|
|
|
#include "Templates/Utility.h"
|
|
|
|
#include "Templates/TypeHash.h"
|
|
|
|
#include "Templates/Container.h"
|
|
|
|
#include "Containers/Iterator.h"
|
|
|
|
#include "Containers/ArrayView.h"
|
|
|
|
#include "TypeTraits/TypeTraits.h"
|
2024-10-28 17:51:21 +08:00
|
|
|
#include "Memory/MemoryOperator.h"
|
2024-10-27 11:29:45 +08:00
|
|
|
#include "Miscellaneous/AssertionMacros.h"
|
2024-10-28 17:51:21 +08:00
|
|
|
#include "Miscellaneous/ConstantIterator.h"
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
#include <cwchar>
|
|
|
|
|
|
|
|
NAMESPACE_REDCRAFT_BEGIN
|
|
|
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
|
|
|
NAMESPACE_MODULE_BEGIN(Utility)
|
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
template <CCharType T, CAllocator<T> Allocator>
|
|
|
|
class TString;
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/**
|
|
|
|
* The class template TStringView describes an object that can refer to a constant contiguous sequence of char-like objects
|
|
|
|
* with the first element of the sequence at position zero. Provides a set of convenient string processing functions.
|
|
|
|
*/
|
|
|
|
template <CCharType T>
|
2024-11-03 10:45:54 +08:00
|
|
|
class TStringView : public TArrayView<const T>
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
private:
|
|
|
|
|
|
|
|
using Super = TArrayView<const T>;
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
using ElementType = T;
|
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
using Reference = typename Super::Reference;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
using Iterator = typename Super::Iterator;
|
|
|
|
using ReverseIterator = typename Super::ReverseIterator;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
static_assert(CContiguousIterator<Iterator>);
|
|
|
|
|
|
|
|
/** Constructs an empty string view. */
|
|
|
|
FORCEINLINE constexpr TStringView() = default;
|
|
|
|
|
|
|
|
/** Constructs a string view that is a view over the range ['InFirst', 'InFirst' + 'Count'). */
|
|
|
|
template <CContiguousIterator I> requires (CConvertibleTo<TIteratorElementType<I>(*)[], const ElementType(*)[]>)
|
2024-11-03 10:45:54 +08:00
|
|
|
FORCEINLINE constexpr TStringView(I InFirst, size_t InCount) : Super(InFirst, InCount) { }
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
/** Constructs a string view that is a view over the range ['InFirst', 'InLast'). */
|
|
|
|
template <CContiguousIterator I, CSizedSentinelFor<I> S> requires (CConvertibleTo<TIteratorElementType<I>(*)[], const ElementType(*)[]>)
|
2024-11-03 10:45:54 +08:00
|
|
|
FORCEINLINE constexpr TStringView(I InFirst, S InLast) : Super(InFirst, InLast) { }
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
/** Constructs a string view that is a view over the string 'InString'. */
|
|
|
|
template <typename Allocator>
|
|
|
|
FORCEINLINE constexpr TStringView(const TString<ElementType, Allocator>& InString);
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Constructs a string view that is a view over the range ['InPtr', 'InPtr' + 'Count'). */
|
2024-11-03 10:45:54 +08:00
|
|
|
FORCEINLINE constexpr TStringView(const ElementType* InPtr, size_t Count) : Super(InPtr, Count)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
checkf(InPtr != nullptr, TEXT("TStringView cannot be initialized by nullptr. Please check the pointer."));
|
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE constexpr TStringView(nullptr_t, size_t) = delete;
|
|
|
|
|
|
|
|
/** Constructs a string view that is a view over the range ['InPtr', '\0'). */
|
|
|
|
FORCEINLINE constexpr TStringView(const ElementType* InPtr)
|
|
|
|
{
|
|
|
|
checkf(InPtr != nullptr, TEXT("TStringView cannot be initialized by nullptr. Please check the pointer."));
|
|
|
|
|
|
|
|
size_t Length = 0;
|
|
|
|
|
|
|
|
if constexpr (CSameAs<ElementType, char>)
|
|
|
|
{
|
|
|
|
Length = NAMESPACE_STD::strlen(InPtr);
|
|
|
|
}
|
|
|
|
else if constexpr (CSameAs<ElementType, wchar>)
|
|
|
|
{
|
|
|
|
Length = NAMESPACE_STD::wcslen(InPtr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (InPtr[Length] != LITERAL(ElementType, '\0')) ++Length;
|
|
|
|
}
|
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
*this = TStringView(InPtr, Length);
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
FORCEINLINE constexpr TStringView(nullptr_t) = delete;
|
|
|
|
|
|
|
|
/** Defaulted copy constructor copies the size and data pointer. */
|
|
|
|
FORCEINLINE constexpr TStringView(const TStringView&) = default;
|
|
|
|
|
|
|
|
/** Assigns other to *this. This defaulted assignment operator performs a shallow copy of the data pointer and the size. */
|
|
|
|
FORCEINLINE constexpr TStringView& operator=(const TStringView&) noexcept = default;
|
|
|
|
|
|
|
|
/** Compares the contents of two string views. */
|
2024-11-03 10:45:54 +08:00
|
|
|
NODISCARD friend constexpr bool operator==(TStringView LHS, TStringView RHS) { return static_cast<Super>(LHS) == static_cast<Super>(RHS); }
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
/** Compares the contents of a string view and a character. */
|
|
|
|
NODISCARD friend constexpr bool operator==(TStringView LHS, ElementType RHS) { return LHS == TStringView(&RHS, 1); }
|
|
|
|
NODISCARD friend constexpr bool operator==(ElementType LHS, TStringView RHS) { return TStringView(&LHS, 1) == RHS; }
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Compares the contents of two string views. */
|
2024-11-03 10:45:54 +08:00
|
|
|
NODISCARD friend constexpr auto operator<=>(TStringView LHS, TStringView RHS) { return static_cast<Super>(LHS) <=> static_cast<Super>(RHS); }
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
/** Compares the contents of a string view and a character. */
|
|
|
|
NODISCARD friend constexpr auto operator<=>(TStringView LHS, ElementType RHS) { return LHS <=> TStringView(&RHS, 1); }
|
|
|
|
NODISCARD friend constexpr auto operator<=>(ElementType LHS, TStringView RHS) { return TStringView(&LHS, 1) <=> RHS; }
|
|
|
|
|
2024-11-12 17:21:36 +08:00
|
|
|
public:
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Shrinks the view by moving its start forward. */
|
2024-11-12 17:21:36 +08:00
|
|
|
FORCEINLINE constexpr TStringView RemovePrefix(size_t Count)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
checkf(Count <= this->Num(), TEXT("Illegal subview range. Please check Count."));
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
*this = Substr(Count);
|
2024-11-12 17:21:36 +08:00
|
|
|
|
|
|
|
return *this;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Shrinks the view by moving its end backward. */
|
2024-11-12 17:21:36 +08:00
|
|
|
FORCEINLINE constexpr TStringView RemoveSuffix(size_t Count)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
checkf(Count <= this->Num(), TEXT("Illegal subview range. Please check Count."));
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
*this = Substr(0, this->Num() - Count);
|
2024-11-12 17:21:36 +08:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes whitespace characters from the start of this string. */
|
|
|
|
FORCEINLINE constexpr TStringView TrimStart()
|
|
|
|
{
|
|
|
|
auto Index = Find([](ElementType Char) { return !TChar<ElementType>::IsSpace(Char); });
|
|
|
|
|
|
|
|
if (Index != INDEX_NONE)
|
|
|
|
{
|
|
|
|
RemovePrefix(Index);
|
|
|
|
}
|
|
|
|
else *this = TStringView();
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes whitespace characters from the end of this string. */
|
|
|
|
FORCEINLINE constexpr TStringView TrimEnd()
|
|
|
|
{
|
|
|
|
auto Index = RFind([](ElementType Char) { return !TChar<ElementType>::IsSpace(Char); });
|
|
|
|
|
|
|
|
if (Index != INDEX_NONE)
|
|
|
|
{
|
|
|
|
RemoveSuffix(this->Num() - Index - 1);
|
|
|
|
}
|
|
|
|
else *this = TStringView();
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes whitespace characters from the start and end of this string. */
|
|
|
|
FORCEINLINE constexpr TStringView TrimStartAndEnd()
|
|
|
|
{
|
|
|
|
TrimStart();
|
|
|
|
TrimEnd();
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes characters after the first null-terminator. */
|
|
|
|
FORCEINLINE constexpr TStringView TrimToNullTerminator()
|
|
|
|
{
|
|
|
|
auto Index = Find(LITERAL(ElementType, '\0'));
|
|
|
|
|
|
|
|
if (Index != INDEX_NONE)
|
|
|
|
{
|
|
|
|
*this = Substr(0, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
2024-11-12 17:21:36 +08:00
|
|
|
public:
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Copies the elements of this string view to the destination buffer without null-termination. */
|
|
|
|
FORCEINLINE constexpr size_t Copy(ElementType* Dest, size_t Count = DynamicExtent, size_t Offset = 0) const
|
|
|
|
{
|
2024-10-28 17:51:21 +08:00
|
|
|
checkf(Dest != nullptr, TEXT("Illegal destination buffer. Please check the pointer."));
|
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
checkf(Offset <= this->Num() && (Count == DynamicExtent || Offset + Count <= this->Num()), TEXT("Illegal subview range. Please check Offset and Count."));
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
if (Count == DynamicExtent)
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
Count = this->Num() - Offset;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
Memory::CopyAssign(Dest, this->GetData() + Offset, Count);
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
return Count;
|
|
|
|
}
|
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
FORCEINLINE constexpr size_t Copy(nullptr_t, size_t Count = DynamicExtent, size_t Offset = 0) const = delete;
|
|
|
|
|
2024-11-03 21:22:01 +08:00
|
|
|
/** Obtains an array view that is a view over the first 'Count' elements of this array view. */
|
|
|
|
NODISCARD FORCEINLINE constexpr TStringView First(size_t Count) const
|
|
|
|
{
|
|
|
|
checkf(Count <= this->Num(), TEXT("Illegal subview range. Please check Count."));
|
|
|
|
|
|
|
|
return Substr(0, Count);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Obtains an array view that is a view over the last 'Count' elements of this array view. */
|
|
|
|
NODISCARD FORCEINLINE constexpr TStringView Last(size_t Count) const
|
|
|
|
{
|
|
|
|
checkf(Count <= this->Num(), TEXT("Illegal subview range. Please check Count."));
|
|
|
|
|
|
|
|
return Substr(this->Num() - Count);
|
|
|
|
}
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Obtains a string view that is a view over the 'Count' elements of this string view starting at 'Offset'. */
|
|
|
|
NODISCARD FORCEINLINE constexpr TStringView Substr(size_t Offset, size_t Count = DynamicExtent) const
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
checkf(Offset <= this->Num() && (Count == DynamicExtent || Offset + Count <= this->Num()), TEXT("Illegal subview range. Please check Offset and Count."));
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
Super Temp = this->Subview(Offset, Count);
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
return TStringView(Temp.GetData(), Temp.Num());
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view starts with the given prefix, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool StartsWith(TStringView Prefix) const
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
return this->Num() >= Prefix.Num() && Substr(0, Prefix.Num()) == Prefix;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view starts with the given prefix, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool StartsWith(ElementType Prefix) const
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
return this->Num() >= 1 && this->Front() == Prefix;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view ends with the given suffix, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool EndsWith(TStringView Suffix) const
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
return this->Num() >= Suffix.Num() && Substr(this->Num() - Suffix.Num(), Suffix.Num()) == Suffix;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view ends with the given suffix, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool EndsWith(ElementType Suffix) const
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
return this->Num() >= 1 && this->Back() == Suffix;
|
2024-10-27 11:29:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view contains the given substring, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool Contains(TStringView View) const
|
|
|
|
{
|
|
|
|
return Find(View) != INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view contains the given character, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool Contains(ElementType Char) const
|
|
|
|
{
|
|
|
|
return Find(Char) != INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string view contains character that satisfy the given predicate, false otherwise. */
|
|
|
|
template <CPredicate<ElementType> F>
|
|
|
|
NODISCARD FORCEINLINE constexpr bool Contains(F&& InPredicate) const
|
|
|
|
{
|
|
|
|
return Find(Forward<F>(InPredicate)) != INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first occurrence of the given substring, or INDEX_NONE if not found. */
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t Find(TStringView View, size_t Index = 0) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
if (View.Num() > this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
if (View.Num() == 0) return Index;
|
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
for (; Index != this->Num() - View.Num() + 1; ++Index)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
if (Substr(Index).StartsWith(View))
|
|
|
|
{
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first occurrence of the given character, or INDEX_NONE if not found. */
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t Find(ElementType Char, size_t Index = 0) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
for (; Index != this->Num(); ++Index)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
if ((*this)[Index] == Char)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first occurrence of the character that satisfy the given predicate, or INDEX_NONE if not found. */
|
|
|
|
template <CPredicate<ElementType> F>
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t Find(F&& InPredicate, size_t Index = 0) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
for (; Index != this->Num(); ++Index)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
if (InvokeResult<bool>(Forward<F>(InPredicate), (*this)[Index]))
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last occurrence of the given substring, or INDEX_NONE if not found. */
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t RFind(TStringView View, size_t Index = INDEX_NONE) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index != INDEX_NONE && Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
if (View.Num() > this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
if (Index == INDEX_NONE) Index = this->Num();
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
if (View.Num() == 0) return Index;
|
|
|
|
|
|
|
|
for (; Index != View.Num() - 1; --Index)
|
|
|
|
{
|
|
|
|
if (Substr(0, Index).EndsWith(View))
|
|
|
|
{
|
|
|
|
return Index - View.Num();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last occurrence of the given character, or INDEX_NONE if not found. */
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t RFind(ElementType Char, size_t Index = INDEX_NONE) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index != INDEX_NONE && Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
if (Index == INDEX_NONE) Index = this->Num();
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
for (; Index != 0; --Index)
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
if ((*this)[Index - 1] == Char)
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
return Index - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last occurrence of the character that satisfy the given predicate, or INDEX_NONE if not found. */
|
|
|
|
template <CPredicate<ElementType> F>
|
2024-10-28 17:51:21 +08:00
|
|
|
NODISCARD constexpr size_t RFind(F&& InPredicate, size_t Index = INDEX_NONE) const
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
2024-11-08 17:55:35 +08:00
|
|
|
if (Index != INDEX_NONE && Index >= this->Num()) return INDEX_NONE;
|
2024-10-27 11:29:45 +08:00
|
|
|
|
2024-11-03 10:45:54 +08:00
|
|
|
if (Index == INDEX_NONE) Index = this->Num();
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
for (; Index != 0; --Index)
|
|
|
|
{
|
2024-11-03 10:45:54 +08:00
|
|
|
if (InvokeResult<bool>(Forward<F>(InPredicate), (*this)[Index - 1]))
|
2024-10-27 11:29:45 +08:00
|
|
|
{
|
|
|
|
return Index - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return INDEX_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first occurrence of the character contained in the given view, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindFirstOf(TStringView View, size_t Index = 0) const
|
|
|
|
{
|
|
|
|
return Find([View](ElementType Char) { return View.Contains(Char); }, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first occurrence of the given character, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindFirstOf(ElementType Char, size_t Index = 0) const
|
|
|
|
{
|
|
|
|
return Find(Char, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last occurrence of the character contained in the given view, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindLastOf(TStringView View, size_t Index = INDEX_NONE) const
|
|
|
|
{
|
|
|
|
return RFind([View](ElementType Char) { return View.Contains(Char); }, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last occurrence of the given character, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindLastOf(ElementType Char, size_t Index = INDEX_NONE) const
|
|
|
|
{
|
|
|
|
return RFind(Char, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first absence of the character contained in the given view, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindFirstNotOf(TStringView View, size_t Index = 0) const
|
|
|
|
{
|
|
|
|
return Find([View](ElementType Char) { return !View.Contains(Char); }, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the first absence of the given character, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindFirstNotOf(ElementType Char, size_t Index = 0) const
|
|
|
|
{
|
|
|
|
return Find([Char](ElementType C) { return C != Char; }, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last absence of the character contained in the given view, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindLastNotOf(TStringView View, size_t Index = INDEX_NONE) const
|
|
|
|
{
|
|
|
|
return RFind([View](ElementType Char) { return !View.Contains(Char); }, Index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return The index of the last absence of the given character, or INDEX_NONE if not found. */
|
|
|
|
NODISCARD FORCEINLINE constexpr size_t FindLastNotOf(ElementType Char, size_t Index = INDEX_NONE) const
|
|
|
|
{
|
|
|
|
return RFind([Char](ElementType C) { return C != Char; }, Index);
|
|
|
|
}
|
|
|
|
|
2024-11-10 19:24:40 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
/** @return true if the string only contains valid characters, false otherwise. */
|
|
|
|
NODISCARD constexpr bool IsValid() const
|
|
|
|
{
|
|
|
|
for (ElementType Char : *this)
|
|
|
|
{
|
|
|
|
if (!TChar<ElementType>::IsValid(Char)) return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string only contains ASCII characters, false otherwise. */
|
|
|
|
NODISCARD constexpr bool IsASCII() const
|
|
|
|
{
|
|
|
|
for (ElementType Char : *this)
|
|
|
|
{
|
|
|
|
if (!TChar<ElementType>::IsASCII(Char)) return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-11-12 17:21:36 +08:00
|
|
|
/** @return true if the string can be fully represented as a boolean value, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool IsBoolean() const
|
2024-11-10 19:24:40 +08:00
|
|
|
{
|
2024-11-12 17:21:36 +08:00
|
|
|
TStringView View = *this;
|
|
|
|
|
|
|
|
Ignore = View.ToBoolAndTrim();
|
|
|
|
|
|
|
|
return View.IsEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string can be fully represented as an integer value, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool IsInteger(unsigned Base = 10, bool bSigned = true) const
|
|
|
|
{
|
|
|
|
TStringView View = *this;
|
|
|
|
|
|
|
|
if (View.StartsWith(LITERAL(ElementType, '-')))
|
2024-11-10 19:24:40 +08:00
|
|
|
{
|
2024-11-12 17:21:36 +08:00
|
|
|
if (bSigned) View.RemovePrefix(1);
|
|
|
|
else return false;
|
2024-11-10 19:24:40 +08:00
|
|
|
}
|
|
|
|
|
2024-11-12 17:21:36 +08:00
|
|
|
Ignore = View.ToIntAndTrim(Base);
|
|
|
|
|
|
|
|
return View.IsEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return true if the string can be fully represented as a floating-point value, false otherwise. */
|
|
|
|
NODISCARD FORCEINLINE constexpr bool IsFloatingPoint(bool bFixed = true, bool bScientific = true, bool bSigned = true) const
|
|
|
|
{
|
|
|
|
TStringView View = *this;
|
|
|
|
|
|
|
|
if (View.StartsWith(LITERAL(ElementType, '-')))
|
|
|
|
{
|
|
|
|
if (bSigned) View.RemovePrefix(1);
|
|
|
|
else return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ignore = View.ToFloatAndTrim(bFixed, bScientific);
|
|
|
|
|
|
|
|
return View.IsEmpty();
|
2024-11-10 19:24:40 +08:00
|
|
|
}
|
|
|
|
|
2024-11-11 22:30:15 +08:00
|
|
|
public:
|
2024-11-10 19:24:40 +08:00
|
|
|
|
2024-11-11 22:30:15 +08:00
|
|
|
/**
|
|
|
|
* Converts a string into a boolean value.
|
|
|
|
*
|
|
|
|
* - 1, "true", "True", "TRUE" and non-zero integers become true.
|
|
|
|
* - 0, "false", "False", "FALSE" and unparsable values become false.
|
|
|
|
*
|
|
|
|
* @return The boolean value.
|
|
|
|
*/
|
2024-11-12 17:21:36 +08:00
|
|
|
NODISCARD constexpr bool ToBool() const
|
|
|
|
{
|
|
|
|
return TStringView(*this).ToBoolAndTrim();
|
|
|
|
}
|
2024-11-11 22:30:15 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a string into an integer value.
|
|
|
|
*
|
|
|
|
* - "0x" or "0X" prefixes are not recognized if base is 16.
|
|
|
|
* - Only the minus sign is recognized (not the plus sign), and only for signed integer types of value.
|
|
|
|
* - Leading whitespace is not ignored.
|
|
|
|
*
|
|
|
|
* Ensure that the entire string can be parsed if IsNumeric(Base, false, true, false) is true.
|
|
|
|
*
|
|
|
|
* @param Base - The base of the number, between [2, 36].
|
|
|
|
*
|
|
|
|
* @return The integer value.
|
|
|
|
*/
|
|
|
|
template <CIntegral U = int> requires (!CSameAs<U, bool> && !CConst<U> && !CVolatile<U>)
|
2024-11-12 17:21:36 +08:00
|
|
|
NODISCARD constexpr U ToInt(unsigned Base = 10) const
|
|
|
|
{
|
|
|
|
return TStringView(*this).ToIntAndTrim<U>(Base);
|
|
|
|
}
|
2024-11-11 22:30:15 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a string into a floating-point value.
|
|
|
|
*
|
|
|
|
* - "0x" or "0X" prefixes are not recognized if base is 16.
|
|
|
|
* - The plus sign is not recognized outside the exponent (only the minus sign is permitted at the beginning).
|
|
|
|
* - Leading whitespace is not ignored.
|
|
|
|
*
|
|
|
|
* Ensure that the entire string can be parsed if bFixed and IsNumeric(10, false) is true.
|
|
|
|
* Parsers hex floating-point values if bFixed and bScientific are false.
|
|
|
|
*
|
|
|
|
* @param bFixed - The fixed-point format.
|
|
|
|
* @param bScientific - The scientific notation.
|
|
|
|
*
|
|
|
|
* @return The floating-point value.
|
|
|
|
*/
|
|
|
|
template <CFloatingPoint U = float> requires (!CConst<U> && !CVolatile<U>)
|
2024-11-12 17:21:36 +08:00
|
|
|
NODISCARD constexpr U ToFloat(bool bFixed = true, bool bScientific = true) const
|
|
|
|
{
|
|
|
|
return TStringView(*this).ToFloatAndTrim<U>(bFixed, bScientific);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Converts a string into a boolean value and remove the parsed substring. */
|
|
|
|
NODISCARD constexpr bool ToBoolAndTrim();
|
|
|
|
|
|
|
|
/** Converts a string into an integer value and remove the parsed substring. */
|
|
|
|
template <CIntegral U = int> requires (!CSameAs<U, bool> && !CConst<U> && !CVolatile<U>)
|
|
|
|
NODISCARD constexpr U ToIntAndTrim(unsigned Base = 10);
|
|
|
|
|
|
|
|
/** Converts a string into a floating-point value and remove the parsed substring. */
|
|
|
|
template <CFloatingPoint U = float> requires (!CConst<U> && !CVolatile<U>)
|
|
|
|
NODISCARD constexpr U ToFloatAndTrim(bool bFixed = true, bool bScientific = true);
|
2024-11-10 19:24:40 +08:00
|
|
|
|
2024-11-08 20:38:19 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse a string using a format string to objects.
|
|
|
|
*
|
|
|
|
* @param Fmt - The format string.
|
|
|
|
* @param Args - The objects to parse.
|
|
|
|
*
|
|
|
|
* @return The number of objects successfully parsed.
|
|
|
|
*/
|
|
|
|
template <typename... Ts>
|
2024-11-12 17:21:36 +08:00
|
|
|
size_t Parse(TStringView Fmt, Ts&... Args) const
|
|
|
|
{
|
|
|
|
return TStringView(*this).ParseAndTrim(Fmt, Args...);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Parse a string using a format string to objects and remove the parsed substring. */
|
|
|
|
template <typename... Ts>
|
|
|
|
size_t ParseAndTrim(TStringView Fmt, Ts&... Args);
|
2024-11-08 20:38:19 +08:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
/** Overloads the GetTypeHash algorithm for TStringView. */
|
2024-11-03 10:45:54 +08:00
|
|
|
NODISCARD friend FORCEINLINE constexpr size_t GetTypeHash(TStringView A) { return GetTypeHash(static_cast<Super>(A)); }
|
2024-10-27 11:29:45 +08:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename I, typename S>
|
|
|
|
TStringView(I, S) -> TStringView<TIteratorElementType<I>>;
|
|
|
|
|
2024-10-28 17:51:21 +08:00
|
|
|
template<typename T, typename Allocator>
|
|
|
|
TStringView(TString<T, Allocator>) -> TStringView<T>;
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
using FStringView = TStringView<char>;
|
|
|
|
using FWStringView = TStringView<wchar>;
|
|
|
|
using FU8StringView = TStringView<u8char>;
|
|
|
|
using FU16StringView = TStringView<u16char>;
|
|
|
|
using FU32StringView = TStringView<u32char>;
|
|
|
|
using FUnicodeStringView = TStringView<unicodechar>;
|
|
|
|
|
2024-11-03 15:56:38 +08:00
|
|
|
#define TEXT_VIEW(X) TStringView(TEXT(X))
|
|
|
|
#define WTEXT_VIEW(X) TStringView(WTEXT(X))
|
|
|
|
#define U8TEXT_VIEW(X) TStringView(U8TEXT(X))
|
|
|
|
#define U16TEXT_VIEW(X) TStringView(U16TEXT(X))
|
|
|
|
#define U32TEXT_VIEW(X) TStringView(U32TEXT(X))
|
|
|
|
#define UNICODETEXT_VIEW(X) TStringView(UNICODETEXT(X))
|
|
|
|
|
|
|
|
#define LITERAL_VIEW(T, X) TStringView(LITERAL(T, X))
|
|
|
|
|
2024-10-27 11:29:45 +08:00
|
|
|
NAMESPACE_MODULE_END(Utility)
|
|
|
|
NAMESPACE_MODULE_END(Redcraft)
|
|
|
|
NAMESPACE_REDCRAFT_END
|