mirror of
https://github.com/kwsch/PKHeX.git
synced 2026-03-21 17:48:28 -05:00
335 lines
12 KiB
C#
335 lines
12 KiB
C#
using System;
|
|
using static PKHeX.Core.RandomCorrelationRating;
|
|
|
|
namespace PKHeX.Core;
|
|
|
|
/// <summary>
|
|
/// Generation 3 Static Encounter
|
|
/// </summary>
|
|
public sealed record EncounterStatic3(ushort Species, byte Level, GameVersion Version)
|
|
: IEncounterable, IEncounterMatch, IEncounterConvertible<PK3>, IFatefulEncounterReadOnly, IRandomCorrelation, IMoveset
|
|
{
|
|
public byte Generation => 3;
|
|
public EntityContext Context => EntityContext.Gen3;
|
|
public bool IsRoaming { get; init; }
|
|
public bool IsRoamingTruncatedIVs => IsRoaming && Version != GameVersion.E;
|
|
ushort ILocation.EggLocation => 0;
|
|
ushort ILocation.Location => Location;
|
|
public bool IsShiny => false;
|
|
private bool Gift => FixedBall == Ball.Poke;
|
|
public Shiny Shiny => Shiny.Random;
|
|
|
|
public AbilityPermission Ability => AbilityPermission.Any12;
|
|
|
|
public Ball FixedBall { get; init; }
|
|
public bool FatefulEncounter { get; init; }
|
|
|
|
public required byte Location { get; init; }
|
|
public byte Form { get; init; }
|
|
public bool IsEgg { get; init; }
|
|
public Moveset Moves { get; init; }
|
|
|
|
public string Name => "Static Encounter";
|
|
public string LongName => Name;
|
|
public byte LevelMin => Level;
|
|
public byte LevelMax => Level;
|
|
|
|
#region Generating
|
|
PKM IEncounterConvertible.ConvertToPKM(ITrainerInfo tr, EncounterCriteria criteria) => ConvertToPKM(tr, criteria);
|
|
PKM IEncounterConvertible.ConvertToPKM(ITrainerInfo tr) => ConvertToPKM(tr);
|
|
public PK3 ConvertToPKM(ITrainerInfo tr) => ConvertToPKM(tr, EncounterCriteria.Unrestricted);
|
|
|
|
public PK3 ConvertToPKM(ITrainerInfo tr, EncounterCriteria criteria)
|
|
{
|
|
int language = GetTemplateLanguage(tr);
|
|
var version = this.GetCompatibleVersion(tr.Version);
|
|
var pi = PersonalTable.E[Species];
|
|
var pk = new PK3
|
|
{
|
|
Species = Species,
|
|
CurrentLevel = LevelMin,
|
|
OriginalTrainerFriendship = pi.BaseFriendship,
|
|
|
|
MetLocation = Location,
|
|
MetLevel = LevelMin,
|
|
Version = version,
|
|
Ball = (byte)(FixedBall != Ball.None ? FixedBall : Ball.Poke),
|
|
FatefulEncounter = FatefulEncounter,
|
|
|
|
Language = language,
|
|
OriginalTrainerGender = tr.Gender,
|
|
ID32 = tr.ID32,
|
|
Nickname = SpeciesName.GetSpeciesNameGeneration(Species, language, Generation),
|
|
};
|
|
// Copy from SaveFile's OT name. Trash bytes here should be pure, but our OT name might not always source from a PK3/SAV3.
|
|
// Condition the buffer as if it came from a correct SAV3 named after the OT.
|
|
var ot = pk.OriginalTrainerTrash;
|
|
ot[..(language == 1 ? 6 : 7)].Fill(0xFF);
|
|
pk.OriginalTrainerName = EncounterUtil.GetTrainerName(tr, language);
|
|
|
|
if (IsEgg)
|
|
{
|
|
// Fake as hatched.
|
|
pk.MetLevel = EggStateLegality.EggMetLevel34;
|
|
pk.MetLocation = version is GameVersion.FR or GameVersion.LG
|
|
? Locations.HatchLocationFRLG
|
|
: Locations.HatchLocationRSE;
|
|
}
|
|
|
|
SetPINGA(pk, criteria, pi);
|
|
if (Moves.HasMoves)
|
|
pk.SetMoves(Moves);
|
|
else
|
|
EncounterUtil.SetEncounterMoves(pk, Version, LevelMin);
|
|
|
|
pk.ResetPartyStats();
|
|
return pk;
|
|
}
|
|
|
|
private int GetTemplateLanguage(ITrainerInfo tr)
|
|
{
|
|
// Old Sea Map was only distributed to Japanese games.
|
|
if (Species is (ushort)Core.Species.Mew)
|
|
return (int)LanguageID.Japanese;
|
|
|
|
// Deoxys for Emerald was not available for Japanese games.
|
|
if (Species is (ushort)Core.Species.Deoxys && tr.Language == 1)
|
|
return (int)LanguageID.English;
|
|
|
|
return (int)Language.GetSafeLanguage3((LanguageID)tr.Language);
|
|
}
|
|
|
|
private void SetPINGA(PK3 pk, in EncounterCriteria criteria, PersonalInfo3 pi)
|
|
{
|
|
var gr = pi.Gender;
|
|
if (IsRoamingTruncatedIVs)
|
|
{
|
|
SetRoamerPINGA(pk, criteria);
|
|
return;
|
|
}
|
|
if (criteria.IsSpecifiedIVsAll())
|
|
{
|
|
if (TrySetMethod1(pk, criteria, gr))
|
|
return;
|
|
}
|
|
SetMethod1(pk, criteria, gr, Util.Rand32());
|
|
}
|
|
|
|
private static bool SetRoamerPINGA(PK3 pk, in EncounterCriteria criteria)
|
|
{
|
|
// For every possible 8-bit IV combination, check if it meets the criteria.
|
|
var id32 = pk.ID32;
|
|
for (uint i = 0; i <= byte.MaxValue; i++)
|
|
{
|
|
var iv32 = i;
|
|
|
|
// IVs can only ever be Hidden Power: Fighting. Don't bother checking if it matches.
|
|
if (!criteria.IsSatisfiedIVs(iv32))
|
|
continue;
|
|
|
|
// Satisfactory IVs; now check PID/nature/shininess.
|
|
var frame = iv32 << 16;
|
|
for (uint hi = 0; hi <= byte.MaxValue; hi++)
|
|
{
|
|
for (uint lo = 0; lo <= ushort.MaxValue; lo++)
|
|
{
|
|
var state = frame | (hi << 24) | lo;
|
|
var rand2 = LCRNG.Prev16(ref state);
|
|
var rand1 = LCRNG.Prev16(ref state);
|
|
var pid = (rand2 << 16) | rand1;
|
|
if (criteria.IsSpecifiedNature() && !criteria.IsSatisfiedNature((Nature)(pid % 25)))
|
|
continue;
|
|
bool shiny = ShinyUtil.GetIsShiny3(id32, pid);
|
|
if (criteria.Shiny.IsShiny() != shiny)
|
|
continue;
|
|
|
|
pk.PID = pid;
|
|
pk.IV32 = iv32;
|
|
pk.RefreshAbility(0);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
// Should never happen, but just in case.
|
|
SetMethod1(pk, criteria, 255, Util.Rand32());
|
|
pk.IV32 &= 0xFF; // truncate to 8 bits (value storage fail)
|
|
return false;
|
|
}
|
|
|
|
private static bool TrySetMethod1(PK3 pk, in EncounterCriteria criteria, byte gr)
|
|
{
|
|
criteria.GetCombinedIVs(out var iv1, out var iv2);
|
|
|
|
Span<uint> seeds = stackalloc uint[LCRNG.MaxCountSeedsIV];
|
|
var count = LCRNGReversal.GetSeedsIVs(seeds, iv1 << 16, iv2 << 16);
|
|
foreach (var s in seeds[..count])
|
|
{
|
|
var seed = LCRNG.Prev2(s); // Unwind the RNG to get the real origin seed for the PID/IV
|
|
var pid = ClassicEraRNG.GetSequentialPID(seed);
|
|
if (criteria.IsSpecifiedNature() && !criteria.IsSatisfiedNature((Nature)(pid % 25)))
|
|
continue;
|
|
|
|
var gender = EntityGender.GetFromPIDAndRatio(pid, gr);
|
|
if (criteria.IsSpecifiedGender() && !criteria.IsSatisfiedGender(gender))
|
|
continue;
|
|
|
|
var abit = (int)(pid & 1);
|
|
if (criteria.IsSpecifiedAbility() && !criteria.IsSatisfiedAbility(abit))
|
|
continue;
|
|
|
|
pk.PID = pid;
|
|
pk.IV32 |= iv2 << 15 | iv1;
|
|
pk.Gender = gender;
|
|
pk.RefreshAbility(abit);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private static void SetMethod1(PK3 pk, in EncounterCriteria criteria, byte gr, uint seed)
|
|
{
|
|
var id32 = pk.ID32;
|
|
bool filterIVs = criteria.IsSpecifiedIVs(2);
|
|
while (true)
|
|
{
|
|
var pid = ClassicEraRNG.GetSequentialPID(ref seed);
|
|
var shiny = ShinyUtil.GetIsShiny3(id32, pid);
|
|
if (criteria.Shiny.IsShiny() != shiny)
|
|
continue;
|
|
|
|
if (criteria.IsSpecifiedNature() && !criteria.IsSatisfiedNature((Nature)(pid % 25)))
|
|
continue;
|
|
|
|
var gender = EntityGender.GetFromPIDAndRatio(pid, gr);
|
|
if (criteria.IsSpecifiedGender() && !criteria.IsSatisfiedGender(gender))
|
|
continue;
|
|
|
|
var abit = (int)(pid & 1);
|
|
if (criteria.IsSpecifiedAbility() && !criteria.IsSatisfiedAbility(abit))
|
|
continue;
|
|
|
|
var iv32 = ClassicEraRNG.GetSequentialIVs(ref seed);
|
|
if (criteria.IsSpecifiedHiddenPower() && !criteria.IsSatisfiedHiddenPower(iv32))
|
|
continue;
|
|
if (filterIVs && !criteria.IsSatisfiedIVs(iv32))
|
|
continue;
|
|
|
|
pk.PID = pid;
|
|
pk.IV32 |= iv32;
|
|
pk.Gender = gender;
|
|
pk.RefreshAbility(abit);
|
|
break;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Matching
|
|
public bool IsMatchExact(PKM pk, EvoCriteria evo)
|
|
{
|
|
if (!IsMatchEggLocation(pk))
|
|
return false;
|
|
if (!IsMatchLocation(pk))
|
|
return false;
|
|
if (!IsMatchLevel(pk, evo))
|
|
return false;
|
|
if (Form != evo.Form && !FormInfo.IsFormChangeable(Species, Form, pk.Form, Context, pk.Context))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
public EncounterMatchRating GetMatchRating(PKM pk)
|
|
{
|
|
if (IsMatchPartial(pk))
|
|
return EncounterMatchRating.PartialMatch;
|
|
return EncounterMatchRating.Match;
|
|
}
|
|
|
|
private bool IsDeferredSafari3(bool isSafariBall) => isSafariBall != Locations.IsSafariZoneLocation3(Location);
|
|
|
|
private static bool IsMatchEggLocation(PKM pk)
|
|
{
|
|
if (pk.Format == 3)
|
|
return true;
|
|
|
|
var expect = pk is PB8 ? Locations.Default8bNone : 0;
|
|
return pk.EggLocation == expect;
|
|
}
|
|
|
|
private bool IsMatchLevel(PKM pk, EvoCriteria evo)
|
|
{
|
|
if (pk.Format != 3) // Met Level lost on PK3=>PK4
|
|
return evo.LevelMax >= Level;
|
|
if (!IsEgg)
|
|
return pk.MetLevel == Level;
|
|
return pk is { MetLevel: EggStateLegality.EggMetLevel34, CurrentLevel: >= 5 }; // met level 0, origin level 5
|
|
}
|
|
|
|
private bool IsMatchLocation(PKM pk)
|
|
{
|
|
if (pk.Format != 3)
|
|
return true; // transfer location verified later
|
|
|
|
if (IsEgg)
|
|
return !pk.IsEgg || pk.MetLocation == Location;
|
|
|
|
var met = pk.MetLocation;
|
|
if (!IsRoaming)
|
|
return Location == met;
|
|
|
|
// Route 101-138
|
|
if (Version <= GameVersion.E)
|
|
return met is >= 16 and <= 49;
|
|
// Route 1-25 encounter is possible either in grass or on water
|
|
return met is >= 101 and <= 125;
|
|
}
|
|
|
|
private bool IsMatchPartial(PKM pk)
|
|
{
|
|
if (IsDeferredSafari3(pk.Ball == (int)Ball.Safari))
|
|
return true;
|
|
if (Gift && pk.Ball != (byte)FixedBall)
|
|
return true;
|
|
if (FatefulEncounter != pk.FatefulEncounter)
|
|
return true;
|
|
return false;
|
|
}
|
|
#endregion
|
|
|
|
public RandomCorrelationRating IsCompatible(PIDType type, PKM pk)
|
|
{
|
|
var version = pk.Version;
|
|
if (version is GameVersion.E)
|
|
return type is PIDType.Method_1 ? Match : Mismatch;
|
|
|
|
if (IsRoaming) // Glitched IVs
|
|
return IsRoamerPIDIV(type, pk) ? Match : Mismatch;
|
|
|
|
if (type is PIDType.Method_1)
|
|
return Match;
|
|
// RS: Only Method 1, but RSBox s/w emulation can yield Method 4.
|
|
if (version is GameVersion.R or GameVersion.S)
|
|
return type is PIDType.Method_4 ? NotIdeal : Mismatch;
|
|
// FR/LG: Only Method 1, but Togepi gift can be Method 4 via PID modulo VBlank abuse
|
|
return type is PIDType.Method_4 && Species is (ushort)Core.Species.Togepi ? NotIdeal : Mismatch;
|
|
}
|
|
|
|
private static bool IsRoamerPIDIV(PIDType val, PKM pk)
|
|
{
|
|
// Roamer PID/IV is always Method 1.
|
|
// M1 is checked before M1R. A M1R PID/IV can also be a M1 PID/IV, so check that collision.
|
|
if (PIDType.Method_1_Roamer == val)
|
|
return true;
|
|
if (PIDType.Method_1 != val)
|
|
return false;
|
|
|
|
// only 8 bits are stored instead of 32 -- 5 bits HP, 3 bits for ATK.
|
|
// return pk.IV32 <= 0xFF; -- not always in right order, and can have nickname flagged.
|
|
var ivs = pk.GetIVs();
|
|
return ivs <= 0xFF;
|
|
}
|
|
|
|
public PIDType GetSuggestedCorrelation() => PIDType.Method_1;
|
|
}
|