mirror of
https://github.com/kwsch/PKHeX
synced 2025-01-10 19:48:53 +00:00
47071b41f3
Existing `get`/`set` logic is flawed in that it doesn't work on Big Endian operating systems, and it allocates heap objects when it doesn't need to. `System.Buffers.Binary.BinaryPrimitives` in the `System.Memory` NuGet package provides both Little Endian and Big Endian methods to read and write data; all the `get`/`set` operations have been reworked to use this new API. This removes the need for PKHeX's manual `BigEndian` class, as all functions are already covered by the BinaryPrimitives API. The `StringConverter` has now been rewritten to accept a Span to read from & write to, no longer requiring a temporary StringBuilder. Other Fixes included: - The Super Training UI for Gen6 has been reworked according to the latest block structure additions. - Cloning a Stadium2 Save File now works correctly (opening from the Folder browser list). - Checksum & Sanity properties removed from parent PKM class, and is now implemented via interface.
147 lines
6.3 KiB
C#
147 lines
6.3 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using static PKHeX.Core.LegalityCheckStrings;
|
|
using static PKHeX.Core.ParseSettings;
|
|
|
|
namespace PKHeX.Core
|
|
{
|
|
/// <summary>
|
|
/// Logic to verify the current <see cref="PKM.RelearnMoves"/>.
|
|
/// </summary>
|
|
public static class VerifyRelearnMoves
|
|
{
|
|
internal static readonly CheckResult DummyValid = new(CheckIdentifier.RelearnMove);
|
|
private static readonly CheckResult DummyNone = new(Severity.Invalid, LMoveRelearnNone, CheckIdentifier.RelearnMove);
|
|
|
|
public static CheckResult[] VerifyRelearn(PKM pkm, IEncounterTemplate enc, CheckResult[] result)
|
|
{
|
|
if (ShouldNotHaveRelearnMoves(enc, pkm))
|
|
return VerifyRelearnNone(pkm, result);
|
|
|
|
return enc switch
|
|
{
|
|
IRelearn s when s.Relearn.Count != 0 => VerifyRelearnSpecifiedMoveset(pkm, s.Relearn, result),
|
|
EncounterEgg e => VerifyEggMoveset(e, result, pkm.RelearnMoves),
|
|
EncounterSlot6AO z when pkm.RelearnMove1 != 0 && z.CanDexNav => VerifyRelearnDexNav(pkm, result, z),
|
|
EncounterSlot8b {IsUnderground:true} u => VerifyRelearnUnderground(pkm, result, u),
|
|
_ => VerifyRelearnNone(pkm, result),
|
|
};
|
|
}
|
|
|
|
public static bool ShouldNotHaveRelearnMoves(IGeneration enc, PKM pkm) => enc.Generation < 6 || pkm is IBattleVersion {BattleVersion: not 0};
|
|
|
|
private static CheckResult[] VerifyRelearnSpecifiedMoveset(PKM pkm, IReadOnlyList<int> required, CheckResult[] result)
|
|
{
|
|
result[3] = CheckResult(pkm.RelearnMove4, required[3]);
|
|
result[2] = CheckResult(pkm.RelearnMove3, required[2]);
|
|
result[1] = CheckResult(pkm.RelearnMove2, required[1]);
|
|
result[0] = CheckResult(pkm.RelearnMove1, required[0]);
|
|
return result;
|
|
|
|
static CheckResult CheckResult(int move, int require)
|
|
{
|
|
if (move == require)
|
|
return DummyValid;
|
|
return new CheckResult(Severity.Invalid, string.Format(LMoveFExpect_0, MoveStrings[require]), CheckIdentifier.RelearnMove);
|
|
}
|
|
}
|
|
|
|
private static CheckResult[] VerifyRelearnDexNav(PKM pkm, CheckResult[] result, EncounterSlot6AO slot)
|
|
{
|
|
// DexNav Pokémon can have 1 random egg move as a relearn move.
|
|
result[0] = !slot.CanBeDexNavMove(pkm.RelearnMove1) // not found
|
|
? new CheckResult(Severity.Invalid, LMoveRelearnDexNav, CheckIdentifier.RelearnMove)
|
|
: DummyValid;
|
|
|
|
// All other relearn moves must be empty.
|
|
result[3] = pkm.RelearnMove4 == 0 ? DummyValid : DummyNone;
|
|
result[2] = pkm.RelearnMove3 == 0 ? DummyValid : DummyNone;
|
|
result[1] = pkm.RelearnMove2 == 0 ? DummyValid : DummyNone;
|
|
|
|
return result;
|
|
}
|
|
|
|
private static CheckResult[] VerifyRelearnUnderground(PKM pkm, CheckResult[] result, EncounterSlot8b slot)
|
|
{
|
|
// Underground Pokémon can have 1 random egg move as a relearn move.
|
|
result[0] = !slot.CanBeUndergroundMove(pkm.RelearnMove1) // not found
|
|
? new CheckResult(Severity.Invalid, LMoveRelearnUnderground, CheckIdentifier.RelearnMove)
|
|
: DummyValid;
|
|
|
|
// All other relearn moves must be empty.
|
|
result[3] = pkm.RelearnMove4 == 0 ? DummyValid : DummyNone;
|
|
result[2] = pkm.RelearnMove3 == 0 ? DummyValid : DummyNone;
|
|
result[1] = pkm.RelearnMove2 == 0 ? DummyValid : DummyNone;
|
|
|
|
return result;
|
|
}
|
|
|
|
private static CheckResult[] VerifyRelearnNone(PKM pkm, CheckResult[] result)
|
|
{
|
|
// No relearn moves should be present.
|
|
result[3] = pkm.RelearnMove4 == 0 ? DummyValid : DummyNone;
|
|
result[2] = pkm.RelearnMove3 == 0 ? DummyValid : DummyNone;
|
|
result[1] = pkm.RelearnMove2 == 0 ? DummyValid : DummyNone;
|
|
result[0] = pkm.RelearnMove1 == 0 ? DummyValid : DummyNone;
|
|
return result;
|
|
}
|
|
|
|
internal static CheckResult[] VerifyEggMoveset(EncounterEgg e, CheckResult[] result, int[] moves, CheckIdentifier type = CheckIdentifier.RelearnMove)
|
|
{
|
|
int gen = e.Generation;
|
|
var origins = MoveBreed.Process(gen, e.Species, e.Form, e.Version, moves, out var valid);
|
|
if (valid)
|
|
{
|
|
for (int i = 0; i < result.Length; i++)
|
|
{
|
|
var msg = EggSourceUtil.GetSource(origins, gen, i);
|
|
result[i] = new CheckMoveResult(MoveSource.EggMove, gen, Severity.Valid, msg, type);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
var expected = MoveBreed.GetExpectedMoves(moves, e);
|
|
origins = MoveBreed.Process(gen, e.Species, e.Form, e.Version, expected, out _);
|
|
for (int i = 0; i < moves.Length; i++)
|
|
{
|
|
var msg = EggSourceUtil.GetSource(origins, gen, i);
|
|
var expect = expected[i];
|
|
CheckMoveResult line;
|
|
if (moves[i] == expect)
|
|
{
|
|
line = new CheckMoveResult(MoveSource.EggMove, gen, Severity.Valid, msg, type);
|
|
}
|
|
else
|
|
{
|
|
msg = string.Format(LMoveRelearnFExpect_0, MoveStrings[expect], msg);
|
|
line = new CheckMoveResult(MoveSource.EggMove, gen, Severity.Invalid, msg, type);
|
|
}
|
|
result[i] = line;
|
|
}
|
|
}
|
|
|
|
var dupe = IsAnyMoveDuplicate(moves);
|
|
if (dupe != NO_DUPE)
|
|
result[dupe] = new CheckMoveResult(MoveSource.EggMove, gen, Severity.Invalid, LMoveSourceDuplicate, type);
|
|
return result;
|
|
}
|
|
|
|
private const int NO_DUPE = -1;
|
|
|
|
private static int IsAnyMoveDuplicate(ReadOnlySpan<int> move)
|
|
{
|
|
int m1 = move[0];
|
|
int m2 = move[1];
|
|
|
|
if (m1 != 0 && m1 == m2)
|
|
return 1;
|
|
int m3 = move[2];
|
|
if (m3 != 0 && (m1 == m3 || m2 == m3))
|
|
return 2;
|
|
int m4 = move[3];
|
|
if (m4 != 0 && (m1 == m4 || m2 == m4 || m3 == m4))
|
|
return 3;
|
|
return NO_DUPE;
|
|
}
|
|
}
|
|
}
|