mirror of
https://github.com/kwsch/PKHeX
synced 2024-11-27 06:20:25 +00:00
114 lines
4 KiB
C#
114 lines
4 KiB
C#
using System;
|
|
|
|
namespace PKHeX.Core;
|
|
|
|
/// <summary>
|
|
/// Logic for calculating the characteristic behavior of Pokémon.
|
|
/// </summary>
|
|
public static class EntityCharacteristic
|
|
{
|
|
/// <summary>
|
|
/// Gets the characteristic index from the input parameters.
|
|
/// </summary>
|
|
/// <param name="maxStatIndex">Index of the highest IV.</param>
|
|
/// <param name="maxStatValue">Value of the highest IV.</param>
|
|
/// <returns>Characteristic index.</returns>
|
|
public static int GetCharacteristic(int maxStatIndex, int maxStatValue) => (maxStatIndex * 5) + (maxStatValue % 5);
|
|
|
|
/// <summary>
|
|
/// Gets the characteristic index of the given IVs in Little Endian format.
|
|
/// </summary>
|
|
/// <param name="ec">Encryption Constant.</param>
|
|
/// <param name="iv32">Lumped IVs in Little Endian format.</param>
|
|
/// <returns>Characteristic index.</returns>
|
|
public static int GetCharacteristic(uint ec, uint iv32)
|
|
{
|
|
int index = (int)(ec % 6);
|
|
|
|
// Get individual IVs from the lumped value.
|
|
// The IVs are stored in the following order: HP, Atk, Def, Spe, SpA, SpD
|
|
// Check all IVs, get the highest IV and its index. If there are multiple highest IVs, the first index checked is chosen.
|
|
int maxStatIndex = index;
|
|
var maxStatValue = 0u;
|
|
do
|
|
{
|
|
var value = iv32 >> (index * 5) & 0x1F;
|
|
if (value > maxStatValue)
|
|
{
|
|
maxStatIndex = index;
|
|
maxStatValue = value;
|
|
}
|
|
if (index != 5)
|
|
index++;
|
|
else
|
|
index = 0;
|
|
} while (maxStatIndex != index);
|
|
|
|
return GetCharacteristic(maxStatIndex, (int)maxStatValue);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the characteristic index of the given unpacked IVs.
|
|
/// </summary>
|
|
/// <param name="ec">Encryption Constant.</param>
|
|
/// <param name="ivs">Unpacked IVs.</param>
|
|
/// <returns>Characteristic index.</returns>
|
|
public static int GetCharacteristic(uint ec, Span<int> ivs)
|
|
{
|
|
int index = (int)(ec % 6);
|
|
|
|
// Get individual IVs from the lumped value.
|
|
// The IVs are stored in the following order: HP, Atk, Def, Spe, SpA, SpD
|
|
// Check all IVs, get the highest IV and its index. If there are multiple highest IVs, the first index checked is chosen.
|
|
int maxStatIndex = index;
|
|
var maxStatValue = 0;
|
|
do
|
|
{
|
|
var value = ivs[index];
|
|
if (value > maxStatValue)
|
|
{
|
|
maxStatIndex = index;
|
|
maxStatValue = value;
|
|
}
|
|
if (index != 5)
|
|
index++;
|
|
else
|
|
index = 0;
|
|
} while (maxStatIndex != index);
|
|
|
|
return GetCharacteristic(maxStatIndex, maxStatValue);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the characteristic index of the given IVs in Big Endian format.
|
|
/// </summary>
|
|
/// <param name="ec">Encryption Constant.</param>
|
|
/// <param name="iv32">Lumped IVs in Big Endian format.</param>
|
|
/// <returns>Characteristic index.</returns>
|
|
public static int GetCharacteristicInvertFields(uint ec, uint iv32)
|
|
{
|
|
int index = (int)(ec % 6);
|
|
|
|
// Get individual IVs from the lumped value.
|
|
// The IVs are stored in the following order: SpD, SpA, Spe, Def, Atk, HP
|
|
// Check all IVs, get the highest IV and its index. If there are multiple highest IVs, the first index checked is chosen.
|
|
int maxStatIndex = index;
|
|
var maxStatValue = 0u;
|
|
do
|
|
{
|
|
// IVs are stored in reverse order, get the bits from the end of the IV value
|
|
var value = iv32 >> (27 - (index * 5)) & 0x1F;
|
|
if (value > maxStatValue)
|
|
{
|
|
maxStatIndex = index;
|
|
maxStatValue = value;
|
|
}
|
|
if (index != 5)
|
|
index++;
|
|
else
|
|
index = 0;
|
|
} while (maxStatIndex != index);
|
|
|
|
return GetCharacteristic(maxStatIndex, (int)maxStatValue);
|
|
}
|
|
}
|