#pragma once #include "CoreTypes.h" #include "TypeTraits/TypeTraits.h" #include "Miscellaneous/AssertionMacros.h" #include #include NAMESPACE_REDCRAFT_BEGIN NAMESPACE_MODULE_BEGIN(Redcraft) NAMESPACE_MODULE_BEGIN(Utility) template concept CCharType = CSameAs || CSameAs || CSameAs || CSameAs || CSameAs; NAMESPACE_PRIVATE_BEGIN template struct TLiteral; template <> struct TLiteral { NODISCARD FORCEINLINE static constexpr char Select(const char X, const wchar , const u8char , const u16char , const u32char ) { return X; } NODISCARD FORCEINLINE static constexpr const char* Select(const char* X, const wchar*, const u8char*, const u16char*, const u32char*) { return X; } }; template <> struct TLiteral { NODISCARD FORCEINLINE static constexpr wchar Select(const char , const wchar X, const u8char , const u16char , const u32char ) { return X; } NODISCARD FORCEINLINE static constexpr const wchar* Select(const char*, const wchar* X, const u8char*, const u16char*, const u32char*) { return X; } }; template <> struct TLiteral { NODISCARD FORCEINLINE static constexpr u8char Select(const char , const wchar , const u8char X, const u16char , const u32char ) { return X; } NODISCARD FORCEINLINE static constexpr const u8char* Select(const char*, const wchar*, const u8char* X, const u16char*, const u32char*) { return X; } }; template <> struct TLiteral { NODISCARD FORCEINLINE static constexpr u16char Select(const char , const wchar , const u8char , const u16char X, const u32char ) { return X; } NODISCARD FORCEINLINE static constexpr const u16char* Select(const char*, const wchar*, const u8char*, const u16char* X, const u32char*) { return X; } }; template <> struct TLiteral { NODISCARD FORCEINLINE static constexpr u32char Select(const char , const wchar , const u8char , const u16char , const u32char X) { return X; } NODISCARD FORCEINLINE static constexpr const u32char* Select(const char*, const wchar*, const u8char*, const u16char*, const u32char* X) { return X; } }; NAMESPACE_PRIVATE_END /** Templated literal struct to allow selection of string literals based on the character type provided, and not on compiler switches. */ #define LITERAL(CharType, StringLiteral) NAMESPACE_PRIVATE::TLiteral::Select(TEXT(StringLiteral), WTEXT(StringLiteral), U8TEXT(StringLiteral), U16TEXT(StringLiteral), U32TEXT(StringLiteral)) static_assert(CUnsigned, "TChar assumes u8char is an unsigned integer"); static_assert(CUnsigned, "TChar assumes u16char is an unsigned integer"); static_assert(CUnsigned, "TChar assumes u32char is an unsigned integer"); static_assert(CUnsigned, "TChar assumes unicodechar is an unsigned integer"); /** Set of utility functions operating on a single character. Implemented based on user-preferred locale and ISO 30112 "i18n". */ template struct TChar { using CharType = T; /** The maximum number of code units required to represent a single character. if unknown, guess 1. */ static constexpr size_t MaxCodeUnitLength = CSameAs ? MB_LEN_MAX : CSameAs ? PLATFORM_WINDOWS ? 2 : PLATFORM_LINUX ? 1 : 1 : CSameAs ? 4 : CSameAs ? 2 : CSameAs ? 1 : 1; /** Whether the character type is fixed-length. */ static constexpr bool bIsFixedLength = MaxCodeUnitLength == 1; NODISCARD FORCEINLINE static constexpr bool IsValid(CharType InChar) { if constexpr (CSameAs) { if ((InChar & 0b10000000) == 0b00000000) return true; return false; } else if constexpr (CSameAs || CSameAs) { if (InChar >= 0xD800 && InChar <= 0xDBFF) return false; if (InChar >= 0xDC00 && InChar <= 0xDFFF) return false; return InChar <= 0x10FFFF; } // Windows uses UTF-16 encoding for wchar. else if constexpr (PLATFORM_WINDOWS && (CSameAs)) { return TChar::IsValid(static_cast(InChar)); } // Linux uses UTF-32 encoding for wchar. else if constexpr (PLATFORM_LINUX && (CSameAs)) { return TChar::IsValid(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsNonch(CharType InChar) { if constexpr (CSameAs) { return false; } else if constexpr (CSameAs) { if (InChar >= U16TEXT('\uFDD0') && InChar <= U16TEXT('\uFDEF')) return true; if (InChar == U16TEXT('\uFFFE')) return true; if (InChar == U16TEXT('\uFFFF')) return true; return false; } else if constexpr (CSameAs) { if (InChar >= U32TEXT('\uFDD0') && InChar <= U32TEXT('\uFDEF')) return true; if ((InChar & 0x0000FFFE) == 0x0000FFFE) return TChar::IsValid(InChar); return false; } // Windows uses UTF-16 encoding for wchar. else if constexpr (PLATFORM_WINDOWS && (CSameAs)) { return TChar::IsNonch(static_cast(InChar)); } // Linux uses UTF-32 encoding for wchar. else if constexpr (PLATFORM_LINUX && (CSameAs)) { return TChar::IsNonch(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsASCII(CharType InChar = LITERAL(CharType, '\0')) { if constexpr (CSameAs) { constexpr bool ASCIICompatible = []() -> bool { constexpr char ASCIITable[] = TEXT("\u0000\u0001\u0002\u0003\u0004\u0005\u0006") TEXT("\a\b\t\n\v\f\r") TEXT("\u000E\u000F") TEXT("\u0010\u0011\u0012\u0013\u0014\u0015\u0016\u0017") TEXT("\u0018\u0019\u001A\u001B\u001C\u001D\u001E\u001F") TEXT(" !\"#$%&'()*+,-./0123456789:;<=>?") TEXT("@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_") TEXT("`abcdefghijklmnopqrstuvwxyz{|}~\u007F"); for (size_t Index = 0; Index <= 0x7F; ++Index) { if (ASCIITable[Index] != static_cast(Index)) return false; } return true; } (); return ASCIICompatible && 0x00 <= InChar && InChar <= 0x7F; } else if constexpr (CSameAs) { constexpr bool ASCIICompatible = []() -> bool { constexpr wchar ASCIITable[] = WTEXT("\u0000\u0001\u0002\u0003\u0004\u0005\u0006") WTEXT("\a\b\t\n\v\f\r") WTEXT("\u000E\u000F") WTEXT("\u0010\u0011\u0012\u0013\u0014\u0015\u0016\u0017") WTEXT("\u0018\u0019\u001A\u001B\u001C\u001D\u001E\u001F") WTEXT(" !\"#$%&'()*+,-./0123456789:;<=>?") WTEXT("@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_") WTEXT("`abcdefghijklmnopqrstuvwxyz{|}~\u007F"); for (size_t Index = 0; Index <= 0x7F; ++Index) { if (ASCIITable[Index] != static_cast(Index)) return false; } return true; } (); return ASCIICompatible && 0x00 <= InChar && InChar <= 0x7F; } else if constexpr (CSameAs || CSameAs || CSameAs || CSameAs) { return InChar <= 0x7F; } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsAlnum(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isalnum(InChar, Loc); } else { return TChar::IsAlpha(InChar) || TChar::IsDigit(InChar); } } NODISCARD FORCEINLINE static constexpr bool IsAlpha(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isalpha(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..;..; */ if ((InChar >= U8TEXT('\u0041') && InChar <= U8TEXT('\u005A')) || (InChar >= U8TEXT('\u0061') && InChar <= U8TEXT('\u007A'))) return true; return false; } else if constexpr (CSameAs || CSameAs) { checkf(InChar <= LITERAL(CharType, '\u007F'), TEXT("TChar::IsAlpha() only supports basic latin block.")); if (InChar > LITERAL(CharType, '\u007F')) return false; return TChar::IsAlpha(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsLower(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::islower(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..; */ if (InChar >= U8TEXT('\u0061') && InChar <= U8TEXT('\u007A')) return true; return false; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::IsLower() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return TChar::IsLower(static_cast(InChar)); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::IsLower() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return TChar::IsLower(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsUpper(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isupper(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..; */ if (InChar >= U8TEXT('\u0041') && InChar <= U8TEXT('\u005A')) return true; return false; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::IsUpper() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return TChar::IsUpper(static_cast(InChar)); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::IsUpper() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return TChar::IsUpper(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsDigit(CharType InChar) { static_assert(TChar::IsASCII()); /* ..; */ return InChar >= LITERAL(CharType, '0') && InChar <= LITERAL(CharType, '9'); } NODISCARD FORCEINLINE static constexpr bool IsDigit(CharType InChar, unsigned Base) { checkf(Base >= 2 && Base <= 36, TEXT("Base must be in the range [2, 36].")); static_assert(TChar::IsASCII()); bool bResult = false; /* ..; */ bResult |= InChar >= LITERAL(CharType, '0') && InChar < LITERAL(CharType, '0') + static_cast(Base); /* ..; */ bResult |= InChar >= LITERAL(CharType, 'a') && InChar < LITERAL(CharType, 'a') + static_cast(Base) - 10; /* ..; */ bResult |= InChar >= LITERAL(CharType, 'A') && InChar < LITERAL(CharType, 'A') + static_cast(Base) - 10; return bResult; } NODISCARD FORCEINLINE static constexpr bool IsCntrl(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::iscntrl(InChar, Loc); } else if constexpr (CSameAs) { /* ..;; */ return (InChar >= U8TEXT('\u0000') && InChar <= U8TEXT('\u001F')) || InChar == U8TEXT('\u007F'); } else if constexpr (CSameAs) { /* ..;..;;; */ return (InChar >= U16TEXT('\u0000') && InChar <= U16TEXT('\u001F')) || (InChar >= U16TEXT('\u007F') && InChar <= U16TEXT('\u009F')) || (InChar == U16TEXT('\u2028') || InChar == U16TEXT('\u2029')); } else if constexpr (CSameAs) { /* ..;..;;; */ return (InChar >= U32TEXT('\u0000') && InChar <= U32TEXT('\u001F')) || (InChar >= U32TEXT('\u007F') && InChar <= U32TEXT('\u009F')) || (InChar == U32TEXT('\u2028') || InChar == U32TEXT('\u2029')); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsGraph(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isgraph(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..; */ if (InChar >= U8TEXT('\u0021') && InChar <= U8TEXT('\u007E')) return true; return false; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::IsGraph() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return TChar::IsGraph(static_cast(InChar)); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::IsGraph() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return TChar::IsGraph(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsSpace(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isspace(InChar, Loc); } else if constexpr (CSameAs) { /* * ISO/IEC 6429 * ..; */ if (InChar >= U8TEXT('\u0009') && InChar <= U8TEXT('\u000D')) return true; /* * BASIC LATIN * ; */ if (InChar == U8TEXT('\u0020')) return true; return false; } else if constexpr (CSameAs) { /* * ISO/IEC 6429 * ..; */ if (InChar >= U16TEXT('\u0009') && InChar <= U16TEXT('\u000D')) return true; /* * BASIC LATIN * ; */ if (InChar == U16TEXT('\u0020')) return true; /* * OGHAM * ; */ if (InChar == U16TEXT('\u1680')) return true; /* * MONGOL * ; */ if (InChar == U16TEXT('\u180E')) return true; /* * GENERAL PUNCTUATION * ..;..;;;; */ if ((InChar >= U16TEXT('\u2000') && InChar <= U16TEXT('\u2006')) || (InChar >= U16TEXT('\u2008') && InChar <= U16TEXT('\u200A')) || (InChar == U16TEXT('\u2028') || InChar == U16TEXT('\u2029')) || (InChar == U16TEXT('\u205F'))) return true; /* * CJK SYMBOLS AND PUNCTUATION, HIRAGANA * ; */ if (InChar == U16TEXT('\u3000')) return true; return false; } else if constexpr (CSameAs) { /* * ISO/IEC 6429 * ..; */ if (InChar >= U32TEXT('\u0009') && InChar <= U32TEXT('\u000D')) return true; /* * BASIC LATIN * ; */ if (InChar == U32TEXT('\u0020')) return true; /* * OGHAM * ; */ if (InChar == U32TEXT('\u1680')) return true; /* * MONGOL * ; */ if (InChar == U32TEXT('\u180E')) return true; /* * GENERAL PUNCTUATION * ..;..;;;; */ if ((InChar >= U32TEXT('\u2000') && InChar <= U32TEXT('\u2006')) || (InChar >= U32TEXT('\u2008') && InChar <= U32TEXT('\u200A')) || (InChar == U32TEXT('\u2028') || InChar == U32TEXT('\u2029')) || (InChar == U32TEXT('\u205F'))) return true; /* * CJK SYMBOLS AND PUNCTUATION, HIRAGANA * ; */ if (InChar == U32TEXT('\u3000')) return true; return false; } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsBlank(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isblank(InChar, Loc); } else if constexpr (CSameAs) { /* ;; */ return InChar == U8TEXT('\u0009') || InChar == U8TEXT('\u0020'); } else if constexpr (CSameAs) { /* ;;;;..;..;;; */ return (InChar >= U16TEXT('\u2000') && InChar <= U16TEXT('\u2006')) || (InChar == U16TEXT('\u0009') || InChar == U16TEXT('\u0020')) || (InChar == U16TEXT('\u1680') || InChar == U16TEXT('\u180E')) || (InChar == U16TEXT('\u2008') || InChar == U16TEXT('\u200A')) || (InChar == U16TEXT('\u205F') || InChar == U16TEXT('\u3000')); } else if constexpr (CSameAs) { /* ;;;;..;..;;; */ return (InChar >= U32TEXT('\u2000') && InChar <= U32TEXT('\u2006')) || (InChar == U32TEXT('\u0009') || InChar == U32TEXT('\u0020')) || (InChar == U32TEXT('\u1680') || InChar == U32TEXT('\u180E')) || (InChar == U32TEXT('\u2008') || InChar == U32TEXT('\u200A')) || (InChar == U32TEXT('\u205F') || InChar == U32TEXT('\u3000')); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsPrint(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::isprint(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..; */ if (InChar >= U8TEXT('\u0020') && InChar <= U8TEXT('\u007E')) return true; return false; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::IsPrint() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return TChar::IsPrint(static_cast(InChar)); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::IsPrint() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return TChar::IsPrint(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr bool IsPunct(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return NAMESPACE_STD::ispunct(InChar, Loc); } else if constexpr (CSameAs) { /* * BASIC LATIN * ..;..;..;..; */ if ((InChar >= U8TEXT('\u0021') && InChar <= U8TEXT('\u002F')) || (InChar >= U8TEXT('\u003A') && InChar <= U8TEXT('\u0040')) || (InChar >= U8TEXT('\u005B') && InChar <= U8TEXT('\u0060')) || (InChar >= U8TEXT('\u007B') && InChar <= U8TEXT('\u007E'))) return true; return false; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::IsPunct() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return TChar::IsPunct(static_cast(InChar)); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::IsPunct() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return TChar::IsPunct(static_cast(InChar)); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return false; } NODISCARD FORCEINLINE static constexpr CharType ToLower(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return static_cast(NAMESPACE_STD::tolower(InChar, Loc)); } else if constexpr (CSameAs) { /* * BASIC LATIN * (,);(,);(,);(,); * (,);(,);(,);(,); * (,);(,);(,);(,); * (,);(,);(,);(,); * (,);(,);(,);(,); * (,);(,);(,);(,); * (,);(,); */ if (InChar >= U8TEXT('\u0041') && InChar <= U8TEXT('\u005A')) return InChar + U8TEXT('\u0020'); return InChar; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::ToLower() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return static_cast(TChar::ToLower(static_cast(InChar))); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::ToLower() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return static_cast(TChar::ToLower(static_cast(InChar))); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return InChar; } NODISCARD FORCEINLINE static constexpr CharType ToUpper(CharType InChar) { if constexpr (CSameAs || CSameAs) { NAMESPACE_STD::locale Loc(""); return static_cast(NAMESPACE_STD::toupper(InChar, Loc)); } else if constexpr (CSameAs) { /* * BASIC LATIN * (,);(,);(,);(,);/ * (,);(,);(,);(,);/ * (,);(,);(,);(,);/ * (,);(,);(,);(,);/ * (,);(,);(,);(,);/ * (,);(,);(,);(,);/ * (,);(,); */ if (InChar >= U8TEXT('\u0061') && InChar <= U8TEXT('\u007A')) return InChar - U8TEXT('\u0020'); return InChar; } else if constexpr (CSameAs) { checkf(InChar <= U16TEXT('\u007F'), TEXT("TChar::ToUpper() only supports basic latin block.")); if (InChar > U16TEXT('\u007F')) return false; return static_cast(TChar::ToUpper(static_cast(InChar))); } else if constexpr (CSameAs) { checkf(InChar <= U32TEXT('\u007F'), TEXT("TChar::ToUpper() only supports basic latin block.")); if (InChar > U32TEXT('\u007F')) return false; return static_cast(TChar::ToUpper(static_cast(InChar))); } else static_assert(sizeof(CharType) == -1, "Unsupported character type"); return InChar; } NODISCARD FORCEINLINE static constexpr unsigned ToDigit(CharType InChar) { static_assert(TChar::IsASCII()); constexpr uint8 DigitFromChar[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, }; static_assert(sizeof(DigitFromChar) == 256); if constexpr (sizeof(CharType) > 1) if (InChar >> 8) return DigitFromChar[0]; return DigitFromChar[InChar]; } NODISCARD FORCEINLINE static constexpr CharType FromDigit(unsigned InDigit) { checkf(InDigit < 36, TEXT("Digit must be in the range [0, 35].")); return LITERAL(CharType, "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ")[InDigit]; } NODISCARD FORCEINLINE static constexpr CharType FromDigit(unsigned InDigit, bool bLowercase) { checkf(InDigit < 36, TEXT("Digit must be in the range [0, 35].")); if (bLowercase) return LITERAL(CharType, "0123456789abcdefghijklmnopqrstuvwxyz")[InDigit]; return LITERAL(CharType, "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ")[InDigit]; } }; using FChar = TChar; using FWChar = TChar; using FU8Char = TChar; using FU16Char = TChar; using FU32Char = TChar; using FUnicodeChar = TChar; static_assert(FUnicodeChar::bIsFixedLength); NAMESPACE_MODULE_END(Utility) NAMESPACE_MODULE_END(Redcraft) NAMESPACE_REDCRAFT_END