2016-03-12 17:16:41 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2017-03-25 17:24:56 +00:00
|
|
|
|
using static PKHeX.Core.LegalityCheckStrings;
|
2016-03-11 04:36:32 +00:00
|
|
|
|
|
2017-01-08 07:54:09 +00:00
|
|
|
|
namespace PKHeX.Core
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2016-03-14 03:19:04 +00:00
|
|
|
|
public partial class LegalityAnalysis
|
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyGender()
|
2016-07-04 01:36:04 +00:00
|
|
|
|
{
|
2016-10-29 06:41:22 +00:00
|
|
|
|
if (pkm.PersonalInfo.Gender == 255 && pkm.Gender != 2)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V203, CheckIdentifier.Gender);
|
2017-03-25 17:24:56 +00:00
|
|
|
|
|
|
|
|
|
// Check for PID relationship to Gender & Nature if applicable
|
|
|
|
|
int gen = pkm.GenNumber;
|
|
|
|
|
|
|
|
|
|
bool PIDGender = 3 <= gen && gen <= 5;
|
|
|
|
|
if (!PIDGender)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
bool genderValid = pkm.getGenderIsValid();
|
|
|
|
|
if (!genderValid && pkm.Format > 5 && (pkm.Species == 183 || pkm.Species == 184))
|
|
|
|
|
{
|
|
|
|
|
var gv = pkm.PID & 0xFF;
|
|
|
|
|
if (gv > 63 && pkm.Gender == 1) // evolved from azurill after transferring to keep gender
|
|
|
|
|
genderValid = true;
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
|
|
|
|
|
if (genderValid)
|
|
|
|
|
AddLine(Severity.Valid, V250, CheckIdentifier.Gender);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V251, CheckIdentifier.Gender);
|
|
|
|
|
|
|
|
|
|
bool PIDNature = gen != 5;
|
|
|
|
|
if (!PIDNature)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-05-04 06:14:34 +00:00
|
|
|
|
if (pkm.EncryptionConstant % 25 == pkm.Nature)
|
2017-03-25 17:24:56 +00:00
|
|
|
|
AddLine(Severity.Valid, V252, CheckIdentifier.Nature);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V253, CheckIdentifier.Nature);
|
2016-07-04 01:36:04 +00:00
|
|
|
|
}
|
2017-02-26 15:55:22 +00:00
|
|
|
|
private void verifyItem()
|
|
|
|
|
{
|
2017-02-26 19:43:26 +00:00
|
|
|
|
if (!Legal.getHeldItemAllowed(pkm.Format, pkm.HeldItem))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V204, CheckIdentifier.Form);
|
2017-04-10 22:59:16 +00:00
|
|
|
|
if (pkm.Format == 3 && pkm.HeldItem == 175)
|
|
|
|
|
verifyEReaderBerry();
|
|
|
|
|
}
|
|
|
|
|
private void verifyEReaderBerry()
|
|
|
|
|
{
|
2017-04-11 02:00:58 +00:00
|
|
|
|
if (Legal.EReaderBerryIsEnigma) // no E-Reader berry data provided, can't hold berry.
|
2017-04-10 22:59:16 +00:00
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V204, CheckIdentifier.Form);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-04-11 02:00:58 +00:00
|
|
|
|
|
2017-04-10 22:59:16 +00:00
|
|
|
|
var matchUSA = Legal.EReaderBerriesNames_USA.Contains(Legal.EReaderBerryName);
|
|
|
|
|
var matchJP = Legal.EReaderBerriesNames_JP.Contains(Legal.EReaderBerryName);
|
2017-04-11 02:00:58 +00:00
|
|
|
|
if (!matchJP && !matchUSA) // Does not match any released E-Reader berry
|
2017-04-10 22:59:16 +00:00
|
|
|
|
AddLine(Severity.Invalid, V369, CheckIdentifier.Form);
|
2017-04-11 02:00:58 +00:00
|
|
|
|
else if (matchJP && !Legal.SavegameJapanese) // E-Reader is region locked
|
2017-04-10 22:59:16 +00:00
|
|
|
|
AddLine(Severity.Invalid, V370, CheckIdentifier.Form);
|
2017-04-11 02:00:58 +00:00
|
|
|
|
else if (matchUSA && Legal.SavegameJapanese) // E-Reader is region locked
|
2017-04-10 22:59:16 +00:00
|
|
|
|
AddLine(Severity.Invalid, V371, CheckIdentifier.Form);
|
2017-02-26 15:55:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyECPID()
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2017-04-11 02:00:58 +00:00
|
|
|
|
if (pkm.Format >= 6)
|
|
|
|
|
verifyEC();
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch.Species == 265)
|
2017-04-11 02:00:58 +00:00
|
|
|
|
verifyECPIDWurmple();
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
if (pkm.PID == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V207, CheckIdentifier.PID);
|
2016-03-11 04:36:32 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.GenNumber >= 6 && pkm.PID == pkm.EncryptionConstant)
|
2017-04-11 02:00:58 +00:00
|
|
|
|
AddLine(Severity.Invalid, V208, CheckIdentifier.PID); // better to flag than 1:2^32 odds since RNG is not feasible to yield match
|
2016-03-11 04:36:32 +00:00
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterStatic s)
|
2016-06-20 04:22:43 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (s.Shiny != null && (bool) s.Shiny ^ pkm.IsShiny)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V209, CheckIdentifier.Shiny);
|
2016-06-20 04:22:43 +00:00
|
|
|
|
}
|
2017-05-28 04:17:53 +00:00
|
|
|
|
else if (EncounterMatch is EncounterSlot w)
|
2017-04-02 13:39:39 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (pkm.IsShiny && w.Type == SlotType.HiddenGrotto)
|
2017-04-02 13:39:39 +00:00
|
|
|
|
AddLine(Severity.Invalid, V221, CheckIdentifier.Shiny);
|
|
|
|
|
}
|
2017-04-11 02:00:58 +00:00
|
|
|
|
}
|
|
|
|
|
private void verifyECPIDWurmple()
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
uint evoVal = EncounterGenerator.getWurmpleEvoVal(pkm);
|
2016-06-20 04:22:43 +00:00
|
|
|
|
|
2017-04-11 02:00:58 +00:00
|
|
|
|
if (pkm.Species == 265)
|
|
|
|
|
AddLine(Severity.Valid, string.Format(V212, evoVal == 0 ? specieslist[267] : specieslist[269]), CheckIdentifier.EC);
|
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
int wIndex = Array.IndexOf(Legal.WurmpleEvolutions, pkm.Species) / 2;
|
2017-04-11 02:00:58 +00:00
|
|
|
|
if (evoVal != wIndex)
|
|
|
|
|
AddLine(Severity.Invalid, V210, CheckIdentifier.EC);
|
|
|
|
|
}
|
|
|
|
|
private void verifyEC()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.EncryptionConstant == 0)
|
|
|
|
|
AddLine(Severity.Fishy, V201, CheckIdentifier.EC);
|
|
|
|
|
if (3 <= pkm.GenNumber && pkm.GenNumber <= 5)
|
|
|
|
|
verifyTransferEC();
|
|
|
|
|
else
|
2016-04-08 00:56:39 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
int xor = pkm.TSV ^ pkm.PSV;
|
|
|
|
|
if (xor < 16 && xor >= 8 && (pkm.PID ^ 0x80000000) == pkm.EncryptionConstant)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V211, CheckIdentifier.EC);
|
2016-04-08 00:56:39 +00:00
|
|
|
|
}
|
2017-04-11 02:00:58 +00:00
|
|
|
|
}
|
|
|
|
|
private void verifyTransferEC()
|
|
|
|
|
{
|
2016-03-11 04:36:32 +00:00
|
|
|
|
// When transferred to Generation 6, the Encryption Constant is copied from the PID.
|
|
|
|
|
// The PID is then checked to see if it becomes shiny with the new Shiny rules (>>4 instead of >>3)
|
|
|
|
|
// If the PID is nonshiny->shiny, the top bit is flipped.
|
|
|
|
|
|
|
|
|
|
// Check to see if the PID and EC are properly configured.
|
2016-12-23 02:18:35 +00:00
|
|
|
|
bool xorPID = ((pkm.TID ^ pkm.SID ^ (int)(pkm.PID & 0xFFFF) ^ (int)(pkm.PID >> 16)) & ~0x7) == 8;
|
2016-03-11 04:36:32 +00:00
|
|
|
|
bool valid = xorPID
|
2016-10-23 19:48:49 +00:00
|
|
|
|
? pkm.EncryptionConstant == (pkm.PID ^ 0x8000000)
|
|
|
|
|
: pkm.EncryptionConstant == pkm.PID;
|
2016-03-11 04:36:32 +00:00
|
|
|
|
|
|
|
|
|
if (!valid)
|
2017-04-11 02:00:58 +00:00
|
|
|
|
AddLine(Severity.Invalid, xorPID ? V215 : V216, CheckIdentifier.ECPID);
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyNickname
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyNickname()
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
|
|
|
|
// If the Pokémon is not nicknamed, it should match one of the language strings.
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Nickname.Length == 0)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V2, CheckIdentifier.Nickname);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (pkm.Species > PKX.SpeciesLang[0].Length)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Indeterminate, V3, CheckIdentifier.Nickname);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-27 05:35:26 +00:00
|
|
|
|
if (pkm.VC)
|
|
|
|
|
{
|
|
|
|
|
string pk = pkm.Nickname;
|
|
|
|
|
var langset = PKX.SpeciesLang.FirstOrDefault(s => s.Contains(pk)) ?? PKX.SpeciesLang[2];
|
|
|
|
|
int lang = Array.IndexOf(PKX.SpeciesLang, langset);
|
|
|
|
|
|
|
|
|
|
if (pk.Length > (lang == 2 ? 10 : 5))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V1, CheckIdentifier.Nickname);
|
2017-04-09 23:38:29 +00:00
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
else if (Type == typeof(MysteryGift))
|
2017-04-09 23:38:29 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.IsNicknamed && (!(EncounterMatch as MysteryGift)?.IsEgg ?? false))
|
|
|
|
|
AddLine(Severity.Fishy, V0, CheckIdentifier.Nickname);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-20 01:06:02 +00:00
|
|
|
|
if (!Encounter.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (pkm.Format <= 6 && pkm.Language > 8)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Indeterminate, V4, CheckIdentifier.Language);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-11-09 16:23:47 +00:00
|
|
|
|
if (pkm.Format <= 7 && pkm.Language > 10)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Indeterminate, V5, CheckIdentifier.Language);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-22 20:04:12 +00:00
|
|
|
|
if (Type == typeof(EncounterTrade))
|
2016-03-27 00:23:53 +00:00
|
|
|
|
{
|
2017-03-25 17:24:56 +00:00
|
|
|
|
verifyNicknameTrade();
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-03-27 00:23:53 +00:00
|
|
|
|
}
|
2016-03-20 01:06:02 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.IsEgg)
|
2016-03-16 04:15:40 +00:00
|
|
|
|
{
|
2017-03-25 17:24:56 +00:00
|
|
|
|
verifyNicknameEgg();
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-03-16 04:15:40 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
string nickname = pkm.Nickname.Replace("'", "’");
|
|
|
|
|
if (pkm.IsNicknamed)
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2016-03-20 01:06:02 +00:00
|
|
|
|
for (int i = 0; i < PKX.SpeciesLang.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
string[] lang = PKX.SpeciesLang[i];
|
|
|
|
|
int index = Array.IndexOf(lang, nickname);
|
|
|
|
|
if (index < 0)
|
|
|
|
|
continue;
|
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
AddLine(Severity.Fishy, index == pkm.Species && i != pkm.Language
|
2017-03-21 07:18:38 +00:00
|
|
|
|
? V15
|
|
|
|
|
: V16, CheckIdentifier.Nickname);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-03-20 01:06:02 +00:00
|
|
|
|
}
|
2017-04-04 03:02:40 +00:00
|
|
|
|
if (nickname.Any(c => 0x4E00 <= c && c <= 0x9FFF)) // East Asian Scripts
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V222, CheckIdentifier.Nickname);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V17, CheckIdentifier.Nickname);
|
2016-03-16 01:56:18 +00:00
|
|
|
|
}
|
2017-02-13 01:00:03 +00:00
|
|
|
|
else if (pkm.Format < 3)
|
|
|
|
|
{
|
|
|
|
|
// pk1/pk2 IsNicknamed getter checks for match, logic should only reach here if matches.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V18, CheckIdentifier.Nickname);
|
2017-02-13 01:00:03 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2016-03-16 01:56:18 +00:00
|
|
|
|
{
|
2016-07-11 05:22:55 +00:00
|
|
|
|
// Can't have another language name if it hasn't evolved or wasn't a language-traded egg.
|
2017-04-10 01:28:22 +00:00
|
|
|
|
bool evolved = Legal.getHasEvolved(pkm);
|
2017-03-19 22:40:49 +00:00
|
|
|
|
bool match = PKX.getSpeciesNameGeneration(pkm.Species, pkm.Language, pkm.Format) == nickname;
|
2017-04-10 01:28:22 +00:00
|
|
|
|
if (pkm.WasTradedEgg || evolved)
|
2017-03-25 17:24:56 +00:00
|
|
|
|
match |= !PKX.getIsNicknamedAnyLanguage(pkm.Species, nickname, pkm.Format);
|
2017-04-10 01:28:22 +00:00
|
|
|
|
if (!match && pkm.Format == 5 && !pkm.IsNative) // transfer
|
|
|
|
|
{
|
|
|
|
|
if (evolved)
|
|
|
|
|
match |= !PKX.getIsNicknamedAnyLanguage(pkm.Species, nickname, 4);
|
|
|
|
|
else
|
|
|
|
|
match |= PKX.getSpeciesNameGeneration(pkm.Species, pkm.Language, 4) == nickname;
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
if (!match)
|
2017-02-21 06:03:07 +00:00
|
|
|
|
{
|
|
|
|
|
if ((EncounterMatch as MysteryGift)?.CardID == 2046 && (pkm.SID << 16 | pkm.TID) == 0x79F57B49)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V19, CheckIdentifier.Nickname);
|
2017-02-21 06:03:07 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V20, CheckIdentifier.Nickname);
|
2017-02-21 06:03:07 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V18, CheckIdentifier.Nickname);
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
private void verifyNicknameEgg()
|
|
|
|
|
{
|
2017-05-10 04:11:13 +00:00
|
|
|
|
switch (pkm.Format)
|
|
|
|
|
{
|
|
|
|
|
case 4:
|
|
|
|
|
if (pkm.IsNicknamed) // gen4 doesn't use the nickname flag for eggs
|
|
|
|
|
AddLine(Severity.Invalid, V224, CheckIdentifier.Egg);
|
|
|
|
|
break;
|
|
|
|
|
case 7:
|
|
|
|
|
if (EncounterMatch is EncounterStatic ^ !pkm.IsNicknamed) // gen7 doesn't use for ingame gifts
|
|
|
|
|
AddLine(Severity.Invalid, pkm.IsNicknamed ? V224 : V12, CheckIdentifier.Egg);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
if (!pkm.IsNicknamed)
|
|
|
|
|
AddLine(Severity.Invalid, V12, CheckIdentifier.Egg);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (PKX.getSpeciesNameGeneration(0, pkm.Language, pkm.GenNumber) != pkm.Nickname)
|
2017-03-25 17:24:56 +00:00
|
|
|
|
AddLine(Severity.Invalid, V13, CheckIdentifier.Egg);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V14, CheckIdentifier.Egg);
|
|
|
|
|
}
|
|
|
|
|
private void verifyNicknameTrade()
|
|
|
|
|
{
|
|
|
|
|
string[] validOT = new string[0];
|
|
|
|
|
int index = -1;
|
|
|
|
|
if (pkm.XY)
|
|
|
|
|
{
|
|
|
|
|
validOT = Legal.TradeXY[pkm.Language];
|
|
|
|
|
index = Array.IndexOf(Legal.TradeGift_XY, EncounterMatch);
|
|
|
|
|
}
|
|
|
|
|
else if (pkm.AO)
|
|
|
|
|
{
|
|
|
|
|
validOT = Legal.TradeAO[pkm.Language];
|
|
|
|
|
index = Array.IndexOf(Legal.TradeGift_AO, EncounterMatch);
|
|
|
|
|
}
|
|
|
|
|
else if (pkm.SM)
|
|
|
|
|
{
|
|
|
|
|
// TODO
|
|
|
|
|
AddLine(Severity.Valid, V194, CheckIdentifier.Nickname);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else if (pkm.Format <= 2 || pkm.VC)
|
|
|
|
|
{
|
Gen 1 move analysis improvement. Adapted the valid moves to take into account that move deleter and move reminder do not exits in generation 1 (#1037)
* Fix getMoves with min level, when SkipWhile and TakeWhile is used together the index i in TakeWhile is calculated from the enumerator that results of the SkipWhile function, not the index of the initial array, those giving an incorrect index to check Levels array in the TakeWhile
* Fix getMoves when levelmin or levelmax is above max level in the levels array, TakeWhile and SkipWhile return empty list if the while goes beyond the last element of the array
* Include player hatched egg in the list of possible encounters for parseMoves only if the pokemon was an egg
Also change the valur of WasEgg for gen1,2,3 pokemon if the encounter analyzed is not an egg
add the non egg encounters to the list instead of checking the non-egg encounter inside parseMovesWasEggPreRelearn
* Fix for gen3 egg encounters
Remove non-egg encounters without special moves if there is an egg encounter because egg encounter already cover every possible move combination
Do not add daycare egg encounter for gen3 unhatched egg if there is another encounter, that means is an event or gift egg, not a daycare egg
Remove duplicate encounters
* Gift egg should not allow inherited moves even it they dont have special moves
Those gift eggs are hatched only with the species base moves
* Added getEncounterMoves functions, to be used for generation 1 encounters to find what moves have a pokemon at the moment of being caught because there is no move reminder in generation 1
* Added GBEncounterData, structure for refactor the tuples used in generation 1 and 2 encounters
* Add LevelMin and LevelMax to IEncounterable to get the encounter moves by the min level of the generation 1 EncounterLink
Add iGeneration to difference generation 1 and generation 2 encounters for GB Era pokemon
* Mark generation in gen 1 and 2 encounters
There is no need to mark the generation in gen 3 to 7 encounters because in that generations it match the pokemon generation number
* Add min level for generation 1 moves in getMoves functions
Add function to return the default moves for a GB encounters, for generation 1 games that included both moves from level up table and level 1 moves from personal table
Fix getMoves with min level when the moves list is empty, like Raichu generation 1
* Add maxSpecies to getBaseSpecies function for gen1 pokemon with a gen2 egg encounter
Refactor VC Encounter changing Tuples for GBData class
* Fixed for gen 2 Checks
Also do not search for generation1 encounter if the gen2 pokemon have met location from crystal
* Fix VC wild encounters, should be stored as array or else other verifyEncounter functions wont work
* Add generation 1 parse moves function including default moves
* Clean-up get encounters
* Verify empty moves for generation 1 encounters, in generation 1 does not exits move deleter
That means when a move slot have been used by a level up move or a TM/HM/Tutor move it cant be empty again
Does not apply if generation 2 tradeback is allowed, in generation 2 there is a move deleter
* Added two edge cases for pokemon that learn in red/blue and yellow different moves at the same level, this combinations can not exits until a later level when they learn again one of the levels in the other game, only happen for flareon and vaporeon
* Check incompatible moves between evolution species, it is for species that learn a move in a level as an evolved species and a move at a upper level as a preevolution
Also added most edge cases for the min slots used for generation 1 games, i think every weird combination is already covered
* Fix gen 1 eevee and evolutions move checks
* Cleanup
* Move the code to change valid moves for generation 1 to a function
* Fix getMoveMinLevelGBEncounter
* getUsedMoveSlots, removed wild Butterfree edge case, it is not possible
* Filter the min level of the encounter by the possible games the pokemon could be originated, yellow pikachu and kadabra can be detected
2017-04-09 00:17:20 +00:00
|
|
|
|
var et = EncounterOriginalGB as EncounterTrade;
|
2017-03-25 17:24:56 +00:00
|
|
|
|
if (et?.TID == 0) // Gen1 Trade
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (!EncounterGenerator.getEncounterTrade1Valid(pkm))
|
2017-03-25 17:24:56 +00:00
|
|
|
|
AddLine(Severity.Invalid, V10, CheckIdentifier.Trainer);
|
|
|
|
|
}
|
|
|
|
|
else // Gen2
|
|
|
|
|
{
|
|
|
|
|
return; // already checked all relevant properties when fetching with getValidEncounterTradeVC2
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
else if (3 <= pkm.GenNumber && pkm.GenNumber <= 5)
|
2017-03-30 06:31:02 +00:00
|
|
|
|
{
|
|
|
|
|
// Suppressing temporarily
|
2017-03-28 09:02:46 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-25 17:24:56 +00:00
|
|
|
|
if (validOT.Length == 0)
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Indeterminate, V7, CheckIdentifier.Trainer);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (index == -1 || validOT.Length < index * 2)
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Indeterminate, V8, CheckIdentifier.Trainer);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string nick = validOT[index];
|
|
|
|
|
string OT = validOT[validOT.Length / 2 + index];
|
|
|
|
|
|
|
|
|
|
if (nick != pkm.Nickname)
|
|
|
|
|
AddLine(Severity.Fishy, V9, CheckIdentifier.Nickname);
|
|
|
|
|
else if (OT != pkm.OT_Name)
|
|
|
|
|
AddLine(Severity.Invalid, V10, CheckIdentifier.Trainer);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V11, CheckIdentifier.Nickname);
|
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#endregion
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyEVs()
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
var evs = pkm.EVs;
|
2016-03-14 03:19:04 +00:00
|
|
|
|
int sum = evs.Sum();
|
2017-04-12 03:55:34 +00:00
|
|
|
|
if (sum > 0 && pkm.IsEgg)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V22, CheckIdentifier.EVs);
|
2017-04-12 03:55:34 +00:00
|
|
|
|
if (sum > 510)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V25, CheckIdentifier.EVs);
|
2017-04-12 03:55:34 +00:00
|
|
|
|
if (pkm.Format >= 6 && evs.Any(ev => ev > 252))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V26, CheckIdentifier.EVs);
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (pkm.Format == 4 && pkm.Gen4 && EncounterMatch.LevelMin == 100)
|
2017-04-10 12:11:07 +00:00
|
|
|
|
{
|
2017-04-12 03:55:34 +00:00
|
|
|
|
// Cannot EV train at level 100 -- Certain events are distributed at level 100.
|
|
|
|
|
if (evs.Any(ev => ev > 100)) // EVs can only be increased by vitamins to a max of 100.
|
2017-04-10 12:11:07 +00:00
|
|
|
|
AddLine(Severity.Invalid, V367, CheckIdentifier.EVs);
|
|
|
|
|
}
|
2017-04-12 03:55:34 +00:00
|
|
|
|
|
|
|
|
|
// Only one of the following can be true: 0, 508, and x%6!=0
|
|
|
|
|
if (sum == 0 && pkm.CurrentLevel - Math.Max(1, pkm.Met_Level) > 0)
|
|
|
|
|
AddLine(Severity.Fishy, V23, CheckIdentifier.EVs);
|
|
|
|
|
else if (sum == 508)
|
|
|
|
|
AddLine(Severity.Fishy, V24, CheckIdentifier.EVs);
|
|
|
|
|
else if (evs[0] != 0 && evs.All(ev => evs[0] == ev))
|
|
|
|
|
AddLine(Severity.Fishy, V27, CheckIdentifier.EVs);
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyIVs()
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is EncounterStatic s && s.IV3)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-10 04:27:03 +00:00
|
|
|
|
int IVCount = 3;
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (s.Version == GameVersion.RBY && pkm.Species == 151)
|
2017-03-10 04:27:03 +00:00
|
|
|
|
IVCount = 5; // VC Mew
|
|
|
|
|
if (pkm.IVs.Count(iv => iv == 31) < IVCount)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V28, IVCount), CheckIdentifier.IVs);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot w && w.Type == SlotType.FriendSafari)
|
2017-01-01 01:20:13 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.IVs.Count(iv => iv == 31) < 2)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V29, CheckIdentifier.IVs);
|
2017-01-01 01:20:13 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is MysteryGift g)
|
2017-02-02 05:27:14 +00:00
|
|
|
|
{
|
|
|
|
|
int[] IVs;
|
2017-05-28 05:40:21 +00:00
|
|
|
|
switch (g.Format)
|
2017-02-02 05:27:14 +00:00
|
|
|
|
{
|
|
|
|
|
case 7: IVs = ((WC7)EncounterMatch).IVs; break;
|
2017-03-10 04:27:03 +00:00
|
|
|
|
case 6: IVs = ((WC6)EncounterMatch).IVs; break;
|
|
|
|
|
case 5: IVs = ((PGF)EncounterMatch).IVs; break;
|
2017-02-02 05:27:14 +00:00
|
|
|
|
default: IVs = null; break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (IVs != null)
|
|
|
|
|
{
|
|
|
|
|
var pkIVs = pkm.IVs;
|
|
|
|
|
bool valid = true;
|
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
|
if (IVs[i] <= 31 && IVs[i] != pkIVs[i])
|
|
|
|
|
valid = false;
|
|
|
|
|
if (!valid)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V30, CheckIdentifier.IVs);
|
2017-04-10 04:59:44 +00:00
|
|
|
|
bool IV3 = IVs[0] == 0xFE;
|
|
|
|
|
if (IV3 && pkm.IVs.Count(iv => iv == 31) < 3)
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V28, 3), CheckIdentifier.IVs);
|
2017-02-02 05:27:14 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.IVs.Sum() == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V31, CheckIdentifier.IVs);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (pkm.IVs[0] < 30 && pkm.IVs.All(iv => pkm.IVs[0] == iv))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V32, CheckIdentifier.IVs);
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
2017-02-13 01:00:03 +00:00
|
|
|
|
private void verifyDVs()
|
|
|
|
|
{
|
|
|
|
|
// todo
|
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyOT
|
2017-01-27 05:35:26 +00:00
|
|
|
|
private void verifyOT()
|
2016-03-11 04:36:32 +00:00
|
|
|
|
{
|
2017-04-22 20:04:12 +00:00
|
|
|
|
if (Type == typeof(EncounterTrade))
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return; // Already matches Encounter Trade information
|
|
|
|
|
|
|
|
|
|
if (pkm.TID == 0 && pkm.SID == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V33, CheckIdentifier.Trainer);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
else if (pkm.VC)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.SID != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V34, CheckIdentifier.Trainer);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (pkm.TID == pkm.SID)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V35, CheckIdentifier.Trainer);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (pkm.TID == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V36, CheckIdentifier.Trainer);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (pkm.SID == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V37, CheckIdentifier.Trainer);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
|
|
|
|
|
if (pkm.VC)
|
2017-02-13 01:00:03 +00:00
|
|
|
|
verifyG1OT();
|
|
|
|
|
}
|
|
|
|
|
private void verifyG1OT()
|
|
|
|
|
{
|
|
|
|
|
string tr = pkm.OT_Name;
|
|
|
|
|
string pk = pkm.Nickname;
|
|
|
|
|
var langset = PKX.SpeciesLang.FirstOrDefault(s => s.Contains(pk)) ?? PKX.SpeciesLang[2];
|
|
|
|
|
int lang = Array.IndexOf(PKX.SpeciesLang, langset);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
|
2017-02-13 01:00:03 +00:00
|
|
|
|
if (tr.Length > (lang == 2 ? 7 : 5))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V38, CheckIdentifier.Trainer);
|
2017-02-13 01:00:03 +00:00
|
|
|
|
if (pkm.Species == 151)
|
|
|
|
|
{
|
|
|
|
|
if (tr != "GF" && tr != "ゲーフリ") // if there are more events with special OTs, may be worth refactoring
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V39, CheckIdentifier.Trainer);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
}
|
2017-05-01 15:25:20 +00:00
|
|
|
|
if ((EncounterMatch as EncounterStatic)?.Version == GameVersion.Stadium)
|
Encounter Type fix and detection of pokemon that should have evolve on trade (#1105)
* Detect encounter trades that evolve on trade and have not been evolved
Detect generation 1 pokemon with special catch rates : krabby trade and Pokemon Stadium
Detect generation 1 pokemon that evolve on trade and have been traded but not evolved
Detect pokemon with tradeback status any but with only encounters from the other GB generation, that means they are WasTradeback
Detect pokemon with moves from the other GB generation, change tradebackstatus to WasTradeback
* Fix dppt surfing and fishing encounter type, is is surfing because the battle background is the same as other surfingfishing encounters
Fix headbutt encounter type, the encounter type depends on the battle background used when battle a pokemon, for headbutt it changes with the player tile, is the player is in a city it will be building encounter, in a grass tile tall grass, in water it is surfingfishing and cave inside a cave. Some locations have more than one possible encounter type, for example routes with trees near the grass, near the water and near non-combat tiles.
Also added slot type headbutt special for the special trees, those trees are all in routes and are only adjacent to non-combat tiles
* Fix encounter type for missing areas with multiple grass encounters types: Mt Coronet, Mt Silver Cave and Stark Mountain (Issue # 1095)
* Fixes and typos
* Check for non-japanese e-reader pokemon, is unreleased
2017-05-01 15:07:20 +00:00
|
|
|
|
{
|
2017-05-01 15:25:20 +00:00
|
|
|
|
if (tr == "STADIUM" && pkm.TID == 2000)
|
Encounter Type fix and detection of pokemon that should have evolve on trade (#1105)
* Detect encounter trades that evolve on trade and have not been evolved
Detect generation 1 pokemon with special catch rates : krabby trade and Pokemon Stadium
Detect generation 1 pokemon that evolve on trade and have been traded but not evolved
Detect pokemon with tradeback status any but with only encounters from the other GB generation, that means they are WasTradeback
Detect pokemon with moves from the other GB generation, change tradebackstatus to WasTradeback
* Fix dppt surfing and fishing encounter type, is is surfing because the battle background is the same as other surfingfishing encounters
Fix headbutt encounter type, the encounter type depends on the battle background used when battle a pokemon, for headbutt it changes with the player tile, is the player is in a city it will be building encounter, in a grass tile tall grass, in water it is surfingfishing and cave inside a cave. Some locations have more than one possible encounter type, for example routes with trees near the grass, near the water and near non-combat tiles.
Also added slot type headbutt special for the special trees, those trees are all in routes and are only adjacent to non-combat tiles
* Fix encounter type for missing areas with multiple grass encounters types: Mt Coronet, Mt Silver Cave and Stark Mountain (Issue # 1095)
* Fixes and typos
* Check for non-japanese e-reader pokemon, is unreleased
2017-05-01 15:07:20 +00:00
|
|
|
|
AddLine(Severity.Valid, V403, CheckIdentifier.Trainer);
|
|
|
|
|
else if (tr == "スタジアム" && pkm.TID == 1999)
|
|
|
|
|
AddLine(Severity.Valid, V404, CheckIdentifier.Trainer);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V402, CheckIdentifier.Trainer);
|
|
|
|
|
}
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#endregion
|
2016-11-11 10:31:50 +00:00
|
|
|
|
private void verifyHyperTraining()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Format < 7)
|
|
|
|
|
return; // No Hyper Training before Gen VII
|
|
|
|
|
|
|
|
|
|
var IVs = new[] { pkm.IV_HP, pkm.IV_ATK, pkm.IV_DEF, pkm.IV_SPA, pkm.IV_SPD, pkm.IV_SPE };
|
|
|
|
|
var HTs = new[] { pkm.HT_HP, pkm.HT_ATK, pkm.HT_DEF, pkm.HT_SPA, pkm.HT_SPD, pkm.HT_SPE };
|
2016-11-11 11:23:38 +00:00
|
|
|
|
|
|
|
|
|
if (HTs.Any(ht => ht) && pkm.CurrentLevel != 100)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V40, CheckIdentifier.IVs);
|
2016-11-11 11:23:38 +00:00
|
|
|
|
|
2016-11-11 10:31:50 +00:00
|
|
|
|
if (IVs.All(iv => iv == 31) && HTs.Any(ht => ht))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V41, CheckIdentifier.IVs);
|
2016-11-11 10:31:50 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < 6; i++) // Check individual IVs
|
|
|
|
|
{
|
|
|
|
|
if ((IVs[i] == 31) && HTs[i])
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V42, CheckIdentifier.IVs);
|
2016-11-11 10:31:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyEncounter
|
2017-05-28 04:17:53 +00:00
|
|
|
|
private void verifyFormFriendSafari()
|
2017-02-13 01:00:03 +00:00
|
|
|
|
{
|
2017-03-20 07:03:31 +00:00
|
|
|
|
switch (pkm.Species)
|
|
|
|
|
{
|
|
|
|
|
case 670: // Floette
|
|
|
|
|
case 671: // Florges
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (!new[] { 0, 1, 3 }.Contains(pkm.AltForm)) // 0/1/3 - RBY
|
|
|
|
|
AddLine(Severity.Invalid, V64, CheckIdentifier.Form);
|
2017-03-20 07:03:31 +00:00
|
|
|
|
break;
|
|
|
|
|
case 710: // Pumpkaboo
|
|
|
|
|
case 711: // Goregeist
|
2017-03-25 17:24:56 +00:00
|
|
|
|
if (pkm.AltForm != 0) // Average
|
2017-05-28 04:17:53 +00:00
|
|
|
|
AddLine(Severity.Invalid, V6, CheckIdentifier.Form);
|
2017-03-20 07:03:31 +00:00
|
|
|
|
break;
|
|
|
|
|
case 586: // Sawsbuck
|
|
|
|
|
if (pkm.AltForm != 0)
|
2017-05-28 04:17:53 +00:00
|
|
|
|
AddLine(Severity.Invalid, V65, CheckIdentifier.Form);
|
2017-03-20 07:03:31 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
}
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
private void verifyEncounterType()
|
|
|
|
|
{
|
2017-04-23 04:00:06 +00:00
|
|
|
|
if (pkm.Format >= 7)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (!Encounter.Valid)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-04-22 20:04:12 +00:00
|
|
|
|
EncounterType type = EncounterType.None;
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
// Encounter type data is only stored for gen 4 encounters
|
2017-05-11 01:52:14 +00:00
|
|
|
|
// Gen 6 -> 7 transfer deletes encounter type data
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
// All eggs have encounter type none, even if they are from static encounters
|
2017-04-23 04:00:06 +00:00
|
|
|
|
if (pkm.Gen4 && !pkm.WasEgg)
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot w)
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
// If there is more than one slot, the get wild encounter have filter for the pkm type encounter like safari/sports ball
|
2017-05-28 04:17:53 +00:00
|
|
|
|
type = w.TypeEncounter;
|
|
|
|
|
if (EncounterMatch is EncounterStaticTyped s)
|
|
|
|
|
type = s.TypeEncounter;
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
}
|
2017-04-23 04:00:06 +00:00
|
|
|
|
|
Encounter Type fix and detection of pokemon that should have evolve on trade (#1105)
* Detect encounter trades that evolve on trade and have not been evolved
Detect generation 1 pokemon with special catch rates : krabby trade and Pokemon Stadium
Detect generation 1 pokemon that evolve on trade and have been traded but not evolved
Detect pokemon with tradeback status any but with only encounters from the other GB generation, that means they are WasTradeback
Detect pokemon with moves from the other GB generation, change tradebackstatus to WasTradeback
* Fix dppt surfing and fishing encounter type, is is surfing because the battle background is the same as other surfingfishing encounters
Fix headbutt encounter type, the encounter type depends on the battle background used when battle a pokemon, for headbutt it changes with the player tile, is the player is in a city it will be building encounter, in a grass tile tall grass, in water it is surfingfishing and cave inside a cave. Some locations have more than one possible encounter type, for example routes with trees near the grass, near the water and near non-combat tiles.
Also added slot type headbutt special for the special trees, those trees are all in routes and are only adjacent to non-combat tiles
* Fix encounter type for missing areas with multiple grass encounters types: Mt Coronet, Mt Silver Cave and Stark Mountain (Issue # 1095)
* Fixes and typos
* Check for non-japanese e-reader pokemon, is unreleased
2017-05-01 15:07:20 +00:00
|
|
|
|
if (!type.Contains(pkm.EncounterType))
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
AddLine(Severity.Invalid, V381, CheckIdentifier.Encounter);
|
2017-04-23 04:00:06 +00:00
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V380, CheckIdentifier.Encounter);
|
Special Eggs improvement and Generation 4 Encounter Type legal analysis (#1083)
* Ignore relearn level 2-100 moves from Phione
* Cave encounter types DPPt
* Generation 4 EncounterType filter and validation
Not every generation 4 static encounter have yet their encounter type defined, i temporally included Any to those encounters
Generation 4 roaaming static encounters have been splitted in two, grass and surf
* Added new legality texts
* Added unreleased event DP Darkai, added check for surf in jhoto route 45, is impossible
Moved unreleased DP locations to valid Platinum locations only
* Improved generation 3 special egg check.
Only check special egg if pokemon knows any of the special egg moves, also in that case do check for normal egg before special egg
because special eggs will explicitly check for normal egg moves but normal eggs will not check special egg moves, it will improve the error output
* Clean up
* Fix gen 5 pokemon from issue #1011
Those pokemon have generation 4 static gift encounters and also wild encounters, the analysis was selecting the static encounter, but if there is a valid wild encounter and the static encounter does not match the pokemon ball the willd encounter should be selected instead
Also move the transfer legality to check it before the static encounters and make that check to work like generation 3 transfer legality
* Another fix for Issue 1011, suppress temporally OT and pokemon name analysis for generations 3 to 5 instead of format 3 to 5, there is no data stored yet to make those analysis
* Do not make wild encounters prevail static encounter if the pokemon was an egg
* Changed type of WildEncounter variable to EncounterSlot[]
* Fix Jhoto Route 45 to avoid return error with fishing encounters
2017-04-22 18:49:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
private void verifyTransferLegalityG3()
|
2017-03-26 22:37:56 +00:00
|
|
|
|
{
|
2017-03-31 07:04:29 +00:00
|
|
|
|
if (pkm.Format == 4 && pkm.Met_Location != 0x37) // Pal Park
|
|
|
|
|
AddLine(Severity.Invalid, V60, CheckIdentifier.Encounter);
|
|
|
|
|
if (pkm.Format != 4 && pkm.Met_Location != 30001)
|
|
|
|
|
AddLine(Severity.Invalid, V61, CheckIdentifier.Encounter);
|
2017-03-26 22:37:56 +00:00
|
|
|
|
}
|
2017-04-25 00:42:15 +00:00
|
|
|
|
private void verifyTransferLegalityG4()
|
|
|
|
|
{
|
|
|
|
|
// Transfer Legality
|
|
|
|
|
int loc = pkm.Met_Location;
|
|
|
|
|
if (loc != 30001) // PokéTransfer
|
|
|
|
|
{
|
|
|
|
|
// Crown
|
|
|
|
|
switch (pkm.Species)
|
|
|
|
|
{
|
|
|
|
|
case 251: // Celebi
|
|
|
|
|
if (loc != 30010 && loc != 30011) // unused || used
|
|
|
|
|
AddLine(Severity.Invalid, V351, CheckIdentifier.Encounter);
|
|
|
|
|
break;
|
|
|
|
|
case 243: // Raikou
|
|
|
|
|
case 244: // Entei
|
|
|
|
|
case 245: // Suicune
|
|
|
|
|
if (loc != 30012 && loc != 30013) // unused || used
|
|
|
|
|
AddLine(Severity.Invalid, V351, CheckIdentifier.Encounter);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
AddLine(Severity.Invalid, V61, CheckIdentifier.Encounter);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-28 04:17:53 +00:00
|
|
|
|
private static IEnumerable<CheckResult> verifyVCEncounter(PKM pkm, int baseSpecies, GBEncounterData encounter)
|
2017-01-27 05:35:26 +00:00
|
|
|
|
{
|
|
|
|
|
// Sanitize Species to non-future species#
|
|
|
|
|
int species = pkm.Species;
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (pkm.VC1 && species > Legal.MaxSpeciesID_1 ||
|
|
|
|
|
pkm.VC2 && species > Legal.MaxSpeciesID_2)
|
2017-01-27 05:35:26 +00:00
|
|
|
|
species = baseSpecies;
|
2017-02-14 02:06:01 +00:00
|
|
|
|
|
|
|
|
|
// Check existing EncounterMatch
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (encounter == null)
|
|
|
|
|
yield break; // Avoid duplicate invaild message
|
2017-02-15 06:06:15 +00:00
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (encounter.Encounter is EncounterStatic v && (GameVersion.GBCartEraOnly.Contains(v.Version) || v.Version == GameVersion.VCEvents))
|
2017-02-15 06:06:15 +00:00
|
|
|
|
{
|
|
|
|
|
bool exceptions = false;
|
2017-05-28 04:17:53 +00:00
|
|
|
|
exceptions |= v.Version == GameVersion.VCEvents && baseSpecies == 151 && pkm.TID == 22796;
|
2017-02-15 06:06:15 +00:00
|
|
|
|
if (!exceptions)
|
2017-05-28 04:17:53 +00:00
|
|
|
|
yield return new CheckResult(Severity.Invalid, V79, CheckIdentifier.Encounter);
|
2017-02-15 06:06:15 +00:00
|
|
|
|
}
|
2017-03-10 04:27:03 +00:00
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
var ematch = EncounterGenerator.getRBYStaticTransfer(species);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
if (pkm.Met_Location != ematch.Location)
|
2017-05-28 04:17:53 +00:00
|
|
|
|
yield return new CheckResult(Severity.Invalid, V81, CheckIdentifier.Encounter);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
if (pkm.Egg_Location != ematch.EggLocation)
|
2017-05-28 04:17:53 +00:00
|
|
|
|
yield return new CheckResult(Severity.Invalid, V59, CheckIdentifier.Encounter);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
|
2017-03-26 08:25:21 +00:00
|
|
|
|
if (species == 150 && pkm.Moves.Contains(6)) // pay day
|
2017-05-28 04:17:53 +00:00
|
|
|
|
yield return new CheckResult(Severity.Invalid, V82, CheckIdentifier.Encounter);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#endregion
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyLevel()
|
2016-03-19 05:49:21 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
MysteryGift MatchedGift = EncounterMatch as MysteryGift;
|
|
|
|
|
if (MatchedGift != null && MatchedGift.Level != pkm.Met_Level)
|
|
|
|
|
{
|
2017-03-26 08:25:21 +00:00
|
|
|
|
if (pkm.HasOriginalMetLocation && (!(MatchedGift is WC7) || ((WC7) MatchedGift).MetLevel != pkm.Met_Level))
|
2016-11-13 23:40:34 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(new CheckResult(Severity.Invalid, V83, CheckIdentifier.Level));
|
2016-11-13 23:40:34 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
2016-12-12 01:13:59 +00:00
|
|
|
|
if (MatchedGift != null && MatchedGift.Level > pkm.CurrentLevel)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(new CheckResult(Severity.Invalid, V84, CheckIdentifier.Level));
|
2016-12-12 01:13:59 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
int lvl = pkm.CurrentLevel;
|
2017-03-19 23:52:56 +00:00
|
|
|
|
if (pkm.IsEgg)
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
int elvl = pkm.Format <= 3 ? 5 : 1;
|
|
|
|
|
if (elvl != lvl)
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V52, elvl), CheckIdentifier.Level);
|
2017-03-19 23:52:56 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (lvl < pkm.Met_Level)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V85, CheckIdentifier.Level);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else if (lvl > pkm.Met_Level && lvl > 1 && lvl != 100 && pkm.EXP == PKX.getEXP(pkm.Stat_Level, pkm.Species))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Fishy, V87, CheckIdentifier.Level);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V88, CheckIdentifier.Level);
|
Encounter Type fix and detection of pokemon that should have evolve on trade (#1105)
* Detect encounter trades that evolve on trade and have not been evolved
Detect generation 1 pokemon with special catch rates : krabby trade and Pokemon Stadium
Detect generation 1 pokemon that evolve on trade and have been traded but not evolved
Detect pokemon with tradeback status any but with only encounters from the other GB generation, that means they are WasTradeback
Detect pokemon with moves from the other GB generation, change tradebackstatus to WasTradeback
* Fix dppt surfing and fishing encounter type, is is surfing because the battle background is the same as other surfingfishing encounters
Fix headbutt encounter type, the encounter type depends on the battle background used when battle a pokemon, for headbutt it changes with the player tile, is the player is in a city it will be building encounter, in a grass tile tall grass, in water it is surfingfishing and cave inside a cave. Some locations have more than one possible encounter type, for example routes with trees near the grass, near the water and near non-combat tiles.
Also added slot type headbutt special for the special trees, those trees are all in routes and are only adjacent to non-combat tiles
* Fix encounter type for missing areas with multiple grass encounters types: Mt Coronet, Mt Silver Cave and Stark Mountain (Issue # 1095)
* Fixes and typos
* Check for non-japanese e-reader pokemon, is unreleased
2017-05-01 15:07:20 +00:00
|
|
|
|
|
2017-05-01 15:25:20 +00:00
|
|
|
|
// There is no way to prevent a gen1 trade evolution as held items (everstone) did not exist.
|
|
|
|
|
// Machoke, Graveler, Haunter and Kadabra captured in the second phase evolution, excluding in-game trades, are already checked
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (pkm.Format <= 2 && Type != typeof (EncounterTrade) && EncounterMatch.Species == pkm.Species && Legal.Trade_Evolution1.Contains(EncounterMatch.Species))
|
2017-05-01 15:25:20 +00:00
|
|
|
|
verifyG1TradeEvo();
|
|
|
|
|
}
|
|
|
|
|
private void verifyG1TradeEvo()
|
|
|
|
|
{
|
|
|
|
|
var mustevolve = pkm.TradebackStatus == TradebackType.WasTradeback || (pkm.Format == 1 && Legal.IsOutsider(pkm));
|
|
|
|
|
if (!mustevolve)
|
|
|
|
|
return;
|
|
|
|
|
// Pokemon have been traded but it is not evolved, trade evos are sequential dex numbers
|
|
|
|
|
var unevolved = specieslist[pkm.Species];
|
|
|
|
|
var evolved = specieslist[pkm.Species + 1];
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V405, unevolved, evolved), CheckIdentifier.Level);
|
2016-03-19 05:49:21 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyMedals
|
2016-11-08 16:43:57 +00:00
|
|
|
|
private void verifyMedals()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Format < 6)
|
|
|
|
|
return;
|
2017-03-25 17:24:56 +00:00
|
|
|
|
|
|
|
|
|
verifyMedalsRegular();
|
|
|
|
|
verifyMedalsEvent();
|
|
|
|
|
}
|
|
|
|
|
private void verifyMedalsRegular()
|
|
|
|
|
{
|
2017-04-15 19:22:29 +00:00
|
|
|
|
uint data = BitConverter.ToUInt32(pkm.Data, 0x2C);
|
|
|
|
|
if ((data & 3) != 0) // 2 unused flags
|
|
|
|
|
AddLine(Severity.Invalid, V98, CheckIdentifier.Training);
|
|
|
|
|
|
|
|
|
|
int TrainCount = 0;
|
|
|
|
|
data >>= 2;
|
|
|
|
|
for (int i = 2; i < 32; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((data & 1) != 0)
|
|
|
|
|
TrainCount++;
|
|
|
|
|
data >>= 1;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-08 16:43:57 +00:00
|
|
|
|
if (pkm.IsEgg && TrainCount > 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V89, CheckIdentifier.Training); }
|
|
|
|
|
else if (TrainCount > 0 && pkm.GenNumber > 6)
|
|
|
|
|
{ AddLine(Severity.Invalid, V90, CheckIdentifier.Training); }
|
2017-01-25 17:17:20 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Format >= 7)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.SecretSuperTrainingUnlocked)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V91, CheckIdentifier.Training); }
|
2017-01-25 17:17:20 +00:00
|
|
|
|
if (pkm.SecretSuperTrainingComplete)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V92, CheckIdentifier.Training); }
|
2017-01-25 17:17:20 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (TrainCount == 30 ^ pkm.SecretSuperTrainingComplete)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V93, CheckIdentifier.Training); }
|
2017-01-25 17:17:20 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
}
|
|
|
|
|
private void verifyMedalsEvent()
|
|
|
|
|
{
|
2017-04-15 19:22:29 +00:00
|
|
|
|
byte data = pkm.Data[0x3A];
|
|
|
|
|
if ((data & 0xC0) != 0) // 2 unused flags highest bits
|
|
|
|
|
AddLine(Severity.Invalid, V98, CheckIdentifier.Training);
|
|
|
|
|
|
|
|
|
|
int TrainCount = 0;
|
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((data & 1) != 0)
|
|
|
|
|
TrainCount++;
|
|
|
|
|
data >>= 1;
|
|
|
|
|
}
|
|
|
|
|
if (pkm.IsEgg && TrainCount > 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V89, CheckIdentifier.Training); }
|
2017-04-15 19:22:29 +00:00
|
|
|
|
else if (TrainCount > 0 && pkm.GenNumber > 6)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V90, CheckIdentifier.Training); }
|
2017-04-15 19:22:29 +00:00
|
|
|
|
else if (TrainCount > 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Fishy, V94, CheckIdentifier.Training); }
|
2016-11-08 16:43:57 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#endregion
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyRibbons()
|
2016-03-20 22:20:11 +00:00
|
|
|
|
{
|
|
|
|
|
if (!Encounter.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-09-17 19:21:23 +00:00
|
|
|
|
|
2016-03-20 22:20:11 +00:00
|
|
|
|
List<string> missingRibbons = new List<string>();
|
|
|
|
|
List<string> invalidRibbons = new List<string>();
|
2016-09-03 15:10:22 +00:00
|
|
|
|
|
2017-04-18 02:11:01 +00:00
|
|
|
|
// Check Event Ribbons
|
|
|
|
|
var encounterContent = (EncounterMatch as MysteryGift)?.Content ?? EncounterMatch;
|
|
|
|
|
var set1 = pkm as IRibbonSet1;
|
|
|
|
|
var set2 = pkm as IRibbonSet2;
|
|
|
|
|
if (set1 != null)
|
|
|
|
|
verifyRibbonSet1(set1, encounterContent, missingRibbons, invalidRibbons);
|
|
|
|
|
if (set2 != null)
|
|
|
|
|
verifyRibbonSet2(set2, encounterContent, missingRibbons, invalidRibbons);
|
|
|
|
|
|
|
|
|
|
// Check Unobtainable Ribbons
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.IsEgg)
|
2016-09-03 15:10:22 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
var RibbonNames = ReflectUtil.getPropertiesStartWithPrefix(pkm.GetType(), "Ribbon");
|
2017-04-18 02:11:01 +00:00
|
|
|
|
if (set1 != null)
|
|
|
|
|
RibbonNames = RibbonNames.Except(RibbonSetHelper.getRibbonNames(set1));
|
|
|
|
|
if (set2 != null)
|
|
|
|
|
RibbonNames = RibbonNames.Except(RibbonSetHelper.getRibbonNames(set2));
|
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
foreach (object RibbonValue in RibbonNames.Select(RibbonName => ReflectUtil.GetValue(pkm, RibbonName)))
|
2016-09-03 15:10:22 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (RibbonValue as bool? == true) // Boolean
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V95, CheckIdentifier.Ribbon); return; }
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if ((RibbonValue as int?) > 0) // Count
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V95, CheckIdentifier.Ribbon); return; }
|
2016-09-03 15:10:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-09-03 15:10:22 +00:00
|
|
|
|
}
|
2016-03-20 22:20:11 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
// Unobtainable ribbons for Gen Origin
|
|
|
|
|
if (pkm.GenNumber > 3)
|
|
|
|
|
{
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK3.RibbonChampionG3Hoenn)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V96); // RSE HoF
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK3.RibbonArtist)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V97); // RSE Master Rank Portrait
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
2017-04-18 02:11:01 +00:00
|
|
|
|
if (pkm.Format >= 4 && pkm.GenNumber > 4)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK4.RibbonChampionSinnoh)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V99); // DPPt HoF
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK4.RibbonLegend)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V100); // HGSS Defeat Red @ Mt.Silver
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
|
|
|
|
if (pkm.Format >= 6 && pkm.GenNumber >= 6)
|
|
|
|
|
{
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK6.RibbonCountMemoryContest)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V106); // Gen3/4 Contest
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK6.RibbonCountMemoryBattle)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V105); // Gen3/4 Battle
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
2016-12-11 18:30:41 +00:00
|
|
|
|
if (ReflectUtil.getBooleanState(pkm, nameof(PK6.RibbonRecord)) == true)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
invalidRibbons.Add(V104); // Unobtainable
|
2016-03-20 22:20:11 +00:00
|
|
|
|
|
|
|
|
|
if (missingRibbons.Count + invalidRibbons.Count == 0)
|
2016-09-03 15:10:22 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V103, CheckIdentifier.Ribbon);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-09-03 15:10:22 +00:00
|
|
|
|
}
|
2016-03-20 22:20:11 +00:00
|
|
|
|
|
|
|
|
|
string[] result = new string[2];
|
|
|
|
|
if (missingRibbons.Count > 0)
|
2017-04-15 08:20:29 +00:00
|
|
|
|
result[0] = string.Format(V101, string.Join(", ", missingRibbons.Select(z => z.Replace("Ribbon", ""))));
|
2016-03-20 22:20:11 +00:00
|
|
|
|
if (invalidRibbons.Count > 0)
|
2017-04-15 08:20:29 +00:00
|
|
|
|
result[1] = string.Format(V102, string.Join(", ", invalidRibbons.Select(z => z.Replace("Ribbon", ""))));
|
2017-03-17 06:16:11 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Join(Environment.NewLine, result.Where(s => !string.IsNullOrEmpty(s))), CheckIdentifier.Ribbon);
|
2016-03-20 22:20:11 +00:00
|
|
|
|
}
|
2017-04-18 02:11:01 +00:00
|
|
|
|
private void verifyRibbonSet1(IRibbonSet1 set1, object encounterContent, List<string> missingRibbons, List<string> invalidRibbons)
|
|
|
|
|
{
|
|
|
|
|
var names = RibbonSetHelper.getRibbonNames(set1);
|
|
|
|
|
var sb = RibbonSetHelper.getRibbonBits(set1);
|
|
|
|
|
var eb = RibbonSetHelper.getRibbonBits(encounterContent as IRibbonSet1);
|
|
|
|
|
|
|
|
|
|
if (pkm.Gen3)
|
2017-04-30 23:53:54 +00:00
|
|
|
|
{
|
2017-04-18 02:11:01 +00:00
|
|
|
|
eb[0] = sb[0]; // permit Earth Ribbon
|
2017-05-01 05:11:51 +00:00
|
|
|
|
if (pkm.Version == 15 && MatchedType == typeof(EncounterStaticShadow)) // only require national ribbon if no longer on C/XD
|
2017-04-30 23:53:54 +00:00
|
|
|
|
eb[1] = (pkm as CK3)?.RibbonNational ?? (pkm as XK3)?.RibbonNational ?? true;
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-18 02:11:01 +00:00
|
|
|
|
for (int i = 0; i < sb.Length; i++)
|
|
|
|
|
if (sb[i] != eb[i])
|
|
|
|
|
(eb[i] ? missingRibbons : invalidRibbons).Add(names[i]);
|
|
|
|
|
}
|
|
|
|
|
private void verifyRibbonSet2(IRibbonSet2 set2, object encounterContent, List<string> missingRibbons, List<string> invalidRibbons)
|
|
|
|
|
{
|
|
|
|
|
var names = RibbonSetHelper.getRibbonNames(set2);
|
|
|
|
|
var sb = RibbonSetHelper.getRibbonBits(set2);
|
|
|
|
|
var eb = RibbonSetHelper.getRibbonBits(encounterContent as IRibbonSet2);
|
|
|
|
|
|
|
|
|
|
if (EncounterMatch is EncounterLink)
|
|
|
|
|
eb[0] = true; // require Classic Ribbon
|
|
|
|
|
if ((EncounterMatch as EncounterStatic)?.RibbonWishing ?? false)
|
|
|
|
|
eb[1] = true; // require Wishing Ribbon
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < sb.Length; i++)
|
|
|
|
|
if (sb[i] != eb[i])
|
|
|
|
|
(eb[i] ? missingRibbons : invalidRibbons).Add(names[i]);
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-30 23:53:54 +00:00
|
|
|
|
private void verifyCXD()
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot w) // pokespot
|
2017-04-30 23:53:54 +00:00
|
|
|
|
{
|
|
|
|
|
// find origin info, if nothing returned then the PID is unobtainable
|
2017-05-28 04:17:53 +00:00
|
|
|
|
var slot = w.SlotNumber;
|
2017-04-30 23:53:54 +00:00
|
|
|
|
var pidiv = MethodFinder.getPokeSpotSeeds(pkm, slot);
|
|
|
|
|
if (!pidiv.Any())
|
|
|
|
|
AddLine(Severity.Invalid, V400, CheckIdentifier.PID);
|
|
|
|
|
}
|
|
|
|
|
else if (Type == typeof (EncounterStatic))
|
|
|
|
|
{
|
2017-05-11 07:32:31 +00:00
|
|
|
|
var pidiv = MethodFinder.Analyze(pkm);
|
|
|
|
|
if (pidiv == null || pidiv.Type != PIDType.CXD)
|
|
|
|
|
AddLine(Severity.Invalid, V400, CheckIdentifier.PID);
|
2017-05-13 07:03:35 +00:00
|
|
|
|
else // Starters have a correlation to the Trainer ID numbers
|
|
|
|
|
verifyCXDStarterCorrelation(pidiv);
|
2017-04-30 23:53:54 +00:00
|
|
|
|
}
|
2017-05-03 02:08:16 +00:00
|
|
|
|
else if (pkm.WasEgg) // can't obtain eggs in CXD
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V80, CheckIdentifier.Encounter); // invalid encounter
|
|
|
|
|
}
|
2017-04-30 23:53:54 +00:00
|
|
|
|
}
|
2017-05-13 07:03:35 +00:00
|
|
|
|
private void verifyCXDStarterCorrelation(PIDIV pidiv)
|
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
var spec = EncounterMatch.Species;
|
2017-05-28 05:18:04 +00:00
|
|
|
|
int rev; // pidiv reversed 2x yields SID, 3x yields TID. shift by 7 if another PKM is generated prior
|
2017-05-13 07:03:35 +00:00
|
|
|
|
switch (spec)
|
|
|
|
|
{
|
|
|
|
|
// XD
|
|
|
|
|
case 133: // Eevee
|
2017-05-28 05:18:04 +00:00
|
|
|
|
rev = 2;
|
2017-05-13 07:03:35 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
// Colosseum
|
|
|
|
|
case 197: // Umbreon (generated before Espeon)
|
2017-05-28 05:18:04 +00:00
|
|
|
|
rev = 2;
|
2017-05-13 07:03:35 +00:00
|
|
|
|
break;
|
|
|
|
|
case 196: // Espeon (generated after Umbreon)
|
2017-05-28 05:18:04 +00:00
|
|
|
|
rev = 2+7;
|
2017-05-13 07:03:35 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
var seed = pidiv.OriginSeed;
|
|
|
|
|
var SIDf = pidiv.RNG.Reverse(seed, rev);
|
|
|
|
|
var TIDf = pidiv.RNG.Prev(SIDf);
|
|
|
|
|
if (SIDf >> 16 != pkm.SID || TIDf >> 16 != pkm.TID)
|
|
|
|
|
AddLine(Severity.Invalid, V400, CheckIdentifier.PID);
|
|
|
|
|
}
|
2017-04-30 23:53:54 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyAbility()
|
2016-03-21 15:01:08 +00:00
|
|
|
|
{
|
2016-10-29 06:41:22 +00:00
|
|
|
|
int[] abilities = pkm.PersonalInfo.Abilities;
|
2017-03-18 23:17:42 +00:00
|
|
|
|
if (abilities[1] == 0)
|
|
|
|
|
abilities[1] = abilities[0];
|
2016-10-23 19:48:49 +00:00
|
|
|
|
int abilval = Array.IndexOf(abilities, pkm.Ability);
|
2016-03-21 15:01:08 +00:00
|
|
|
|
if (abilval < 0)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V107, CheckIdentifier.Ability);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-03-21 15:01:08 +00:00
|
|
|
|
|
2017-04-14 23:24:41 +00:00
|
|
|
|
bool? AbilityUnchanged = true;
|
|
|
|
|
// 3 states flag: true for unchanged, false for changed, null for uncertain/allowing PID mismatch
|
|
|
|
|
// if true, check encounter ability
|
|
|
|
|
// if true or false, check PID/AbilityNumber
|
|
|
|
|
if (3 <= pkm.Format && pkm.Format <= 5 && abilities[0] != abilities[1]) // 3-5 and have 2 distinct ability now
|
|
|
|
|
AbilityUnchanged = verifyAbilityPreCapsule(abilities, abilval);
|
|
|
|
|
|
|
|
|
|
if (EncounterMatch != null)
|
2016-03-23 02:47:13 +00:00
|
|
|
|
{
|
2017-04-10 04:59:44 +00:00
|
|
|
|
// Check Ability Mismatches
|
|
|
|
|
int? EncounterAbility = (EncounterMatch as EncounterStatic)?.Ability ??
|
|
|
|
|
(EncounterMatch as EncounterTrade)?.Ability ??
|
|
|
|
|
(EncounterMatch as EncounterLink)?.Ability;
|
2017-04-12 03:55:34 +00:00
|
|
|
|
|
2017-05-09 03:35:06 +00:00
|
|
|
|
if (EncounterAbility != null && verifySetAbility(EncounterAbility, AbilityUnchanged, abilities, abilval))
|
|
|
|
|
return; // result added via verifySetAbility
|
2017-04-10 04:59:44 +00:00
|
|
|
|
|
2017-04-12 03:55:34 +00:00
|
|
|
|
switch (pkm.GenNumber)
|
2017-03-22 23:47:14 +00:00
|
|
|
|
{
|
2017-04-12 03:55:34 +00:00
|
|
|
|
case 5: verifyAbility5(abilities); break;
|
|
|
|
|
case 6: verifyAbility6(abilities); break;
|
|
|
|
|
case 7: verifyAbility7(abilities); break;
|
2016-03-31 02:55:27 +00:00
|
|
|
|
}
|
2016-03-23 02:47:13 +00:00
|
|
|
|
}
|
2016-03-22 04:50:39 +00:00
|
|
|
|
|
2017-04-12 03:11:58 +00:00
|
|
|
|
if (3 <= pkm.GenNumber && pkm.GenNumber <= 4 && pkm.AbilityNumber == 4)
|
|
|
|
|
AddLine(Severity.Invalid, V112, CheckIdentifier.Ability);
|
2017-04-14 23:24:41 +00:00
|
|
|
|
else if (AbilityUnchanged != null && abilities[pkm.AbilityNumber >> 1] != pkm.Ability)
|
|
|
|
|
AddLine(Severity.Invalid, pkm.Format < 6 ? V113 : V114, CheckIdentifier.Ability);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-04-12 03:55:34 +00:00
|
|
|
|
AddLine(Severity.Valid, V115, CheckIdentifier.Ability);
|
2016-03-21 15:01:08 +00:00
|
|
|
|
}
|
2017-05-09 03:35:06 +00:00
|
|
|
|
private bool verifySetAbility(int? EncounterAbility, bool? AbilityUnchanged, int[] abilities, int abilval)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.AbilityNumber == 4 && EncounterAbility != 4)
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V108, CheckIdentifier.Ability);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(AbilityUnchanged ?? false) || EncounterAbility == 0 || pkm.AbilityNumber == EncounterAbility)
|
|
|
|
|
return false;
|
|
|
|
|
|
2017-05-24 03:10:57 +00:00
|
|
|
|
if (EncounterMatch is EncounterTrade z && EncounterAbility == 1 << abilval && z.Species == pkm.Species) // Edge case (Static PID?)
|
2017-05-09 03:35:06 +00:00
|
|
|
|
AddLine(Severity.Valid, V115, CheckIdentifier.Ability);
|
2017-05-24 03:10:57 +00:00
|
|
|
|
else if (pkm.Format >= 6 && abilities[0] != abilities[1] && pkm.AbilityNumber < 4) // Ability Capsule
|
|
|
|
|
AddLine(Severity.Valid, V109, CheckIdentifier.Ability);
|
2017-05-09 03:35:06 +00:00
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V223, CheckIdentifier.Ability);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2017-04-14 23:24:41 +00:00
|
|
|
|
private bool? verifyAbilityPreCapsule(int[] abilities, int abilval)
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{
|
2017-04-15 02:55:40 +00:00
|
|
|
|
// CXD pokemon could have any ability without maching PID
|
2017-04-14 23:24:41 +00:00
|
|
|
|
if (pkm.Version == (int)GameVersion.CXD && pkm.Format == 3)
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
// gen3 native or gen4/5 origin
|
|
|
|
|
if (pkm.Format == 3 || !pkm.InhabitedGeneration(3))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
// Evovled in gen4/5
|
|
|
|
|
if (pkm.Species > Legal.MaxSpeciesID_3)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// gen3Species will be zero for pokemon with illegal gen 3 encounters, like Infernape with gen 3 "origin"
|
2017-05-28 04:17:53 +00:00
|
|
|
|
var gen3Species = info.EvoChainsAllGens[3].FirstOrDefault()?.Species ?? 0;
|
2017-04-14 23:24:41 +00:00
|
|
|
|
if (gen3Species == 0)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
// Fall through when gen3 pkm transferred to gen4/5
|
|
|
|
|
return verifyAbilityGen3Transfer(abilities, abilval, gen3Species);
|
2017-04-12 03:11:58 +00:00
|
|
|
|
}
|
2017-04-14 23:24:41 +00:00
|
|
|
|
private bool? verifyAbilityGen3Transfer(int[] abilities, int abilval, int Species_g3)
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{
|
2017-04-14 23:24:41 +00:00
|
|
|
|
var abilities_g3 = PersonalTable.E[Species_g3].Abilities.Where(a => a != 0).Distinct().ToArray();
|
2017-05-09 03:35:06 +00:00
|
|
|
|
if (abilities_g3.Length == 2) // Excluding Colosseum/XD, a gen3 pkm must match PID if it has 2 unique abilities
|
2017-04-12 03:11:58 +00:00
|
|
|
|
return pkm.Version != (int)GameVersion.CXD;
|
2017-04-14 23:24:41 +00:00
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
int Species_g4 = info.EvoChainsAllGens[4].FirstOrDefault()?.Species ?? 0;
|
|
|
|
|
int Species_g5 = pkm.Format == 5 ? info.EvoChainsAllGens[5].FirstOrDefault()?.Species ?? 0 : 0;
|
2017-05-09 03:35:06 +00:00
|
|
|
|
if (Math.Max(Species_g5, Species_g4) > Species_g3) // it has evolved in either gen 4 or gen 5; the ability must match PID
|
2017-04-14 23:24:41 +00:00
|
|
|
|
return false;
|
2017-04-12 03:11:58 +00:00
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
var Evolutions_g45 = Math.Max(info.EvoChainsAllGens[4].Length, pkm.Format == 5 ? info.EvoChainsAllGens[5].Length : 0);
|
2017-04-12 03:11:58 +00:00
|
|
|
|
if (Evolutions_g45 > 1)
|
|
|
|
|
{
|
|
|
|
|
// Evolutions_g45 > 1 and Species_g45 = Species_g3 with means both options, evolve in gen 4-5 or not evolve, are possible
|
|
|
|
|
if (pkm.Ability == abilities_g3[0])
|
|
|
|
|
// It could evolve in gen 4-5 an have generation 3 only ability
|
|
|
|
|
// that means it have not actually evolved in gen 4-5, ability do not need to match PID
|
2017-04-14 23:24:41 +00:00
|
|
|
|
return null;
|
2017-04-12 03:11:58 +00:00
|
|
|
|
if (pkm.Ability == abilities[1])
|
|
|
|
|
// It could evolve in gen4-5 an have generation 4 second ability
|
|
|
|
|
// that means it have actually evolved in gen 4-5, ability must match PID
|
2017-04-14 23:24:41 +00:00
|
|
|
|
return false;
|
2017-04-12 03:11:58 +00:00
|
|
|
|
}
|
|
|
|
|
// Evolutions_g45 == 1 means it have not evolved in gen 4-5 games,
|
|
|
|
|
// ability do not need to match PID, but only generation 3 ability is allowed
|
|
|
|
|
if (pkm.Ability != abilities_g3[0])
|
|
|
|
|
// Not evolved in gen4-5 but do not have generation 3 only ability
|
|
|
|
|
AddLine(Severity.Invalid, V373, CheckIdentifier.Ability);
|
2017-04-14 23:24:41 +00:00
|
|
|
|
return null;
|
2017-04-12 03:11:58 +00:00
|
|
|
|
}
|
2017-04-12 03:55:34 +00:00
|
|
|
|
private void verifyAbility5(int[] abilities)
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot w)
|
2017-04-12 03:55:34 +00:00
|
|
|
|
{
|
|
|
|
|
// Hidden Abilities for Wild Encounters are only available at a Hidden Grotto
|
2017-05-28 04:17:53 +00:00
|
|
|
|
bool grotto = w.Type == SlotType.HiddenGrotto;
|
2017-04-12 03:55:34 +00:00
|
|
|
|
if (pkm.AbilityNumber == 4 ^ grotto)
|
|
|
|
|
AddLine(Severity.Invalid, grotto ? V217 : V108, CheckIdentifier.Ability);
|
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
else if (Type == typeof(MysteryGift))
|
2017-04-12 03:55:34 +00:00
|
|
|
|
verifyAbilityMG456(abilities, ((PGF)EncounterMatch).AbilityType);
|
|
|
|
|
}
|
|
|
|
|
private void verifyAbility6(int[] abilities)
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot slot && pkm.AbilityNumber == 4)
|
2017-04-12 03:55:34 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
bool valid = slot.DexNav || slot.Type == SlotType.FriendSafari || slot.Type == SlotType.Horde;
|
2017-04-12 03:55:34 +00:00
|
|
|
|
if (!valid)
|
|
|
|
|
AddLine(Severity.Invalid, V300, CheckIdentifier.Ability);
|
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
else if (Type == typeof(MysteryGift))
|
2017-04-12 03:55:34 +00:00
|
|
|
|
verifyAbilityMG456(abilities, ((WC6)EncounterMatch).AbilityType);
|
|
|
|
|
else if (Legal.Ban_NoHidden6.Contains(pkm.SpecForm) && pkm.AbilityNumber == 4)
|
|
|
|
|
AddLine(Severity.Invalid, V112, CheckIdentifier.Ability);
|
|
|
|
|
}
|
|
|
|
|
private void verifyAbility7(int[] abilities)
|
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot slot && pkm.AbilityNumber == 4)
|
2017-04-12 03:55:34 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
bool valid = slot.Type == SlotType.SOS;
|
2017-04-12 03:55:34 +00:00
|
|
|
|
|
|
|
|
|
if (!valid)
|
|
|
|
|
AddLine(Severity.Invalid, V111, CheckIdentifier.Ability);
|
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
else if (Type == typeof(MysteryGift))
|
2017-04-12 03:55:34 +00:00
|
|
|
|
verifyAbilityMG456(abilities, ((WC7)EncounterMatch).AbilityType);
|
|
|
|
|
else if (Legal.Ban_NoHidden7.Contains(pkm.SpecForm) && pkm.AbilityNumber == 4)
|
|
|
|
|
AddLine(Severity.Invalid, V112, CheckIdentifier.Ability);
|
|
|
|
|
}
|
|
|
|
|
private void verifyAbilityMG456(int[] abilities, int cardtype)
|
|
|
|
|
{
|
|
|
|
|
int abilNumber = pkm.AbilityNumber;
|
|
|
|
|
if (cardtype < 3 && abilNumber != 1 << cardtype) // set number
|
|
|
|
|
{
|
|
|
|
|
// Ability can be flipped 0/1 if Ability Capsule is available, is not Hidden Ability, and Abilities are different.
|
|
|
|
|
if (pkm.Format >= 6 && cardtype < 2 && abilNumber < 3 && abilities[0] != abilities[1])
|
|
|
|
|
AddLine(Severity.Valid, V109, CheckIdentifier.Ability);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V110, CheckIdentifier.Ability);
|
|
|
|
|
}
|
|
|
|
|
else if (cardtype == 3 && abilNumber == 4) // 1/2 only
|
|
|
|
|
AddLine(Severity.Invalid, V110, CheckIdentifier.Ability);
|
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyBall
|
2017-03-21 07:18:38 +00:00
|
|
|
|
private void verifyBallEquals(params int[] balls)
|
|
|
|
|
{
|
|
|
|
|
int ball = pkm.Ball;
|
|
|
|
|
if (balls.Any(b => b == ball))
|
|
|
|
|
AddLine(Severity.Valid, V119, CheckIdentifier.Ball);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V118, CheckIdentifier.Ball);
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyBall()
|
2016-03-21 15:01:08 +00:00
|
|
|
|
{
|
2017-02-28 04:57:24 +00:00
|
|
|
|
if (pkm.Format < 3)
|
|
|
|
|
return; // no ball info saved
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2016-03-23 02:47:13 +00:00
|
|
|
|
if (!Encounter.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is MysteryGift g)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (pkm.Species == 490 && g.Ball == 0)
|
2017-03-30 18:58:30 +00:00
|
|
|
|
// there is no ball data in Manaphy Mystery Gift
|
|
|
|
|
verifyBallEquals(4); // Pokeball
|
|
|
|
|
else
|
2017-05-28 05:40:21 +00:00
|
|
|
|
verifyBallEquals(g.Ball);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is EncounterLink l)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
verifyBallEquals(l.Ball);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-04-22 20:04:12 +00:00
|
|
|
|
if (Type == typeof (EncounterTrade))
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Pokeball
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-13 19:12:55 +00:00
|
|
|
|
|
2017-04-08 00:45:22 +00:00
|
|
|
|
if (pkm.Species == 292 && pkm.GenNumber > 3) // Shedinja. For gen3, copy the ball from Nincada
|
2017-04-04 03:02:40 +00:00
|
|
|
|
{
|
2017-04-08 00:45:22 +00:00
|
|
|
|
verifyBallEquals(4); // Pokeball Only
|
2017-04-04 03:02:40 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-13 19:28:39 +00:00
|
|
|
|
if (pkm.Ball == 0x14 && pkm.Gen7) // Heavy Ball
|
2017-03-13 19:12:55 +00:00
|
|
|
|
{
|
2017-03-14 01:38:56 +00:00
|
|
|
|
var lineage = Legal.getLineage(pkm);
|
|
|
|
|
if (lineage.Any(e => Legal.AlolanCaptureNoHeavyBall.Contains(e)))
|
2017-03-15 04:13:39 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V116, CheckIdentifier.Ball);
|
2017-03-15 04:13:39 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-13 19:12:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterStatic s)
|
2016-04-21 03:46:18 +00:00
|
|
|
|
{
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (s.Gift)
|
|
|
|
|
verifyBallEquals(s.Ball);
|
2017-03-30 06:31:02 +00:00
|
|
|
|
else if (pkm.Met_Location == 75 && pkm.Gen5) // DreamWorld
|
|
|
|
|
verifyBallEquals(Legal.DreamWorldBalls);
|
2016-11-08 16:43:57 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(Legal.getWildBalls(pkm));
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (EncounterMatch is EncounterSlot w)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-01-25 03:49:40 +00:00
|
|
|
|
if (pkm.Met_Location == 30016 && pkm.Gen7) // Poké Pelago
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Pokeball
|
2017-03-30 11:24:38 +00:00
|
|
|
|
// For gen3/4 safari zones and BCC getValidWildEncounters already filter to not return
|
2017-03-30 11:18:05 +00:00
|
|
|
|
// mixed possible encounters between safari, BCC and other encounters
|
|
|
|
|
// That means is the first encounter is not safari then there is no safari encounter in the array
|
2017-05-28 04:17:53 +00:00
|
|
|
|
else if (3 <= pkm.GenNumber && pkm.GenNumber <= 4 && EncounterGenerator.IsSafariSlot(w.Type))
|
2017-03-31 08:12:49 +00:00
|
|
|
|
verifyBallEquals(5); // Safari Ball
|
2017-05-28 04:17:53 +00:00
|
|
|
|
else if (pkm.GenNumber == 4 && w.Type == SlotType.BugContest)
|
2017-03-31 08:12:49 +00:00
|
|
|
|
verifyBallEquals(0x18); // Sport Ball
|
2016-11-08 16:43:57 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(Legal.getWildBalls(pkm));
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pkm.WasEgg)
|
|
|
|
|
{
|
2017-02-23 05:19:29 +00:00
|
|
|
|
verifyBallEgg();
|
|
|
|
|
return;
|
2016-11-11 05:10:28 +00:00
|
|
|
|
}
|
2016-03-28 05:05:51 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Pokeball
|
2016-11-11 05:10:28 +00:00
|
|
|
|
}
|
2017-02-23 05:19:29 +00:00
|
|
|
|
private void verifyBallEgg()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.GenNumber < 6) // No inheriting Balls
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Must be Pokéball -- no ball inheritance.
|
2017-02-23 05:19:29 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pkm.Ball == 0x01) // Master Ball
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V117, CheckIdentifier.Ball); return; }
|
2017-02-23 05:19:29 +00:00
|
|
|
|
if (pkm.Ball == 0x10) // Cherish Ball
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V120, CheckIdentifier.Ball); return; }
|
2017-02-23 05:19:29 +00:00
|
|
|
|
if (pkm.Ball == 0x04) // Poké Ball
|
2017-03-21 07:18:38 +00:00
|
|
|
|
{ AddLine(Severity.Valid, V119, CheckIdentifier.Ball); return; }
|
2017-02-23 05:19:29 +00:00
|
|
|
|
|
|
|
|
|
switch (pkm.GenNumber)
|
|
|
|
|
{
|
|
|
|
|
case 6: // Gen6 Inheritance Rules
|
|
|
|
|
verifyBallEggGen6();
|
|
|
|
|
return;
|
|
|
|
|
case 7: // Gen7 Inheritance Rules
|
|
|
|
|
verifyBallEggGen7();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
private void verifyBallEggGen6()
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.Gender == 2) // Genderless
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Must be Pokéball as ball can only pass via mother (not Ditto!)
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (Legal.BreedMaleOnly.Contains(pkm.Species))
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4); // Must be Pokéball as ball can only pass via mother (not Ditto!)
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
int ball = pkm.Ball;
|
|
|
|
|
|
|
|
|
|
if (ball >= 26)
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V126, CheckIdentifier.Ball);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (ball == 0x05) // Safari Ball
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
|
|
|
|
if (Legal.getLineage(pkm).All(e => !Legal.Inherit_Safari.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x10 < ball && ball < 0x18) // Apricorn Ball
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
2017-02-04 18:42:56 +00:00
|
|
|
|
if (Legal.getLineage(pkm).All(e => !Legal.Inherit_Apricorn6.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
if (pkm.AbilityNumber == 4)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-03-22 04:31:06 +00:00
|
|
|
|
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball == 0x18) // Sport Ball
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
|
|
|
|
if (Legal.getLineage(pkm).All(e => !Legal.Inherit_Sport.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-03-23 02:47:13 +00:00
|
|
|
|
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball == 0x19) // Dream Ball
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (Legal.getLineage(pkm).Any(e => Legal.Inherit_Dream.Contains(e)))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2016-11-26 01:54:32 +00:00
|
|
|
|
if (pkm.AbilityNumber == 4 && Legal.Ban_DreamHidden.Contains(pkm.Species))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
2016-11-26 01:54:32 +00:00
|
|
|
|
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x0D <= ball && ball <= 0x0F)
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (!Legal.Ban_Gen4Ball_6.Contains(pkm.Species))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
|
|
|
|
|
return;
|
2016-03-29 05:30:23 +00:00
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x02 <= ball && ball <= 0x0C) // Don't worry, Ball # 0x05 was already checked.
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
|
|
|
|
if (Legal.Ban_Gen3Ball.Contains(pkm.Species))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2017-04-02 13:39:39 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4 && Legal.Ban_Gen3BallHidden.Contains(pkm.SpecForm))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-03-21 15:01:08 +00:00
|
|
|
|
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pkm.Species > 650 && pkm.Species != 700) // Sylveon
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (Legal.getWildBalls(pkm).Contains(pkm.Ball))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-11-12 14:52:40 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V125, CheckIdentifier.Ball);
|
2016-03-21 15:01:08 +00:00
|
|
|
|
}
|
2017-02-23 05:19:29 +00:00
|
|
|
|
private void verifyBallEggGen7()
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
2016-11-12 05:23:33 +00:00
|
|
|
|
var Lineage = Legal.getLineage(pkm).ToArray();
|
2016-12-02 02:48:38 +00:00
|
|
|
|
if (722 <= pkm.Species && pkm.Species <= 730) // G7 Starters
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(4);
|
2016-12-02 02:48:38 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
int ball = pkm.Ball;
|
|
|
|
|
if (ball == 0x05) // Safari Ball
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-04-04 03:02:40 +00:00
|
|
|
|
if (!Lineage.Any(e => Legal.Inherit_Safari.Contains(e) || Legal.Inherit_SafariMale.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2017-04-04 03:02:40 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4 && Lineage.Any(e => Legal.Ban_SafariBallHidden_7.Contains(e)))
|
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x10 < ball && ball < 0x18) // Apricorn Ball
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-04-04 03:02:40 +00:00
|
|
|
|
if (!Lineage.Any(e => Legal.Inherit_Apricorn7.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2017-04-04 03:02:40 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4 && (Lineage.Contains(029) || Lineage.Contains(032))) // Nido
|
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball == 0x18) // Sport Ball
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-04-04 03:02:40 +00:00
|
|
|
|
if (!Lineage.Any(e => Legal.Inherit_Sport.Contains(e)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2017-04-04 03:02:40 +00:00
|
|
|
|
else if (pkm.AbilityNumber == 4 && (Lineage.Contains(313) || Lineage.Contains(314))) // Volbeat/Illumise
|
|
|
|
|
AddLine(Severity.Invalid, V122, CheckIdentifier.Ball);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball == 0x19) // Dream Ball
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (Lineage.Any(e => Legal.Inherit_Dream.Contains(e) || Legal.Inherit_DreamMale.Contains(e)))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x0D <= ball && ball <= 0x0F) // Dusk Heal Quick
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (!Legal.Ban_Gen4Ball_7.Contains(pkm.Species))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (0x02 <= ball && ball <= 0x0C) // Don't worry, Ball # 0x05 was already checked.
|
2016-11-12 05:23:33 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (!Legal.Ban_Gen3Ball_7.Contains(pkm.Species))
|
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V121, CheckIdentifier.Ball);
|
2016-11-12 05:23:33 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-11-12 14:52:40 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball == 26)
|
2016-11-12 14:52:40 +00:00
|
|
|
|
{
|
2017-05-10 02:27:30 +00:00
|
|
|
|
if ((pkm.Species > 731 && pkm.Species <= 785) || Lineage.Any(e => Legal.PastGenAlolanNatives.Contains(e) && !Legal.PastGenAlolanNativesUncapturable.Contains(e)))
|
2016-11-12 14:52:40 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 14:52:40 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (Lineage.Any(e => Legal.PastGenAlolanScans.Contains(e)))
|
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, V123, CheckIdentifier.Ball);
|
2016-11-12 14:52:40 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// next statement catches all new alolans
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pkm.Species > 721)
|
2016-11-11 05:10:28 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyBallEquals(Legal.getWildBalls(pkm));
|
2016-11-11 05:10:28 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2016-11-12 14:52:40 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (ball >= 27)
|
|
|
|
|
{
|
|
|
|
|
AddLine(Severity.Invalid, V126, CheckIdentifier.Ball);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
AddLine(Severity.Invalid, V125, CheckIdentifier.Ball);
|
2016-11-11 05:10:28 +00:00
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#endregion
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private CheckResult verifyHistory()
|
2016-03-22 04:31:06 +00:00
|
|
|
|
{
|
2016-03-23 02:47:13 +00:00
|
|
|
|
if (!Encounter.Valid)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Valid, V127, CheckIdentifier.History);
|
2016-12-08 02:28:24 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.GenNumber < 6)
|
2016-12-08 02:28:24 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.Format < 6)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Valid, V128, CheckIdentifier.History);
|
2016-12-08 02:28:24 +00:00
|
|
|
|
|
|
|
|
|
if (pkm.OT_Affection > 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V129, CheckIdentifier.History);
|
2016-12-08 02:28:24 +00:00
|
|
|
|
if (pkm.OT_Memory > 0 || pkm.OT_Feeling > 0 || pkm.OT_Intensity > 0 || pkm.OT_TextVar > 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V130, CheckIdentifier.History);
|
2016-12-08 02:28:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-07 06:19:10 +00:00
|
|
|
|
if (pkm.Format >= 6 && pkm.GenNumber != pkm.Format && pkm.CurrentHandler != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V124, CheckIdentifier.History);
|
2017-02-07 06:19:10 +00:00
|
|
|
|
|
2016-12-08 02:28:24 +00:00
|
|
|
|
if (pkm.HT_Gender > 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V131, pkm.HT_Gender), CheckIdentifier.History);
|
2017-01-25 17:17:20 +00:00
|
|
|
|
|
2017-01-11 01:14:48 +00:00
|
|
|
|
MysteryGift mg = EncounterMatch as MysteryGift;
|
2016-03-23 05:49:46 +00:00
|
|
|
|
WC6 MatchedWC6 = EncounterMatch as WC6;
|
2017-01-11 01:14:48 +00:00
|
|
|
|
WC7 MatchedWC7 = EncounterMatch as WC7;
|
2016-03-22 04:31:06 +00:00
|
|
|
|
if (MatchedWC6?.OT.Length > 0) // Has Event OT -- null propagation yields false if MatchedWC6=null
|
|
|
|
|
{
|
2017-01-11 01:14:48 +00:00
|
|
|
|
if (pkm.OT_Friendship != PersonalTable.AO[MatchedWC6.Species].BaseFriendship)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V132, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.OT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V133, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.CurrentHandler != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V134, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
}
|
2017-01-11 01:14:48 +00:00
|
|
|
|
else if (MatchedWC7?.OT.Length > 0) // Has Event OT -- null propagation yields false if MatchedWC7=null
|
|
|
|
|
{
|
|
|
|
|
if (pkm.OT_Friendship != PersonalTable.SM[MatchedWC7.Species].BaseFriendship)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V132, CheckIdentifier.History);
|
2017-01-11 01:14:48 +00:00
|
|
|
|
if (pkm.OT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V133, CheckIdentifier.History);
|
2017-01-11 01:14:48 +00:00
|
|
|
|
if (pkm.CurrentHandler != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V134, CheckIdentifier.History);
|
2017-01-11 01:14:48 +00:00
|
|
|
|
}
|
|
|
|
|
else if (mg != null && mg.Format < 6 && pkm.Format >= 6)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.OT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V133, CheckIdentifier.History);
|
2017-01-11 01:14:48 +00:00
|
|
|
|
if (pkm.CurrentHandler != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V134, CheckIdentifier.History);
|
2017-01-11 01:14:48 +00:00
|
|
|
|
}
|
2017-01-31 01:49:28 +00:00
|
|
|
|
|
|
|
|
|
// Geolocations
|
|
|
|
|
var geo = new[]
|
2016-11-11 03:29:00 +00:00
|
|
|
|
{
|
2017-01-31 01:49:28 +00:00
|
|
|
|
pkm.Geo1_Country, pkm.Geo2_Country, pkm.Geo3_Country, pkm.Geo4_Country, pkm.Geo5_Country,
|
|
|
|
|
pkm.Geo1_Region, pkm.Geo2_Region, pkm.Geo3_Region, pkm.Geo4_Region, pkm.Geo5_Region,
|
|
|
|
|
};
|
2017-01-27 13:59:08 +00:00
|
|
|
|
|
2017-01-31 01:49:28 +00:00
|
|
|
|
// Check sequential order (no zero gaps)
|
|
|
|
|
bool geoEnd = false;
|
|
|
|
|
for (int i = 0; i < 5; i++)
|
|
|
|
|
{
|
|
|
|
|
if (geoEnd && geo[i] != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V135, CheckIdentifier.History);
|
2017-01-27 13:59:08 +00:00
|
|
|
|
|
2017-01-31 01:49:28 +00:00
|
|
|
|
if (geo[i] != 0)
|
|
|
|
|
continue;
|
|
|
|
|
if (geo[i + 5] != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V136, CheckIdentifier.History);
|
2017-01-31 01:49:28 +00:00
|
|
|
|
geoEnd = true;
|
|
|
|
|
}
|
|
|
|
|
if (pkm.Format >= 7)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.VC1)
|
2017-01-27 13:59:08 +00:00
|
|
|
|
{
|
|
|
|
|
var hasGeo = geo.Any(d => d != 0);
|
|
|
|
|
|
|
|
|
|
if (!hasGeo)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V137, CheckIdentifier.History);
|
2017-01-27 13:59:08 +00:00
|
|
|
|
}
|
2016-12-03 03:50:02 +00:00
|
|
|
|
|
2017-01-25 17:17:20 +00:00
|
|
|
|
if (pkm.GenNumber >= 7 && pkm.CNTs.Any(stat => stat > 0))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V138, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
|
|
|
|
|
if (!pkm.WasEvent && pkm.HT_Name.Length == 0) // Is not Traded
|
|
|
|
|
{
|
|
|
|
|
if (pkm.CurrentHandler != 0) // Badly edited; PKHeX doesn't trip this.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V139, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
if (pkm.HT_Friendship != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V140, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
if (pkm.HT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V141, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
|
|
|
|
|
// We know it is untraded (HT is empty), if it must be trade evolved flag it.
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (Legal.getHasTradeEvolved(pkm) && EncounterMatch.Species != pkm.Species)
|
2016-12-03 03:50:02 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.Species != 350) // Milotic
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V142, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
if (pkm.CNT_Beauty < 170) // Beauty Contest Stat Requirement
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V143, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
if (pkm.CurrentLevel == 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V144, CheckIdentifier.History);
|
2016-12-03 03:50:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Valid, V145, CheckIdentifier.History);
|
2016-11-11 03:29:00 +00:00
|
|
|
|
}
|
2017-04-01 21:20:00 +00:00
|
|
|
|
|
|
|
|
|
// Determine if we should check for Handling Trainer Memories
|
|
|
|
|
// A Pokémon is untraded if...
|
|
|
|
|
bool untraded = pkm.HT_Name.Length == 0 || pkm.Geo1_Country == 0;
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (Type == typeof(MysteryGift))
|
2017-04-01 21:20:00 +00:00
|
|
|
|
{
|
|
|
|
|
untraded |= !pkm.WasEventEgg;
|
|
|
|
|
untraded &= pkm.WasEgg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pkm.WasLink && (EncounterMatch as EncounterLink)?.OT == false)
|
|
|
|
|
untraded = false;
|
|
|
|
|
else if (pkm.GenNumber < 6)
|
|
|
|
|
untraded = false;
|
|
|
|
|
|
|
|
|
|
if (untraded) // Is not Traded
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.HT_Name.Length != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V146, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Geo1_Country != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V147, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.HT_Memory != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V148, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.CurrentHandler != 0) // Badly edited; PKHeX doesn't trip this.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V139, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.HT_Friendship != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V140, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.HT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V141, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.XY && pkm.CNTs.Any(stat => stat > 0))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V138, CheckIdentifier.History);
|
2016-03-22 04:31:06 +00:00
|
|
|
|
|
|
|
|
|
// We know it is untraded (HT is empty), if it must be trade evolved flag it.
|
2017-05-28 04:17:53 +00:00
|
|
|
|
if (Legal.getHasTradeEvolved(pkm) && EncounterMatch.Species != pkm.Species)
|
2016-03-22 04:31:06 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Species != 350) // Milotic
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V142, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.CNT_Beauty < 170) // Beauty Contest Stat Requirement
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V143, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.CurrentLevel == 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V144, CheckIdentifier.History);
|
2016-03-22 04:31:06 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else // Is Traded
|
|
|
|
|
{
|
2017-02-23 05:19:29 +00:00
|
|
|
|
if (pkm.Format == 6 && pkm.HT_Memory == 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V150, CheckIdentifier.History);
|
2016-03-22 04:31:06 +00:00
|
|
|
|
}
|
2016-03-31 01:53:34 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
// Memory ChecksResult
|
|
|
|
|
if (pkm.IsEgg)
|
2016-03-31 01:53:34 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.HT_Memory != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V149, CheckIdentifier.History);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.OT_Memory != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V151, CheckIdentifier.History);
|
2016-03-31 01:53:34 +00:00
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
else if (MatchedType != typeof(WC6))
|
2016-03-31 01:53:34 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.OT_Memory == 0 ^ !pkm.Gen6)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V152, CheckIdentifier.History);
|
2016-11-11 06:11:53 +00:00
|
|
|
|
if (pkm.GenNumber < 6 && pkm.OT_Affection != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, V129, CheckIdentifier.History);
|
2016-03-31 01:53:34 +00:00
|
|
|
|
}
|
2016-03-22 04:31:06 +00:00
|
|
|
|
// Unimplemented: Ingame Trade Memories
|
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Valid, V145, CheckIdentifier.History);
|
2016-03-22 04:31:06 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private CheckResult verifyCommonMemory(int handler)
|
2016-05-07 06:16:35 +00:00
|
|
|
|
{
|
|
|
|
|
int m = 0;
|
2016-05-09 23:09:40 +00:00
|
|
|
|
int t = 0;
|
2016-05-07 06:16:35 +00:00
|
|
|
|
string resultPrefix = "";
|
|
|
|
|
switch (handler)
|
|
|
|
|
{
|
|
|
|
|
case 0:
|
2016-10-23 19:48:49 +00:00
|
|
|
|
m = pkm.OT_Memory;
|
|
|
|
|
t = pkm.OT_TextVar;
|
2017-03-21 07:18:38 +00:00
|
|
|
|
resultPrefix = V205;
|
2016-05-07 06:16:35 +00:00
|
|
|
|
break;
|
|
|
|
|
case 1:
|
2016-10-23 19:48:49 +00:00
|
|
|
|
m = pkm.HT_Memory;
|
|
|
|
|
t = pkm.HT_TextVar;
|
2017-03-21 07:18:38 +00:00
|
|
|
|
resultPrefix = V206;
|
2016-05-07 06:16:35 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
int matchingMoveMemory = Array.IndexOf(Legal.MoveSpecificMemories[0], m);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
if (matchingMoveMemory != -1 && pkm.Species != 235 && !Legal.getCanLearnMachineMove(pkm, Legal.MoveSpecificMemories[1][matchingMoveMemory], 6))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V153, resultPrefix), CheckIdentifier.Memory);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
|
2016-05-09 23:09:40 +00:00
|
|
|
|
if (m == 6 && !Legal.LocationsWithPKCenter[0].Contains(t))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V154, resultPrefix), CheckIdentifier.Memory);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
|
2016-05-10 01:19:31 +00:00
|
|
|
|
if (m == 21) // {0} saw {2} carrying {1} on its back. {4} that {3}.
|
2017-02-21 03:40:50 +00:00
|
|
|
|
if (!Legal.getCanLearnMachineMove(new PK6 {Species = t, EXP = PKX.getEXP(100, t)}, 19, 6))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V153, resultPrefix), CheckIdentifier.Memory);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
|
|
|
|
|
if ((m == 16 || m == 48) && (t == 0 || !Legal.getCanKnowMove(pkm, t, 6)))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V153, resultPrefix), CheckIdentifier.Memory);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
|
|
|
|
|
if (m == 49 && (t == 0 || !Legal.getCanRelearnMove(pkm, t, 6))) // {0} was able to remember {2} at {1}'s instruction. {4} that {3}.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Invalid, string.Format(V153, resultPrefix), CheckIdentifier.Memory);
|
2017-02-21 03:40:50 +00:00
|
|
|
|
|
2017-03-21 07:18:38 +00:00
|
|
|
|
return new CheckResult(Severity.Valid, string.Format(V155, resultPrefix), CheckIdentifier.Memory);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void verifyOTMemoryIs(int[] values)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.OT_Memory != values[0])
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V197, V205, values[0]), CheckIdentifier.Memory);
|
|
|
|
|
if (pkm.OT_Intensity != values[1])
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V198, V205, values[1]), CheckIdentifier.Memory);
|
|
|
|
|
if (pkm.OT_TextVar != values[2])
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V199, V205, values[2]), CheckIdentifier.Memory);
|
|
|
|
|
if (pkm.OT_Feeling != values[3])
|
|
|
|
|
AddLine(Severity.Invalid, string.Format(V200, V205, values[3]), CheckIdentifier.Memory);
|
2016-05-07 06:16:35 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyOTMemory()
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2016-11-26 01:30:39 +00:00
|
|
|
|
if (pkm.Format < 6)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-11-26 01:30:39 +00:00
|
|
|
|
|
|
|
|
|
if (!History.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
|
2016-11-26 01:30:39 +00:00
|
|
|
|
if (pkm.GenNumber < 6)
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2017-05-03 02:08:16 +00:00
|
|
|
|
verifyOTMemoryIs(new [] {0, 0, 0, 0}); // empty
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2016-11-26 01:30:39 +00:00
|
|
|
|
|
2017-04-22 20:04:12 +00:00
|
|
|
|
if (Type == typeof(EncounterTrade))
|
2017-01-25 17:17:20 +00:00
|
|
|
|
{
|
|
|
|
|
// Undocumented, uncommon, and insignificant -- don't bother.
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (MatchedType == typeof(WC6))
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
WC6 g = EncounterMatch as WC6;
|
|
|
|
|
verifyOTMemoryIs(new[] {g.OT_Memory, g.OT_Intensity, g.OT_TextVar, g.OT_Feeling});
|
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (MatchedType == typeof(WC7))
|
2016-11-26 01:30:39 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
WC7 g = EncounterMatch as WC7;
|
|
|
|
|
verifyOTMemoryIs(new[] {g.OT_Memory, g.OT_Intensity, g.OT_TextVar, g.OT_Feeling});
|
|
|
|
|
return;
|
2016-11-26 01:30:39 +00:00
|
|
|
|
}
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (pkm.GenNumber >= 7)
|
2016-11-26 01:30:39 +00:00
|
|
|
|
{
|
2017-03-21 07:18:38 +00:00
|
|
|
|
verifyOTMemoryIs(new[] {0, 0, 0, 0}); // empty
|
2016-11-26 01:30:39 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
switch (pkm.OT_Memory)
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2016-05-25 05:20:19 +00:00
|
|
|
|
case 2: // {0} hatched from an Egg and saw {1} for the first time at... {2}. {4} that {3}.
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (!pkm.WasEgg && pkm.Egg_Location != 60004)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V160, V205), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
break;
|
|
|
|
|
|
2016-05-06 03:05:22 +00:00
|
|
|
|
case 4: // {0} became {1}’s friend when it arrived via Link Trade at... {2}. {4} that {3}.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V161, V205), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2016-05-09 23:09:40 +00:00
|
|
|
|
case 6: // {0} went to the Pokémon Center in {2} with {1} and had its tired body healed there. {4} that {3}.
|
2016-10-23 19:48:49 +00:00
|
|
|
|
int matchingOriginGame = Array.IndexOf(Legal.LocationsWithPKCenter[0], pkm.OT_TextVar);
|
2016-05-10 00:51:35 +00:00
|
|
|
|
if (matchingOriginGame != -1)
|
|
|
|
|
{
|
|
|
|
|
int gameID = Legal.LocationsWithPKCenter[1][matchingOriginGame];
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.XY && gameID != 0 || pkm.AO && gameID != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V162, V205), CheckIdentifier.Memory);
|
2016-05-10 00:51:35 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
AddLine(verifyCommonMemory(0));
|
|
|
|
|
return;
|
|
|
|
|
|
2016-05-06 03:35:18 +00:00
|
|
|
|
case 14:
|
2016-11-09 06:10:32 +00:00
|
|
|
|
if (!Legal.getCanBeCaptured(pkm.OT_TextVar, pkm.GenNumber, (GameVersion)pkm.Version))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V165, V205), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Valid, string.Format(V164, V205), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.XY && Legal.Memory_NotXY.Contains(pkm.OT_Memory))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V163, V205), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.AO && Legal.Memory_NotAO.Contains(pkm.OT_Memory))
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V163, V205), CheckIdentifier.Memory);
|
2016-05-06 03:05:22 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
AddLine(verifyCommonMemory(0));
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyHTMemory()
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Format < 6)
|
|
|
|
|
return;
|
|
|
|
|
|
2016-05-06 03:05:22 +00:00
|
|
|
|
if (!History.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
|
2017-03-12 23:36:23 +00:00
|
|
|
|
if (pkm.Format >= 7)
|
2016-11-26 01:30:39 +00:00
|
|
|
|
{
|
2017-03-12 23:36:23 +00:00
|
|
|
|
/*
|
|
|
|
|
* Bank Transfer adds in the Link Trade Memory.
|
|
|
|
|
* Trading 7<->7 between games (not Bank) clears this data.
|
|
|
|
|
*/
|
|
|
|
|
if (pkm.HT_Memory == 0)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.HT_TextVar != 0 || pkm.HT_Intensity != 0 || pkm.HT_Feeling != 0)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V329, CheckIdentifier.Memory);
|
2017-01-27 13:59:08 +00:00
|
|
|
|
return;
|
2017-03-12 23:36:23 +00:00
|
|
|
|
}
|
2017-01-27 04:13:27 +00:00
|
|
|
|
|
2017-05-19 00:38:26 +00:00
|
|
|
|
// Transfer 6->7 & withdraw to same HT => keeps past gen memory
|
|
|
|
|
// Don't require link trade memory for these past gen cases
|
|
|
|
|
int gen = pkm.GenNumber;
|
|
|
|
|
if (3 <= gen && gen < 7 && pkm.CurrentHandler == 1)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-01-27 13:59:08 +00:00
|
|
|
|
if (pkm.HT_Memory != 4)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V156, CheckIdentifier.Memory);
|
2017-01-27 05:35:26 +00:00
|
|
|
|
if (pkm.HT_TextVar != 0)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V157, CheckIdentifier.Memory);
|
2017-01-27 13:59:08 +00:00
|
|
|
|
if (pkm.HT_Intensity != 1)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V158, CheckIdentifier.Memory);
|
2017-01-27 13:59:08 +00:00
|
|
|
|
if (pkm.HT_Feeling > 10)
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V159, CheckIdentifier.Memory);
|
2016-11-26 01:30:39 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
switch (pkm.HT_Memory)
|
2016-05-06 03:05:22 +00:00
|
|
|
|
{
|
2017-01-27 04:13:27 +00:00
|
|
|
|
case 0:
|
2017-01-27 13:59:08 +00:00
|
|
|
|
if (string.IsNullOrEmpty(pkm.HT_Name))
|
2017-01-27 04:13:27 +00:00
|
|
|
|
return;
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, V150, CheckIdentifier.Memory); return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
case 1: // {0} met {1} at... {2}. {1} threw a Poké Ball at it, and they started to travel together. {4} that {3}.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V202, V206), CheckIdentifier.Memory); return;
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2016-05-06 03:05:22 +00:00
|
|
|
|
case 2: // {0} hatched from an Egg and saw {1} for the first time at... {2}. {4} that {3}.
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V160, V206), CheckIdentifier.Memory); return;
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
2016-05-06 03:35:18 +00:00
|
|
|
|
case 14:
|
2017-03-21 07:18:38 +00:00
|
|
|
|
if (Legal.getCanBeCaptured(pkm.HT_TextVar, pkm.GenNumber))
|
|
|
|
|
AddLine(Severity.Valid, string.Format(V164, V206), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-21 07:18:38 +00:00
|
|
|
|
AddLine(Severity.Invalid, string.Format(V165, V206), CheckIdentifier.Memory);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
AddLine(verifyCommonMemory(1));
|
2016-05-06 03:05:22 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyRegion()
|
2016-05-11 23:20:31 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Format < 6)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
bool pass;
|
|
|
|
|
switch (pkm.ConsoleRegion)
|
2016-05-11 23:20:31 +00:00
|
|
|
|
{
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 0: // Japan
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = pkm.Country == 1;
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 1: // Americas
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = 8 <= pkm.Country && pkm.Country <= 52 || new[] {153, 156, 168, 174, 186}.Contains(pkm.Country);
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 2: // Europe
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = 64 <= pkm.Country && pkm.Country <= 127 || new[] {169, 184, 185}.Contains(pkm.Country);
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 4: // China
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = pkm.Country == 144 || pkm.Country == 160;
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 5: // Korea
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = pkm.Country == 136;
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-05-12 04:18:05 +00:00
|
|
|
|
case 6: // Taiwan
|
2016-10-23 19:48:49 +00:00
|
|
|
|
pass = pkm.Country == 128;
|
2016-05-11 23:20:31 +00:00
|
|
|
|
break;
|
2016-10-23 19:48:49 +00:00
|
|
|
|
default:
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(new CheckResult(Severity.Invalid, V301, CheckIdentifier.Geography));
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-05-11 23:20:31 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
if (!pass)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V302, CheckIdentifier.Geography);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V303, CheckIdentifier.Geography);
|
2016-05-11 23:20:31 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyForm()
|
2016-04-14 10:17:03 +00:00
|
|
|
|
{
|
|
|
|
|
if (!Encounter.Valid)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
2016-04-14 10:17:03 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.Format < 4)
|
|
|
|
|
return;
|
|
|
|
|
|
2016-11-09 05:43:22 +00:00
|
|
|
|
if (pkm.AltForm > pkm.PersonalInfo.FormeCount)
|
|
|
|
|
{
|
2016-11-14 15:33:39 +00:00
|
|
|
|
bool valid = false;
|
|
|
|
|
int species = pkm.Species;
|
|
|
|
|
if (species == 201) // Unown
|
|
|
|
|
{
|
2017-03-25 17:24:56 +00:00
|
|
|
|
int maxCount = pkm.GenNumber == 2 ? 26 : 28; // A-Z : A-Z?!
|
|
|
|
|
if (pkm.AltForm < maxCount)
|
2016-11-14 15:33:39 +00:00
|
|
|
|
valid = true;
|
|
|
|
|
}
|
|
|
|
|
if (species == 414 && pkm.AltForm < 3) // Wormadam base form kept
|
2017-03-25 17:24:56 +00:00
|
|
|
|
valid = true;
|
2016-11-14 15:33:39 +00:00
|
|
|
|
|
|
|
|
|
if ((species == 664 || species == 665) && pkm.AltForm < 18) // Vivillon Pre-evolutions
|
|
|
|
|
valid = true;
|
|
|
|
|
|
|
|
|
|
if (!valid) // ignore list
|
2017-03-23 03:12:45 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, string.Format(V304, pkm.PersonalInfo.FormeCount, pkm.AltForm), CheckIdentifier.Form); return; }
|
2016-11-09 05:43:22 +00:00
|
|
|
|
}
|
2017-05-28 04:17:53 +00:00
|
|
|
|
|
|
|
|
|
if (EncounterMatch is EncounterSlot w && w.Type == SlotType.FriendSafari)
|
|
|
|
|
verifyFormFriendSafari();
|
2016-11-09 05:43:22 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
switch (pkm.Species)
|
2016-04-14 10:17:03 +00:00
|
|
|
|
{
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 25: // Pikachu
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (pkm.Format == 6 && pkm.AltForm != 0 ^ Type == typeof(EncounterStatic))
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-04-24 00:53:22 +00:00
|
|
|
|
string msg = Type == typeof(EncounterStatic) ? V305 : V306;
|
2017-02-21 03:40:50 +00:00
|
|
|
|
AddLine(Severity.Invalid, msg, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (pkm.Format == 7 && pkm.AltForm != 0 ^ Type == typeof(MysteryGift))
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-05-13 17:20:25 +00:00
|
|
|
|
if (EncounterMatch is WC7 gift && gift.Form != pkm.AltForm)
|
2016-10-23 19:48:49 +00:00
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V307, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-04-23 21:16:12 +00:00
|
|
|
|
break;
|
2017-01-31 22:01:32 +00:00
|
|
|
|
case 487: // Giratina
|
2017-02-03 03:30:17 +00:00
|
|
|
|
if (pkm.AltForm == 1 ^ pkm.HeldItem == 112) // Origin form only with Griseous Orb
|
2017-01-31 22:01:32 +00:00
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V308, CheckIdentifier.Form);
|
2017-01-31 22:01:32 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2017-02-01 04:11:11 +00:00
|
|
|
|
case 493: // Arceus
|
|
|
|
|
{
|
|
|
|
|
int item = pkm.HeldItem;
|
|
|
|
|
int form = 0;
|
|
|
|
|
if ((298 <= item && item <= 313) || item == 644)
|
|
|
|
|
form = Array.IndexOf(Legal.Arceus_Plate, item) + 1;
|
|
|
|
|
else if (777 <= item && item <= 793)
|
|
|
|
|
form = Array.IndexOf(Legal.Arceus_ZCrystal, item) + 1;
|
|
|
|
|
if (form != pkm.AltForm)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V308, CheckIdentifier.Form);
|
2017-02-01 04:11:11 +00:00
|
|
|
|
else if (form != 0)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V309, CheckIdentifier.Form);
|
2017-02-01 04:11:11 +00:00
|
|
|
|
}
|
2017-02-13 01:06:09 +00:00
|
|
|
|
break;
|
2017-03-23 06:34:35 +00:00
|
|
|
|
case 647: // Keldeo
|
|
|
|
|
{
|
|
|
|
|
int index = Array.IndexOf(pkm.Moves, 548); // Secret Sword
|
|
|
|
|
bool noSword = index < 0;
|
|
|
|
|
if (pkm.AltForm == 0 ^ noSword) // mismatch
|
2017-05-28 04:17:53 +00:00
|
|
|
|
info.vMoves[noSword ? 0 : index] = new CheckResult(Severity.Invalid, V169, CheckIdentifier.Move);
|
2017-03-23 06:34:35 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2017-02-13 01:06:09 +00:00
|
|
|
|
case 649: // Genesect
|
|
|
|
|
{
|
|
|
|
|
int item = pkm.HeldItem;
|
|
|
|
|
int form = 0;
|
|
|
|
|
if (116 <= item && item <= 119)
|
|
|
|
|
form = item - 115;
|
|
|
|
|
|
|
|
|
|
if (form != pkm.AltForm)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V308, CheckIdentifier.Form);
|
2017-02-13 01:06:09 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V309, CheckIdentifier.Form);
|
2017-02-13 01:06:09 +00:00
|
|
|
|
}
|
2017-02-01 04:11:11 +00:00
|
|
|
|
break;
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 658: // Greninja
|
2016-11-09 05:43:22 +00:00
|
|
|
|
if (pkm.AltForm > 1) // Ash Battle Bond active
|
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V310, CheckIdentifier.Form);
|
2016-11-09 05:43:22 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 664: // Scatterbug
|
|
|
|
|
case 665: // Spewpa
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.AltForm > 17) // Fancy & Pokéball
|
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V311, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
if (!Legal.CheckVivillonPattern(pkm.AltForm, pkm.Country, pkm.Region))
|
|
|
|
|
AddLine(Severity.Fishy, V312, CheckIdentifier.Form);
|
2016-04-14 10:17:03 +00:00
|
|
|
|
break;
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 666: // Vivillon
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.AltForm > 17) // Fancy & Pokéball
|
|
|
|
|
{
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (Type != typeof(MysteryGift))
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V312, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V313, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-03-25 17:24:56 +00:00
|
|
|
|
if (!Legal.CheckVivillonPattern(pkm.AltForm, pkm.Country, pkm.Region))
|
|
|
|
|
AddLine(Severity.Fishy, V312, CheckIdentifier.Form);
|
2016-04-14 10:17:03 +00:00
|
|
|
|
break;
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 670: // Floette
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.AltForm == 5) // Eternal Flower -- Never Released
|
|
|
|
|
{
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (Type != typeof(MysteryGift))
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V314, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V315, CheckIdentifier.Form);
|
2016-10-23 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-04-14 10:17:03 +00:00
|
|
|
|
break;
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 718: // Zygarde
|
2016-11-09 05:43:22 +00:00
|
|
|
|
if (pkm.AltForm >= 4)
|
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V310, CheckIdentifier.Form);
|
2016-11-09 05:43:22 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2017-02-01 04:11:11 +00:00
|
|
|
|
case 773: // Silvally
|
|
|
|
|
{
|
|
|
|
|
int item = pkm.HeldItem;
|
|
|
|
|
int form = 0;
|
|
|
|
|
if ((904 <= item && item <= 920) || item == 644)
|
|
|
|
|
form = item - 903;
|
|
|
|
|
if (form != pkm.AltForm)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V308, CheckIdentifier.Form);
|
2017-02-01 04:11:11 +00:00
|
|
|
|
else if (form != 0)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V309, CheckIdentifier.Form);
|
2017-02-01 04:11:11 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2016-11-14 02:25:33 +00:00
|
|
|
|
case 774: // Minior
|
|
|
|
|
if (pkm.AltForm < 7)
|
2016-11-09 05:43:22 +00:00
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V310, CheckIdentifier.Form);
|
2016-11-09 05:43:22 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2017-02-03 03:27:48 +00:00
|
|
|
|
|
|
|
|
|
// Party Only Forms
|
|
|
|
|
case 492: // Shaymin
|
|
|
|
|
case 676: // Furfrou
|
|
|
|
|
case 720: // Hoopa
|
2017-04-15 00:49:13 +00:00
|
|
|
|
if (pkm.AltForm != 0 && pkm.Box > -1 && pkm.Format <= 6) // has form but stored in box
|
2017-02-03 03:30:17 +00:00
|
|
|
|
{
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V316, CheckIdentifier.Form);
|
2017-02-03 03:30:17 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-02-03 03:27:48 +00:00
|
|
|
|
break;
|
2016-04-14 10:17:03 +00:00
|
|
|
|
}
|
2016-11-14 02:19:09 +00:00
|
|
|
|
|
2017-02-02 05:59:39 +00:00
|
|
|
|
if (pkm.Format >= 7 && pkm.GenNumber < 7 && pkm.AltForm != 0)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Species == 25 || Legal.AlolanOriginForms.Contains(pkm.Species))
|
2017-03-23 03:12:45 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V317, CheckIdentifier.Form); return; }
|
2017-02-02 05:59:39 +00:00
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.AltForm > 0 && new[] {Legal.BattleForms, Legal.BattleMegas, Legal.BattlePrimals}.Any(arr => arr.Contains(pkm.Species)))
|
2017-03-23 03:12:45 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V310, CheckIdentifier.Form); return; }
|
2016-04-14 10:17:03 +00:00
|
|
|
|
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V318, CheckIdentifier.Form);
|
2016-04-14 10:17:03 +00:00
|
|
|
|
}
|
Generation 1 and 2 legal Improvements (#1099)
* Refactor parseMovesForEncounter to gather valid moves for species encounter, some Pokemon can have valid encounters with different source species from the encounter, the valid moves change if the encounter species change because some preevolutions moves are illegal if pokemon caught already evolved.
Example, generation 1 Pikachu that can have a RBY Pikachu encounter and GSC Pichu encounter, the valid moves for the first encounters should not have any Pichu exclusive evolution moves
Also assign the encounter match from gb when parsing moves like the variable Encounter Match, to store the encounter that is valid for the pokemon moves instead the first encounter.
Store the species encounter, this will be needed to check if the evolution is valid for species that evolve leveling with a given learned move
* Add Tradeback Status to the pokemon, this variable for generations 1 and 2 use data like the catch rate to determine if trade between generations 1 and 2 was possible.
If analysis is for VC games tradeback have value NotTradeback for every gen 1 pokemon, but for cart saves some pokemon can be determine that have not been tradeback, if catch rate match species catch rate but do not match a valid generation 2 held item that means the pokemon habe been never traded to generation 2 games, that allow to discart encounters and moves from generation 2.
Also if is not tradeback catch rate is used to filter encounters, catch rate determine in what species was captured the pokemon discarting some preevolutions moves
Also add option for generation 1 cart save analysis to check legal status not allowing generation 2 games, like VC games but with Stadium allowed, like the generation 1 non tradeback rules from Smogon
Also change evolution chains to included generation 2 preevolutions for gen 1 pokemon if tradeback was possible, it is needed to avoid parsemoves to check illegal pokemon like Hitmonchan with Tyrogue level up moves
* Check legal values of generation 1 type and catch rate
Replace pokemon catch rate after changind pokemon species always if pokemon was not tradeback from generation 2, the catch rate will keep unchanged only if it can be a held item and do not match species catch rate (default item)
Also if catch rate is changed use base species catch rate to avoid legal errors if the catch rate of the evolution species if is not possible with the current moves
* Filter ingame trades and static encounters with catch rate for generation 1 non tradeback
* Fix min moves for generation 1 metapod encounter
* Clean up
* Fix encounter level for generation 1, valid moves are those with one level after the encounter level, pokemon can not learn a new move until level up
Clean up type validation
Fix generation 3 fatefull encounter eggs, the pokemon lost the fatefull mark when it hatch
* Clean-up
* Use new variable EncounterSpecies when it is needed to detect the species of the encounter, the old code wont work if the encounter is a wild slots array
* Fix generation 1 evolution chains and catch rate as default held item
* Fix Generation 1 Yellow Pikachu and Kadabra catch rates
2017-04-27 04:27:59 +00:00
|
|
|
|
private void verifyMiscG1()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Format > 1)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
var Type_A = (pkm as PK1).Type_A;
|
|
|
|
|
var Type_B = (pkm as PK1).Type_B;
|
|
|
|
|
if (pkm.Species == 137)
|
|
|
|
|
{
|
|
|
|
|
// Porygon can have any type combination of any generation 1 species because of the move Conversion,
|
|
|
|
|
// that change Porygon type to match the oponent types
|
|
|
|
|
var Type_A_Match = Legal.Types_Gen1.Any(t => t == Type_A);
|
|
|
|
|
var Type_B_Match = Legal.Types_Gen1.Any(t => t == Type_B);
|
|
|
|
|
if (!Type_A_Match)
|
|
|
|
|
AddLine(Severity.Invalid, V386, CheckIdentifier.Misc);
|
|
|
|
|
if (!Type_B_Match)
|
|
|
|
|
AddLine(Severity.Invalid, V387, CheckIdentifier.Misc);
|
|
|
|
|
if (Type_A_Match && Type_B_Match)
|
|
|
|
|
{
|
|
|
|
|
var TypesAB_Match = PersonalTable.RB.IsValidTypeCombination(Type_A, Type_B);
|
|
|
|
|
if (TypesAB_Match)
|
|
|
|
|
AddLine(Severity.Valid, V391, CheckIdentifier.Misc);
|
|
|
|
|
else
|
|
|
|
|
AddLine(Severity.Invalid, V388, CheckIdentifier.Misc);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else // Types must match species types
|
|
|
|
|
{
|
|
|
|
|
var Type_A_Match = Type_A == PersonalTable.RB[pkm.Species].Types[0];
|
|
|
|
|
var Type_B_Match = Type_B == PersonalTable.RB[pkm.Species].Types[1];
|
|
|
|
|
|
|
|
|
|
AddLine(Type_A_Match ? Severity.Valid : Severity.Invalid, Type_A_Match ? V392 : V389, CheckIdentifier.Misc);
|
|
|
|
|
|
|
|
|
|
AddLine(Type_B_Match ? Severity.Valid : Severity.Invalid, Type_B_Match ? V393 : V390, CheckIdentifier.Misc);
|
|
|
|
|
}
|
|
|
|
|
var catch_rate =(pkm as PK1).Catch_Rate;
|
|
|
|
|
switch (pkm.TradebackStatus)
|
|
|
|
|
{
|
|
|
|
|
case TradebackType.Any:
|
|
|
|
|
case TradebackType.WasTradeback:
|
|
|
|
|
if (catch_rate == 0 || Legal.HeldItems_GSC.Any(h => h == catch_rate))
|
|
|
|
|
{ AddLine(Severity.Valid, V394, CheckIdentifier.Misc); }
|
|
|
|
|
else if (pkm.TradebackStatus == TradebackType.WasTradeback)
|
|
|
|
|
{ AddLine(Severity.Invalid, V395, CheckIdentifier.Misc); }
|
|
|
|
|
else
|
|
|
|
|
goto case TradebackType.Gen1_NotTradeback;
|
|
|
|
|
break;
|
|
|
|
|
case TradebackType.Gen1_NotTradeback:
|
2017-05-01 15:25:20 +00:00
|
|
|
|
if ((EncounterMatch as EncounterStatic)?.Version == GameVersion.Stadium || EncounterMatch is EncounterTradeCatchRate)
|
Encounter Type fix and detection of pokemon that should have evolve on trade (#1105)
* Detect encounter trades that evolve on trade and have not been evolved
Detect generation 1 pokemon with special catch rates : krabby trade and Pokemon Stadium
Detect generation 1 pokemon that evolve on trade and have been traded but not evolved
Detect pokemon with tradeback status any but with only encounters from the other GB generation, that means they are WasTradeback
Detect pokemon with moves from the other GB generation, change tradebackstatus to WasTradeback
* Fix dppt surfing and fishing encounter type, is is surfing because the battle background is the same as other surfingfishing encounters
Fix headbutt encounter type, the encounter type depends on the battle background used when battle a pokemon, for headbutt it changes with the player tile, is the player is in a city it will be building encounter, in a grass tile tall grass, in water it is surfingfishing and cave inside a cave. Some locations have more than one possible encounter type, for example routes with trees near the grass, near the water and near non-combat tiles.
Also added slot type headbutt special for the special trees, those trees are all in routes and are only adjacent to non-combat tiles
* Fix encounter type for missing areas with multiple grass encounters types: Mt Coronet, Mt Silver Cave and Stark Mountain (Issue # 1095)
* Fixes and typos
* Check for non-japanese e-reader pokemon, is unreleased
2017-05-01 15:07:20 +00:00
|
|
|
|
// Encounters detected by the catch rate, cant be invalid if match this encounters
|
|
|
|
|
{ AddLine(Severity.Valid, V398, CheckIdentifier.Misc); }
|
2017-05-13 17:20:25 +00:00
|
|
|
|
if (pkm.Species == 149 && catch_rate == PersonalTable.Y[149].CatchRate ||
|
|
|
|
|
Legal.Species_NotAvailable_CatchRate.Contains(pkm.Species) && catch_rate == PersonalTable.RB[pkm.Species].CatchRate)
|
Generation 1 and 2 legal Improvements (#1099)
* Refactor parseMovesForEncounter to gather valid moves for species encounter, some Pokemon can have valid encounters with different source species from the encounter, the valid moves change if the encounter species change because some preevolutions moves are illegal if pokemon caught already evolved.
Example, generation 1 Pikachu that can have a RBY Pikachu encounter and GSC Pichu encounter, the valid moves for the first encounters should not have any Pichu exclusive evolution moves
Also assign the encounter match from gb when parsing moves like the variable Encounter Match, to store the encounter that is valid for the pokemon moves instead the first encounter.
Store the species encounter, this will be needed to check if the evolution is valid for species that evolve leveling with a given learned move
* Add Tradeback Status to the pokemon, this variable for generations 1 and 2 use data like the catch rate to determine if trade between generations 1 and 2 was possible.
If analysis is for VC games tradeback have value NotTradeback for every gen 1 pokemon, but for cart saves some pokemon can be determine that have not been tradeback, if catch rate match species catch rate but do not match a valid generation 2 held item that means the pokemon habe been never traded to generation 2 games, that allow to discart encounters and moves from generation 2.
Also if is not tradeback catch rate is used to filter encounters, catch rate determine in what species was captured the pokemon discarting some preevolutions moves
Also add option for generation 1 cart save analysis to check legal status not allowing generation 2 games, like VC games but with Stadium allowed, like the generation 1 non tradeback rules from Smogon
Also change evolution chains to included generation 2 preevolutions for gen 1 pokemon if tradeback was possible, it is needed to avoid parsemoves to check illegal pokemon like Hitmonchan with Tyrogue level up moves
* Check legal values of generation 1 type and catch rate
Replace pokemon catch rate after changind pokemon species always if pokemon was not tradeback from generation 2, the catch rate will keep unchanged only if it can be a held item and do not match species catch rate (default item)
Also if catch rate is changed use base species catch rate to avoid legal errors if the catch rate of the evolution species if is not possible with the current moves
* Filter ingame trades and static encounters with catch rate for generation 1 non tradeback
* Fix min moves for generation 1 metapod encounter
* Clean up
* Fix encounter level for generation 1, valid moves are those with one level after the encounter level, pokemon can not learn a new move until level up
Clean up type validation
Fix generation 3 fatefull encounter eggs, the pokemon lost the fatefull mark when it hatch
* Clean-up
* Use new variable EncounterSpecies when it is needed to detect the species of the encounter, the old code wont work if the encounter is a wild slots array
* Fix generation 1 evolution chains and catch rate as default held item
* Fix Generation 1 Yellow Pikachu and Kadabra catch rates
2017-04-27 04:27:59 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V396, CheckIdentifier.Misc); }
|
2017-05-28 04:17:53 +00:00
|
|
|
|
else if (!info.EvoChainsAllGens[1].Any(e => catch_rate == PersonalTable.RB[e.Species].CatchRate || catch_rate == PersonalTable.Y[e.Species].CatchRate))
|
Generation 1 and 2 legal Improvements (#1099)
* Refactor parseMovesForEncounter to gather valid moves for species encounter, some Pokemon can have valid encounters with different source species from the encounter, the valid moves change if the encounter species change because some preevolutions moves are illegal if pokemon caught already evolved.
Example, generation 1 Pikachu that can have a RBY Pikachu encounter and GSC Pichu encounter, the valid moves for the first encounters should not have any Pichu exclusive evolution moves
Also assign the encounter match from gb when parsing moves like the variable Encounter Match, to store the encounter that is valid for the pokemon moves instead the first encounter.
Store the species encounter, this will be needed to check if the evolution is valid for species that evolve leveling with a given learned move
* Add Tradeback Status to the pokemon, this variable for generations 1 and 2 use data like the catch rate to determine if trade between generations 1 and 2 was possible.
If analysis is for VC games tradeback have value NotTradeback for every gen 1 pokemon, but for cart saves some pokemon can be determine that have not been tradeback, if catch rate match species catch rate but do not match a valid generation 2 held item that means the pokemon habe been never traded to generation 2 games, that allow to discart encounters and moves from generation 2.
Also if is not tradeback catch rate is used to filter encounters, catch rate determine in what species was captured the pokemon discarting some preevolutions moves
Also add option for generation 1 cart save analysis to check legal status not allowing generation 2 games, like VC games but with Stadium allowed, like the generation 1 non tradeback rules from Smogon
Also change evolution chains to included generation 2 preevolutions for gen 1 pokemon if tradeback was possible, it is needed to avoid parsemoves to check illegal pokemon like Hitmonchan with Tyrogue level up moves
* Check legal values of generation 1 type and catch rate
Replace pokemon catch rate after changind pokemon species always if pokemon was not tradeback from generation 2, the catch rate will keep unchanged only if it can be a held item and do not match species catch rate (default item)
Also if catch rate is changed use base species catch rate to avoid legal errors if the catch rate of the evolution species if is not possible with the current moves
* Filter ingame trades and static encounters with catch rate for generation 1 non tradeback
* Fix min moves for generation 1 metapod encounter
* Clean up
* Fix encounter level for generation 1, valid moves are those with one level after the encounter level, pokemon can not learn a new move until level up
Clean up type validation
Fix generation 3 fatefull encounter eggs, the pokemon lost the fatefull mark when it hatch
* Clean-up
* Use new variable EncounterSpecies when it is needed to detect the species of the encounter, the old code wont work if the encounter is a wild slots array
* Fix generation 1 evolution chains and catch rate as default held item
* Fix Generation 1 Yellow Pikachu and Kadabra catch rates
2017-04-27 04:27:59 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, pkm.Gen1_NotTradeback? V397: V399, CheckIdentifier.Misc); }
|
|
|
|
|
else
|
|
|
|
|
{ AddLine(Severity.Valid, V398, CheckIdentifier.Misc); }
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-10-23 19:48:49 +00:00
|
|
|
|
private void verifyMisc()
|
2016-04-22 02:32:22 +00:00
|
|
|
|
{
|
2017-01-26 19:16:21 +00:00
|
|
|
|
if (pkm.Format == 7 && ((PK7)pkm).PelagoEventStatus != 0)
|
|
|
|
|
{
|
|
|
|
|
// TODO: Figure out what PelagoEventStati are legal.
|
|
|
|
|
}
|
2017-01-25 07:34:50 +00:00
|
|
|
|
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.IsEgg)
|
2016-09-03 15:10:22 +00:00
|
|
|
|
{
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (new[] {pkm.Move1_PPUps, pkm.Move2_PPUps, pkm.Move3_PPUps, pkm.Move4_PPUps}.Any(ppup => ppup > 0))
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V319, CheckIdentifier.Misc); }
|
2016-10-23 19:48:49 +00:00
|
|
|
|
if (pkm.CNTs.Any(stat => stat > 0))
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V320, CheckIdentifier.Misc); }
|
2017-04-15 02:55:40 +00:00
|
|
|
|
if (pkm.Format == 2 && (pkm.PKRS_Cured || pkm.PKRS_Infected))
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V368, CheckIdentifier.Misc); }
|
|
|
|
|
var HatchCycles = (EncounterMatch as EncounterStatic)?.EggCycles;
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (HatchCycles == 0 || HatchCycles == null)
|
2017-04-12 03:11:58 +00:00
|
|
|
|
HatchCycles = pkm.PersonalInfo.HatchCycles;
|
2017-05-28 05:50:10 +00:00
|
|
|
|
if (pkm.CurrentFriendship > HatchCycles)
|
2017-04-12 03:11:58 +00:00
|
|
|
|
{ AddLine(Severity.Invalid, V374, CheckIdentifier.Misc); }
|
2016-09-03 15:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-11 02:55:04 +00:00
|
|
|
|
if (Encounter.Valid)
|
2016-04-22 02:32:22 +00:00
|
|
|
|
{
|
2017-04-24 00:53:22 +00:00
|
|
|
|
if (Type == typeof(MysteryGift))
|
2016-12-11 02:55:04 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.FatefulEncounter)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V321, CheckIdentifier.Fateful);
|
2016-12-11 02:55:04 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V322, CheckIdentifier.Fateful);
|
2016-12-11 02:55:04 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is EncounterStatic s)
|
2016-11-13 17:37:28 +00:00
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
var fateful = s.Fateful;
|
2017-05-11 07:15:17 +00:00
|
|
|
|
var shadow = EncounterMatch is EncounterStaticShadow;
|
Generation 1 and 2 legal Improvements (#1099)
* Refactor parseMovesForEncounter to gather valid moves for species encounter, some Pokemon can have valid encounters with different source species from the encounter, the valid moves change if the encounter species change because some preevolutions moves are illegal if pokemon caught already evolved.
Example, generation 1 Pikachu that can have a RBY Pikachu encounter and GSC Pichu encounter, the valid moves for the first encounters should not have any Pichu exclusive evolution moves
Also assign the encounter match from gb when parsing moves like the variable Encounter Match, to store the encounter that is valid for the pokemon moves instead the first encounter.
Store the species encounter, this will be needed to check if the evolution is valid for species that evolve leveling with a given learned move
* Add Tradeback Status to the pokemon, this variable for generations 1 and 2 use data like the catch rate to determine if trade between generations 1 and 2 was possible.
If analysis is for VC games tradeback have value NotTradeback for every gen 1 pokemon, but for cart saves some pokemon can be determine that have not been tradeback, if catch rate match species catch rate but do not match a valid generation 2 held item that means the pokemon habe been never traded to generation 2 games, that allow to discart encounters and moves from generation 2.
Also if is not tradeback catch rate is used to filter encounters, catch rate determine in what species was captured the pokemon discarting some preevolutions moves
Also add option for generation 1 cart save analysis to check legal status not allowing generation 2 games, like VC games but with Stadium allowed, like the generation 1 non tradeback rules from Smogon
Also change evolution chains to included generation 2 preevolutions for gen 1 pokemon if tradeback was possible, it is needed to avoid parsemoves to check illegal pokemon like Hitmonchan with Tyrogue level up moves
* Check legal values of generation 1 type and catch rate
Replace pokemon catch rate after changind pokemon species always if pokemon was not tradeback from generation 2, the catch rate will keep unchanged only if it can be a held item and do not match species catch rate (default item)
Also if catch rate is changed use base species catch rate to avoid legal errors if the catch rate of the evolution species if is not possible with the current moves
* Filter ingame trades and static encounters with catch rate for generation 1 non tradeback
* Fix min moves for generation 1 metapod encounter
* Clean up
* Fix encounter level for generation 1, valid moves are those with one level after the encounter level, pokemon can not learn a new move until level up
Clean up type validation
Fix generation 3 fatefull encounter eggs, the pokemon lost the fatefull mark when it hatch
* Clean-up
* Use new variable EncounterSpecies when it is needed to detect the species of the encounter, the old code wont work if the encounter is a wild slots array
* Fix generation 1 evolution chains and catch rate as default held item
* Fix Generation 1 Yellow Pikachu and Kadabra catch rates
2017-04-27 04:27:59 +00:00
|
|
|
|
if (pkm.Gen3 && pkm.WasEgg && !pkm.IsEgg)
|
2017-05-11 07:15:17 +00:00
|
|
|
|
fateful = false; // lost after hatching
|
|
|
|
|
else if (shadow && !(pkm is XK3 || pkm is CK3))
|
|
|
|
|
fateful = true; // purification required for transfer
|
|
|
|
|
|
Generation 1 and 2 legal Improvements (#1099)
* Refactor parseMovesForEncounter to gather valid moves for species encounter, some Pokemon can have valid encounters with different source species from the encounter, the valid moves change if the encounter species change because some preevolutions moves are illegal if pokemon caught already evolved.
Example, generation 1 Pikachu that can have a RBY Pikachu encounter and GSC Pichu encounter, the valid moves for the first encounters should not have any Pichu exclusive evolution moves
Also assign the encounter match from gb when parsing moves like the variable Encounter Match, to store the encounter that is valid for the pokemon moves instead the first encounter.
Store the species encounter, this will be needed to check if the evolution is valid for species that evolve leveling with a given learned move
* Add Tradeback Status to the pokemon, this variable for generations 1 and 2 use data like the catch rate to determine if trade between generations 1 and 2 was possible.
If analysis is for VC games tradeback have value NotTradeback for every gen 1 pokemon, but for cart saves some pokemon can be determine that have not been tradeback, if catch rate match species catch rate but do not match a valid generation 2 held item that means the pokemon habe been never traded to generation 2 games, that allow to discart encounters and moves from generation 2.
Also if is not tradeback catch rate is used to filter encounters, catch rate determine in what species was captured the pokemon discarting some preevolutions moves
Also add option for generation 1 cart save analysis to check legal status not allowing generation 2 games, like VC games but with Stadium allowed, like the generation 1 non tradeback rules from Smogon
Also change evolution chains to included generation 2 preevolutions for gen 1 pokemon if tradeback was possible, it is needed to avoid parsemoves to check illegal pokemon like Hitmonchan with Tyrogue level up moves
* Check legal values of generation 1 type and catch rate
Replace pokemon catch rate after changind pokemon species always if pokemon was not tradeback from generation 2, the catch rate will keep unchanged only if it can be a held item and do not match species catch rate (default item)
Also if catch rate is changed use base species catch rate to avoid legal errors if the catch rate of the evolution species if is not possible with the current moves
* Filter ingame trades and static encounters with catch rate for generation 1 non tradeback
* Fix min moves for generation 1 metapod encounter
* Clean up
* Fix encounter level for generation 1, valid moves are those with one level after the encounter level, pokemon can not learn a new move until level up
Clean up type validation
Fix generation 3 fatefull encounter eggs, the pokemon lost the fatefull mark when it hatch
* Clean-up
* Use new variable EncounterSpecies when it is needed to detect the species of the encounter, the old code wont work if the encounter is a wild slots array
* Fix generation 1 evolution chains and catch rate as default held item
* Fix Generation 1 Yellow Pikachu and Kadabra catch rates
2017-04-27 04:27:59 +00:00
|
|
|
|
if (fateful)
|
2016-12-11 02:55:04 +00:00
|
|
|
|
{
|
|
|
|
|
if (pkm.FatefulEncounter)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Valid, V323, CheckIdentifier.Fateful);
|
2016-12-11 02:55:04 +00:00
|
|
|
|
else
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V324, CheckIdentifier.Fateful);
|
2016-12-11 02:55:04 +00:00
|
|
|
|
}
|
2017-05-11 07:15:17 +00:00
|
|
|
|
else if (pkm.FatefulEncounter && !shadow)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V325, CheckIdentifier.Fateful);
|
2016-11-13 17:37:28 +00:00
|
|
|
|
}
|
2017-04-02 13:39:39 +00:00
|
|
|
|
else if (pkm.FatefulEncounter)
|
2017-03-30 22:14:52 +00:00
|
|
|
|
AddLine(Severity.Invalid, V325, CheckIdentifier.Fateful);
|
|
|
|
|
|
2017-04-02 13:39:39 +00:00
|
|
|
|
if (pkm.GenNumber == 5)
|
2017-03-19 17:44:46 +00:00
|
|
|
|
{
|
|
|
|
|
var enc = EncounterMatch as EncounterStatic;
|
|
|
|
|
bool req = enc?.NSparkle ?? false;
|
2017-04-02 13:39:39 +00:00
|
|
|
|
if (pkm.Format == 5)
|
|
|
|
|
{
|
|
|
|
|
bool has = ((PK5)pkm).NPokémon;
|
|
|
|
|
if (req && !has)
|
|
|
|
|
AddLine(Severity.Invalid, V326, CheckIdentifier.Fateful);
|
|
|
|
|
if (!req && has)
|
|
|
|
|
AddLine(Severity.Invalid, V327, CheckIdentifier.Fateful);
|
|
|
|
|
}
|
|
|
|
|
if (req)
|
|
|
|
|
{
|
|
|
|
|
if (pkm.IVs.Any(iv => iv != 30))
|
|
|
|
|
AddLine(Severity.Invalid, V218, CheckIdentifier.IVs);
|
|
|
|
|
if (pkm.OT_Name != "N" || pkm.TID != 00002 || pkm.SID != 00000)
|
|
|
|
|
AddLine(Severity.Invalid, V219, CheckIdentifier.Trainer);
|
|
|
|
|
if (pkm.IsShiny)
|
|
|
|
|
AddLine(Severity.Invalid, V220, CheckIdentifier.Shiny);
|
|
|
|
|
}
|
2017-03-19 17:44:46 +00:00
|
|
|
|
}
|
2016-04-22 02:32:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-12-01 03:37:53 +00:00
|
|
|
|
private void verifyVersionEvolution()
|
|
|
|
|
{
|
|
|
|
|
if (pkm.Format < 7)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// No point using the evolution tree. Just handle certain species.
|
|
|
|
|
switch (pkm.Species)
|
|
|
|
|
{
|
|
|
|
|
case 745: // Lycanroc
|
|
|
|
|
if (!pkm.WasEgg)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (pkm.AltForm == 0 && pkm.Version == 31 // Moon
|
|
|
|
|
|| pkm.AltForm == 1 && pkm.Version == 30) // Sun
|
|
|
|
|
if (pkm.IsUntraded)
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V328, CheckIdentifier.Evolution);
|
2016-12-01 03:37:53 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 791: // Solgaleo
|
|
|
|
|
if (pkm.Version == 31 && pkm.IsUntraded)
|
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is MysteryGift g && g.Species == pkm.Species) // Gifted via Mystery Gift
|
2016-12-01 03:37:53 +00:00
|
|
|
|
break;
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V328, CheckIdentifier.Evolution);
|
2016-12-01 03:37:53 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 792: // Lunala
|
|
|
|
|
if (pkm.Version == 30 && pkm.IsUntraded)
|
|
|
|
|
{
|
2017-05-28 05:40:21 +00:00
|
|
|
|
if (EncounterMatch is MysteryGift g && g.Species == pkm.Species) // Gifted via Mystery Gift
|
2016-12-01 03:37:53 +00:00
|
|
|
|
break;
|
2017-03-23 03:12:45 +00:00
|
|
|
|
AddLine(Severity.Invalid, V328, CheckIdentifier.Evolution);
|
2016-12-01 03:37:53 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-04-07 00:41:25 +00:00
|
|
|
|
#region verifyMoves
|
|
|
|
|
#endregion
|
2017-01-08 07:54:09 +00:00
|
|
|
|
public static string[] movelist = Util.getMovesList("en");
|
2017-04-10 00:43:05 +00:00
|
|
|
|
public static string[] specieslist = Util.getMovesList("en");
|
2017-03-25 23:38:40 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Converts a Check result Severity determination (Valid/Invalid/etc) to the localized string.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="s"><see cref="Severity"/> value to convert to string.</param>
|
|
|
|
|
/// <returns>Localized <see cref="string"/>.</returns>
|
|
|
|
|
private static string getString(Severity s)
|
|
|
|
|
{
|
|
|
|
|
switch (s)
|
|
|
|
|
{
|
|
|
|
|
case Severity.Indeterminate: return V500;
|
|
|
|
|
case Severity.Invalid: return V501;
|
|
|
|
|
case Severity.Fishy: return V502;
|
|
|
|
|
case Severity.Valid: return V503;
|
|
|
|
|
default: return V504;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-03-11 04:36:32 +00:00
|
|
|
|
}
|
|
|
|
|
}
|