mirror of
https://github.com/kwsch/PKHeX
synced 2025-01-05 09:08:45 +00:00
6441bdadd8
`Moveset` struct stores 4 moves, and exposes methods to interact with a moveset. `IndividualValueSet` stores a 6 IV template (signed). Performance impact: * Less allocating on the heap: Moves - (8 bytes member ptr, 20 bytes heap->8 bytes member) * Less allocating on the heap: IVs - (8 bytes member ptr, 28 bytes heap->8 bytes member) * No heap pointers, no need to jump to grab data. * Easy to inline logic for checking if moves are present (no linq usage with temporary collections). End result is faster ctor times, less memory used, faster program.
200 lines
7.7 KiB
C#
200 lines
7.7 KiB
C#
using System;
|
|
|
|
namespace PKHeX.Core;
|
|
|
|
/// <summary>
|
|
/// Group that checks the source of a move in <see cref="GameVersion.Gen4"/>.
|
|
/// </summary>
|
|
public sealed class LearnGroup4 : ILearnGroup
|
|
{
|
|
public static readonly LearnGroup4 Instance = new();
|
|
private const int Generation = 4;
|
|
|
|
public ILearnGroup? GetPrevious(PKM pk, EvolutionHistory history, IEncounterTemplate enc, LearnOption option) => enc.Generation is Generation ? null : LearnGroup3.Instance;
|
|
public bool HasVisited(PKM pk, EvolutionHistory history) => history.HasVisitedGen4;
|
|
|
|
public bool Check(Span<MoveResult> result, ReadOnlySpan<int> current, PKM pk, EvolutionHistory history,
|
|
IEncounterTemplate enc, MoveSourceType types = MoveSourceType.All, LearnOption option = LearnOption.Current)
|
|
{
|
|
var evos = history.Gen4;
|
|
for (var i = 0; i < evos.Length; i++)
|
|
Check(result, current, pk, evos[i], i);
|
|
|
|
if (types.HasFlagFast(MoveSourceType.Encounter) && enc is EncounterEgg { Generation: Generation } egg)
|
|
CheckEncounterMoves(result, current, egg);
|
|
|
|
if (types.HasFlagFast(MoveSourceType.LevelUp) && enc.Species is (int)Species.Nincada && evos.Length == 2 && evos[0].Species == (int)Species.Shedinja)
|
|
CheckNincadaMoves(result, current, evos[^1]);
|
|
|
|
return MoveResult.AllParsed(result);
|
|
}
|
|
|
|
private static void CheckNincadaMoves(Span<MoveResult> result, ReadOnlySpan<int> current, EvoCriteria nincada)
|
|
{
|
|
if (MoveResult.AllParsed(result))
|
|
return;
|
|
|
|
// If a result is not valid, check to see if it is a Shedinja move.
|
|
// Ninjask's Gen4 learnset is a superset of Gen3, so don't bother re-checking in that group.
|
|
var shedinja = LearnSource4Pt.Instance;
|
|
var moves = shedinja.GetLearnset((int)Species.Ninjask, 0);
|
|
for (var i = 0; i < result.Length; i++)
|
|
{
|
|
if (result[i].Valid)
|
|
continue;
|
|
var move = current[i];
|
|
if (move == 0)
|
|
break;
|
|
|
|
var level = moves.GetLevelLearnMove(move);
|
|
if (level == -1 || !nincada.InsideLevelRange(level))
|
|
continue;
|
|
|
|
var info = new MoveLearnInfo(LearnMethod.ShedinjaEvo, LearnEnvironment.Pt, (byte)level);
|
|
result[i] = new MoveResult(info, 0, Generation);
|
|
break; // Can only have one Ninjask move.
|
|
}
|
|
}
|
|
|
|
private static void CheckEncounterMoves(Span<MoveResult> result, ReadOnlySpan<int> current, EncounterEgg egg)
|
|
{
|
|
ReadOnlySpan<int> eggMoves, levelMoves;
|
|
if (egg.Version <= GameVersion.SS) // HG/SS
|
|
{
|
|
var inst = LearnSource4HGSS.Instance;
|
|
eggMoves = inst.GetEggMoves(egg.Species, egg.Form);
|
|
levelMoves = egg.CanInheritMoves
|
|
? inst.GetLearnset(egg.Species, egg.Form).Moves
|
|
: ReadOnlySpan<int>.Empty;
|
|
}
|
|
else if (egg.Version is GameVersion.Pt)
|
|
{
|
|
var inst = LearnSource4Pt.Instance;
|
|
eggMoves = inst.GetEggMoves(egg.Species, egg.Form);
|
|
levelMoves = egg.CanInheritMoves
|
|
? inst.GetLearnset(egg.Species, egg.Form).Moves
|
|
: ReadOnlySpan<int>.Empty;
|
|
}
|
|
else
|
|
{
|
|
var inst = LearnSource4DP.Instance;
|
|
eggMoves = inst.GetEggMoves(egg.Species, egg.Form);
|
|
levelMoves = egg.CanInheritMoves
|
|
? inst.GetLearnset(egg.Species, egg.Form).Moves
|
|
: ReadOnlySpan<int>.Empty;
|
|
}
|
|
|
|
for (var i = result.Length - 1; i >= 0; i--)
|
|
{
|
|
if (result[i].Valid)
|
|
continue;
|
|
var move = current[i];
|
|
if (eggMoves.Contains(move))
|
|
result[i] = new(LearnMethod.EggMove);
|
|
else if (levelMoves.Contains(move))
|
|
result[i] = new(LearnMethod.InheritLevelUp);
|
|
else if (move is (int)Move.VoltTackle && egg.CanHaveVoltTackle)
|
|
result[i] = new(LearnMethod.SpecialEgg);
|
|
}
|
|
}
|
|
|
|
private static void Check(Span<MoveResult> result, ReadOnlySpan<int> current, PKM pk, EvoCriteria evo, int stage)
|
|
{
|
|
if (evo.Species is not ((int)Species.Deoxys or (int)Species.Giratina or (int)Species.Shaymin))
|
|
{
|
|
CheckInternal(result, current, pk, evo, stage);
|
|
return;
|
|
}
|
|
|
|
// Check all forms
|
|
var inst = LearnSource4HGSS.Instance;
|
|
if (!inst.TryGetPersonal(evo.Species, evo.Form, out var pi))
|
|
return;
|
|
|
|
var fc = pi.FormCount;
|
|
for (int i = 0; i < fc; i++)
|
|
CheckInternal(result, current, pk, evo with { Form = (byte)i }, stage);
|
|
}
|
|
|
|
private static void CheckInternal(Span<MoveResult> result, ReadOnlySpan<int> current, PKM pk, EvoCriteria evo, int stage)
|
|
{
|
|
var hgss = LearnSource4HGSS.Instance;
|
|
var species = evo.Species;
|
|
if (!hgss.TryGetPersonal(species, evo.Form, out var hgss_pi))
|
|
return; // should never happen.
|
|
|
|
var pt = LearnSource4Pt.Instance;
|
|
if (!pt.TryGetPersonal(species, evo.Form, out var pt_pi))
|
|
return; // should never happen.
|
|
|
|
var dp = LearnSource4DP.Instance;
|
|
|
|
for (int i = result.Length - 1; i >= 0; i--)
|
|
{
|
|
if (result[i].Valid)
|
|
continue;
|
|
|
|
// Level Up moves are different for each game, but others (TM/Tutor) are same (except Pt Defog).
|
|
var move = current[i];
|
|
var chk = hgss.GetCanLearn(pk, hgss_pi, evo, move);
|
|
if (chk != default)
|
|
{
|
|
result[i] = new(chk, (byte)stage, Generation);
|
|
continue;
|
|
}
|
|
|
|
chk = pt.GetCanLearn(pk, pt_pi, evo, move, MoveSourceType.LevelUp | MoveSourceType.AllMachines);
|
|
if (chk != default)
|
|
{
|
|
result[i] = new(chk, (byte)stage, Generation);
|
|
continue;
|
|
}
|
|
chk = dp.GetCanLearn(pk, pt_pi, evo, move, MoveSourceType.LevelUp);
|
|
if (chk != default)
|
|
{
|
|
result[i] = new(chk, (byte)stage, Generation);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void GetAllMoves(Span<bool> result, PKM pk, EvolutionHistory history, IEncounterTemplate enc, MoveSourceType types = MoveSourceType.All, LearnOption option = LearnOption.Current)
|
|
{
|
|
if (types.HasFlagFast(MoveSourceType.Encounter) && enc.Generation == Generation)
|
|
FlagEncounterMoves(enc, result);
|
|
|
|
var evos = history.Gen4;
|
|
foreach (var evo in evos)
|
|
GetAllMoves(result, pk, evo, types);
|
|
|
|
if (enc.Species is (int)Species.Nincada && evos.Length == 2 && evos[0].Species == (int)Species.Shedinja)
|
|
{
|
|
var shedinja = LearnSource4Pt.Instance;
|
|
var moves = shedinja.GetLearnset((int)Species.Ninjask, 0);
|
|
(bool HasMoves, int start, int end) = moves.GetMoveRange(evos[0].LevelMax, 20);
|
|
if (HasMoves)
|
|
{
|
|
var all = moves.Moves;
|
|
for (int i = start; i < end; i++)
|
|
result[all[i]] = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void GetAllMoves(Span<bool> result, PKM pk, EvoCriteria evo, MoveSourceType types)
|
|
{
|
|
LearnSource4HGSS.Instance.GetAllMoves(result, pk, evo, types);
|
|
LearnSource4Pt.Instance.GetAllMoves(result, pk, evo, types & (MoveSourceType.LevelUp | MoveSourceType.AllMachines));
|
|
LearnSource4DP.Instance.GetAllMoves(result, pk, evo, types & (MoveSourceType.LevelUp));
|
|
}
|
|
|
|
private static void FlagEncounterMoves(IEncounterTemplate enc, Span<bool> result)
|
|
{
|
|
if (enc is IMoveset { Moves: { HasMoves: true } x })
|
|
{
|
|
result[x.Move4] = true;
|
|
result[x.Move3] = true;
|
|
result[x.Move2] = true;
|
|
result[x.Move1] = true;
|
|
}
|
|
}
|
|
}
|