mirror of
https://github.com/kwsch/PKHeX.git
synced 2026-04-24 23:57:12 -05:00
Detect base algorithm (regular, regular-anti, anti, shiny) and check for sub-patterns that differ per encounter (such as OT gender). Implement PIDIV generators within the Gift3* template, remove old branches Add Table Weight detection (still need to test) for PCJP 5anniv Add PCJP GCEA campaign 1-6 templates Differentiate Mystry Mew seeds; disallow Mews that were deleted b4 cart cloning Need to double check some gen3 events (esp hadou titans), but is 99% there Thanks Sabresite, for your continued support
187 lines
6.7 KiB
C#
187 lines
6.7 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using static PKHeX.Core.LegalityCheckStrings;
|
|
|
|
namespace PKHeX.Core;
|
|
|
|
/// <summary>
|
|
/// Formatting logic for <see cref="LegalityAnalysis"/> to create a human-readable <see cref="T:System.String"/>.
|
|
/// </summary>
|
|
public static class LegalityFormatting
|
|
{
|
|
/// <summary>
|
|
/// Creates a report message with optional verbosity for in-depth analysis.
|
|
/// </summary>
|
|
/// <param name="la">Legality result to format</param>
|
|
/// <param name="verbose">Include all details in the parse, including valid check messages.</param>
|
|
/// <returns>Single line string</returns>
|
|
public static string Report(this LegalityAnalysis la, bool verbose = false) => verbose ? GetVerboseLegalityReport(la) : GetLegalityReport(la);
|
|
|
|
public static ILegalityFormatter Formatter { private get; set; } = new BaseLegalityFormatter();
|
|
|
|
public static string GetLegalityReport(LegalityAnalysis la) => Formatter.GetReport(la);
|
|
public static string GetVerboseLegalityReport(LegalityAnalysis la) => Formatter.GetReportVerbose(la);
|
|
|
|
public static void AddSecondaryChecksValid(IEnumerable<CheckResult> results, List<string> lines)
|
|
{
|
|
var outputLines = results
|
|
.Where(chk => chk.Valid && chk.Comment != L_AValid)
|
|
.OrderBy(chk => chk.Judgement) // Fishy sorted to top
|
|
.Select(chk => chk.Format(L_F0_1));
|
|
lines.AddRange(outputLines);
|
|
}
|
|
|
|
public static void AddSecondaryChecksInvalid(IReadOnlyList<CheckResult> results, List<string> lines)
|
|
{
|
|
foreach (var chk in results)
|
|
{
|
|
if (chk.Valid)
|
|
continue;
|
|
lines.Add(chk.Format(L_F0_1));
|
|
}
|
|
}
|
|
|
|
public static void AddRelearn(ReadOnlySpan<MoveResult> relearn, List<string> lines, bool state, PKM pk, EvolutionHistory history)
|
|
{
|
|
for (int i = 0; i < relearn.Length; i++)
|
|
{
|
|
var move = relearn[i];
|
|
if (move.Valid == state)
|
|
lines.Add(move.Format(L_F0_RM_1_2, i + 1, pk, history));
|
|
}
|
|
}
|
|
|
|
public static void AddMoves(ReadOnlySpan<MoveResult> moves, List<string> lines, in int currentFormat, bool state, PKM pk, EvolutionHistory history)
|
|
{
|
|
for (int i = 0; i < moves.Length; i++)
|
|
{
|
|
var move = moves[i];
|
|
if (move.Valid != state)
|
|
continue;
|
|
var msg = move.Format(L_F0_M_1_2, i + 1, pk, history);
|
|
var gen = move.Generation;
|
|
if (currentFormat != gen && gen != 0)
|
|
msg += $" [Gen{gen}]";
|
|
lines.Add(msg);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds information about the <see cref="LegalityAnalysis.EncounterMatch"/> to the <see cref="lines"/>.
|
|
/// </summary>
|
|
public static void AddEncounterInfo(LegalityAnalysis la, List<string> lines)
|
|
{
|
|
var enc = la.EncounterOriginal;
|
|
|
|
// Name
|
|
lines.Add(string.Format(L_FEncounterType_0, enc.GetEncounterName()));
|
|
if (enc is MysteryGift g)
|
|
lines.Add(g.CardHeader);
|
|
|
|
// Location
|
|
var loc = enc.GetEncounterLocation();
|
|
if (!string.IsNullOrEmpty(loc))
|
|
lines.Add(string.Format(L_F0_1, L_XLocation, loc));
|
|
|
|
// Version
|
|
if (enc.Generation <= 2)
|
|
lines.Add(string.Format(L_F0_1, nameof(GameVersion), enc.Version));
|
|
|
|
// PIDIV
|
|
AddEncounterInfoPIDIV(la, lines);
|
|
}
|
|
|
|
public static void AddEncounterInfoPIDIV(LegalityAnalysis la, List<string> lines)
|
|
{
|
|
var info = la.Info;
|
|
if (!info.PIDParsed)
|
|
info.PIDIV = MethodFinder.Analyze(la.Entity);
|
|
AddEncounterInfoPIDIV(lines, info);
|
|
}
|
|
|
|
private static void AddEncounterInfoPIDIV(List<string> lines, LegalInfo info)
|
|
{
|
|
var pidiv = info.PIDIV;
|
|
var type = string.Format(L_FPIDType_0, pidiv.Type);
|
|
if (info is { EncounterMatch: IRandomCorrelationEvent3 enc})
|
|
{
|
|
var mainType = enc.GetSuggestedCorrelation();
|
|
if (mainType != pidiv.Type)
|
|
type += $" [{mainType}]";
|
|
if (enc is EncounterGift3 { Method: PIDType.BACD_M } && info.PIDIVMatches) // mystry
|
|
{
|
|
var detail = MystryMew.GetSeedIndexes(pidiv.OriginSeed);
|
|
type += $" ({detail.Index}-{detail.SubIndex})";
|
|
}
|
|
}
|
|
lines.Add(type);
|
|
if (pidiv.NoSeed)
|
|
return;
|
|
|
|
if (pidiv.IsSeed64())
|
|
{
|
|
var seed = pidiv.Seed64;
|
|
lines.Add(string.Format(L_FOriginSeed_0, seed.ToString("X16")));
|
|
return;
|
|
}
|
|
if (info is { EncounterMatch: IEncounterSlot34 s })
|
|
{
|
|
var lead = pidiv.Lead;
|
|
var seed = !info.FrameMatches || lead == LeadRequired.Invalid ? pidiv.OriginSeed : pidiv.EncounterSeed;
|
|
var line = string.Format(L_FOriginSeed_0, seed.ToString("X8"));
|
|
if (lead != LeadRequired.None)
|
|
{
|
|
if (lead is LeadRequired.Static)
|
|
line += $" [{s.StaticIndex}/{s.StaticCount}]";
|
|
else if (lead is LeadRequired.MagnetPull)
|
|
line += $" [{s.MagnetPullIndex}/{s.MagnetPullCount}]";
|
|
else
|
|
line += $" [{s.SlotNumber}]";
|
|
|
|
line += $" ({lead.Localize()})";
|
|
}
|
|
else
|
|
{
|
|
line += $" [{s.SlotNumber}]";
|
|
}
|
|
lines.Add(line);
|
|
}
|
|
else
|
|
{
|
|
var seed = pidiv.OriginSeed;
|
|
var line = string.Format(L_FOriginSeed_0, seed.ToString("X8"));
|
|
if (info is { EncounterMatch: IRandomCorrelationEvent3 } && pidiv.Mutated is not 0 && pidiv.OriginSeed != pidiv.EncounterSeed)
|
|
line += $" [{pidiv.EncounterSeed:X8}]";
|
|
lines.Add(line);
|
|
}
|
|
}
|
|
|
|
private static string Localize(this LeadRequired lead)
|
|
{
|
|
if (lead is LeadRequired.Invalid)
|
|
return "❌";
|
|
var (ability, isFail, condition) = lead.GetDisplayAbility();
|
|
var abilities = GameInfo.Strings.Ability;
|
|
var name = abilities[(int)ability];
|
|
var result = isFail ? string.Format(L_F0_1, name, "❌") : name;
|
|
if (condition != EncounterTriggerCondition.None)
|
|
result += $"-{condition}";
|
|
return result;
|
|
}
|
|
|
|
public static string GetEncounterName(this IEncounterable enc)
|
|
{
|
|
var str = ParseSettings.SpeciesStrings;
|
|
var name = (uint) enc.Species < str.Count ? str[enc.Species] : enc.Species.ToString();
|
|
return $"{enc.LongName} ({name})";
|
|
}
|
|
|
|
public static string? GetEncounterLocation(this IEncounterTemplate enc)
|
|
{
|
|
if (enc is not ILocation loc)
|
|
return null;
|
|
return loc.GetEncounterLocation(enc.Generation, enc.Version);
|
|
}
|
|
}
|