mirror of
https://github.com/kwsch/PKHeX
synced 2025-01-13 04:58:50 +00:00
47071b41f3
Existing `get`/`set` logic is flawed in that it doesn't work on Big Endian operating systems, and it allocates heap objects when it doesn't need to. `System.Buffers.Binary.BinaryPrimitives` in the `System.Memory` NuGet package provides both Little Endian and Big Endian methods to read and write data; all the `get`/`set` operations have been reworked to use this new API. This removes the need for PKHeX's manual `BigEndian` class, as all functions are already covered by the BinaryPrimitives API. The `StringConverter` has now been rewritten to accept a Span to read from & write to, no longer requiring a temporary StringBuilder. Other Fixes included: - The Super Training UI for Gen6 has been reworked according to the latest block structure additions. - Cloning a Stadium2 Save File now works correctly (opening from the Folder browser list). - Checksum & Sanity properties removed from parent PKM class, and is now implemented via interface.
121 lines
4.9 KiB
C#
121 lines
4.9 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
|
|
namespace PKHeX.Core
|
|
{
|
|
/// <summary>
|
|
/// Utility for searching strings within arrays or within another string.
|
|
/// </summary>
|
|
public static class StringUtil
|
|
{
|
|
private static readonly CompareInfo CompareInfo = CultureInfo.CurrentCulture.CompareInfo;
|
|
|
|
/// <summary>
|
|
/// Finds the index of the string within the array by ignoring casing, spaces, and punctuation.
|
|
/// </summary>
|
|
/// <param name="arr">Array of strings to search in</param>
|
|
/// <param name="value">Value to search for</param>
|
|
/// <returns>Index within <see cref="arr"/></returns>
|
|
public static int FindIndexIgnoreCase(string[] arr, string value) => Array.FindIndex(arr, z => IsMatchIgnoreCase(z, value));
|
|
|
|
/// <summary>
|
|
/// Gets the indexes by calling <see cref="FindIndexIgnoreCase"/> for all <see cref="items"/>.
|
|
/// </summary>
|
|
/// <param name="arr">Array of strings to search in</param>
|
|
/// <param name="items">Items to search for</param>
|
|
/// <returns>Index within <see cref="arr"/></returns>
|
|
public static int[] GetIndexes(string[] arr, IReadOnlyList<string> items) => GetIndexes(arr, items, 0, items.Count);
|
|
|
|
/// <summary>
|
|
/// Gets the indexes by calling <see cref="FindIndexIgnoreCase"/> for all <see cref="items"/>.
|
|
/// </summary>
|
|
/// <param name="arr">Array of strings to search in</param>
|
|
/// <param name="items">Items to search for</param>
|
|
/// <param name="start">Starting index within <see cref="items"/></param>
|
|
/// <param name="length">Amount to convert within <see cref="items"/></param>
|
|
/// <returns>Index within <see cref="arr"/></returns>
|
|
public static int[] GetIndexes(string[] arr, IReadOnlyList<string> items, int start, int length)
|
|
{
|
|
if (length < 0)
|
|
length = items.Count - start;
|
|
var result = new int[length];
|
|
for (int i = 0; i < result.Length; i++)
|
|
result[i] = FindIndexIgnoreCase(arr, items[start + i]);
|
|
return result;
|
|
}
|
|
|
|
private static bool IsMatchIgnoreCase(string string1, string string2)
|
|
{
|
|
if (string1.Length != string2.Length)
|
|
return false;
|
|
const CompareOptions options = CompareOptions.IgnoreCase | CompareOptions.IgnoreNonSpace | CompareOptions.IgnoreSymbols | CompareOptions.IgnoreWidth;
|
|
var compare = CompareInfo.Compare(string1, string2, options);
|
|
return compare == 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the <see cref="nth"/> string entry within the input <see cref="line"/>, based on the <see cref="separator"/> and <see cref="start"/> position.
|
|
/// </summary>
|
|
public static string GetNthEntry(ReadOnlySpan<char> line, int nth, int start, char separator = ',')
|
|
{
|
|
if (nth != 1)
|
|
start = line.IndexOfNth(separator, nth - 1, start + 1);
|
|
var end = line.IndexOfNth(separator, 1, start + 1);
|
|
if (end == -1)
|
|
return new string(line[(start + 1)..].ToArray());
|
|
return new string(line[(start + 1)..end].ToArray());
|
|
}
|
|
|
|
private static int IndexOfNth(this ReadOnlySpan<char> s, char t, int n, int start)
|
|
{
|
|
int count = 0;
|
|
for (int i = start; i < s.Length; i++)
|
|
{
|
|
if (s[i] != t)
|
|
continue;
|
|
if (++count == n)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts an all-caps hex string to a byte array.
|
|
/// </summary>
|
|
public static byte[] ToByteArray(this string toTransform)
|
|
{
|
|
var result = new byte[toTransform.Length / 2];
|
|
for (int i = 0; i < result.Length; i++)
|
|
{
|
|
var ofs = i << 1;
|
|
var _0 = toTransform[ofs + 0];
|
|
var _1 = toTransform[ofs + 1];
|
|
result[i] = DecodeTuple(_0, _1);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
private static bool IsNum(char c) => (uint)(c - '0') <= 9;
|
|
private static bool IsHexUpper(char c) => (uint)(c - 'A') <= 5;
|
|
|
|
private static byte DecodeTuple(char _0, char _1)
|
|
{
|
|
byte result;
|
|
if (IsNum(_0))
|
|
result = (byte)((_0 - '0') << 4);
|
|
else if (IsHexUpper(_0))
|
|
result = (byte)((_0 - 'A' + 10) << 4);
|
|
else
|
|
throw new ArgumentOutOfRangeException(nameof(_0));
|
|
|
|
if (IsNum(_1))
|
|
result |= (byte)(_1 - '0');
|
|
else if (IsHexUpper(_1))
|
|
result |= (byte)(_1 - 'A' + 10);
|
|
else
|
|
throw new ArgumentOutOfRangeException(nameof(_1));
|
|
return result;
|
|
}
|
|
}
|
|
}
|