PKHeX/PKHeX.Core/Editing/Pokerus.cs
Kurt 841fcde2b2 Enhance Pokerus api functions
Fixes #3480 ; (0,0) is a valid state regardless of format.
2022-04-14 14:05:32 -07:00

92 lines
3.7 KiB
C#

namespace PKHeX.Core;
/// <summary>
/// Logic pertaining to Pokérus -- the virus that doubles EVs gained from battle.
/// </summary>
public static class Pokerus
{
/// <summary>
/// Gets the max duration (in days) that a strain can be infectious.
/// </summary>
/// <param name="strain">Strain number</param>
/// <returns>Initial duration (in days). When the value decrements to zero, the Pokémon is no longer infectious.</returns>
public static int GetMaxDuration(int strain) => (strain & 3) + 1;
/// <summary>
/// Checks if any Pokérus values are possible to have on the input entity.
/// </summary>
/// <param name="pk">Entity to check</param>
/// <returns>True if Pokérus exists in the game format, or can be transmitted to the entity via another game.</returns>
public static bool IsObtainable(PKM pk) => pk is not PA8; // don't care about PK1, not stored in the data.
/// <summary>
/// Checks if the Pokérus value for Strain is possible to have on the input entity.
/// </summary>
/// <param name="pk">Entity to check</param>
/// <param name="strain">Strain number</param>
/// <param name="days">Duration remaining</param>
/// <returns>True if valid</returns>
public static bool IsStrainValid(PKM pk, int strain, int days)
{
if (!IsObtainable(pk))
return IsSusceptible(strain, days);
return IsStrainValid(strain, days);
}
/// <inheritdoc cref="IsStrainValid(PKM,int,int)"/>
public static bool IsStrainValid(int strain, int days) => strain switch
{
0 when days is not 0 => false,
8 => false,
_ => true,
};
/// <summary>
/// Checks if the Pokérus value for Duration is possible to have on the input entity.
/// </summary>
/// <param name="strain">Strain number</param>
/// <param name="days">Duration remaining</param>
/// <param name="max">Maximum value permitted</param>
/// <returns>True if valid</returns>
public static bool IsDurationValid(int strain, int days, out int max)
{
max = GetMaxDuration(strain);
return (uint)days <= max;
}
/// <summary>
/// Checks if the Pokémon is immune to the Pokérus.
/// </summary>
/// <param name="strain">Strain number</param>
/// <param name="days">Duration remaining</param>
/// <returns>True if immune (cannot be infected).</returns>
/// <remarks>An immune Pokémon must have been infected and cured prior to being "immune".</remarks>
public static bool IsImmune(int strain, int days) => strain != 0 && days == 0;
/// <summary>
/// Checks if the Pokémon is currently infected with the Pokérus.
/// </summary>
/// <param name="strain">Strain number</param>
/// <param name="days">Duration remaining</param>
/// <returns>True if currently infected, and infectious to others.</returns>
public static bool IsInfectuous(int strain, int days) => strain != 0 && days != 0;
/// <summary>
/// Checks if the Pokémon can be infected with the Pokérus.
/// </summary>
/// <param name="strain">Strain number</param>
/// <param name="days">Duration remaining</param>
/// <returns>True if can be infected by another infectious individual.</returns>
public static bool IsSusceptible(int strain, int days) => strain == 0 && days == 0;
/// <summary>
/// Vaccinates the Pokémon so it will never be infectious in the format it exists in.
/// </summary>
/// <param name="pk">Entity to modify.</param>
/// <remarks>Overwrites all Pokérus values even if already legal.</remarks>
public static void Vaccinate(this PKM pk)
{
pk.PKRS_Strain = IsObtainable(pk) ? 1 : 0;
pk.PKRS_Days = 0;
}
}