#pragma once #include "CoreTypes.h" #include "String/Char.h" #include "Containers/Array.h" #include "String/StringView.h" #include "Templates/Utility.h" #include "Templates/Optional.h" #include "Templates/Container.h" #include "Containers/Iterator.h" #include "TypeTraits/TypeTraits.h" #include "Templates/Noncopyable.h" #include "Miscellaneous/AssertionMacros.h" NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) NAMESPACE_PRIVATE_BEGIN template class TCStringFromTString final : FNoncopyable { public: FORCEINLINE TCStringFromTString(const T* InPtr, bool bInDelete) : Ptr(InPtr), bDelete(bInDelete) { } FORCEINLINE TCStringFromTString(TCStringFromTString&& InValue) : Ptr(InValue.Ptr), bDelete(Exchange(InValue.bDelete, false)) { } FORCEINLINE ~TCStringFromTString() { if (bDelete) delete[] Ptr; } FORCEINLINE TCStringFromTString& operator=(TCStringFromTString&& InValue) { if (bDelete) delete[] Ptr; Ptr = InValue.Ptr; bDelete = Exchange(InValue.bDelete, false); return *this; } NODISCARD FORCEINLINE operator const T*() const { return Ptr; } private: const T* Ptr; bool bDelete; }; NAMESPACE_PRIVATE_END /** The default string allocator that uses SSO and can be placed right into FAny without dynamically allocating memory. */ template using TDefaultStringAllocator = TInlineAllocator<(40 - 3 * sizeof(size_t)) / sizeof(T)>; /** A string class that stores and manipulates sequences of characters. Unlike std::basic_string, it is not null-terminated. */ template Allocator = TDefaultStringAllocator> class TString : public TArray { private: using Super = TArray; public: using ElementType = typename Super::ElementType; using AllocatorType = typename Super::AllocatorType; using Reference = typename Super:: Reference; using ConstReference = typename Super::ConstReference; using Iterator = typename Super:: Iterator; using ConstIterator = typename Super::ConstIterator; using ReverseIterator = typename Super:: ReverseIterator; using ConstReverseIterator = typename Super::ConstReverseIterator; static_assert(CContiguousIterator< Iterator>); static_assert(CContiguousIterator); /** Default constructor. Constructs an empty string. */ FORCEINLINE TString() = default; /** Constructs the string with 'Count' copies of characters with 'InValue'. */ FORCEINLINE TString(size_t Count, ElementType InChar) : Super(Count, InChar) { } /** Constructs a string with the contents of the range ['InPtr', 'InPtr' + 'Count'). */ FORCEINLINE TString(const ElementType* InPtr, size_t Count) : TString(TStringView(InPtr, Count)) { checkf(InPtr != nullptr, TEXT("TString cannot be initialized by nullptr. Please check the pointer.")); } FORCEINLINE TString(nullptr_t, size_t) = delete; /** Constructs a string with the contents of the range ['InPtr', '\0'). */ FORCEINLINE TString(const ElementType* InPtr) : TString(TStringView(InPtr)) { checkf(InPtr != nullptr, TEXT("TString cannot be initialized by nullptr. Please check the pointer.")); } FORCEINLINE TString(nullptr_t) = delete; /** Constructs the string with the contents of the 'View'. */ FORCEINLINE TString(TStringView View) : TString(View.Begin(), View.End()) { } /** Constructs the string with the contents of the range ['First', 'Last'). */ template S> requires (CConstructibleFrom>) FORCEINLINE TString(I First, S Last) : Super(MoveTemp(First), MoveTemp(Last)) { } /** Copy constructor. Constructs the string with the copy of the contents of 'InValue'. */ FORCEINLINE TString(const TString&) = default; /** Move constructor. After the move, 'InValue' is guaranteed to be empty. */ FORCEINLINE TString(TString&&) = default; /** Constructs the string with the contents of the initializer list. */ FORCEINLINE TString(initializer_list IL) : TString(Iteration::Begin(IL), Iteration::End(IL)) { } /** Copy assignment operator. Replaces the contents with a copy of the contents of 'InValue'. */ FORCEINLINE TString& operator=(const TString&) = default; /** Move assignment operator. After the move, 'InValue' is guaranteed to be empty. */ FORCEINLINE TString& operator=(TString&&) = default; /** Compares the contents of two strings. */ NODISCARD friend FORCEINLINE bool operator==(const TString& LHS, const TString& RHS) { return TStringView(LHS) == TStringView(RHS); } /** Compares the contents of a string and a character. */ NODISCARD friend FORCEINLINE bool operator==(const TString& LHS, ElementType RHS) { return TStringView(LHS) == RHS; } NODISCARD friend FORCEINLINE bool operator==(const TString& LHS, const ElementType* RHS) { return TStringView(LHS) == RHS; } NODISCARD friend FORCEINLINE bool operator==( ElementType LHS, const TString& RHS) { return LHS == TStringView(RHS); } NODISCARD friend FORCEINLINE bool operator==(const ElementType* LHS, const TString& RHS) { return LHS == TStringView(RHS); } /** Compares the contents of 'LHS' and 'RHS' lexicographically. */ NODISCARD friend FORCEINLINE auto operator<=>(const TString& LHS, const TString& RHS) { return TStringView(LHS) <=> TStringView(RHS); } /** Compares the contents of 'LHS' and 'RHS' lexicographically. */ NODISCARD friend FORCEINLINE auto operator<=>(const TString& LHS, ElementType RHS) { return TStringView(LHS) <=> RHS; } NODISCARD friend FORCEINLINE auto operator<=>(const TString& LHS, const ElementType* RHS) { return TStringView(LHS) <=> RHS; } NODISCARD friend FORCEINLINE auto operator<=>( ElementType LHS, const TString& RHS) { return LHS <=> TStringView(RHS); } NODISCARD friend FORCEINLINE auto operator<=>(const ElementType* LHS, const TString& RHS) { return LHS <=> TStringView(RHS); } public: /** Inserts 'InValue' before 'Index' in the string. */ FORCEINLINE Iterator Insert(size_t Index, ElementType InValue) { checkf(Index <= this->Num(), TEXT("Illegal index. Please check Index <= Num().")); return Insert(this->Begin() + Index, InValue); } /** Inserts 'InValue' before 'Iter' in the string. */ FORCEINLINE Iterator Insert(ConstIterator Iter, ElementType InValue) { checkf(this->IsValidIterator(Iter), TEXT("Read access violation. Please check IsValidIterator().")); return Super::Insert(Iter, InValue); } /** Inserts 'Count' copies of the 'InValue' before 'Index' in the string. */ FORCEINLINE Iterator Insert(size_t Index, size_t Count, ElementType InValue) { checkf(Index <= this->Num(), TEXT("Illegal index. Please check Index <= Num().")); return Insert(this->Begin() + Index, Count, InValue); } /** Inserts 'Count' copies of the 'InValue' before 'Iter' in the string. */ FORCEINLINE Iterator Insert(ConstIterator Iter, size_t Count, ElementType InValue) { checkf(this->IsValidIterator(Iter), TEXT("Read access violation. Please check IsValidIterator().")); return Super::Insert(Iter, Count, InValue); } /** Inserts characters from the 'View' before 'Index' in the string. */ FORCEINLINE Iterator Insert(size_t Index, TStringView View) { checkf(Index <= this->Num(), TEXT("Illegal index. Please check Index <= Num().")); return Insert(this->Begin() + Index, View); } /** Inserts characters from the 'View' before 'Iter' in the string. */ FORCEINLINE Iterator Insert(ConstIterator Iter, TStringView View) { checkf(this->IsValidIterator(Iter), TEXT("Read access violation. Please check IsValidIterator().")); return Insert(Iter, View.Begin(), View.End()); } /** Inserts characters from the range ['First', 'Last') before 'Index' in the string. */ template S> requires (CConstructibleFrom>) FORCEINLINE Iterator Insert(size_t Index, I First, S Last) { checkf(Index <= this->Num(), TEXT("Illegal index. Please check Index <= Num().")); return Insert(this->Begin() + Index, MoveTemp(First), MoveTemp(Last)); } /** Inserts characters from the range ['First', 'Last') before 'Iter'. */ template S> requires (CConstructibleFrom>) FORCEINLINE Iterator Insert(ConstIterator Iter, I First, S Last) { checkf(this->IsValidIterator(Iter), TEXT("Read access violation. Please check IsValidIterator().")); return Super::Insert(Iter, MoveTemp(First), MoveTemp(Last)); } /** Inserts characters from the initializer list before 'Index' in the string. */ FORCEINLINE Iterator Insert(size_t Index, initializer_list IL) { checkf(Index <= this->Num(), TEXT("Illegal index. Please check Index <= Num().")); return Insert(this->Begin() + Index, IL); } /** Inserts characters from the initializer list before 'Iter' in the string. */ FORCEINLINE Iterator Insert(ConstIterator Iter, initializer_list IL) { checkf(this->IsValidIterator(Iter), TEXT("Read access violation. Please check IsValidIterator().")); return Super::Insert(Iter, IL); } /** Erases the character at 'Index' in the string. But it may change the order of characters. */ FORCEINLINE Iterator Erase(size_t Index, bool bAllowShrinking = true) { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index < Num().")); return Erase(this->Begin() + Index, bAllowShrinking); } /** Erases the character at 'Iter' in the string. But it may change the order of characters. */ FORCEINLINE Iterator Erase(ConstIterator Iter, bool bAllowShrinking = true) { checkf(this->IsValidIterator(Iter) && Iter != this->End(), TEXT("Read access violation. Please check IsValidIterator().")); return Super::StableErase(Iter, bAllowShrinking); } /** Erases 'CountToErase' characters starting from 'Index' in the string. But it may change the order of characters. */ FORCEINLINE Iterator Erase(size_t Index, size_t CountToErase, bool bAllowShrinking = true) { checkf(Index <= this->Num() && Index + CountToErase <= this->Num(), TEXT("Illegal substring range. Please check Index and CountToErase.")); return Erase(this->Begin() + Index, this->Begin() + Index + CountToErase, bAllowShrinking); } /** Erases the characters in the range ['First', 'Last') in the string. But it may change the order of characters. */ FORCEINLINE Iterator Erase(ConstIterator First, ConstIterator Last, bool bAllowShrinking = true) { checkf(this->IsValidIterator(First) && this->IsValidIterator(Last) && First <= Last, TEXT("Read access violation. Please check IsValidIterator().")); return Super::StableErase(First, Last, bAllowShrinking); } /** Here, the 'Erase' is already stable and there is no need to provide 'StableErase'. */ void StableErase(...) = delete; /** Appends 'Count' copies of the 'InValue' to the end of the string. */ TString& Append(size_t Count, ElementType InChar) { return Append(MakeCountedConstantIterator(InChar, Count), DefaultSentinel); } /** Appends the contents of the 'View' to the end of the string. */ FORCEINLINE TString& Append(TStringView View) { return Append(View.Begin(), View.End()); } /** Appends the contents of the range ['First', 'Last') to the end of the string. */ template S> requires (CConstructibleFrom>) TString& Append(I First, S Last) { if constexpr (CForwardIterator) { if constexpr (CSizedSentinelFor) { checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); } const size_t Count = Iteration::Distance(First, Last); const size_t CurrentNum = this->Num(); this->SetNum(CurrentNum + Count); for (size_t Index = CurrentNum; Index != CurrentNum + Count; ++Index) { (*this)[Index] = ElementType(*First++); } } else { Insert(this->End(), MoveTemp(First), MoveTemp(Last)); } return *this; } /** Appends the contents of the initializer list to the end of the string. */ FORCEINLINE TString& Append(initializer_list IL) { return Append(Iteration::Begin(IL), Iteration::End(IL)); } /** Appends the given character value to the end of the string. */ FORCEINLINE TString& operator+=(ElementType InChar) { return Append(1, InChar); } /** Appends the contents of the 'View' to the end of the string. */ FORCEINLINE TString& operator+=(TStringView View) { return Append(View); } /** Appends the contents of the range ['First', 'Last') to the end of the string. */ FORCEINLINE TString& operator+=(initializer_list IL) { return Append(IL); } /** Concatenates two strings. */ NODISCARD friend FORCEINLINE TString operator+(const TString& LHS, const TString& RHS) { return TString(LHS).Append(RHS); } /** Concatenates the string with the given character value. */ NODISCARD friend FORCEINLINE TString operator+(const TString& LHS, ElementType RHS) { return TString(LHS).Append(1, RHS); } NODISCARD friend FORCEINLINE TString operator+(const TString& LHS, const ElementType* RHS) { return TString(LHS).Append( RHS); } NODISCARD friend FORCEINLINE TString operator+(const TString& LHS, TStringView RHS) { return TString(LHS).Append( RHS); } NODISCARD friend FORCEINLINE TString operator+( ElementType LHS, const TString& RHS) { return TString(1, LHS).Append(RHS); } NODISCARD friend FORCEINLINE TString operator+( const ElementType* LHS, const TString& RHS) { return TString( LHS).Append(RHS); } NODISCARD friend FORCEINLINE TString operator+(TStringView LHS, const TString& RHS) { return TString( LHS).Append(RHS); } /** Concatenates two strings. The rvalue maybe modified. */ NODISCARD friend FORCEINLINE TString operator+(TString&& LHS, TString&& RHS) { LHS.Append(MoveTemp(RHS)); return LHS; } /** Concatenates two strings. The rvalue maybe modified. */ NODISCARD friend FORCEINLINE TString operator+(TString&& LHS, ElementType RHS) { LHS.Append(1, RHS); return LHS; } NODISCARD friend FORCEINLINE TString operator+(TString&& LHS, const ElementType* RHS) { LHS.Append( RHS); return LHS; } NODISCARD friend FORCEINLINE TString operator+(TString&& LHS, TStringView RHS) { LHS.Append( RHS); return LHS; } NODISCARD friend FORCEINLINE TString operator+(TString&& LHS, const TString& RHS) { LHS.Append( RHS); return LHS; } NODISCARD friend FORCEINLINE TString operator+( ElementType LHS, TString&& RHS) { RHS.Insert(0, LHS); return RHS; } NODISCARD friend FORCEINLINE TString operator+( const ElementType* LHS, TString&& RHS) { RHS.Insert(0, LHS); return RHS; } NODISCARD friend FORCEINLINE TString operator+( TStringView LHS, TString&& RHS) { RHS.Insert(0, LHS); return RHS; } NODISCARD friend FORCEINLINE TString operator+(const TString& LHS, TString&& RHS) { RHS.Insert(0, LHS); return RHS; } public: /** @return true if the string view starts with the given prefix, false otherwise. */ NODISCARD FORCEINLINE bool StartsWith(TStringView Prefix) const { return TStringView(*this).StartsWith(Prefix); } /** @return true if the string view starts with the given prefix, false otherwise. */ NODISCARD FORCEINLINE bool StartsWith(ElementType Prefix) const { return TStringView(*this).StartsWith(Prefix); } /** @return true if the string view ends with the given suffix, false otherwise. */ NODISCARD FORCEINLINE bool EndsWith(TStringView Suffix) const { return TStringView(*this).EndsWith(Suffix); } /** @return true if the string view ends with the given suffix, false otherwise. */ NODISCARD FORCEINLINE bool EndsWith(ElementType Suffix) const { return TStringView(*this).EndsWith(Suffix); } /** @return true if the string view contains the given substring, false otherwise. */ NODISCARD FORCEINLINE bool Contains(TStringView View) const { return TStringView(*this).Contains(View); } /** @return true if the string view contains the given character, false otherwise. */ NODISCARD FORCEINLINE bool Contains(ElementType Char) const { return TStringView(*this).Contains(Char); } /** @return true if the string view contains character that satisfy the given predicate, false otherwise. */ template F> NODISCARD FORCEINLINE bool Contains(F&& InPredicate) const { return TStringView(*this).Contains(Forward(InPredicate)); } /** Replace the substring [Index, Index + CountToReplace) with 'Count' copies of the 'InChar'. */ FORCEINLINE TString& Replace(size_t Index, size_t CountToReplace, size_t Count, ElementType InChar) { checkf(Index <= this->Num() && Index + CountToReplace <= this->Num(), TEXT("Illegal substring range. Please check Index and CountToReplace.")); return Replace(this->Begin() + Index, this->Begin() + Index + CountToReplace, Count, InChar); } /** Replace the substring ['First', 'Last') with 'Count' copies of the 'InChar'. */ FORCEINLINE TString& Replace(ConstIterator First, ConstIterator Last, size_t Count, ElementType InChar) { checkf(this->IsValidIterator(First) && this->IsValidIterator(Last) && First <= Last, TEXT("Read access violation. Please check IsValidIterator().")); return Replace(First, Last, MakeCountedConstantIterator(InChar, Count), DefaultSentinel); } /** Replace the substring [Index, Index + CountToReplace) with the contents of the 'View'. */ FORCEINLINE TString& Replace(size_t Index, size_t CountToReplace, TStringView View) { checkf(Index <= this->Num() && Index + CountToReplace <= this->Num(), TEXT("Illegal substring range. Please check Index and CountToReplace.")); return Replace(this->Begin() + Index, this->Begin() + Index + CountToReplace, View); } /** Replace the substring ['First', 'Last') with the contents of the 'View'. */ FORCEINLINE TString& Replace(ConstIterator First, ConstIterator Last, TStringView View) { checkf(this->IsValidIterator(First) && this->IsValidIterator(Last) && First <= Last, TEXT("Read access violation. Please check IsValidIterator().")); return Replace(First, Last, View.Begin(), View.End()); } /** Replace the substring [Index, Index + CountToReplace) with the contents of the range ['First', 'Last'). */ template S> requires (CConstructibleFrom>) FORCEINLINE TString& Replace(size_t Index, size_t CountToReplace, I InString, S Sentinel) { checkf(Index <= this->Num() && Index + CountToReplace <= this->Num(), TEXT("Illegal substring range. Please check Index and CountToReplace.")); return Replace(this->Begin() + Index, this->Begin() + Index + CountToReplace, MoveTemp(InString), MoveTemp(Sentinel)); } /** Replace the substring ['First', 'Last') with the contents of the range ['InString', 'Sentinel'). */ template S> requires (CConstructibleFrom>) TString& Replace(ConstIterator First, ConstIterator Last, I InString, S Sentinel) { checkf(this->IsValidIterator(First) && this->IsValidIterator(Last) && First <= Last, TEXT("Read access violation. Please check IsValidIterator().")); if constexpr (CForwardIterator) { if (CSizedSentinelFor) { checkf(First <= Last, TEXT("Illegal range iterator. Please check First <= Last.")); } const size_t InsertIndex = First - this->Begin(); const size_t RemoveCount = Iteration::Distance(First, Last); const size_t InsertCount = Iteration::Distance(InString, Sentinel); const size_t NumToReset = this->Num() - RemoveCount + InsertCount; if (InsertCount < RemoveCount) { for (size_t Index = InsertIndex; Index != InsertIndex + InsertCount; ++Index) { (*this)[Index] = ElementType(*InString++); } for (size_t Index = InsertIndex + InsertCount; Index != NumToReset; ++Index) { (*this)[Index] = (*this)[Index + (RemoveCount - InsertCount)]; } this->SetNum(NumToReset); } else { this->SetNum(NumToReset); for (size_t Index = this->Num(); Index != InsertIndex + InsertCount - 1; --Index) { (*this)[Index - 1] = (*this)[Index + (RemoveCount - InsertCount) - 1]; } for (size_t Index = InsertIndex; Index != InsertIndex + InsertCount; ++Index) { (*this)[Index] = ElementType(*InString++); } } } else { TString Temp(MoveTemp(First), MoveTemp(Last)); return Replace(First, Last, Temp.Begin(), Temp.End()); } return *this; } /** Replace the substring [Index, Index + CountToReplace) with the contents of the initializer list. */ FORCEINLINE TString& Replace(size_t Index, size_t CountToReplace, initializer_list IL) { checkf(Index <= this->Num() && Index + CountToReplace <= this->Num(), TEXT("Illegal substring range. Please check Index and CountToReplace.")); return Replace(this->Begin() + Index, this->Begin() + Index + CountToReplace, IL); } /** Replace the substring ['First', 'Last') with the contents of the initializer list. */ FORCEINLINE TString& Replace(ConstIterator First, ConstIterator Last, initializer_list IL) { checkf(this->IsValidIterator(First) && this->IsValidIterator(Last) && First <= Last, TEXT("Read access violation. Please check IsValidIterator().")); return Replace(First, Last, Iteration::Begin(IL), Iteration::End(IL)); } /** Obtains a string that is a view over the 'Count' characters of this string view starting at 'Offset'. */ FORCEINLINE TString Substr(size_t Offset = 0, size_t Count = DynamicExtent) const { checkf(Offset <= this->Num() && Offset + Count <= this->Num(), TEXT("Illegal substring range. Please check Offset and Count.")); return TStringView(*this).Substr(Offset, Count); } /** Copies the characters of this string to the destination buffer without null-termination. */ FORCEINLINE size_t Copy(ElementType* Dest, size_t Count = DynamicExtent, size_t Offset = 0) const { checkf(Dest != nullptr, TEXT("Illegal destination buffer. Please check the pointer.")); checkf(Offset <= this->Num() && (Count == DynamicExtent || Offset + Count <= this->Num()), TEXT("Illegal subview range. Please check Offset and Count.")); return TStringView(*this).Copy(Dest, Count, Offset); } FORCEINLINE size_t Copy(nullptr_t, size_t = DynamicExtent, size_t = 0) const = delete; /** @return The index of the first occurrence of the given substring, or INDEX_NONE if not found. */ NODISCARD size_t Find(TStringView View, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).Find(View, Index); } /** @return The index of the first occurrence of the given character, or INDEX_NONE if not found. */ NODISCARD size_t Find(ElementType Char, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).Find(Char, Index); } /** @return The index of the first occurrence of the character that satisfy the given predicate, or INDEX_NONE if not found. */ template F> NODISCARD size_t Find(F&& InPredicate, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).Find(Forward(InPredicate), Index); } /** @return The index of the last occurrence of the given substring, or INDEX_NONE if not found. */ NODISCARD size_t RFind(TStringView View, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).RFind(View, Index); } /** @return The index of the last occurrence of the given character, or INDEX_NONE if not found. */ NODISCARD size_t RFind(ElementType Char, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).RFind(Char, Index); } /** @return The index of the last occurrence of the character that satisfy the given predicate, or INDEX_NONE if not found. */ template F> NODISCARD size_t RFind(F&& InPredicate, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).RFind(Forward(InPredicate), Index); } /** @return The index of the first occurrence of the character contained in the given view, or INDEX_NONE if not found. */ NODISCARD FORCEINLINE size_t FindFirstOf(TStringView View, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindFirstOf(View, Index); } /** @return The index of the first occurrence of the given character, or INDEX_NONE if not found. */ NODISCARD FORCEINLINE size_t FindFirstOf(ElementType Char, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindFirstOf(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 size_t FindLastOf(TStringView View, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindLastOf(View, Index); } /** @return The index of the last occurrence of the given character, or INDEX_NONE if not found. */ NODISCARD FORCEINLINE size_t FindLastOf(ElementType Char, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindLastOf(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 size_t FindFirstNotOf(TStringView View, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindFirstNotOf(View, Index); } /** @return The index of the first absence of the given character, or INDEX_NONE if not found. */ NODISCARD FORCEINLINE size_t FindFirstNotOf(ElementType Char, size_t Index = 0) const { checkf(Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindFirstNotOf(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 size_t FindLastNotOf(TStringView View, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindLastNotOf(View, Index); } /** @return The index of the last absence of the given character, or INDEX_NONE if not found. */ NODISCARD FORCEINLINE size_t FindLastNotOf(ElementType Char, size_t Index = INDEX_NONE) const { checkf(Index == INDEX_NONE || Index < this->Num(), TEXT("Illegal index. Please check Index.")); return TStringView(*this).FindLastNotOf(Char, Index); } public: /** Try to decode the given character using the U-encoded to a string using the T-encoded. */ template bool DecodeFrom(U Char, bool bAllowShrinking = true) { return DecodeFrom(TStringView(&Char, 1), bAllowShrinking); } /** Try to decode the given string using the U-encoded to a string using the T-encoded. */ template A> bool DecodeFrom(const TString& String, bool bAllowShrinking = true) { return DecodeFrom(TStringView(String), bAllowShrinking); } /** Try to decode the given string view using the U-encoded to a string using the T-encoded. */ template bool DecodeFrom(TStringView View, bool bAllowShrinking = true) { this->Reset(false); auto AppendToResult = [this](auto& Self, TStringView View) -> bool { // char -> char // wchar -> wchar if constexpr (CSameAs && CSameAs || CSameAs && CSameAs) { // Unable to determine whether the user-preferred locale encoded character is valid or not, it is assumed to be valid. Insert(this->End(), View.Begin(), View.End()); return true; } // char -> wchar // char -> wchar -> ... else if constexpr (CSameAs) { NAMESPACE_STD::locale Loc = NAMESPACE_STD::locale(""); check((NAMESPACE_STD::has_facet>(Loc))); const auto& Facet = NAMESPACE_STD::use_facet>(Loc); NAMESPACE_STD::mbstate_t State = NAMESPACE_STD::mbstate_t(); const char* BeginFrom = ToAddress(View.Begin()); const char* EndFrom = ToAddress(View.End()); wchar Buffer[FWChar::MaxCodeUnitLength]; const char* NextFrom; wchar* NextTo; do { const auto Result = Facet.in(State, BeginFrom, EndFrom, NextFrom, Iteration::Begin(Buffer), Iteration::End(Buffer), NextTo); if (BeginFrom == NextFrom) return false; if (Result == NAMESPACE_STD::codecvt_base::error) return false; if (Result == NAMESPACE_STD::codecvt_base::noconv) return false; // char -> wchar if constexpr (CSameAs) { for (wchar* Iter = Buffer; Iter != NextTo; ++Iter) { this->PushBack(*Iter); } } else { if (!Self(Self, TStringView(Buffer, NextTo))) return false; } BeginFrom = NextFrom; } while (BeginFrom != EndFrom); return true; } // wchar -> char else if constexpr (CSameAs && CSameAs) { NAMESPACE_STD::locale Loc = NAMESPACE_STD::locale(""); check((NAMESPACE_STD::has_facet>(Loc))); const auto& Facet = NAMESPACE_STD::use_facet>(Loc); NAMESPACE_STD::mbstate_t State = NAMESPACE_STD::mbstate_t(); const wchar* BeginFrom = ToAddress(View.Begin()); const wchar* EndFrom = ToAddress(View.End()); char Buffer[FChar::MaxCodeUnitLength]; const wchar* NextFrom; char* NextTo; do { const auto Result = Facet.out(State, BeginFrom, EndFrom, NextFrom, Iteration::Begin(Buffer), Iteration::End(Buffer), NextTo); if (BeginFrom == NextFrom) return false; if (Result == NAMESPACE_STD::codecvt_base::error) return false; if (Result == NAMESPACE_STD::codecvt_base::noconv) return false; for (char* Iter = Buffer; Iter != NextTo; ++Iter) { this->PushBack(*Iter); } BeginFrom = NextFrom; } while (BeginFrom != EndFrom); return true; } // u8char -> unicodechar -> ... else if constexpr (CSameAs) { auto Iter = View.Begin(); while (Iter != View.End()) { unicodechar Temp = static_cast(*Iter++); unicodechar Unicode; if ((Temp & 0b10000000) == 0b00000000) // 0XXXXXXX { Unicode = Temp; } else if ((Temp & 0b11100000) == 0b11000000) // 110XXXXX 10XXXXXX { if (Iter + 1 > View.End()) return false; Unicode = (Temp & 0b00011111) << 6; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= Temp & 0b00111111; } else if ((Temp & 0b11110000) == 0b11100000) // 1110XXXX 10XXXXXX 10XXXXXX { if (Iter + 2 > View.End()) return false; Unicode = (Temp & 0b00001111) << 12; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= (Temp & 0b00111111) << 6; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= Temp & 0b00111111; } else if ((Temp & 0b11111000) == 0b11110000) // 11110XXX 10XXXXXX 10XXXXXX 10XXXXXX { if (Iter + 3 > View.End()) return false; Unicode = (Temp & 0b00000111) << 18; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= (Temp & 0b00111111) << 12; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= (Temp & 0b00111111) << 6; Temp = static_cast(*Iter++); if ((Temp & 0b11000000) != 0b10000000) return false; else Unicode |= Temp & 0b00111111; } else return false; if (!Self(Self, TStringView(&Unicode, 1))) return false; } return true; } // u16char -> unicodechar -> ... // wchar -> unicodechar -> ... for Windows else if constexpr (CSameAs || PLATFORM_WINDOWS && CSameAs) { auto Iter = View.Begin(); while (Iter != View.End()) { unicodechar Temp = static_cast(*Iter++); unicodechar Unicode; // High Surrogate ..; // Low Surrogate ..; if (Temp >= 0xD800 && Temp <= 0xDBFF) { if (Iter == View.End()) return false; Unicode = (Temp & 0b00000011'11111111) << 10; Temp = static_cast(*Iter++); if (Temp >= 0xDC00 && Temp <= 0xDFFF) { Unicode |= Temp & 0b00000011'11111111; Unicode += 0x10000; } else return false; } else Unicode = Temp; if (!Self(Self, TStringView(&Unicode, 1))) return false; } return true; } // wchar -> unicodechar -> ... for Linux else if constexpr (PLATFORM_LINUX && CSameAs) { return Self(Self, TStringView(reinterpret_cast(View.GetData()), View.Num())); } // unicodechar u32char -> u8char else if constexpr (CSameAs && CSameAs) { for (unicodechar Char : View) { if (!FUnicodeChar::IsValid(Char)) return false; if (!(Char & ~0b0000000'00000000'00000000'01111111)) // 0XXXXXXX { this->PushBack(static_cast(Char)); } else if (!(Char & ~0b0000000'00000000'00000111'11111111)) // 110XXXXX 10XXXXXX { this->PushBack(static_cast(0b11000000 | (Char >> 6 & 0b00011111))); this->PushBack(static_cast(0b10000000 | (Char & 0b00111111))); } else if (!(Char & ~0b0000000'00000000'11111111'11111111)) // 1110XXXX 10XXXXXX 10XXXXXX { this->PushBack(static_cast(0b11100000 | (Char >> 12 & 0b00001111))); this->PushBack(static_cast(0b10000000 | (Char >> 6 & 0b00111111))); this->PushBack(static_cast(0b10000000 | (Char & 0b00111111))); } else if (!(Char & ~0b0000000'11111111'11111111'11111111)) // 11110XXX 10XXXXXX 10XXXXXX 10XXXXXX { this->PushBack(static_cast(0b11110000 | (Char >> 18 & 0b00000111))); this->PushBack(static_cast(0b10000000 | (Char >> 12 & 0b00111111))); this->PushBack(static_cast(0b10000000 | (Char >> 6 & 0b00111111))); this->PushBack(static_cast(0b10000000 | (Char & 0b00111111))); } else check_no_entry(); } return true; } // unicodechar u32char -> u16char // unicodechar u32char -> wchar for Windows // unicodechar u32char -> wchar -> char for Windows else if constexpr (CSameAs && (CSameAs || PLATFORM_WINDOWS && (CSameAs || CSameAs))) { for (unicodechar Char : View) { if (!FUnicodeChar::IsValid(Char)) return false; if (!(Char & ~0b0000000'00000000'11111111'11111111)) // XXXXXXXX'XXXXXXXX { if constexpr (PLATFORM_WINDOWS && (CSameAs || CSameAs)) { wchar WChar = static_cast(Char); if (!Self(Self, TStringView(&WChar, 1))) return false; } else this->PushBack(static_cast(Char)); } else if (!(Char & ~0b0000000'00011111'11111111'11111111)) // 110110XX'XXXXXXXX 110111XX'XXXXXXXX { Char -= 0x10000; u16char Buffer[] = { static_cast(0b11011000'00000000 | (Char >> 10 & 0b00000011'11111111)), static_cast(0b11011100'00000000 | (Char & 0b00000011'11111111)) }; if constexpr (PLATFORM_WINDOWS && (CSameAs || CSameAs)) { if (!Self(Self, TStringView(reinterpret_cast(Buffer), 2))) return false; } else { this->PushBack(Buffer[0]); this->PushBack(Buffer[1]); } } else check_no_entry(); } return true; } // unicodechar u32char -> unicodechar u32char // unicodechar u32char -> wchar for Linux // unicodechar u32char -> wchar -> char for Linux else if constexpr (CSameAs && (CSameAs || PLATFORM_LINUX && (CSameAs || CSameAs))) { for (unicodechar Char : View) { if (!FUnicodeChar::IsValid(Char)) return false; } if constexpr (PLATFORM_LINUX && (CSameAs || CSameAs)) { return Self(Self, TStringView(reinterpret_cast(View.GetData()), View.Num())); } else Insert(this->End(), View.Begin(), View.End()); return true; } else static_assert(sizeof(W) == -1, "Unsupported character type"); return false; }; bool bIsValid = AppendToResult(AppendToResult, View); if (!bIsValid) this->Reset(bAllowShrinking); else if (bAllowShrinking) this->Shrink(); return bIsValid; } /** Try to encode a T-encoded string to a U-encoded string. */ template A = TDefaultStringAllocator> NODISCARD TOptional> EncodeTo() const { TString Result; bool bIsValid = Result.DecodeFrom(*this); if (!bIsValid) return Invalid; return Result; } /** @return The target-encoded string from the T-encoded string. */ NODISCARD FORCEINLINE auto ToString() const { return EncodeTo(); } NODISCARD FORCEINLINE auto ToWString() const { return EncodeTo(); } NODISCARD FORCEINLINE auto ToU8String() const { return EncodeTo(); } NODISCARD FORCEINLINE auto ToU16String() const { return EncodeTo(); } NODISCARD FORCEINLINE auto ToU32String() const { return EncodeTo(); } NODISCARD FORCEINLINE auto ToUnicodeString() const { return EncodeTo(); } /** @return The non-modifiable standard C character string version of the string. */ NODISCARD FORCEINLINE auto operator*() const& { if (this->Max() >= this->Num() + 1) { const_cast(this->GetData())[this->Num()] = LITERAL(ElementType, '\0'); return NAMESPACE_PRIVATE::TCStringFromTString(this->GetData(), false); } if (this->Back() == LITERAL(ElementType, '\0')) { return NAMESPACE_PRIVATE::TCStringFromTString(this->GetData(), false); } ElementType* Buffer = new ElementType[this->Num() + 1]; Copy(Buffer); Buffer[this->Num()] = LITERAL(ElementType, '\0'); return NAMESPACE_PRIVATE::TCStringFromTString(Buffer, true); } /** @return The non-modifiable standard C character string version of the string. */ NODISCARD FORCEINLINE auto operator*() && { if (this->Back() != LITERAL(T, '\0')) { this->PushBack(LITERAL(T, '\0')); } return AsConst(*this).GetData(); } public: /** @return true if the string only contains valid characters, false otherwise. */ NODISCARD FORCEINLINE constexpr bool IsValid() const { return TStringView(*this).IsValid(); } /** @return true if the string only contains ASCII characters, false otherwise. */ NODISCARD FORCEINLINE constexpr bool IsASCII() const { return TStringView(*this).IsASCII(); } /** @return true if the string only contains numeric characters, false otherwise. */ NODISCARD FORCEINLINE constexpr bool IsNumeric() const { return TStringView(*this).IsNumeric(); } /** @return true if the string only contains numeric characters, false otherwise. */ NODISCARD FORCEINLINE constexpr bool IsNumeric(unsigned Base) const { return TStringView(*this).IsNumeric(Base); } public: /** * Format some objects using a format string and append to the string. * * @param Fmt - The format string. * @param Args - The objects to format. * * @return The formatted string containing the objects. */ template NODISCARD static TString Format(TStringView Fmt, const Ts&... Args); /** * 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 size_t Parse(TStringView Fmt, Ts&... Args) const { return TStringView(*this).Parse(Fmt, Args...); } public: /** Overloads the GetTypeHash algorithm for TString. */ NODISCARD friend FORCEINLINE size_t GetTypeHash(const TString& A) { return GetTypeHash(TStringView(A)); } /** Overloads the Swap algorithm for TString. */ friend FORCEINLINE void Swap(TString& A, TString& B) { Swap(static_cast(A), static_cast(B)); } }; template TString(const T*) -> TString; template TString(TStringView) -> TString; template TString(I, S) -> TString>; template TString(initializer_list) -> TString; using FString = TString; using FWString = TString; using FU8String = TString; using FU16String = TString; using FU32String = TString; using FUnicodeString = TString; template template constexpr TStringView::TStringView(const TString& InString) : TStringView(InString.GetData(), InString.Num()) { } NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END #include "String/Conversion.h.inl"