2018-05-13 01:11:47 +00:00
|
|
|
|
using System;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
using System.Collections.Generic;
|
2018-05-13 01:11:47 +00:00
|
|
|
|
using System.Linq;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
using static PKHeX.Core.GameVersion;
|
2018-05-13 01:11:47 +00:00
|
|
|
|
|
|
|
|
|
namespace PKHeX.Core
|
2016-10-29 22:41:55 +00:00
|
|
|
|
{
|
2017-10-24 06:12:58 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Utility class for <see cref="GameVersion"/> logic.
|
|
|
|
|
/// </summary>
|
2016-10-29 22:41:55 +00:00
|
|
|
|
public static class GameUtil
|
|
|
|
|
{
|
2018-06-15 23:00:28 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// List of possible <see cref="GameVersion"/> values a <see cref="PKM.Version"/> can have.
|
|
|
|
|
/// </summary>
|
2018-07-12 02:19:19 +00:00
|
|
|
|
/// <remarks>Ordered roughly by most recent games first.</remarks>
|
2019-02-02 07:08:03 +00:00
|
|
|
|
public static readonly IReadOnlyList<GameVersion> GameVersions = ((GameVersion[])Enum.GetValues(typeof(GameVersion))).Where(z => z < RB && z > 0).Reverse().ToArray();
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Indicates if the <see cref="GameVersion"/> value is a value used by the games or is an aggregate indicator.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="game">Game to check</param>
|
|
|
|
|
public static bool IsValidSavedVersion(this GameVersion game) => 0 < game && game <= RB;
|
|
|
|
|
|
2019-02-04 04:28:03 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Most recent game ID utilized by official games.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public const GameVersion HighestGameID = RB - 1;
|
|
|
|
|
|
2016-10-29 22:41:55 +00:00
|
|
|
|
/// <summary>Determines the Version Grouping of an input Version ID</summary>
|
2020-06-17 02:46:22 +00:00
|
|
|
|
/// <param name="version">Version of which to determine the group</param>
|
2016-10-29 22:41:55 +00:00
|
|
|
|
/// <returns>Version Group Identifier or Invalid if type cannot be determined.</returns>
|
2020-06-17 02:46:22 +00:00
|
|
|
|
public static GameVersion GetMetLocationVersionGroup(GameVersion version)
|
2016-10-29 22:41:55 +00:00
|
|
|
|
{
|
2020-06-17 02:46:22 +00:00
|
|
|
|
switch (version)
|
2016-10-29 22:41:55 +00:00
|
|
|
|
{
|
2020-06-17 02:46:22 +00:00
|
|
|
|
// Side games
|
2018-06-15 23:00:28 +00:00
|
|
|
|
case CXD:
|
|
|
|
|
return CXD;
|
|
|
|
|
case GO:
|
|
|
|
|
return GO;
|
|
|
|
|
|
2019-07-28 20:26:35 +00:00
|
|
|
|
// VC Transfers
|
|
|
|
|
case RD: case BU: case YW: case GN:
|
|
|
|
|
case GD: case SV: case C:
|
|
|
|
|
return USUM;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
2019-07-28 20:26:35 +00:00
|
|
|
|
// Gen2 -- PK2
|
|
|
|
|
case GS: case GSC:
|
2018-06-15 23:00:28 +00:00
|
|
|
|
return GSC;
|
|
|
|
|
|
|
|
|
|
// Gen3
|
|
|
|
|
case R: case S:
|
|
|
|
|
return RS;
|
|
|
|
|
case E:
|
|
|
|
|
return E;
|
|
|
|
|
case FR: case LG:
|
|
|
|
|
return FR;
|
|
|
|
|
|
|
|
|
|
// Gen4
|
|
|
|
|
case D: case P:
|
|
|
|
|
return DP;
|
|
|
|
|
case Pt:
|
|
|
|
|
return Pt;
|
|
|
|
|
case HG: case SS:
|
|
|
|
|
return HGSS;
|
|
|
|
|
|
|
|
|
|
// Gen5
|
|
|
|
|
case B: case W:
|
|
|
|
|
return BW;
|
|
|
|
|
case B2: case W2:
|
|
|
|
|
return B2W2;
|
|
|
|
|
|
|
|
|
|
// Gen6
|
|
|
|
|
case X: case Y:
|
|
|
|
|
return XY;
|
|
|
|
|
case OR: case AS:
|
|
|
|
|
return ORAS;
|
|
|
|
|
|
|
|
|
|
// Gen7
|
|
|
|
|
case SN: case MN:
|
|
|
|
|
return SM;
|
|
|
|
|
case US: case UM:
|
|
|
|
|
return USUM;
|
2018-07-15 04:55:45 +00:00
|
|
|
|
case GP: case GE:
|
|
|
|
|
return GG;
|
2017-09-02 15:41:36 +00:00
|
|
|
|
|
2019-02-27 14:37:11 +00:00
|
|
|
|
// Gen8
|
|
|
|
|
case SW: case SH:
|
|
|
|
|
return SWSH;
|
|
|
|
|
|
2016-10-29 22:41:55 +00:00
|
|
|
|
default:
|
2018-06-15 23:00:28 +00:00
|
|
|
|
return Invalid;
|
2016-10-29 22:41:55 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-01-14 21:10:36 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a Version ID from the end of that Generation
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="generation">Generation ID</param>
|
2018-09-03 17:30:35 +00:00
|
|
|
|
/// <returns>Version ID from requested generation. If none, return <see cref="Invalid"/>.</returns>
|
2017-06-18 01:37:19 +00:00
|
|
|
|
public static GameVersion GetVersion(int generation)
|
2017-01-14 21:10:36 +00:00
|
|
|
|
{
|
2019-10-05 03:10:50 +00:00
|
|
|
|
return generation switch
|
2017-01-14 21:10:36 +00:00
|
|
|
|
{
|
2019-10-05 03:10:50 +00:00
|
|
|
|
1 => RBY,
|
|
|
|
|
2 => C,
|
|
|
|
|
3 => E,
|
|
|
|
|
4 => SS,
|
|
|
|
|
5 => W2,
|
|
|
|
|
6 => AS,
|
|
|
|
|
7 => UM,
|
|
|
|
|
8 => SH,
|
|
|
|
|
_ => Invalid
|
|
|
|
|
};
|
2017-01-14 21:10:36 +00:00
|
|
|
|
}
|
2017-12-19 00:17:21 +00:00
|
|
|
|
|
2018-06-15 23:00:28 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the Generation the <see cref="GameVersion"/> belongs to.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="game">Game to retrieve the generation for</param>
|
|
|
|
|
/// <returns>Generation ID</returns>
|
|
|
|
|
public static int GetGeneration(this GameVersion game)
|
|
|
|
|
{
|
|
|
|
|
if (Gen1.Contains(game)) return 1;
|
|
|
|
|
if (Gen2.Contains(game)) return 2;
|
|
|
|
|
if (Gen3.Contains(game)) return 3;
|
|
|
|
|
if (Gen4.Contains(game)) return 4;
|
|
|
|
|
if (Gen5.Contains(game)) return 5;
|
|
|
|
|
if (Gen6.Contains(game)) return 6;
|
|
|
|
|
if (Gen7.Contains(game)) return 7;
|
2019-09-23 23:56:47 +00:00
|
|
|
|
if (Gen8.Contains(game)) return 8;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
2017-12-19 00:17:21 +00:00
|
|
|
|
|
2018-06-18 04:52:52 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the Generation the <see cref="GameVersion"/> belongs to.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="game">Game to retrieve the generation for</param>
|
|
|
|
|
/// <returns>Generation ID</returns>
|
|
|
|
|
public static int GetMaxSpeciesID(this GameVersion game)
|
|
|
|
|
{
|
|
|
|
|
if (Gen1.Contains(game)) return Legal.MaxSpeciesID_1;
|
|
|
|
|
if (Gen2.Contains(game)) return Legal.MaxSpeciesID_2;
|
|
|
|
|
if (Gen3.Contains(game)) return Legal.MaxSpeciesID_3;
|
|
|
|
|
if (Gen4.Contains(game)) return Legal.MaxSpeciesID_4;
|
|
|
|
|
if (Gen5.Contains(game)) return Legal.MaxSpeciesID_5;
|
|
|
|
|
if (Gen6.Contains(game)) return Legal.MaxSpeciesID_6;
|
2020-08-07 01:24:20 +00:00
|
|
|
|
if (Gen7b.Contains(game)) return Legal.MaxSpeciesID_7b;
|
2018-06-18 04:52:52 +00:00
|
|
|
|
if (Gen7.Contains(game))
|
|
|
|
|
{
|
|
|
|
|
if (SM.Contains(game))
|
|
|
|
|
return Legal.MaxSpeciesID_7;
|
|
|
|
|
if (USUM.Contains(game))
|
|
|
|
|
return Legal.MaxSpeciesID_7_USUM;
|
|
|
|
|
return Legal.MaxSpeciesID_7_USUM;
|
|
|
|
|
}
|
2019-09-23 23:56:47 +00:00
|
|
|
|
if (Gen8.Contains(game)) return Legal.MaxSpeciesID_8;
|
2018-06-18 04:52:52 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-15 23:00:28 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Checks if the <see cref="g1"/> version (or subset versions) is equivalent to <see cref="g2"/>.
|
2018-10-13 00:45:52 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="g1">Version (set)</param>
|
|
|
|
|
/// <param name="g2">Individual version</param>
|
|
|
|
|
public static bool Contains(this GameVersion g1, int g2) => g1.Contains((GameVersion) g2);
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Checks if the <see cref="g1"/> version (or subset versions) is equivalent to <see cref="g2"/>.
|
2018-06-15 23:00:28 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="g1">Version (set)</param>
|
|
|
|
|
/// <param name="g2">Individual version</param>
|
|
|
|
|
public static bool Contains(this GameVersion g1, GameVersion g2)
|
|
|
|
|
{
|
|
|
|
|
if (g1 == g2 || g1 == Any)
|
|
|
|
|
return true;
|
2017-12-19 00:17:21 +00:00
|
|
|
|
|
2018-06-15 23:00:28 +00:00
|
|
|
|
switch (g1)
|
|
|
|
|
{
|
|
|
|
|
case RB:
|
|
|
|
|
return g2 == RD || g2 == BU || g2 == GN;
|
2018-08-28 03:44:26 +00:00
|
|
|
|
case Stadium:
|
2018-06-15 23:00:28 +00:00
|
|
|
|
case RBY:
|
|
|
|
|
return RB.Contains(g2) || g2 == YW;
|
|
|
|
|
case Gen1:
|
2020-10-07 03:05:38 +00:00
|
|
|
|
return RBY.Contains(g2) || g2 == Stadium;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
|
|
|
|
case GS: return g2 == GD || g2 == SV;
|
2018-08-28 03:44:26 +00:00
|
|
|
|
case Stadium2:
|
2018-06-15 23:00:28 +00:00
|
|
|
|
case GSC:
|
|
|
|
|
return GS.Contains(g2) || g2 == C;
|
|
|
|
|
case Gen2:
|
2020-10-07 03:05:38 +00:00
|
|
|
|
return GSC.Contains(g2) || g2 == Stadium2;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
|
|
|
|
case RS: return g2 == R || g2 == S;
|
|
|
|
|
case RSE:
|
|
|
|
|
return RS.Contains(g2) || g2 == E;
|
|
|
|
|
case FRLG: return g2 == FR || g2 == LG;
|
|
|
|
|
case COLO:
|
|
|
|
|
case XD: return g2 == CXD;
|
|
|
|
|
case CXD: return g2 == COLO || g2 == XD;
|
|
|
|
|
case RSBOX: return RS.Contains(g2) || g2 == E || FRLG.Contains(g2);
|
|
|
|
|
case Gen3:
|
|
|
|
|
return RSE.Contains(g2) || FRLG.Contains(g2) || CXD.Contains(g2) || g2 == RSBOX;
|
|
|
|
|
|
|
|
|
|
case DP: return g2 == D || g2 == P;
|
|
|
|
|
case HGSS: return g2 == HG || g2 == SS;
|
|
|
|
|
case DPPt:
|
|
|
|
|
return DP.Contains(g2) || g2 == Pt;
|
|
|
|
|
case BATREV: return DP.Contains(g2) || g2 == Pt || HGSS.Contains(g2);
|
|
|
|
|
case Gen4:
|
|
|
|
|
return DPPt.Contains(g2) || HGSS.Contains(g2) || g2 == BATREV;
|
|
|
|
|
|
|
|
|
|
case BW: return g2 == B || g2 == W;
|
|
|
|
|
case B2W2: return g2 == B2 || g2 == W2;
|
|
|
|
|
case Gen5:
|
|
|
|
|
return BW.Contains(g2) || B2W2.Contains(g2);
|
|
|
|
|
|
|
|
|
|
case XY: return g2 == X || g2 == Y;
|
|
|
|
|
case ORAS: return g2 == OR || g2 == AS;
|
|
|
|
|
case Gen6:
|
|
|
|
|
return XY.Contains(g2) || ORAS.Contains(g2);
|
|
|
|
|
|
|
|
|
|
case SM:
|
|
|
|
|
return g2 == SN || g2 == MN;
|
|
|
|
|
case USUM:
|
|
|
|
|
return g2 == US || g2 == UM;
|
2018-07-12 02:19:19 +00:00
|
|
|
|
case GG:
|
2020-08-07 01:24:20 +00:00
|
|
|
|
return g2 == GP || g2 == GE;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
case Gen7:
|
2020-08-07 01:24:20 +00:00
|
|
|
|
return SM.Contains(g2) || USUM.Contains(g2);
|
|
|
|
|
case Gen7b:
|
|
|
|
|
return GG.Contains(g2) || GO == g2;
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
2019-02-27 14:37:11 +00:00
|
|
|
|
case SWSH:
|
|
|
|
|
return g2 == SW || g2 == SH;
|
2019-11-16 01:34:18 +00:00
|
|
|
|
case Gen8:
|
|
|
|
|
return SWSH.Contains(g2);
|
2019-02-27 14:37:11 +00:00
|
|
|
|
|
2018-06-15 23:00:28 +00:00
|
|
|
|
default: return false;
|
2017-12-19 00:17:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-05-13 01:11:47 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// List of possible <see cref="GameVersion"/> values within the provided <see cref="generation"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="generation">Generation to look within</param>
|
2018-11-23 18:15:49 +00:00
|
|
|
|
/// <param name="pkVersion"></param>
|
|
|
|
|
public static GameVersion[] GetVersionsInGeneration(int generation, int pkVersion)
|
|
|
|
|
{
|
2020-08-07 01:24:20 +00:00
|
|
|
|
if (Gen7b.Contains(pkVersion))
|
2018-11-23 18:15:49 +00:00
|
|
|
|
return new[] {GO, GP, GE};
|
|
|
|
|
return GameVersions.Where(z => z.GetGeneration() == generation).ToArray();
|
|
|
|
|
}
|
2018-06-15 23:00:28 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// List of possible <see cref="GameVersion"/> values within the provided <see cref="IGameValueLimit"/> criteria.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="obj">Criteria for retrieving versions</param>
|
|
|
|
|
/// <param name="generation">Generation format minimum (necessary for the CXD/Gen4 swap etc)</param>
|
|
|
|
|
public static IEnumerable<GameVersion> GetVersionsWithinRange(IGameValueLimit obj, int generation = -1)
|
|
|
|
|
{
|
2018-11-23 18:15:49 +00:00
|
|
|
|
if (obj.MaxGameID == Legal.MaxGameID_7b) // edge case
|
|
|
|
|
return new[] {GO, GP, GE};
|
2020-06-17 02:46:22 +00:00
|
|
|
|
var versions = GameVersions
|
|
|
|
|
.Where(version => (GameVersion)obj.MinGameID <= version && version <= (GameVersion)obj.MaxGameID);
|
2018-11-23 18:15:49 +00:00
|
|
|
|
if (generation < 0)
|
2020-06-17 02:46:22 +00:00
|
|
|
|
return versions;
|
2018-11-23 18:15:49 +00:00
|
|
|
|
if (obj.MaxGameID == Legal.MaxGameID_7 && generation == 7)
|
2020-06-17 02:46:22 +00:00
|
|
|
|
versions = versions.Where(version => version != GO);
|
|
|
|
|
return versions.Where(version => version.GetGeneration() <= generation);
|
2018-06-15 23:00:28 +00:00
|
|
|
|
}
|
2016-10-29 22:41:55 +00:00
|
|
|
|
}
|
|
|
|
|
}
|