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;
}
}
}