using System;
using System.Collections.Generic;
using System.Linq;
using static PKHeX.Core.Species;
namespace PKHeX.Core
{
///
/// Contains logic that calculates the evolution chain of a , only considering the generation it originated in.
///
public static class EncounterOrigin
{
///
/// Gets possible evolution details for the input
///
/// Current state of the Pokémon
/// Possible origin species-form-levels to match against encounter data.
/// Use if the originated from Generation 1 or 2.
public static IReadOnlyList GetOriginChain(PKM pkm)
{
bool hasOriginMet = pkm.HasOriginalMetLocation;
var maxLevel = GetLevelOriginMax(pkm, hasOriginMet);
var minLevel = GetLevelOriginMin(pkm, hasOriginMet);
return GetOriginChain(pkm, -1, maxLevel, minLevel, hasOriginMet);
}
///
/// Gets possible evolution details for the input originating from Generation 1 or 2.
///
/// Current state of the Pokémon
/// Game/group the originated from. If , it assumes Gen 1, otherwise Gen 2.
/// Possible origin species-form-levels to match against encounter data.
public static IReadOnlyList GetOriginChain12(PKM pkm, GameVersion gameSource)
{
bool rby = gameSource == GameVersion.RBY;
var maxSpecies = rby ? Legal.MaxSpeciesID_1 : Legal.MaxSpeciesID_2;
bool hasOriginMet;
int maxLevel, minLevel;
if (pkm is ICaughtData2 pk2)
{
hasOriginMet = pk2.CaughtData != 0;
maxLevel = rby && Future_LevelUp2.Contains(pkm.Species) ? pkm.CurrentLevel - 1 : pkm.CurrentLevel;
minLevel = !hasOriginMet ? 2 : pkm.IsEgg ? 5 : pk2.Met_Level;
}
else if (pkm is PK1 pk1)
{
hasOriginMet = false;
maxLevel = pk1.CurrentLevel;
minLevel = 2;
}
else if (rby)
{
hasOriginMet = false;
maxLevel = Future_LevelUp2.Contains(pkm.Species) ? pkm.CurrentLevel - 1 : GetLevelOriginMaxTransfer(pkm, pkm.Met_Level, 1);
minLevel = 2;
}
else // GSC
{
hasOriginMet = false;
maxLevel = GetLevelOriginMaxTransfer(pkm, pkm.Met_Level, 2);
minLevel = 2;
}
return GetOriginChain(pkm, maxSpecies, maxLevel, minLevel, hasOriginMet);
}
private static IReadOnlyList GetOriginChain(PKM pkm, int maxSpecies, int maxLevel, int minLevel, bool hasOriginMet)
{
if (maxLevel < minLevel)
return Array.Empty();
if (hasOriginMet)
return EvolutionChain.GetValidPreEvolutions(pkm, maxSpecies, maxLevel, minLevel);
// Permit the maximum to be all the way up to Current Level; we'll trim these impossible evolutions out later.
var tempMax = pkm.CurrentLevel;
var chain = EvolutionChain.GetValidPreEvolutions(pkm, maxSpecies, tempMax, minLevel);
for (int i = chain.Count - 1; i >= 0; i--)
{
var evo = chain[i];
if (evo.MinLevel > maxLevel)
{
chain.RemoveAt(i);
if (chain.Any(z => z.Level >= maxLevel))
continue;
chain.Clear();
break;
}
if (evo.Level > maxLevel)
evo.Level = maxLevel;
}
return chain;
}
private static int GetLevelOriginMin(PKM pkm, bool hasMet)
{
if (pkm.Format == 3)
{
if (pkm.IsEgg)
return 5;
return Math.Max(2, pkm.Met_Level);
}
if (!hasMet)
return 1;
return Math.Max(1, pkm.Met_Level);
}
private static int GetLevelOriginMax(PKM pkm, bool hasMet)
{
var met = pkm.Met_Level;
if (hasMet)
return pkm.CurrentLevel;
int generation = pkm.GenNumber;
if (generation >= 4)
return met;
var downLevel = GetLevelOriginMaxTransfer(pkm, pkm.CurrentLevel, generation);
return Math.Min(met, downLevel);
}
private static int GetLevelOriginMaxTransfer(PKM pkm, int met, int generation)
{
var species = pkm.Species;
if (Future_LevelUp.TryGetValue(species | (pkm.AltForm << 11), out var delta))
return met - delta;
if (generation < 4 && Future_LevelUp4.Contains(species) && (pkm.Format <= 7 || !Future_LevelUp4_Not8.Contains(species)))
return met - 1;
return met;
}
///
/// Species introduced in Generation 2 that require a level up to evolve into from a specimen that originated in a previous generation.
///
private static readonly HashSet Future_LevelUp2 = new HashSet
{
(int)Crobat,
(int)Espeon,
(int)Umbreon,
(int)Blissey,
};
///
/// Species introduced in Generation 4 that require a level up to evolve into from a specimen that originated in a previous generation.
///
private static readonly HashSet Future_LevelUp4 = new HashSet
{
(int)Ambipom,
(int)Weavile,
(int)Magnezone,
(int)Lickilicky,
(int)Tangrowth,
(int)Yanmega,
(int)Leafeon,
(int)Glaceon,
(int)Mamoswine,
(int)Gliscor,
(int)Probopass,
};
///
/// Species introduced in Generation 4 that used to require a level up to evolve prior to Generation 8.
///
private static readonly HashSet Future_LevelUp4_Not8 = new HashSet
{
(int)Magnezone, // Thunder Stone
(int)Leafeon, // Leaf Stone
(int)Glaceon, // Ice Stone
};
///
/// Species introduced in Generation 6+ that require a level up to evolve into from a specimen that originated in a previous generation.
///
private static readonly Dictionary Future_LevelUp = new Dictionary
{
// Gen6
{(int)Sylveon, 1},
// Gen7
{(int)Marowak | (1 << 11), 1},
// Gen8
{(int)Weezing | (1 << 11), 1},
{(int)MrMime | (1 << 11), 1},
{(int)MrRime, 2},
};
}
}