2
0
Fork 0
mirror of https://github.com/kwsch/PKHeX synced 2025-01-13 21:18:52 +00:00
PKHeX/PKHeX.Core/Legality/Encounters/EncounterSlot/EncounterSlot8a.cs
Kurt d5be6254f3
Add logic for PLA wild RNG correlation ()
Adds structures to read/write saved spawner data such as seeds, counts.
Adds generator and validator to emulate the FixInitSpec builder used by the game logic

Similar to SW/SH raids, validating these in-process is not feasible due to the number crunching required.

This does not handle the encounter slot call or the follow-up level range call. Just the inner FixInitSpec ctor & fill.

level is calc'd:
randFloat(sum) -> slot float
rand.Next() -> gen_seed (for all the details)
rand.NextInt(delta) +min -> level

Co-Authored-By: Lusamine <30205550+Lusamine@users.noreply.github.com>
2022-02-20 17:59:48 -08:00

122 lines
4 KiB
C#

using System;
namespace PKHeX.Core;
/// <summary>
/// Encounter Slot found in <see cref="GameVersion.SWSH"/>.
/// </summary>
/// <inheritdoc cref="EncounterSlot"/>
public sealed record EncounterSlot8a : EncounterSlot, IAlpha
{
public override int Generation => 8;
public SlotType Type => Area.Type;
public bool IsAlpha { get => AlphaType is not 0; set => throw new InvalidOperationException("Do not mutate this field."); }
public byte FlawlessIVCount { get; }
public Gender Gender { get; }
public byte AlphaType { get; } // 0=Never, 1=Random, 2=Guaranteed
public EncounterSlot8a(EncounterArea8a area, int species, int form, int min, int max, byte alphaType, byte flawlessIVs, Gender gender) : base(area, species, form, min, max)
{
AlphaType = alphaType;
FlawlessIVCount = flawlessIVs;
Gender = gender;
}
protected override void ApplyDetails(ITrainerInfo sav, EncounterCriteria criteria, PKM pk)
{
base.ApplyDetails(sav, criteria, pk);
if (Gender != Gender.Random)
pk.Gender = (int)Gender;
var para = GetParams(criteria);
Overworld8aRNG.ApplyDetails(pk, criteria, para);
if (IsAlpha)
{
if (pk is IAlpha a)
a.IsAlpha = true;
if (Type is not SlotType.Landmark && pk is PA8 pa)
{
var extra = pa.AlphaMove = pa.GetRandomAlphaMove();
pa.SetMasteryFlagMove(extra);
pk.PushMove(extra);
}
}
if (pk is PA8 pa8)
{
pa8.HeightScalarCopy = pa8.HeightScalar;
pa8.SetMasteryFlags();
}
}
protected override void ApplyDetailsBall(PKM pk) => pk.Ball = (int)Ball.LAPoke;
public override EncounterMatchRating GetMatchRating(PKM pkm)
{
if (Gender is not Gender.Random && pkm.Gender != (int)Gender)
return EncounterMatchRating.PartialMatch;
var result = GetMatchRatingInternal(pkm);
var orig = base.GetMatchRating(pkm);
return result > orig ? result : orig;
}
private EncounterMatchRating GetMatchRatingInternal(PKM pkm)
{
if (pkm is IAlpha a && a.IsAlpha != IsAlpha)
return EncounterMatchRating.DeferredErrors;
if (FlawlessIVCount is not 0 && pkm.FlawlessIVCount < FlawlessIVCount)
return EncounterMatchRating.DeferredErrors;
return GetAlphaMoveCompatibility(pkm);
}
private EncounterMatchRating GetAlphaMoveCompatibility(PKM pkm)
{
// Check for Alpha move compatibility.
if (pkm is not PA8 pa)
return EncounterMatchRating.Match;
var alphaMove = pa.AlphaMove;
bool hasAlphaMove = alphaMove != 0;
if (!pa.IsAlpha || Type is SlotType.Landmark)
return !hasAlphaMove ? EncounterMatchRating.Match : EncounterMatchRating.DeferredErrors;
var pi = PersonalTable.LA.GetFormEntry(Species, Form);
var tutors = pi.SpecialTutors[0];
if (alphaMove is 0)
{
bool hasAnyTutor = Array.IndexOf(tutors, true) >= 0;
if (hasAnyTutor)
return EncounterMatchRating.Deferred;
}
else
{
var idx = pa.MoveShopPermitIndexes;
var index = idx.IndexOf(idx);
if (index == -1)
return EncounterMatchRating.Deferred;
if (!tutors[index])
return EncounterMatchRating.Deferred;
}
return EncounterMatchRating.Match;
}
private OverworldParam8a GetParams(EncounterCriteria criteria)
{
var pt = PersonalTable.LA;
var entry = pt.GetFormEntry(Species, Form);
var gender = (byte)entry.Gender;
return new OverworldParam8a
{
IsAlpha = IsAlpha,
FlawlessIVs = FlawlessIVCount,
Shiny = Shiny,
RollCount = criteria.Shiny.IsShiny() ? Type is SlotType.Swarm ? (byte)32 : (byte)7 : (byte)1,
GenderRatio = gender,
};
}
}