using System; namespace PKHeX.Core; public static partial class Util { /// /// Parses the string into a 32-bit integer, skipping all characters except for valid digits. /// /// String to parse /// Parsed value public static int ToInt32(ReadOnlySpan value) { int result = 0; if (value.Length == 0) return result; bool negative = false; foreach (var c in value) { if (char.IsAsciiDigit(c)) { result *= 10; result += c; result -= '0'; } else if (c == '-' && result == 0) { negative = true; } } return negative ? -result : result; } /// /// Parses the string into a , skipping all characters except for valid digits. /// /// String to parse /// Parsed value public static uint ToUInt32(ReadOnlySpan value) { uint result = 0; if (value.Length == 0) return result; foreach (var c in value) { if (!char.IsAsciiDigit(c)) continue; result *= 10; result += (uint)(c - '0'); } return result; } /// /// Parses the hex string into a , skipping all characters except for valid digits. /// /// Hex String to parse /// Parsed value public static uint GetHexValue(ReadOnlySpan value) { uint result = 0; if (value.Length == 0) return result; foreach (var c in value) { if (char.IsAsciiDigit(c)) { result <<= 4; result += (uint)(c - '0'); } else if (char.IsAsciiHexDigitUpper(c)) { result <<= 4; result += (uint)(c - 'A' + 10); } else if (char.IsAsciiHexDigitLower(c)) { result <<= 4; result += (uint)(c - 'a' + 10); } } return result; } /// /// Parses the hex string into a , skipping all characters except for valid digits. /// /// Hex String to parse /// Parsed value public static ulong GetHexValue64(ReadOnlySpan value) { ulong result = 0; if (value.Length == 0) return result; foreach (var c in value) { if (char.IsAsciiDigit(c)) { result <<= 4; result += (uint)(c - '0'); } else if (char.IsAsciiHexDigitUpper(c)) { result <<= 4; result += (uint)(c - 'A' + 10); } else if (char.IsAsciiHexDigitLower(c)) { result <<= 4; result += (uint)(c - 'a' + 10); } } return result; } #if NET9_0_OR_GREATER REPLACE WITH TryFromHexString / TryToHexString #endif /// /// Parses a variable length hex string (non-spaced, bytes in reverse order). /// public static byte[] GetBytesFromHexString(ReadOnlySpan input) { byte[] result = new byte[input.Length / 2]; GetBytesFromHexString(input, result); return result; } /// public static void GetBytesFromHexString(ReadOnlySpan input, Span result) { for (int i = 0; i < result.Length; i++) { var slice = input.Slice(i * 2, 2); result[^(i + 1)] = (byte)GetHexValue(slice); } } private const string HexChars = "0123456789ABCDEF"; /// /// Converts the byte array into a hex string (non-spaced, bytes in reverse order). /// public static string GetHexStringFromBytes(ReadOnlySpan data) { System.Diagnostics.Debug.Assert(data.Length is (4 or 8 or 12 or 16)); Span result = stackalloc char[data.Length * 2]; GetHexStringFromBytes(data, result); return new string(result); } /// public static void GetHexStringFromBytes(ReadOnlySpan data, Span result) { if (result.Length != data.Length * 2) throw new ArgumentException("Result buffer must be twice the size of the input buffer."); for (int i = 0; i < data.Length; i++) { // Write tuples from the opposite side of the result buffer. var offset = (data.Length - i - 1) * 2; result[offset + 0] = HexChars[data[i] >> 4]; result[offset + 1] = HexChars[data[i] & 0xF]; } } /// /// Filters the string down to only valid hex characters, returning a new string. /// /// Input string to filter public static string GetOnlyHex(ReadOnlySpan str) { if (str.IsWhiteSpace()) return string.Empty; Span result = stackalloc char[str.Length]; int ctr = GetOnlyHex(str, result); return new string(result[..ctr]); } /// public static int GetOnlyHex(ReadOnlySpan str, Span result) { int ctr = 0; foreach (var c in str) { if (char.IsAsciiHexDigit(c)) result[ctr++] = c; } return ctr; } /// /// Returns a new string with each word converted to its appropriate title case. /// /// Input string to modify public static string ToTitleCase(ReadOnlySpan span) { if (span.IsEmpty) return string.Empty; Span result = stackalloc char[span.Length]; ToTitleCase(span, result); return new string(result); } /// public static void ToTitleCase(ReadOnlySpan span, Span result) { // Add each word to the string builder. Continue from the first index that isn't a space. // Add the first character as uppercase, then add each successive character as lowercase. bool first = true; for (var i = 0; i < span.Length; i++) { char c = span[i]; if (char.IsWhiteSpace(c)) { first = true; } else if (first) { c = char.ToUpper(c); first = false; } else { c = char.ToLower(c); } result[i] = c; } } }