mirror of
https://github.com/claunia/SabreTools.git
synced 2025-12-16 19:14:27 +00:00
198 lines
7.4 KiB
C#
198 lines
7.4 KiB
C#
using System;
|
|
using System.Linq;
|
|
|
|
namespace SabreTools.Core.Tools
|
|
{
|
|
public static class NumberHelper
|
|
{
|
|
#region Constants
|
|
|
|
#region Byte (1000-based) size comparisons
|
|
|
|
private const long KiloByte = 1000;
|
|
private readonly static long MegaByte = (long)Math.Pow(KiloByte, 2);
|
|
private readonly static long GigaByte = (long)Math.Pow(KiloByte, 3);
|
|
private readonly static long TeraByte = (long)Math.Pow(KiloByte, 4);
|
|
private readonly static long PetaByte = (long)Math.Pow(KiloByte, 5);
|
|
|
|
// The following are too big to be represented in Int64
|
|
// private readonly static long ExaByte = (long)Math.Pow(KiloByte, 6);
|
|
// private readonly static long ZettaByte = (long)Math.Pow(KiloByte, 7);
|
|
// private readonly static long YottaByte = (long)Math.Pow(KiloByte, 8);
|
|
|
|
#endregion
|
|
|
|
#region Byte (1024-based) size comparisons
|
|
|
|
private const long KibiByte = 1024;
|
|
private readonly static long MibiByte = (long)Math.Pow(KibiByte, 2);
|
|
private readonly static long GibiByte = (long)Math.Pow(KibiByte, 3);
|
|
private readonly static long TibiByte = (long)Math.Pow(KibiByte, 4);
|
|
private readonly static long PibiByte = (long)Math.Pow(KibiByte, 5);
|
|
|
|
// The following are too big to be represented in Int64
|
|
// private readonly static long ExiByte = (long)Math.Pow(KibiByte, 6);
|
|
// private readonly static long ZittiByte = (long)Math.Pow(KibiByte, 7);
|
|
// private readonly static long YittiByte = (long)Math.Pow(KibiByte, 8);
|
|
|
|
#endregion
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Convert a string to a Double
|
|
/// </summary>
|
|
public static double? ConvertToDouble(string? numeric)
|
|
{
|
|
// If we don't have a valid string, we can't do anything
|
|
if (string.IsNullOrEmpty(numeric))
|
|
return null;
|
|
|
|
if (!double.TryParse(numeric, out double doubleValue))
|
|
return null;
|
|
|
|
return doubleValue;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert a string to an Int64
|
|
/// </summary>
|
|
public static long? ConvertToInt64(string? numeric)
|
|
{
|
|
// If we don't have a valid string, we can't do anything
|
|
if (string.IsNullOrEmpty(numeric))
|
|
return null;
|
|
|
|
// Normalize the string for easier comparison
|
|
numeric = numeric!.ToLowerInvariant();
|
|
|
|
// Get the multiplication modifier and trim characters
|
|
long multiplier = DetermineMultiplier(numeric);
|
|
numeric = numeric.TrimEnd(['k', 'm', 'g', 't', 'p', 'e', 'z', 'y', 'i', 'b', ' ']);
|
|
|
|
// Parse the numeric string, if possible
|
|
long value;
|
|
if (numeric.StartsWith("0x"))
|
|
value = Convert.ToInt64(numeric, 16);
|
|
else if (long.TryParse(numeric, out long longValue))
|
|
value = longValue;
|
|
else
|
|
return null;
|
|
|
|
// Apply the multiplier and return
|
|
return value * multiplier;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determine the multiplier from a numeric string
|
|
/// </summary>
|
|
public static long DetermineMultiplier(string? numeric)
|
|
{
|
|
if (string.IsNullOrEmpty(numeric))
|
|
return 0;
|
|
|
|
long multiplier = 1;
|
|
if (numeric!.EndsWith("k") || numeric.EndsWith("kb"))
|
|
multiplier = KiloByte;
|
|
else if (numeric.EndsWith("ki") || numeric.EndsWith("kib"))
|
|
multiplier = KibiByte;
|
|
else if (numeric.EndsWith("m") || numeric.EndsWith("mb"))
|
|
multiplier = MegaByte;
|
|
else if (numeric.EndsWith("mi") || numeric.EndsWith("mib"))
|
|
multiplier = MibiByte;
|
|
else if (numeric.EndsWith("g") || numeric.EndsWith("gb"))
|
|
multiplier = GigaByte;
|
|
else if (numeric.EndsWith("gi") || numeric.EndsWith("gib"))
|
|
multiplier = GibiByte;
|
|
else if (numeric.EndsWith("t") || numeric.EndsWith("tb"))
|
|
multiplier = TeraByte;
|
|
else if (numeric.EndsWith("ti") || numeric.EndsWith("tib"))
|
|
multiplier = TibiByte;
|
|
else if (numeric.EndsWith("p") || numeric.EndsWith("pb"))
|
|
multiplier = PetaByte;
|
|
else if (numeric.EndsWith("pi") || numeric.EndsWith("pib"))
|
|
multiplier = PibiByte;
|
|
|
|
// The following are too big to be represented in Int64
|
|
// else if (numeric.EndsWith("e") || numeric.EndsWith("eb"))
|
|
// multiplier = ExaByte;
|
|
// else if (numeric.EndsWith("ei") || numeric.EndsWith("eib"))
|
|
// multiplier = ExiByte;
|
|
// else if (numeric.EndsWith("z") || numeric.EndsWith("zb"))
|
|
// multiplier = ZettaByte;
|
|
// else if (numeric.EndsWith("zi") || numeric.EndsWith("zib"))
|
|
// multiplier = ZittiByte;
|
|
// else if (numeric.EndsWith("y") || numeric.EndsWith("yb"))
|
|
// multiplier = YottaByte;
|
|
// else if (numeric.EndsWith("yi") || numeric.EndsWith("yib"))
|
|
// multiplier = YittiByte;
|
|
|
|
return multiplier;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determine if a string is fully numeric or not
|
|
/// </summary>
|
|
public static bool IsNumeric(string? value)
|
|
{
|
|
// If we have no value, it is not numeric
|
|
if (string.IsNullOrEmpty(value))
|
|
return false;
|
|
|
|
// If we have a hex value
|
|
value = value!.ToLowerInvariant();
|
|
if (value.StartsWith("0x"))
|
|
value = value.Substring(2);
|
|
|
|
// If we have a negative value
|
|
if (value.StartsWith("-"))
|
|
value = value.Substring(1);
|
|
|
|
// If the value has a multiplier
|
|
if (DetermineMultiplier(value) > 1)
|
|
value = value.TrimEnd(['k', 'm', 'g', 't', 'p', 'e', 'z', 'y', 'i', 'b', ' ']);
|
|
|
|
// If the value is empty after trimming
|
|
if (value.Length == 0)
|
|
return false;
|
|
|
|
#if NET7_0_OR_GREATER
|
|
return value.All(c => char.IsAsciiHexDigit(c) || c == '.' || c == ',');
|
|
#else
|
|
return value.All(c => c.IsAsciiHexDigit() || c == '.' || c == ',');
|
|
#endif
|
|
}
|
|
|
|
#if NETFRAMEWORK || NETCOREAPP3_1 || NET5_0 || NET6_0
|
|
/// <summary>
|
|
/// Indicates whether a character is categorized as an ASCII hexademical digit.
|
|
/// </summary>
|
|
/// <param name="c">The character to evaluate.</param>
|
|
/// <returns>true if c is a hexademical digit; otherwise, false.</returns>
|
|
/// <remarks>This method determines whether the character is in the range '0' through '9', inclusive, 'A' through 'F', inclusive, or 'a' through 'f', inclusive.</remarks>
|
|
internal static bool IsAsciiHexDigit(this char c)
|
|
{
|
|
return char.ToLowerInvariant(c) switch
|
|
{
|
|
'0' => true,
|
|
'1' => true,
|
|
'2' => true,
|
|
'3' => true,
|
|
'4' => true,
|
|
'5' => true,
|
|
'6' => true,
|
|
'7' => true,
|
|
'8' => true,
|
|
'9' => true,
|
|
'a' => true,
|
|
'b' => true,
|
|
'c' => true,
|
|
'd' => true,
|
|
'e' => true,
|
|
'f' => true,
|
|
_ => false,
|
|
};
|
|
}
|
|
#endif
|
|
}
|
|
} |