using System; namespace PKHeX.Core; /// /// Generation 8 Static Encounter /// /// public sealed record EncounterStatic8a(GameVersion Version) : EncounterStatic(Version), IAlpha, IMasteryInitialMoveShop8 { public override int Generation => 8; public byte HeightScalar { get; } public byte WeightScalar { get; } public bool IsAlpha { get; set; } public EncounterStatic8aCorrelation Method { get; init; } public bool HasFixedHeight => HeightScalar != NoScalar; public bool HasFixedWeight => WeightScalar != NoScalar; private const byte NoScalar = 0; public EncounterStatic8a(ushort species, ushort form, byte level, byte h = NoScalar, byte w = NoScalar) : this(GameVersion.PLA) { Species = species; Form = form; Level = level; HeightScalar = h; WeightScalar = w; Shiny = Shiny.Never; } protected override void ApplyDetails(ITrainerInfo sav, EncounterCriteria criteria, PKM pk) { base.ApplyDetails(sav, criteria, pk); var pa = (PA8)pk; if (IsAlpha) pa.IsAlpha = true; if (HasFixedHeight) pa.HeightScalar = HeightScalar; if (HasFixedWeight) pa.WeightScalar = WeightScalar; pa.HeightScalarCopy = pa.HeightScalar; pa.ResetHeight(); pa.ResetWeight(); } protected override void SetPINGA(PKM pk, EncounterCriteria criteria) { var para = GetParams(); var (_, slotSeed) = Overworld8aRNG.ApplyDetails(pk, criteria, para, IsAlpha); // We don't override LevelMin, so just handle the two species cases. if (Species == (int)Core.Species.Zorua) pk.CurrentLevel = pk.Met_Level = Overworld8aRNG.GetRandomLevel(slotSeed, 26, 28); else if (Species == (int)Core.Species.Phione) pk.CurrentLevel = pk.Met_Level = Overworld8aRNG.GetRandomLevel(slotSeed, 33, 36); if (Method == EncounterStatic8aCorrelation.Fixed) pk.EncryptionConstant = Util.Rand32(); } protected override void ApplyDetailsBall(PKM pk) => pk.Ball = Gift ? Ball : (int)Core.Ball.LAPoke; public override bool IsMatchExact(PKM pkm, EvoCriteria evo) { if (!base.IsMatchExact(pkm, evo)) return false; if (pkm is IScaledSize s) { if (HasFixedHeight && s.HeightScalar != HeightScalar) return false; if (HasFixedWeight && s.WeightScalar != WeightScalar) return false; } if (pkm is IAlpha a && a.IsAlpha != IsAlpha) return false; return true; } public override EncounterMatchRating GetMatchRating(PKM pkm) { var result = GetMatchRatingInternal(pkm); var orig = base.GetMatchRating(pkm); return result > orig ? result : orig; } private EncounterMatchRating GetMatchRatingInternal(PKM pkm) { if (Shiny != Shiny.Random && !Shiny.IsValid(pkm)) return EncounterMatchRating.DeferredErrors; if (Gift && pkm.Ball != Ball) return EncounterMatchRating.DeferredErrors; var orig = base.GetMatchRating(pkm); if (orig is not EncounterMatchRating.Match) return orig; if (!IsForcedMasteryCorrect(pkm)) return EncounterMatchRating.DeferredErrors; if (IsAlpha && pkm is PA8 { AlphaMove: 0 }) return EncounterMatchRating.Deferred; return EncounterMatchRating.Match; } public bool IsForcedMasteryCorrect(PKM pkm) { ushort alpha = 0; if (IsAlpha && Moves.Count != 0) { if (pkm is PA8 pa && (alpha = pa.AlphaMove) != Moves[0]) return false; } if (pkm is not IMoveShop8Mastery p) return true; const bool allowAlphaPurchaseBug = true; // Everything else Alpha is pre-1.1 var level = pkm.Met_Level; var index = PersonalTable.LA.GetFormIndex(Species, Form); var learn = Legal.LevelUpLA[index]; if (!p.IsValidPurchasedEncounter(learn, level, alpha, allowAlphaPurchaseBug)) return false; Span moves = stackalloc int[4]; var mastery = Legal.MasteryLA[index]; if (Moves.Count != 0) moves = (int[])Moves; else learn.SetEncounterMoves(level, moves); return p.IsValidMasteredEncounter(moves, learn, mastery, level, alpha, allowAlphaPurchaseBug); } protected override void SetEncounterMoves(PKM pk, GameVersion version, int level) { var pa8 = (PA8)pk; Span moves = stackalloc int[4]; var index = PersonalTable.LA.GetFormIndex(Species, Form); var mastery = Legal.MasteryLA[index]; var learn = Legal.LevelUpLA[index]; if (IsAlpha && Moves.Count != 0) { moves = (int[])Moves; pa8.AlphaMove = (ushort)moves[0]; } else { learn.SetEncounterMoves(level, moves); } pk.SetMoves(moves); pk.SetMaximumPPCurrent(moves); pa8.SetEncounterMasteryFlags(moves, mastery, level); } private OverworldParam8a GetParams() { var gender = GetGenderRatio(); return new OverworldParam8a { IsAlpha = IsAlpha, FlawlessIVs = FlawlessIVCount, Shiny = Shiny, RollCount = 1, // Everything is shiny locked anyways GenderRatio = gender, }; } private byte GetGenderRatio() => Gender switch { 0 => PersonalInfo.RatioMagicMale, 1 => PersonalInfo.RatioMagicFemale, 2 => PersonalInfo.RatioMagicGenderless, _ => GetGenderRatioPersonal(), }; private byte GetGenderRatioPersonal() { var pt = PersonalTable.LA; var entry = pt.GetFormEntry(Species, Form); return (byte)entry.Gender; } } public enum EncounterStatic8aCorrelation : byte { WildGroup, Fixed, }