using System;
namespace PKHeX.Core;
///
/// Exposes information about the amount of Awakened stat boosts gained.
///
/// Used only in LGP/E.
public interface IAwakened
{
byte AV_HP { get; set; }
byte AV_ATK { get; set; }
byte AV_DEF { get; set; }
byte AV_SPE { get; set; }
byte AV_SPA { get; set; }
byte AV_SPD { get; set; }
}
public static class AwakeningUtil
{
///
/// Sums all values.
///
/// Data to sum with
public static int AwakeningSum(this IAwakened pk) => pk.AV_HP + pk.AV_ATK + pk.AV_DEF + pk.AV_SPE + pk.AV_SPA + pk.AV_SPD;
///
/// Clears all values.
///
/// Data to clear from
public static void AwakeningClear(this IAwakened pk) => pk.AwakeningSetAllTo(0);
///
/// Sets all values to the maximum value.
///
/// Data to set values for
public static void AwakeningMax(this IAwakened pk) => pk.AwakeningSetAllTo(Legal.AwakeningMax);
///
/// Sets all values to the specified value.
///
/// Data to set values for
/// Value to set all to
public static void AwakeningSetAllTo(this IAwakened pk, byte value) => pk.AV_HP = pk.AV_ATK = pk.AV_DEF = pk.AV_SPE = pk.AV_SPA = pk.AV_SPD = value;
///
/// Sets all values to the specified value.
///
/// Data to set values for
/// Minimum value to set
/// Maximum value to set
public static void AwakeningSetRandom(this IAwakened pk, byte min = 0, int max = Legal.AwakeningMax)
{
if (pk is not PB7 pb7)
return;
Span result = stackalloc byte[6];
GetExpectedMinimumAVs(result, pb7);
var rnd = Util.Rand;
for (int i = 0; i < 6; i++)
{
var realMin = Math.Max(min, result[i]);
var realMax = Math.Min(result[i], max);
result[i] = (byte)rnd.Next(realMin, realMax + 1);
}
AwakeningSetVisual(pb7, result);
}
///
/// Sets the awakening values according to their displayed order.
///
/// Data to set values for
///
public static void AwakeningGetVisual(IAwakened pk, Span value)
{
value[0] = pk.AV_HP;
value[1] = pk.AV_ATK;
value[2] = pk.AV_DEF;
value[3] = pk.AV_SPA;
value[4] = pk.AV_SPD;
value[5] = pk.AV_SPE;
}
///
/// Sets the awakening values according to their displayed order.
///
/// Data to set values for
///
public static void AwakeningSetVisual(IAwakened pk, ReadOnlySpan value)
{
pk.AV_HP = value[0];
pk.AV_ATK = value[1];
pk.AV_DEF = value[2];
pk.AV_SPA = value[3];
pk.AV_SPD = value[4];
pk.AV_SPE = value[5];
}
///
/// Gets if all values are within legal limits.
///
/// Data to check
public static bool AwakeningAllValid(this IAwakened pk)
{
if (pk.AV_HP > Legal.AwakeningMax)
return false;
if (pk.AV_ATK > Legal.AwakeningMax)
return false;
if (pk.AV_DEF > Legal.AwakeningMax)
return false;
if (pk.AV_SPE > Legal.AwakeningMax)
return false;
if (pk.AV_SPA > Legal.AwakeningMax)
return false;
if (pk.AV_SPD > Legal.AwakeningMax)
return false;
return true;
}
///
/// Sets one of the values based on its index within the array.
///
/// Pokémon to modify.
/// Index to set to
/// Value to set
public static byte SetAV(this IAwakened pk, int index, byte value) => index switch
{
0 => pk.AV_HP = value,
1 => pk.AV_ATK = value,
2 => pk.AV_DEF = value,
3 => pk.AV_SPE = value,
4 => pk.AV_SPA = value,
5 => pk.AV_SPD = value,
_ => throw new ArgumentOutOfRangeException(nameof(index)),
};
///
/// Sets one of the values based on its index within the array.
///
/// Pokémon to check.
/// Index to get
public static byte GetAV(this IAwakened pk, int index) => index switch
{
0 => pk.AV_HP,
1 => pk.AV_ATK,
2 => pk.AV_DEF,
3 => pk.AV_SPE,
4 => pk.AV_SPA,
5 => pk.AV_SPD,
_ => throw new ArgumentOutOfRangeException(nameof(index)),
};
///
/// Loads the values to the input span.
///
public static void GetAVs(this IAwakened pk, Span value)
{
if (value.Length != 6)
return;
value[0] = pk.AV_HP;
value[1] = pk.AV_ATK;
value[2] = pk.AV_DEF;
value[3] = pk.AV_SPE;
value[4] = pk.AV_SPA;
value[5] = pk.AV_SPD;
}
///
/// Sets the values based on the current .
///
/// Accessor for setting the values
/// Retriever for IVs
public static void SetSuggestedAwakenedValues(this IAwakened a, PKM pk)
{
Span result = stackalloc byte[6];
GetExpectedMinimumAVs(result, (PB7)a);
a.AV_HP = Legal.AwakeningMax;
a.AV_ATK = pk.IV_ATK == 0 ? result[1] : Legal.AwakeningMax;
a.AV_DEF = Legal.AwakeningMax;
a.AV_SPA = Legal.AwakeningMax;
a.AV_SPD = Legal.AwakeningMax;
a.AV_SPE = pk.IV_SPE == 0 ? result[5] : Legal.AwakeningMax;
}
public static bool IsAwakeningBelow(this IAwakened current, IAwakened initial) => !current.IsAwakeningAboveOrEqual(initial);
///
/// Checks if the has values greater or equal to the .
///
public static bool IsAwakeningAboveOrEqual(this IAwakened current, IAwakened initial)
{
if (current.AV_HP < initial.AV_HP)
return false;
if (current.AV_ATK < initial.AV_ATK)
return false;
if (current.AV_DEF < initial.AV_DEF)
return false;
if (current.AV_SPA < initial.AV_SPA)
return false;
if (current.AV_SPD < initial.AV_SPD)
return false;
if (current.AV_SPE < initial.AV_SPE)
return false;
return true;
}
///
/// Updates the span with the expected minimum values for each index.
///
/// Stat results
/// Entity to check
public static void GetExpectedMinimumAVs(Span result, PB7 pk)
{
// GO Park transfers start with 2 AVs for all stats.
// Every other encounter is either all 0, or can legally start at 0 (trades).
if (pk.Version == (int)GameVersion.GO)
result.Fill(2);
// Leveling up in-game applies 1 AV to a "random" index.
var start = pk.Met_Level;
var end = pk.CurrentLevel;
if (start == end)
return;
// Level up from met level to current level.
var nature = pk.Nature;
var character = pk.Characteristic;
var ec = pk.EncryptionConstant;
for (int i = start + 1; i <= end; i++)
{
var lm10 = i % 10;
var bits = (ec >> (3 * lm10)) & 7;
var index = PB7.GetRandomIndex((int)bits, character, nature);
++result[index];
}
}
}