PKHeX/PKHeX.Core/Legality/Encounters/Templates/Gen6/EncounterSlot6AO.cs
Kurt fd1c538cc5
Changes for Legends: Z-A support (#4596)
Refer to pull request notes and the eventual changelog for a high-level summary.

Co-authored-by: Matt <17801814+sora10pls@users.noreply.github.com>
Co-authored-by: Lusamine <30205550+Lusamine@users.noreply.github.com>
Co-authored-by: SciresM <8676005+SciresM@users.noreply.github.com>
2025-10-26 19:01:44 -05:00

163 lines
6.0 KiB
C#

using System;
using static PKHeX.Core.SlotType6;
namespace PKHeX.Core;
/// <summary>
/// Encounter Slot found in <see cref="GameVersion.ORAS"/>.
/// </summary>
public sealed record EncounterSlot6AO(EncounterArea6AO Parent, ushort Species, byte Form, byte LevelMin, byte LevelMax)
: IEncounterable, IEncounterMatch, IEncounterConvertible<PK6>, IEncounterFormRandom, IEncounterDownlevel
{
public byte Generation => 6;
public EntityContext Context => EntityContext.Gen6;
public bool IsEgg => false;
public Ball FixedBall => Ball.None;
public Shiny Shiny => Shiny.Random;
public bool IsShiny => false;
public ushort EggLocation => 0;
public bool IsRandomUnspecificForm => Form >= EncounterUtil.FormDynamic;
public string Name => $"Wild Encounter ({Version})";
public string LongName => $"{Name} {Type.ToString().Replace('_', ' ')}";
public GameVersion Version => Parent.Version;
public ushort Location => Parent.Location;
public SlotType6 Type => Parent.Type;
public bool CanDexNav => Type != Rock_Smash;
public bool IsHorde => Type == Horde;
private HiddenAbilityPermission IsHiddenAbilitySlot() => CanDexNav || IsHorde ? HiddenAbilityPermission.Possible : HiddenAbilityPermission.Never;
private ReadOnlySpan<ushort> GetDexNavMoves()
{
var et = EvolutionTree.Evolves6;
var baby = et.GetBaseSpeciesForm(Species, Form);
return LearnSource6AO.Instance.GetEggMoves(baby.Species, baby.Form);
}
public bool CanBeDexNavMove(ushort move) => GetDexNavMoves().Contains(move);
public AbilityPermission Ability => IsHiddenAbilitySlot() switch
{
HiddenAbilityPermission.Never => AbilityPermission.Any12,
HiddenAbilityPermission.Always => AbilityPermission.OnlyHidden,
_ => AbilityPermission.Any12H,
};
private bool IsDeferredHiddenAbility(bool IsHidden) => IsHiddenAbilitySlot() switch
{
HiddenAbilityPermission.Never => IsHidden,
HiddenAbilityPermission.Always => !IsHidden,
_ => false,
};
#region Generating
PKM IEncounterConvertible.ConvertToPKM(ITrainerInfo tr, EncounterCriteria criteria) => ConvertToPKM(tr, criteria);
PKM IEncounterConvertible.ConvertToPKM(ITrainerInfo tr) => ConvertToPKM(tr);
public PK6 ConvertToPKM(ITrainerInfo tr) => ConvertToPKM(tr, EncounterCriteria.Unrestricted);
public PK6 ConvertToPKM(ITrainerInfo tr, EncounterCriteria criteria)
{
int language = (int)Language.GetSafeLanguage456((LanguageID)tr.Language);
var pi = PersonalTable.AO[Species];
var geo = tr.GetRegionOrigin(language);
var pk = new PK6
{
Species = Species,
Form = GetWildForm(Form),
CurrentLevel = LevelMin,
OriginalTrainerFriendship = pi.BaseFriendship,
MetLocation = Location,
MetLevel = LevelMin,
Version = Version,
Ball = (byte)Ball.Poke,
MetDate = EncounterDate.GetDate3DS(),
Language = language,
OriginalTrainerName = tr.OT,
OriginalTrainerGender = tr.Gender,
ID32 = tr.ID32,
Nickname = SpeciesName.GetSpeciesNameGeneration(Species, language, Generation),
ConsoleRegion = geo.ConsoleRegion,
Country = geo.Country,
Region = geo.Region,
};
SetPINGA(pk, criteria, pi);
EncounterUtil.SetEncounterMoves(pk, Version, LevelMin);
if (CanDexNav)
{
var eggMoves = GetDexNavMoves();
if (eggMoves.Length != 0)
pk.RelearnMove1 = eggMoves[Util.Rand.Next(eggMoves.Length)];
}
pk.SetRandomMemory6();
pk.ResetPartyStats();
return pk;
}
private byte GetWildForm(byte form)
{
if (form != EncounterUtil.FormRandom)
return form;
// flagged as totally random
return (byte)Util.Rand.Next(PersonalTable.AO[Species].FormCount);
}
private void SetPINGA(PK6 pk, in EncounterCriteria criteria, PersonalInfo6AO pi)
{
var rnd = Util.Rand;
pk.PID = EncounterUtil.GetRandomPID(pk, rnd, criteria.Shiny);
pk.EncryptionConstant = rnd.Rand32();
pk.Nature = criteria.GetNature();
pk.Gender = criteria.GetGender(pi);
pk.RefreshAbility(criteria.GetAbilityFromNumber(Ability));
criteria.SetRandomIVs(pk);
}
#endregion
#region Matching
private const byte FluteBoostMin = 4; // White Flute decreases levels.
private const byte FluteBoostMax = 4; // Black Flute increases levels.
private const byte DexNavBoost = 29 + FluteBoostMax; // Maximum DexNav chain (95) and Flute.
public byte GetDownleveledMin() => (byte)(LevelMin - FluteBoostMin);
public bool IsMatchExact(PKM pk, EvoCriteria evo)
{
var boostMax = Type != Rock_Smash ? DexNavBoost : FluteBoostMax;
const int boostMin = FluteBoostMin;
if (!this.IsLevelWithinRange(pk.MetLevel, boostMin, boostMax))
return false;
if (evo.Form != Form && !IsRandomUnspecificForm)
{
if (Species is not (ushort)Core.Species.Deerling)
return false;
// Deerling can change between forms if imported to a future Gen8+ game.
if (pk.Format < 8 || evo.Form >= 4)
return false;
}
return true;
}
public EncounterMatchRating GetMatchRating(PKM pk)
{
if (IsDeferredWurmple(pk))
return EncounterMatchRating.PartialMatch;
bool isHidden = pk.AbilityNumber == 4;
if (isHidden && this.IsPartialMatchHidden(pk.Species, Species))
return EncounterMatchRating.PartialMatch;
if (IsDeferredHiddenAbility(isHidden))
return EncounterMatchRating.Deferred;
return EncounterMatchRating.Match;
}
private bool IsDeferredWurmple(PKM pk) => Species == (int)Core.Species.Wurmple && pk.Species != (int)Core.Species.Wurmple && !WurmpleUtil.IsWurmpleEvoValid(pk);
#endregion
}