Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
using System;
|
2018-08-19 22:50:15 +00:00
|
|
|
using System.Collections.Generic;
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
namespace PKHeX.Core;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Base format for Generation 1 & 2 <see cref="PKM"/> objects.
|
|
|
|
/// </summary>
|
|
|
|
/// <remarks>
|
|
|
|
/// <see cref="SK2"/> store text buffers with the rest of the data.
|
|
|
|
/// <see cref="PK1"/> and <see cref="PK2"/> store them separately; see <see cref="GBPKML"/>.
|
|
|
|
/// </remarks>
|
|
|
|
public abstract class GBPKM : PKM
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override int MaxBallID => -1;
|
|
|
|
public sealed override int MinGameID => (int)GameVersion.RD;
|
|
|
|
public sealed override int MaxGameID => (int)GameVersion.C;
|
|
|
|
public sealed override int MaxIV => 15;
|
|
|
|
public sealed override int MaxEV => ushort.MaxValue;
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override IReadOnlyList<ushort> ExtraBytes => Array.Empty<ushort>();
|
2018-12-20 06:10:32 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
protected GBPKM(int size) : base(size) { }
|
|
|
|
protected GBPKM(byte[] data) : base(data) { }
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override byte[] EncryptedPartyData => Encrypt();
|
|
|
|
public sealed override byte[] EncryptedBoxData => Encrypt();
|
|
|
|
public sealed override byte[] DecryptedBoxData => Encrypt();
|
|
|
|
public sealed override byte[] DecryptedPartyData => Encrypt();
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public override bool Valid { get => true; set { } }
|
|
|
|
public sealed override void RefreshChecksum() { }
|
2022-01-03 05:35:59 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
protected abstract byte[] GetNonNickname(int language);
|
2020-10-03 01:08:40 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private bool? _isnicknamed;
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override bool IsNicknamed
|
|
|
|
{
|
|
|
|
get => _isnicknamed ??= !Nickname_Trash.SequenceEqual(GetNonNickname(GuessedLanguage()));
|
|
|
|
set
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
_isnicknamed = value;
|
|
|
|
if (_isnicknamed == false)
|
|
|
|
SetNotNicknamed(GuessedLanguage());
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
protected bool IsNicknamedBank
|
|
|
|
{
|
|
|
|
get
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
var spName = SpeciesName.GetSpeciesNameGeneration(Species, GuessedLanguage(), Format);
|
|
|
|
return Nickname != spName;
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override int Language
|
|
|
|
{
|
|
|
|
get
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Japanese)
|
|
|
|
return (int)LanguageID.Japanese;
|
|
|
|
if (Korean)
|
|
|
|
return (int)LanguageID.Korean;
|
|
|
|
if (StringConverter12.IsG12German(OT_Trash))
|
|
|
|
return (int)LanguageID.German; // german
|
|
|
|
int lang = SpeciesName.GetSpeciesNameLanguage(Species, Nickname, Format);
|
|
|
|
if (lang > 0)
|
|
|
|
return lang;
|
|
|
|
return 0;
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
set
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Japanese)
|
|
|
|
return;
|
|
|
|
if (Korean)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (IsNicknamed)
|
|
|
|
return;
|
|
|
|
SetNotNicknamed(value);
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override int Gender
|
|
|
|
{
|
|
|
|
get
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
int gv = PersonalInfo.Gender;
|
|
|
|
return gv switch
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
PersonalInfo.RatioMagicGenderless => 2,
|
|
|
|
PersonalInfo.RatioMagicFemale => 1,
|
|
|
|
PersonalInfo.RatioMagicMale => 0,
|
|
|
|
_ => IV_ATK > gv >> 4 ? 0 : 1,
|
|
|
|
};
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
set { }
|
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
#region Future, Unused Attributes
|
|
|
|
public sealed override bool IsGenderValid() => true; // not a separate property, derived via IVs
|
|
|
|
public sealed override uint EncryptionConstant { get => 0; set { } }
|
|
|
|
public sealed override uint PID { get => 0; set { } }
|
|
|
|
public sealed override int Nature { get => 0; set { } }
|
|
|
|
public sealed override bool ChecksumValid => true;
|
|
|
|
public sealed override bool FatefulEncounter { get => false; set { } }
|
2023-01-22 04:02:33 +00:00
|
|
|
public sealed override uint TSV => 0x0000;
|
|
|
|
public sealed override uint PSV => 0xFFFF;
|
2022-06-18 18:04:24 +00:00
|
|
|
public sealed override int Characteristic => -1;
|
|
|
|
public sealed override int MarkValue { get => 0; set { } }
|
|
|
|
public sealed override int Ability { get => -1; set { } }
|
|
|
|
public sealed override int CurrentHandler { get => 0; set { } }
|
|
|
|
public sealed override int Egg_Location { get => 0; set { } }
|
|
|
|
public sealed override int Ball { get => 0; set { } }
|
2023-01-22 04:02:33 +00:00
|
|
|
public sealed override uint ID32 { get => TID16; set => TID16 = (ushort)value; }
|
|
|
|
public sealed override ushort SID16 { get => 0; set { } }
|
2022-06-18 18:04:24 +00:00
|
|
|
#endregion
|
|
|
|
|
|
|
|
public sealed override bool IsShiny => IV_DEF == 10 && IV_SPE == 10 && IV_SPC == 10 && (IV_ATK & 2) == 2;
|
|
|
|
private int HPBitValPower => ((IV_ATK & 8) >> 0) | ((IV_DEF & 8) >> 1) | ((IV_SPE & 8) >> 2) | ((IV_SPC & 8) >> 3);
|
|
|
|
public sealed override int HPPower => (((5 * HPBitValPower) + (IV_SPC & 3)) >> 1) + 31;
|
|
|
|
|
|
|
|
public sealed override int HPType
|
|
|
|
{
|
|
|
|
get => ((IV_ATK & 3) << 2) | (IV_DEF & 3);
|
|
|
|
set
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
IV_DEF = ((IV_DEF >> 2) << 2) | (value & 3);
|
|
|
|
IV_DEF = ((IV_ATK >> 2) << 2) | ((value >> 2) & 3);
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-08-27 06:43:36 +00:00
|
|
|
public sealed override byte Form
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
|
|
|
get
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Species != 201) // Unown
|
|
|
|
return 0;
|
2022-08-27 06:43:36 +00:00
|
|
|
return GetUnownFormValue(IV_ATK, IV_DEF, IV_SPE, IV_SPC);
|
2019-05-27 18:23:47 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
set
|
2018-08-19 22:50:15 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Species != 201) // Unown
|
|
|
|
return;
|
|
|
|
while (Form != value)
|
|
|
|
SetRandomIVs(0);
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-08-27 06:43:36 +00:00
|
|
|
private static byte GetUnownFormValue(int atk, int def, int spe, int spc)
|
|
|
|
{
|
|
|
|
ushort formeVal = 0;
|
|
|
|
formeVal |= (ushort)((atk & 0x6) << 5);
|
|
|
|
formeVal |= (ushort)((def & 0x6) << 3);
|
|
|
|
formeVal |= (ushort)((spe & 0x6) << 1);
|
|
|
|
formeVal |= (ushort)((spc & 0x6) >> 1);
|
|
|
|
return (byte)(formeVal / 10);
|
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public abstract int EV_SPC { get; set; }
|
|
|
|
public sealed override int EV_SPA { get => EV_SPC; set => EV_SPC = value; }
|
|
|
|
public sealed override int EV_SPD { get => EV_SPC; set { } }
|
|
|
|
public abstract ushort DV16 { get; set; }
|
|
|
|
public sealed override int IV_HP { get => ((IV_ATK & 1) << 3) | ((IV_DEF & 1) << 2) | ((IV_SPE & 1) << 1) | ((IV_SPC & 1) << 0); set { } }
|
|
|
|
public sealed override int IV_ATK { get => (DV16 >> 12) & 0xF; set => DV16 = (ushort)((DV16 & ~(0xF << 12)) | (ushort)((value > 0xF ? 0xF : value) << 12)); }
|
|
|
|
public sealed override int IV_DEF { get => (DV16 >> 8) & 0xF; set => DV16 = (ushort)((DV16 & ~(0xF << 8)) | (ushort)((value > 0xF ? 0xF : value) << 8)); }
|
|
|
|
public sealed override int IV_SPE { get => (DV16 >> 4) & 0xF; set => DV16 = (ushort)((DV16 & ~(0xF << 4)) | (ushort)((value > 0xF ? 0xF : value) << 4)); }
|
|
|
|
public int IV_SPC { get => (DV16 >> 0) & 0xF; set => DV16 = (ushort)((DV16 & ~(0xF << 0)) | (ushort)((value > 0xF ? 0xF : value) << 0)); }
|
|
|
|
public sealed override int IV_SPA { get => IV_SPC; set => IV_SPC = value; }
|
|
|
|
public sealed override int IV_SPD { get => IV_SPC; set { } }
|
|
|
|
public override int MarkingCount => 0;
|
|
|
|
public override int GetMarking(int index) => 0;
|
|
|
|
public override void SetMarking(int index, int value) { }
|
|
|
|
|
|
|
|
public void SetNotNicknamed() => SetNotNicknamed(GuessedLanguage());
|
|
|
|
public abstract void SetNotNicknamed(int language);
|
|
|
|
|
|
|
|
public int GuessedLanguage(int fallback = (int)LanguageID.English)
|
|
|
|
{
|
|
|
|
int lang = Language;
|
|
|
|
if (lang > 0)
|
|
|
|
return lang;
|
|
|
|
if (fallback is (int)LanguageID.French or (int)LanguageID.German) // only other permitted besides English
|
|
|
|
return fallback;
|
|
|
|
return (int)LanguageID.English;
|
|
|
|
}
|
2018-08-19 22:50:15 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Tries to guess the source language ID when transferred to future generations (7+)
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="destLanguage">Destination language ID</param>
|
|
|
|
/// <returns>Source language ID</returns>
|
|
|
|
protected int TransferLanguage(int destLanguage)
|
|
|
|
{
|
|
|
|
// if the Species name of the destination language matches the current nickname, transfer with that language.
|
|
|
|
var expect = SpeciesName.GetSpeciesNameGeneration(Species, destLanguage, 2);
|
|
|
|
if (Nickname == expect)
|
|
|
|
return destLanguage;
|
|
|
|
return GuessedLanguage(destLanguage);
|
|
|
|
}
|
2019-02-26 01:39:41 +00:00
|
|
|
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
public override void LoadStats(IBaseStat p, Span<ushort> stats)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
|
|
|
var lv = Stat_Level;
|
|
|
|
stats[0] = (ushort)(GetStat(p.HP, IV_HP, EV_HP, lv) + (5 + lv)); // HP
|
|
|
|
stats[1] = GetStat(p.ATK, IV_ATK, EV_ATK, lv);
|
|
|
|
stats[2] = GetStat(p.DEF, IV_DEF, EV_DEF, lv);
|
|
|
|
stats[3] = GetStat(p.SPE, IV_SPE, EV_SPE, lv);
|
|
|
|
stats[4] = GetStat(p.SPA, IV_SPA, EV_SPA, lv);
|
|
|
|
stats[5] = GetStat(p.SPD, IV_SPD, EV_SPD, lv);
|
|
|
|
}
|
2021-04-20 08:50:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
protected static ushort GetStat(int baseStat, int iv, int effort, int level)
|
|
|
|
{
|
|
|
|
effort = (ushort)Math.Min(255, Math.Sqrt(effort) + 1) >> 2;
|
|
|
|
return (ushort)((((2 * (baseStat + iv)) + effort) * level / 100) + 5);
|
|
|
|
}
|
2021-02-15 21:52:49 +00:00
|
|
|
|
2022-09-19 06:06:02 +00:00
|
|
|
public sealed override int GetMovePP(ushort move, int ppUpCount)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
|
|
|
var pp = base.GetMovePP(move, 0);
|
|
|
|
return pp + (ppUpCount * Math.Min(7, pp / 5));
|
|
|
|
}
|
|
|
|
|
|
|
|
public void MaxEVs() => EV_HP = EV_ATK = EV_DEF = EV_SPC = EV_SPE = MaxEV;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Applies <see cref="PKM.IVs"/> to the <see cref="PKM"/> to make it shiny.
|
|
|
|
/// </summary>
|
|
|
|
public sealed override void SetShiny()
|
|
|
|
{
|
|
|
|
IV_ATK |= 2;
|
|
|
|
IV_DEF = 10;
|
|
|
|
IV_SPE = 10;
|
|
|
|
IV_SPA = 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void ImportFromFuture(PKM pk)
|
|
|
|
{
|
|
|
|
Nickname = pk.Nickname;
|
|
|
|
OT_Name = pk.OT_Name;
|
|
|
|
IV_ATK = pk.IV_ATK / 2;
|
|
|
|
IV_DEF = pk.IV_DEF / 2;
|
|
|
|
IV_SPC = pk.IV_SPA / 2;
|
|
|
|
//IV_SPD = pk.IV_ATK / 2;
|
|
|
|
IV_SPE = pk.IV_SPE / 2;
|
|
|
|
|
|
|
|
if (pk.HasMove((int)Move.HiddenPower))
|
|
|
|
HPType = pk.HPType;
|
2018-08-19 22:50:15 +00:00
|
|
|
}
|
|
|
|
}
|