using System; using System.Collections.Generic; using System.Linq; using static PKHeX.Core.LegalityCheckStrings; using static PKHeX.Core.ParseSettings; using static PKHeX.Core.MoveSource; using static PKHeX.Core.Severity; using static PKHeX.Core.CheckIdentifier; namespace PKHeX.Core; /// /// Logic to verify the current . /// public static class VerifyCurrentMoves { /// /// Verifies the current moves of the data based on the provided . /// /// Data to check /// Encounter conditions and legality info /// Validity of the public static void VerifyMoves(PKM pk, LegalInfo info) { var parse = info.Moves; Array.ForEach(parse, p => p.Reset()); var currentMoves = pk.Moves; ParseMovesForEncounters(pk, parse, info, currentMoves); // Duplicate Moves Check VerifyNoEmptyDuplicates(currentMoves, parse); if (currentMoves[0] == 0) // Can't have an empty move slot for the first move. parse[0].FlagIllegal(LMoveSourceEmpty, CurrentMove); } private static void ParseMovesForEncounters(PKM pk, CheckMoveResult[] parse, LegalInfo info, int[] currentMoves) { if (pk.Species == (int)Species.Smeargle) // special handling for Smeargle { ParseMovesForSmeargle(pk, parse, currentMoves, info); // Smeargle can have any moves except a few return; } // gather valid moves for encounter species info.EncounterMoves = new ValidEncounterMoves(pk, info.EncounterMatch, info.EvoChainsAllGens); if (info.Generation >= 6) ParseMoves3DS(pk, parse, currentMoves, info); else ParseMovesPre3DS(pk, parse, currentMoves, info); } private static void ParseMovesForSmeargle(PKM pk, CheckMoveResult[] parse, int[] currentMoves, LegalInfo info) { if (!pk.IsEgg) { ParseMovesSketch(pk, parse, currentMoves); return; } // can only know sketch as egg var levelup = new int[info.EvoChainsAllGens.Length][]; levelup[pk.Format] = new[] { (int)Move.Sketch }; info.EncounterMoves = new ValidEncounterMoves(levelup); var source = new MoveParseSource { CurrentMoves = currentMoves }; ParseMoves(pk, source, info, parse); } private static void ParseMovesWasEggPreRelearn(PKM pk, CheckMoveResult[] parse, IReadOnlyList currentMoves, LegalInfo info, EncounterEgg e) { // Level up moves could not be inherited if Ditto is parent, // that means genderless species and male only species (except Nidoran-M and Volbeat; they breed with Nidoran-F and Illumise) could not have level up moves as an egg var pi = pk.PersonalInfo; var AllowLevelUp = !pi.Genderless && !(pi.OnlyMale && Breeding.MixedGenderBreeding.Contains(e.Species)); int BaseLevel = AllowLevelUp ? 100 : e.LevelMin; var LevelUp = MoveList.GetBaseEggMoves(pk, e.Species, e.Form, e.Version, BaseLevel); var TradebackPreevo = pk.Format == 2 && e.Species > 151; var NonTradebackLvlMoves = TradebackPreevo ? MoveList.GetExclusivePreEvolutionMoves(pk, e.Species, info.EvoChainsAllGens.Gen2, 2, e.Version).Where(m => m > Legal.MaxMoveID_1).ToArray() : Array.Empty(); var Egg = MoveEgg.GetEggMoves(pk.PersonalInfo, e.Species, e.Form, e.Version, e.Generation); if (info.Generation < 3 && pk.Format >= 7 && pk.VC1) Egg = Array.FindAll(Egg, m => m <= Legal.MaxMoveID_1); var specialMoves = e.CanHaveVoltTackle ? new[] { (int)Move.VoltTackle } : Array.Empty(); // Volt Tackle for bred Pichu line var source = new MoveParseSource { CurrentMoves = currentMoves, SpecialSource = specialMoves, NonTradeBackLevelUpMoves = NonTradebackLvlMoves, EggLevelUpSource = LevelUp, EggMoveSource = Egg, }; ParseMoves(pk, source, info, parse); } private static void ParseMovesSketch(PKM pk, CheckMoveResult[] parse, ReadOnlySpan currentMoves) { for (int i = parse.Length - 1; i >= 0; i--) { var r = parse[i]; var move = currentMoves[i]; if (move == 0) r.Set(None, pk.Format, Valid, LMoveSourceEmpty, CurrentMove); else if (Legal.IsValidSketch(move, pk.Format)) r.Set(Sketch, pk.Format, Valid, L_AValid, CurrentMove); else r.Set(Unknown, pk.Format, Invalid, LMoveSourceInvalidSketch, CurrentMove); } } private static void ParseMoves3DS(PKM pk, CheckMoveResult[] parse, IReadOnlyList currentMoves, LegalInfo info) { info.EncounterMoves.Relearn = info.Generation >= 6 ? pk.RelearnMoves : Array.Empty(); if (info.EncounterMatch is IMoveset) ParseMovesSpecialMoveset(pk, currentMoves, info, parse); else ParseMovesRelearn(pk, currentMoves, info, parse); } private static void ParseMovesPre3DS(PKM pk, CheckMoveResult[] parse, int[] currentMoves, LegalInfo info) { if (info.EncounterMatch is EncounterEgg e) { if (pk.IsEgg) VerifyRelearnMoves.VerifyEggMoveset(e, parse, currentMoves, CurrentMove); else ParseMovesWasEggPreRelearn(pk, parse, currentMoves, info, e); return; } // Not all games have a re-learner. Initial moves may not fill out all 4 slots. int gen = info.EncounterMatch.Generation; if (gen == 1 || (gen == 2 && !AllowGen2MoveReminder(pk))) ParseMovesGenGB(pk, currentMoves, info, parse); ParseMovesSpecialMoveset(pk, currentMoves, info, parse); } private static void ParseMovesGenGB(PKM pk, IReadOnlyList currentMoves, LegalInfo info, CheckMoveResult[] parse) { var enc = info.EncounterMatch; var evos = info.EvoChainsAllGens[enc.Generation]; var level = evos.Length > 0 ? evos[^1].LevelMin : enc.LevelMin; var InitialMoves = Array.Empty(); var games = enc.Generation == 1 ? GBRestrictions.GetGen1Versions(enc) : GBRestrictions.GetGen2Versions(enc, pk.Korean); foreach (var ver in games) { var VerInitialMoves = enc is IMoveset {Moves.Count: not 0 } x ? (int[])x.Moves : MoveLevelUp.GetEncounterMoves(enc.Species, 0, level, ver); if (VerInitialMoves.SequenceEqual(InitialMoves)) return; // Already checked this combination, and it wasn't valid. Don't bother repeating. var source = new MoveParseSource { CurrentMoves = currentMoves, Base = VerInitialMoves, }; ParseMoves(pk, source, info, parse); // Must have a minimum count of moves, depending on the tradeback state. if (pk is PK1 pk1) { int count = GBRestrictions.GetRequiredMoveCount(pk1, source.CurrentMoves, info, source.Base); if (count == 1) return; // Reverse for loop and break instead of 0..count continue -- early-breaks for the vast majority of cases. // We already flag for empty interstitial moveslots. for (int m = count - 1; m >= 0; m--) { var move = source.CurrentMoves[m]; if (move != 0) break; // There are ways to skip level up moves by leveling up more than once. // https://bulbapedia.bulbagarden.net/wiki/List_of_glitches_(Generation_I)#Level-up_learnset_skipping // Evolution canceling also leads to incorrect assumptions in the above used method, so just indicate them as fishy in that case. // Not leveled up? Not possible to be missing the move slot. var severity = enc.LevelMin == pk.CurrentLevel ? Invalid : Fishy; parse[m].Set(None, pk.Format, severity, LMoveSourceEmpty, CurrentMove); } } if (Array.TrueForAll(parse, z => z.Valid)) return; InitialMoves = VerInitialMoves; } } private static void ParseMovesSpecialMoveset(PKM pk, IReadOnlyList currentMoves, LegalInfo info, CheckMoveResult[] parse) { var source = new MoveParseSource { CurrentMoves = currentMoves, SpecialSource = GetSpecialMoves(info.EncounterMatch), }; ParseMoves(pk, source, info, parse); } private static IReadOnlyList GetSpecialMoves(IEncounterTemplate enc) { if (enc is IMoveset mg) return mg.Moves; return Array.Empty(); } private static void ParseMovesRelearn(PKM pk, IReadOnlyList currentMoves, LegalInfo info, CheckMoveResult[] parse) { var source = new MoveParseSource { CurrentMoves = currentMoves, SpecialSource = GetSpecialMoves(info.EncounterMatch), }; if (info.EncounterMatch is EncounterEgg e) source.EggMoveSource = MoveEgg.GetEggMoves(pk.PersonalInfo, e.Species, e.Form, e.Version, e.Generation); ParseMoves(pk, source, info, parse); var relearn = pk.RelearnMoves; for (int i = parse.Length - 1; i >= 0; i--) { var r = parse[i]; if (!r.IsRelearn && !pk.IsEgg) continue; if (relearn.Contains(currentMoves[i])) continue; r.FlagIllegal(string.Format(LMoveRelearnFMiss_0, r.Comment)); } } private static void ParseMoves(PKM pk, MoveParseSource source, LegalInfo info, CheckMoveResult[] parse) { // Special considerations! const int NoMinGeneration = 0; int minGeneration = NoMinGeneration; if (pk.IsOriginalMovesetDeleted()) { var (_, resetGame) = pk.IsMovesetRestricted(); minGeneration = resetGame.GetGeneration(); source.ResetSources(); } // Check empty moves and relearn moves before generation specific moves for (int m = parse.Length - 1; m >= 0; m--) { var move = source.CurrentMoves[m]; var r = parse[m]; if (move == 0) r.Set(None, pk.Format, Valid, LMoveSourceEmpty, CurrentMove); else if (minGeneration == NoMinGeneration && info.EncounterMoves.Relearn.Contains(move)) r.Set(Relearn, info.Generation, Valid, LMoveSourceRelearn, CurrentMove); } if (Array.TrueForAll(parse, z => z.IsParsed)) return; // Encapsulate arguments to simplify method calls var moveInfo = new LearnInfo(pk, source); // Check moves going backwards, marking the move valid in the most current generation when it can be learned int[] generations = GenerationTraversal.GetVisitedGenerationOrder(pk, info.EncounterOriginal.Generation); if (pk.Format <= 2) generations = Array.FindAll(generations, z => z < info.EncounterMoves.LevelUpMoves.Length); if (minGeneration != NoMinGeneration) generations = Array.FindAll(generations, z => z >= minGeneration); if (generations.Length != 0) { int lastgen = generations[^1]; foreach (var gen in generations) { ParseMovesByGeneration(pk, parse, gen, info, moveInfo, lastgen); if (Array.TrueForAll(parse, z => z.IsParsed)) return; } } if (pk.Species == (int)Species.Shedinja && info.Generation <= 4) ParseShedinjaEvolveMoves(pk, parse, source.CurrentMoves, info.EvoChainsAllGens); foreach (var r in parse) { if (!r.IsParsed) r.Set(Unknown, info.Generation, Invalid, LMoveSourceInvalid, CurrentMove); } } private static void ParseMovesByGeneration(PKM pk, CheckMoveResult[] parse, int gen, LegalInfo info, LearnInfo learnInfo, int last) { Span HMLearned = stackalloc bool[parse.Length]; bool KnowDefogWhirlpool = GetHMCompatibility(pk, parse, gen, learnInfo.Source.CurrentMoves, HMLearned); ParseMovesByGeneration(pk, parse, gen, info, learnInfo); if (gen == last) ParseMovesByGenerationLast(parse, learnInfo, info.EncounterMatch); switch (gen) { case 1 or 2: ParseMovesByGeneration12(pk, parse, learnInfo.Source.CurrentMoves, gen, info, learnInfo); break; case 3 or 4: if (pk.Format > gen) FlagIncompatibleTransferHMs45(parse, learnInfo.Source.CurrentMoves, gen, HMLearned, KnowDefogWhirlpool); break; } } private static void ParseMovesByGeneration(PKM pk, CheckMoveResult[] parse, int gen, LegalInfo info, LearnInfo learnInfo) { var moves = learnInfo.Source.CurrentMoves; bool native = gen == pk.Format; for (int m = parse.Length - 1; m >= 0; m--) { var r = parse[m]; if (IsCheckValid(r)) // already validated with another generation continue; int move = moves[m]; if (move == 0) continue; if (gen <= 2) { if (gen == 2 && !native && move > Legal.MaxMoveID_1 && pk.VC1) { r.Set(Unknown, gen, Invalid, LMoveSourceInvalid, CurrentMove); continue; } if (gen == 2 && learnInfo.Source.EggMoveSource.Contains(move)) r.Set(EggMove, gen, Valid, LMoveSourceEgg, CurrentMove); else if (learnInfo.Source.Base.Contains(move)) r.Set(Initial, gen, Valid, native ? LMoveSourceDefault : string.Format(LMoveFDefault_0, gen), CurrentMove); } if (info.EncounterMoves.LevelUpMoves[gen].Contains(move)) r.Set(LevelUp, gen, Valid, native ? LMoveSourceLevelUp : string.Format(LMoveFLevelUp_0, gen), CurrentMove); else if (info.EncounterMoves.TMHMMoves[gen].Contains(move)) r.Set(TMHM, gen, Valid, native ? LMoveSourceTMHM : string.Format(LMoveFTMHM_0, gen), CurrentMove); else if (info.EncounterMoves.TutorMoves[gen].Contains(move)) r.Set(Tutor, gen, Valid, native ? LMoveSourceTutor : string.Format(LMoveFTutor_0, gen), CurrentMove); else if (gen == info.Generation && learnInfo.Source.SpecialSource.Contains(move)) r.Set(Special, gen, Valid, LMoveSourceSpecial, CurrentMove); else if (gen >= 8 && MoveEgg.GetIsSharedEggMove(pk, gen, move)) r.Set(Shared, gen, Valid, native ? LMoveSourceShared : string.Format(LMoveSourceSharedF, gen), CurrentMove); if (gen >= 3 || !IsCheckValid(r)) continue; // Gen1/Gen2 only below if (gen == 2 && learnInfo.Source.NonTradeBackLevelUpMoves.Contains(m)) { learnInfo.Gen2PreevoMoves.Add(m); } else if (gen == 1) { learnInfo.Gen1Moves.Add(m); if (learnInfo.Gen2PreevoMoves.Count != 0) learnInfo.MixedGen12NonTradeback = true; } } } private static void ParseMovesByGeneration12(PKM pk, CheckMoveResult[] parse, IReadOnlyList currentMoves, int gen, LegalInfo info, LearnInfo learnInfo) { // Mark the gen 1 exclusive moves as illegal because the pokemon also have Non tradeback egg moves. if (learnInfo.MixedGen12NonTradeback) { foreach (int m in learnInfo.Gen1Moves) parse[m].FlagIllegal(LG1MoveExclusive, CurrentMove); foreach (int m in learnInfo.Gen2PreevoMoves) parse[m].FlagIllegal(LG1TradebackPreEvoMove, CurrentMove); } if (gen == 1 && pk.Format == 1 && !AllowGen1Tradeback) { ParseRedYellowIncompatibleMoves(pk, parse, currentMoves); ParseEvolutionsIncompatibleMoves(pk, parse, currentMoves, info.EncounterMoves.TMHMMoves[1]); } } private static void ParseMovesByGenerationLast(CheckMoveResult[] parse, LearnInfo learnInfo, IEncounterTemplate enc) { int gen = enc.Generation; ParseEggMovesInherited(parse, gen, learnInfo); ParseEggMoves(parse, gen, learnInfo); ParseEggMovesRemaining(parse, learnInfo, enc); } private static void ParseEggMovesInherited(CheckMoveResult[] parse, int gen, LearnInfo learnInfo) { var moves = learnInfo.Source.CurrentMoves; // Check higher-level moves after all the moves but just before egg moves to differentiate it from normal level up moves // Also check if the base egg moves is a non tradeback move for (int m = parse.Length - 1; m >= 0; m--) { var r = parse[m]; if (IsCheckValid(r)) // already validated { if (gen == 2 && r.Generation != 1) continue; } int move = moves[m]; if (move == 0) continue; if (!learnInfo.Source.EggLevelUpSource.Contains(move)) // Check if contains level-up egg moves from parents continue; if (learnInfo.IsGen2Pkm && learnInfo.Gen1Moves.Count != 0 && move > Legal.MaxMoveID_1) { r.Set(InheritLevelUp, gen, Invalid, LG1MoveTradeback, CurrentMove); learnInfo.MixedGen12NonTradeback = true; } else { r.Set(InheritLevelUp, gen, Valid, LMoveEggLevelUp, CurrentMove); } learnInfo.LevelUpEggMoves.Add(m); if (gen == 2 && learnInfo.Gen1Moves.Contains(m)) learnInfo.Gen1Moves.Remove(m); } } private static void ParseEggMoves(CheckMoveResult[] parse, int gen, LearnInfo learnInfo) { var moves = learnInfo.Source.CurrentMoves; // Check egg moves after all the generations and all the moves, every move that can't be learned in another source should have preference // the moves that can only be learned from egg moves should in the future check if the move combinations can be breed in gens 2 to 5 for (int m = parse.Length - 1; m >= 0; m--) { var r = parse[m]; if (IsCheckValid(r)) continue; int move = moves[m]; if (move == 0) continue; bool wasEggMove = learnInfo.Source.EggMoveSource.Contains(move); if (wasEggMove) { // To learn exclusive generation 1 moves the pokemon was tradeback, but it can't be trade to generation 1 // without removing moves above MaxMoveID_1, egg moves above MaxMoveID_1 and gen 1 moves are incompatible if (learnInfo.IsGen2Pkm && learnInfo.Gen1Moves.Count != 0 && move > Legal.MaxMoveID_1) { r.Set(EggMove, gen, Invalid, LG1MoveTradeback, CurrentMove); learnInfo.MixedGen12NonTradeback = true; } else { r.Set(EggMove, gen, Valid, LMoveSourceEgg, CurrentMove); } learnInfo.EggMovesLearned.Add(m); } if (!learnInfo.Source.EggEventSource.Contains(move)) continue; if (!wasEggMove) { if (learnInfo.IsGen2Pkm && learnInfo.Gen1Moves.Count != 0 && move > Legal.MaxMoveID_1) { r.Set(SpecialEgg, gen, Invalid, LG1MoveTradeback, CurrentMove); learnInfo.MixedGen12NonTradeback = true; } else { r.Set(SpecialEgg, gen, Valid, LMoveSourceEggEvent, CurrentMove); } } learnInfo.EventEggMoves.Add(m); } } private static void ParseEggMovesRemaining(CheckMoveResult[] parse, LearnInfo learnInfo, IEncounterTemplate enc) { // A pokemon could have normal egg moves and regular egg moves // Only if all regular egg moves are event egg moves or all event egg moves are regular egg moves var RegularEggMovesLearned = learnInfo.EggMovesLearned.FindAll(learnInfo.LevelUpEggMoves.Contains); if (RegularEggMovesLearned.Count != 0 && learnInfo.EventEggMoves.Count != 0) { // Moves that are egg moves or event egg moves but not both var IncompatibleEggMoves = RegularEggMovesLearned.Except(learnInfo.EventEggMoves).Union(learnInfo.EventEggMoves.Except(RegularEggMovesLearned)); foreach (int m in IncompatibleEggMoves) { bool isEvent = learnInfo.EventEggMoves.Contains(m); if (isEvent) { if (!learnInfo.EggMovesLearned.Contains(m)) parse[m].FlagIllegal(LMoveEggIncompatibleEvent, CurrentMove); } else { if (learnInfo.EggMovesLearned.Contains(m)) parse[m].FlagIllegal(LMoveEggIncompatible, CurrentMove); else if (learnInfo.LevelUpEggMoves.Contains(m)) parse[m].FlagIllegal(LMoveEventEggLevelUp, CurrentMove); } } } else if (enc is not EncounterEgg) { // Event eggs cannot inherit moves from parents; they are not bred. var gift = enc is EncounterStatic {Gift: true}; // otherwise, EncounterInvalid foreach (int m in RegularEggMovesLearned) { if (learnInfo.EggMovesLearned.Contains(m)) parse[m].FlagIllegal(gift ? LMoveEggMoveGift : LMoveEggInvalidEvent, CurrentMove); else if (learnInfo.LevelUpEggMoves.Contains(m)) parse[m].FlagIllegal(gift ? LMoveEggInvalidEventLevelUpGift : LMoveEggInvalidEventLevelUp, CurrentMove); } } } private static void ParseRedYellowIncompatibleMoves(PKM pk, CheckMoveResult[] parse, IReadOnlyList currentMoves) { var incompatible = GetIncompatibleRBYMoves(pk, currentMoves); if (incompatible.Count == 0) return; for (int m = parse.Length - 1; m >= 0; m--) { if (incompatible.Contains(currentMoves[m])) parse[m].FlagIllegal(LG1MoveLearnSameLevel, CurrentMove); } } private static IList GetIncompatibleRBYMoves(PKM pk, IReadOnlyList currentMoves) { // Check moves that are learned at the same level in Red/Blue and Yellow, these are illegal because there is no Move Reminder in Gen1. // There are only two incompatibilities for Gen1; there are no illegal combination in Gen2. switch (pk.Species) { // Vaporeon in Yellow learns Mist and Haze at level 42, Mist can only be learned if it leveled up in the daycare // Vaporeon in Red/Blue learns Acid Armor at level 42 and level 47 in Yellow case (int)Species.Vaporeon when pk.CurrentLevel < 47 && currentMoves.Contains((int)Move.AcidArmor): { var incompatible = new List(3); if (currentMoves.Contains((int)Move.Mist)) incompatible.Add((int)Move.Mist); if (currentMoves.Contains((int)Move.Haze)) incompatible.Add((int)Move.Haze); if (incompatible.Count != 0) incompatible.Add((int)Move.AcidArmor); else return Array.Empty(); return incompatible; } // Flareon in Yellow learns Smog at level 42 // Flareon in Red Blue learns Leer at level 42 and level 47 in Yellow case (int)Species.Flareon when pk.CurrentLevel < 47 && currentMoves.Contains((int)Move.Leer) && currentMoves.Contains((int)Move.Smog): return new[] { (int)Move.Leer, (int)Move.Smog }; default: return Array.Empty(); } } private static void ParseEvolutionsIncompatibleMoves(PKM pk, CheckMoveResult[] parse, IReadOnlyList moves, IReadOnlyList tmhm) { GBRestrictions.GetIncompatibleEvolutionMoves(pk, moves, tmhm, out var prevSpeciesID, out var incompatPrev, out var incompatCurr); if (prevSpeciesID == 0) return; var prev = SpeciesStrings[prevSpeciesID]; var curr = SpeciesStrings[pk.Species]; for (int m = parse.Length - 1; m >= 0; m--) { if (incompatCurr.Contains(moves[m])) parse[m].FlagIllegal(string.Format(LMoveEvoFLower, curr, prev), CurrentMove); if (incompatPrev.Contains(moves[m])) parse[m].FlagIllegal(string.Format(LMoveEvoFHigher, curr, prev), CurrentMove); } } private static void ParseShedinjaEvolveMoves(PKM pk, CheckMoveResult[] parse, IReadOnlyList currentMoves, EvolutionHistory evos) { int shedinjaEvoMoveIndex = 0; var format = pk.Format; for (int gen = Math.Min(format, 4); gen >= 3; gen--) { if (evos[gen].Length != 2) continue; // Was not evolved in this generation if (gen == 4 && pk.Ball != 4 && !(pk.Ball == (int)Ball.Sport && pk.HGSS)) continue; // Was definitively evolved in Gen3 var maxLevel = pk.CurrentLevel; var ninjaskMoves = MoveList.GetShedinjaEvolveMoves(pk, gen, maxLevel); bool native = gen == format; for (int m = parse.Length - 1; m >= 0; m--) { var r = parse[m]; if (IsCheckValid(r)) // already validated continue; if (!ninjaskMoves.Contains(currentMoves[m])) continue; // Can't have more than one Ninjask exclusive move on Shedinja if (shedinjaEvoMoveIndex != 0) { r.FlagIllegal(LMoveNincada, CurrentMove); continue; } var msg = native ? LMoveNincadaEvo : string.Format(LMoveNincadaEvoF_0, gen); r.Set(ShedinjaEvo, gen, Valid, msg, CurrentMove); shedinjaEvoMoveIndex = m; } } if (shedinjaEvoMoveIndex == 0) return; // Double check that the Ninjask move level isn't less than any Nincada move level { var r = parse[shedinjaEvoMoveIndex]; if (r.Source != LevelUp) return; var move = currentMoves[shedinjaEvoMoveIndex]; int levelS = MoveList.GetShedinjaMoveLevel((int)Species.Shedinja, move, r.Generation); if (levelS > 0) return; int levelN = MoveList.GetShedinjaMoveLevel((int)Species.Nincada, move, r.Generation); int levelJ = MoveList.GetShedinjaMoveLevel((int)Species.Ninjask, move, r.Generation); if (levelN > levelJ) r.FlagIllegal(string.Format(LMoveEvoFHigher, SpeciesStrings[(int)Species.Nincada], SpeciesStrings[(int)Species.Ninjask]), CurrentMove); } } private static bool GetHMCompatibility(PKM pk, IReadOnlyList parse, int gen, IReadOnlyList moves, Span HMLearned) { // Check if pokemon knows HM moves from generation 3 and 4 but are not valid yet, that means it cant learn the HMs in future generations if (gen == 4 && pk.Format > 4) { FlagIsHMSource(HMLearned, Legal.HM_4_RemovePokeTransfer); return moves.Where((m, i) => IsDefogWhirl(m) && IsCheckInvalid(parse[i])).Count() == 2; } if (gen == 3 && pk.Format > 3) FlagIsHMSource(HMLearned, Legal.HM_3); return false; void FlagIsHMSource(Span flags, ICollection source) { for (int i = parse.Count - 1; i >= 0; i--) flags[i] = IsCheckInvalid(parse[i]) && source.Contains(moves[i]); } } private static bool IsDefogWhirl(int move) => move is (int)Move.Defog or (int)Move.Whirlpool; private static int GetDefogWhirlCount(IReadOnlyList parse, IReadOnlyList moves) { int ctr = 0; for (int i = moves.Count - 1; i >= 0; i--) { if (!IsDefogWhirl(moves[i])) continue; var r = parse[i]; if (!r.Valid || r.Generation >= 5) continue; ctr++; } return ctr; } private static bool IsCheckInvalid(CheckMoveResult chk) => chk.IsParsed && !chk.Valid; private static bool IsCheckValid(CheckMoveResult chk) => chk.IsParsed && chk.Valid; private static void FlagIncompatibleTransferHMs45(IReadOnlyList parse, IReadOnlyList currentMoves, int gen, ReadOnlySpan HMLearned, bool KnowDefogWhirlpool) { // After all the moves from the generations 3 and 4, // including egg moves if is the origin generation because some hidden moves are also special egg moves in gen 3 // Check if the marked hidden moves that were invalid at the start are now marked as valid, that means // the hidden move was learned in gen 3 or 4 but was not removed when transfer to 4 or 5 if (KnowDefogWhirlpool) { int invalidCount = GetDefogWhirlCount(parse, currentMoves); if (invalidCount == 2) // can't know both at the same time { for (int i = parse.Count - 1; i >= 0; i--) // flag both moves { if (IsDefogWhirl(currentMoves[i])) parse[i].FlagIllegal(LTransferMoveG4HM, CurrentMove); } } } // Flag moves that are only legal when learned from a past-gen HM source for (int i = HMLearned.Length - 1; i >= 0; i--) { if (HMLearned[i] && IsCheckValid(parse[i])) parse[i].FlagIllegal(string.Format(LTransferMoveHM, gen, gen + 1), CurrentMove); } } private static void VerifyNoEmptyDuplicates(ReadOnlySpan moves, ReadOnlySpan parse) { bool emptySlot = false; for (int i = 0; i < parse.Length; i++) { var move = moves[i]; if (move == 0) { emptySlot = true; continue; } // If an empty slot was noted for a prior move, flag the empty slots. if (emptySlot) { FlagEmptySlotsBeforeIndex(moves, parse, i); emptySlot = false; continue; } // Check for same move in next move slots FlagDuplicateMovesAfterIndex(moves, parse, i, move); } } private static void FlagDuplicateMovesAfterIndex(ReadOnlySpan moves, ReadOnlySpan parse, int index, int move) { for (int i = parse.Length - 1; i > index; i--) { if (moves[i] != move) continue; parse[index].FlagIllegal(LMoveSourceEmpty); return; } } private static void FlagEmptySlotsBeforeIndex(ReadOnlySpan moves, ReadOnlySpan parse, int index) { for (int i = index - 1; i >= 0; i--) { if (moves[i] != 0) return; parse[i].FlagIllegal(LMoveSourceEmpty); } } }