2017-05-28 04:17:53 +00:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using static PKHeX . Core . LegalityCheckStrings ;
2020-11-28 02:45:06 +00:00
using static PKHeX . Core . ParseSettings ;
2017-05-28 04:17:53 +00:00
2019-02-22 04:44:31 +00:00
using static PKHeX . Core . MoveSource ;
using static PKHeX . Core . Severity ;
using static PKHeX . Core . CheckIdentifier ;
2022-06-18 18:04:24 +00:00
namespace PKHeX.Core ;
/// <summary>
/// Logic to verify the current <see cref="PKM.Moves"/>.
/// </summary>
public static class VerifyCurrentMoves
2017-05-28 04:17:53 +00:00
{
2017-10-24 06:12:58 +00:00
/// <summary>
2022-06-18 18:04:24 +00:00
/// Verifies the current moves of the <see cref="pk"/> data based on the provided <see cref="info"/>.
2017-10-24 06:12:58 +00:00
/// </summary>
2022-06-18 18:04:24 +00:00
/// <param name="pk">Data to check</param>
/// <param name="info">Encounter conditions and legality info</param>
/// <returns>Validity of the <see cref="PKM.Moves"/></returns>
public static void VerifyMoves ( PKM pk , LegalInfo info )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
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 ) ;
}
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesForEncounters ( PKM pk , CheckMoveResult [ ] parse , LegalInfo info , int [ ] currentMoves )
{
if ( pk . Species = = ( int ) Species . Smeargle ) // special handling for Smeargle
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
ParseMovesForSmeargle ( pk , parse , currentMoves , info ) ; // Smeargle can have any moves except a few
return ;
2017-05-28 04:17:53 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
// gather valid moves for encounter species
info . EncounterMoves = new ValidEncounterMoves ( pk , info . EncounterMatch , info . EvoChainsAllGens ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( info . Generation > = 6 )
ParseMoves3DS ( pk , parse , currentMoves , info ) ;
else
ParseMovesPre3DS ( pk , parse , currentMoves , info ) ;
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesForSmeargle ( PKM pk , CheckMoveResult [ ] parse , int [ ] currentMoves , LegalInfo info )
{
if ( ! pk . IsEgg )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
ParseMovesSketch ( pk , parse , currentMoves ) ;
return ;
}
2017-11-18 06:19:23 +00:00
2022-06-18 18:04:24 +00:00
// 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 ) ;
}
2017-11-18 06:19:23 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesWasEggPreRelearn ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > 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 ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
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 < int > ( ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
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 ) ;
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
var specialMoves = e . CanHaveVoltTackle ? new [ ] { ( int ) Move . VoltTackle } : Array . Empty < int > ( ) ; // Volt Tackle for bred Pichu line
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
var source = new MoveParseSource
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
CurrentMoves = currentMoves ,
SpecialSource = specialMoves ,
NonTradeBackLevelUpMoves = NonTradebackLvlMoves ,
EggLevelUpSource = LevelUp ,
EggMoveSource = Egg ,
} ;
ParseMoves ( pk , source , info , parse ) ;
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesSketch ( PKM pk , CheckMoveResult [ ] parse , ReadOnlySpan < int > currentMoves )
{
for ( int i = parse . Length - 1 ; i > = 0 ; i - - )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
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 ) ;
2022-03-13 01:06:03 +00:00
else
2022-06-18 18:04:24 +00:00
r . Set ( Unknown , pk . Format , Invalid , LMoveSourceInvalidSketch , CurrentMove ) ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMoves3DS ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > currentMoves , LegalInfo info )
{
info . EncounterMoves . Relearn = info . Generation > = 6 ? pk . RelearnMoves : Array . Empty < int > ( ) ;
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 )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
if ( pk . IsEgg )
VerifyRelearnMoves . VerifyEggMoveset ( e , parse , currentMoves , CurrentMove ) ;
else
ParseMovesWasEggPreRelearn ( pk , parse , currentMoves , info , e ) ;
return ;
}
2020-05-17 19:32:28 +00:00
2022-06-18 18:04:24 +00:00
// 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 ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
ParseMovesSpecialMoveset ( pk , currentMoves , info , parse ) ;
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesGenGB ( PKM pk , IReadOnlyList < int > 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 < int > ( ) ;
var games = enc . Generation = = 1 ? GBRestrictions . GetGen1Versions ( enc ) : GBRestrictions . GetGen2Versions ( enc , pk . Korean ) ;
foreach ( var ver in games )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
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
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
CurrentMoves = currentMoves ,
Base = VerInitialMoves ,
} ;
ParseMoves ( pk , source , info , parse ) ;
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
// 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 ;
2021-08-20 22:59:54 +00:00
2022-06-18 18:04:24 +00:00
// 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 - - )
2021-08-20 22:59:54 +00:00
{
2022-06-18 18:04:24 +00:00
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 ) ;
2021-08-20 22:59:54 +00:00
}
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
if ( Array . TrueForAll ( parse , z = > z . Valid ) )
return ;
InitialMoves = VerInitialMoves ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesSpecialMoveset ( PKM pk , IReadOnlyList < int > currentMoves , LegalInfo info , CheckMoveResult [ ] parse )
{
var source = new MoveParseSource
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
CurrentMoves = currentMoves ,
SpecialSource = GetSpecialMoves ( info . EncounterMatch ) ,
} ;
ParseMoves ( pk , source , info , parse ) ;
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static IReadOnlyList < int > GetSpecialMoves ( IEncounterTemplate enc )
{
if ( enc is IMoveset mg )
return mg . Moves ;
return Array . Empty < int > ( ) ;
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesRelearn ( PKM pk , IReadOnlyList < int > currentMoves , LegalInfo info , CheckMoveResult [ ] parse )
{
var source = new MoveParseSource
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
CurrentMoves = currentMoves ,
SpecialSource = GetSpecialMoves ( info . EncounterMatch ) ,
} ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( info . EncounterMatch is EncounterEgg e )
source . EggMoveSource = MoveEgg . GetEggMoves ( pk . PersonalInfo , e . Species , e . Form , e . Version , e . Generation ) ;
2017-10-17 00:24:19 +00:00
2022-06-18 18:04:24 +00:00
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 ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
r . FlagIllegal ( string . Format ( LMoveRelearnFMiss_0 , r . Comment ) ) ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMoves ( PKM pk , MoveParseSource source , LegalInfo info , CheckMoveResult [ ] parse )
{
// Special considerations!
const int NoMinGeneration = 0 ;
int minGeneration = NoMinGeneration ;
if ( pk . IsOriginalMovesetDeleted ( ) )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
var ( _ , resetGame ) = pk . IsMovesetRestricted ( ) ;
minGeneration = resetGame . GetGeneration ( ) ;
source . ResetSources ( ) ;
}
2020-06-19 23:51:15 +00:00
2022-06-18 18:04:24 +00:00
// 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 ) ;
}
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( Array . TrueForAll ( parse , z = > z . IsParsed ) )
return ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
// 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 ) ;
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
if ( generations . Length ! = 0 )
{
int lastgen = generations [ ^ 1 ] ;
foreach ( var gen in generations )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
ParseMovesByGeneration ( pk , parse , gen , info , moveInfo , lastgen ) ;
if ( Array . TrueForAll ( parse , z = > z . IsParsed ) )
return ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( pk . Species = = ( int ) Species . Shedinja & & info . Generation < = 4 )
ParseShedinjaEvolveMoves ( pk , parse , source . CurrentMoves , info . EvoChainsAllGens ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
foreach ( var r in parse )
{
if ( ! r . IsParsed )
r . Set ( Unknown , info . Generation , Invalid , LMoveSourceInvalid , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesByGeneration ( PKM pk , CheckMoveResult [ ] parse , int gen , LegalInfo info , LearnInfo learnInfo , int last )
{
Span < bool > HMLearned = stackalloc bool [ parse . Length ] ;
bool KnowDefogWhirlpool = GetHMCompatibility ( pk , parse , gen , learnInfo . Source . CurrentMoves , HMLearned ) ;
ParseMovesByGeneration ( pk , parse , gen , info , learnInfo ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( gen = = last )
ParseMovesByGenerationLast ( parse , learnInfo , info . EncounterMatch ) ;
2017-06-07 03:10:05 +00:00
2022-06-18 18:04:24 +00:00
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 ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
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 - - )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
var r = parse [ m ] ;
if ( IsCheckValid ( r ) ) // already validated with another generation
continue ;
int move = moves [ m ] ;
if ( move = = 0 )
continue ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( gen < = 2 )
{
if ( gen = = 2 & & ! native & & move > Legal . MaxMoveID_1 & & pk . VC1 )
2018-04-01 22:04:53 +00:00
{
2022-06-18 18:04:24 +00:00
r . Set ( Unknown , gen , Invalid , LMoveSourceInvalid , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
continue ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
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 ;
2017-09-02 06:15:57 +00:00
}
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseMovesByGeneration12 ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > 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 )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
foreach ( int m in learnInfo . Gen1Moves )
parse [ m ] . FlagIllegal ( LG1MoveExclusive , CurrentMove ) ;
2017-06-07 03:10:05 +00:00
2022-06-18 18:04:24 +00:00
foreach ( int m in learnInfo . Gen2PreevoMoves )
parse [ m ] . FlagIllegal ( LG1TradebackPreEvoMove , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
if ( gen = = 1 & & pk . Format = = 1 & & ! AllowGen1Tradeback )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
ParseRedYellowIncompatibleMoves ( pk , parse , currentMoves ) ;
ParseEvolutionsIncompatibleMoves ( pk , parse , currentMoves , info . EncounterMoves . TMHMMoves [ 1 ] ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
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 - - )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
var r = parse [ m ] ;
if ( IsCheckValid ( r ) ) // already validated
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
if ( gen = = 2 & & r . Generation ! = 1 )
2017-09-02 06:15:57 +00:00
continue ;
2022-06-18 18:04:24 +00:00
}
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 ) ;
}
}
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
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
2020-10-04 19:21:05 +00:00
if ( learnInfo . IsGen2Pkm & & learnInfo . Gen1Moves . Count ! = 0 & & move > Legal . MaxMoveID_1 )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
r . Set ( EggMove , gen , Invalid , LG1MoveTradeback , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
learnInfo . MixedGen12NonTradeback = true ;
2017-06-07 03:10:05 +00:00
}
2017-09-02 06:15:57 +00:00
else
2018-08-03 03:11:42 +00:00
{
2022-06-18 18:04:24 +00:00
r . Set ( EggMove , gen , Valid , LMoveSourceEgg , CurrentMove ) ;
2018-08-03 03:11:42 +00:00
}
2022-06-18 18:04:24 +00:00
learnInfo . EggMovesLearned . Add ( m ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
if ( ! learnInfo . Source . EggEventSource . Contains ( move ) )
continue ;
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
if ( ! wasEggMove )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
if ( learnInfo . IsGen2Pkm & & learnInfo . Gen1Moves . Count ! = 0 & & move > Legal . MaxMoveID_1 )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
r . Set ( SpecialEgg , gen , Invalid , LG1MoveTradeback , CurrentMove ) ;
learnInfo . MixedGen12NonTradeback = true ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
else
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
r . Set ( SpecialEgg , gen , Valid , LMoveSourceEggEvent , CurrentMove ) ;
2017-05-28 04:17:53 +00:00
}
}
2022-06-18 18:04:24 +00:00
learnInfo . EventEggMoves . Add ( m ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
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 )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
// 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 )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
bool isEvent = learnInfo . EventEggMoves . Contains ( m ) ;
if ( isEvent )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
if ( ! learnInfo . EggMovesLearned . Contains ( m ) )
parse [ m ] . FlagIllegal ( LMoveEggIncompatibleEvent , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
else
2017-09-02 06:15:57 +00:00
{
if ( learnInfo . EggMovesLearned . Contains ( m ) )
2022-06-18 18:04:24 +00:00
parse [ m ] . FlagIllegal ( LMoveEggIncompatible , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
else if ( learnInfo . LevelUpEggMoves . Contains ( m ) )
2022-06-18 18:04:24 +00:00
parse [ m ] . FlagIllegal ( LMoveEventEggLevelUp , CurrentMove ) ;
2017-09-02 06:15:57 +00:00
}
2017-05-28 04:17:53 +00:00
}
}
2022-06-18 18:04:24 +00:00
else if ( enc is not EncounterEgg )
2018-07-02 04:34:17 +00:00
{
2022-06-18 18:04:24 +00:00
// 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 ) ;
2018-08-03 03:11:42 +00:00
}
2018-07-02 04:34:17 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-07-02 04:34:17 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseRedYellowIncompatibleMoves ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > currentMoves )
{
var incompatible = GetIncompatibleRBYMoves ( pk , currentMoves ) ;
if ( incompatible . Count = = 0 )
return ;
for ( int m = parse . Length - 1 ; m > = 0 ; m - - )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
if ( incompatible . Contains ( currentMoves [ m ] ) )
parse [ m ] . FlagIllegal ( LG1MoveLearnSameLevel , CurrentMove ) ;
}
}
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
private static IList < int > GetIncompatibleRBYMoves ( PKM pk , IReadOnlyList < int > 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.
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
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 < int > ( 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 < int > ( ) ;
return incompatible ;
2019-03-17 03:07:30 +00:00
}
2022-06-18 18:04:24 +00:00
// 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 < int > ( ) ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-07-02 04:34:17 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseEvolutionsIncompatibleMoves ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > moves , IReadOnlyList < int > tmhm )
{
GBRestrictions . GetIncompatibleEvolutionMoves ( pk , moves , tmhm ,
out var prevSpeciesID ,
out var incompatPrev ,
out var incompatCurr ) ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( prevSpeciesID = = 0 )
return ;
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
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 ) ;
2018-10-07 22:36:32 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-10-07 22:36:32 +00:00
2022-06-18 18:04:24 +00:00
private static void ParseShedinjaEvolveMoves ( PKM pk , CheckMoveResult [ ] parse , IReadOnlyList < int > currentMoves , EvolutionHistory evos )
{
int shedinjaEvoMoveIndex = 0 ;
var format = pk . Format ;
for ( int gen = Math . Min ( format , 4 ) ; gen > = 3 ; gen - - )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
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 - - )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
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 )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
r . FlagIllegal ( LMoveNincada , CurrentMove ) ;
continue ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
var msg = native ? LMoveNincadaEvo : string . Format ( LMoveNincadaEvoF_0 , gen ) ;
r . Set ( ShedinjaEvo , gen , Valid , msg , CurrentMove ) ;
shedinjaEvoMoveIndex = m ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
if ( shedinjaEvoMoveIndex = = 0 )
return ;
2018-02-02 03:57:02 +00:00
2022-06-18 18:04:24 +00:00
// Double check that the Ninjask move level isn't less than any Nincada move level
{
var r = parse [ shedinjaEvoMoveIndex ] ;
if ( r . Source ! = LevelUp )
return ;
2022-03-13 01:06:03 +00:00
2022-06-18 18:04:24 +00:00
var move = currentMoves [ shedinjaEvoMoveIndex ] ;
int levelS = MoveList . GetShedinjaMoveLevel ( ( int ) Species . Shedinja , move , r . Generation ) ;
if ( levelS > 0 )
return ;
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
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 ) ;
2017-05-28 04:17:53 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static bool GetHMCompatibility ( PKM pk , IReadOnlyList < CheckMoveResult > parse , int gen , IReadOnlyList < int > moves , Span < bool > 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 )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
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 ;
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
void FlagIsHMSource ( Span < bool > flags , ICollection < int > source )
{
for ( int i = parse . Count - 1 ; i > = 0 ; i - - )
flags [ i ] = IsCheckInvalid ( parse [ i ] ) & & source . Contains ( moves [ i ] ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static bool IsDefogWhirl ( int move ) = > move is ( int ) Move . Defog or ( int ) Move . Whirlpool ;
private static int GetDefogWhirlCount ( IReadOnlyList < CheckMoveResult > parse , IReadOnlyList < int > moves )
{
int ctr = 0 ;
for ( int i = moves . Count - 1 ; i > = 0 ; i - - )
2022-03-13 01:06:03 +00:00
{
2022-06-18 18:04:24 +00:00
if ( ! IsDefogWhirl ( moves [ i ] ) )
continue ;
var r = parse [ i ] ;
if ( ! r . Valid | | r . Generation > = 5 )
continue ;
ctr + + ;
2022-03-13 01:06:03 +00:00
}
2022-06-18 18:04:24 +00:00
return ctr ;
}
2022-03-13 01:06:03 +00:00
2022-06-18 18:04:24 +00:00
private static bool IsCheckInvalid ( CheckMoveResult chk ) = > chk . IsParsed & & ! chk . Valid ;
private static bool IsCheckValid ( CheckMoveResult chk ) = > chk . IsParsed & & chk . Valid ;
2018-08-03 03:11:42 +00:00
2022-06-18 18:04:24 +00:00
private static void FlagIncompatibleTransferHMs45 ( IReadOnlyList < CheckMoveResult > parse , IReadOnlyList < int > currentMoves , int gen , ReadOnlySpan < bool > 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 )
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
int invalidCount = GetDefogWhirlCount ( parse , currentMoves ) ;
if ( invalidCount = = 2 ) // can't know both at the same time
2017-09-02 06:15:57 +00:00
{
2022-06-18 18:04:24 +00:00
for ( int i = parse . Count - 1 ; i > = 0 ; i - - ) // flag both moves
2018-08-03 03:11:42 +00:00
{
2022-06-18 18:04:24 +00:00
if ( IsDefogWhirl ( currentMoves [ i ] ) )
parse [ i ] . FlagIllegal ( LTransferMoveG4HM , CurrentMove ) ;
2018-08-03 03:11:42 +00:00
}
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2017-09-02 06:15:57 +00:00
2022-06-18 18:04:24 +00:00
// 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 ) ;
2017-09-02 06:15:57 +00:00
}
2022-06-18 18:04:24 +00:00
}
2017-05-28 04:17:53 +00:00
2022-06-18 18:04:24 +00:00
private static void VerifyNoEmptyDuplicates ( ReadOnlySpan < int > moves , ReadOnlySpan < CheckMoveResult > parse )
{
bool emptySlot = false ;
for ( int i = 0 ; i < parse . Length ; i + + )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
var move = moves [ i ] ;
if ( move = = 0 )
2017-05-28 04:17:53 +00:00
{
2022-06-18 18:04:24 +00:00
emptySlot = true ;
continue ;
}
2020-08-30 22:43:29 +00:00
2022-06-18 18:04:24 +00:00
// If an empty slot was noted for a prior move, flag the empty slots.
if ( emptySlot )
{
FlagEmptySlotsBeforeIndex ( moves , parse , i ) ;
emptySlot = false ;
continue ;
2020-08-30 22:43:29 +00:00
}
2022-06-18 18:04:24 +00:00
// Check for same move in next move slots
FlagDuplicateMovesAfterIndex ( moves , parse , i , move ) ;
2020-08-30 22:43:29 +00:00
}
2022-06-18 18:04:24 +00:00
}
2020-08-30 22:43:29 +00:00
2022-06-18 18:04:24 +00:00
private static void FlagDuplicateMovesAfterIndex ( ReadOnlySpan < int > moves , ReadOnlySpan < CheckMoveResult > parse , int index , int move )
{
for ( int i = parse . Length - 1 ; i > index ; i - - )
2020-08-30 22:43:29 +00:00
{
2022-06-18 18:04:24 +00:00
if ( moves [ i ] ! = move )
continue ;
parse [ index ] . FlagIllegal ( LMoveSourceEmpty ) ;
return ;
2020-08-30 22:43:29 +00:00
}
2022-06-18 18:04:24 +00:00
}
2020-08-30 22:43:29 +00:00
2022-06-18 18:04:24 +00:00
private static void FlagEmptySlotsBeforeIndex ( ReadOnlySpan < int > moves , ReadOnlySpan < CheckMoveResult > parse , int index )
{
for ( int i = index - 1 ; i > = 0 ; i - - )
2020-08-30 22:43:29 +00:00
{
2022-06-18 18:04:24 +00:00
if ( moves [ i ] ! = 0 )
return ;
parse [ i ] . FlagIllegal ( LMoveSourceEmpty ) ;
2017-05-28 04:17:53 +00:00
}
}
}