using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Text;
namespace PKHeX.Core
{
public static partial class Util
{
///
public static int ToInt32(string value) => ToInt32(value.AsSpan());
///
public static uint ToUInt32(string value) => ToUInt32(value.AsSpan());
///
public static uint GetHexValue(string value) => GetHexValue(value.AsSpan());
///
public static ulong GetHexValue64(string value) => GetHexValue64(value.AsSpan());
///
public static byte[] GetBytesFromHexString(string value) => GetBytesFromHexString(value.AsSpan());
///
public static string GetHexStringFromBytes(byte[] data, int offset, int length) => GetHexStringFromBytes(data.AsSpan(offset, length));
///
/// Parses the string into an , 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 (IsNum(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 (!IsNum(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 (IsNum(c))
{
result <<= 4;
result += (uint)(c - '0');
}
else if (IsHexUpper(c))
{
result <<= 4;
result += (uint)(c - 'A' + 10);
}
else if (IsHexLower(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 (IsNum(c))
{
result <<= 4;
result += (uint)(c - '0');
}
else if (IsHexUpper(c))
{
result <<= 4;
result += (uint)(c - 'A' + 10);
}
else if (IsHexLower(c))
{
result <<= 4;
result += (uint)(c - 'a' + 10);
}
}
return result;
}
public static byte[] GetBytesFromHexString(ReadOnlySpan seed)
{
byte[] result = new byte[seed.Length / 2];
for (int i = 0; i < result.Length; i++)
{
var slice = seed.Slice(i * 2, 2);
result[^(i+1)] = (byte)GetHexValue(slice);
}
return result;
}
public static string GetHexStringFromBytes(ReadOnlySpan arr)
{
var sb = new StringBuilder(arr.Length * 2);
for (int i = arr.Length - 1; i >= 0; i--)
sb.AppendFormat("{0:X2}", arr[i]);
return sb.ToString();
}
private static bool IsNum(char c) => (uint)(c - '0') <= 9;
private static bool IsHexUpper(char c) => (uint)(c - 'A') <= 5;
private static bool IsHexLower(char c) => (uint)(c - 'a') <= 5;
private static bool IsHex(char c) => IsNum(c) || IsHexUpper(c) || IsHexLower(c);
private static string TitleCase(string word) => char.ToUpper(word[0]) + word[1..].ToLower();
///
/// Filters the string down to only valid hex characters, returning a new string.
///
/// Input string to filter
public static string GetOnlyHex(string str) => string.IsNullOrWhiteSpace(str) ? string.Empty : string.Concat(str.Where(IsHex));
///
/// Returns a new string with each word converted to its appropriate title case.
///
/// Input string to modify
public static string ToTitleCase(string str) => string.IsNullOrWhiteSpace(str) ? string.Empty : string.Join(" ", str.Split(' ').Select(TitleCase));
///
/// Trims a string at the first instance of a 0x0000 terminator.
///
/// String to trim.
/// Trimmed string.
public static string TrimFromZero(string input) => TrimFromFirst(input, '\0');
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static string TrimFromFirst(string input, char c)
{
int index = input.IndexOf(c);
return index < 0 ? input : input[..index];
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static void TrimFromFirst(StringBuilder input, char c)
{
for (int i = 0; i < input.Length; i++)
{
if (input[i] != c)
continue;
input.Remove(i, input.Length - i);
return;
}
}
public static Dictionary[] GetMultiDictionary(IReadOnlyList> nameArray)
{
var result = new Dictionary[nameArray.Count];
for (int i = 0; i < result.Length; i++)
{
var dict = result[i] = new Dictionary();
var names = nameArray[i];
for (int j = 0; j < names.Count; j++)
dict.Add(names[j], j);
}
return result;
}
}
}