2022-06-18 18:04:24 +00:00
|
|
|
using System;
|
2022-08-16 04:04:30 +00:00
|
|
|
using System.Text;
|
2017-09-04 02:51:29 +00:00
|
|
|
using static PKHeX.Core.LegalityCheckStrings;
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
namespace PKHeX.Core;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Verifies the <see cref="PKM"/> Ribbon values.
|
|
|
|
/// </summary>
|
|
|
|
public sealed class RibbonVerifier : Verifier
|
2017-09-04 02:51:29 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
protected override CheckIdentifier Identifier => CheckIdentifier.Ribbon;
|
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Maximum amount of ribbons to consider when allocating a span for parsing.
|
|
|
|
/// </summary>
|
|
|
|
/// <remarks>
|
|
|
|
/// Minor optimization is to stackalloc as little as possible, without too much calculation.
|
|
|
|
/// <see cref="RibbonIndex3.MAX_COUNT"/> + <see cref="RibbonIndex4.MAX_COUNT"/> is 48, but are not present after Gen5.
|
|
|
|
/// <see cref="PK5"/> only has 80 ribbons implemented.
|
|
|
|
/// Instead of using the sum of all 3 enums, we can use <see cref="RibbonIndex.MAX_COUNT"/> as the true maximum count.
|
|
|
|
/// </remarks>
|
|
|
|
public const int MaxRibbonCount = (int)RibbonIndex.MAX_COUNT;
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public override void Verify(LegalityAnalysis data)
|
2017-09-04 02:51:29 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
// Flag VC (Gen1/2) ribbons using Gen7 origin rules.
|
|
|
|
var enc = data.EncounterMatch;
|
|
|
|
var pk = data.Entity;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
// Check Unobtainable Ribbons
|
2022-08-16 04:04:30 +00:00
|
|
|
var args = new RibbonVerifierArguments(pk, enc, data.Info.EvoChainsAllGens);
|
|
|
|
Span<RibbonResult> result = stackalloc RibbonResult[MaxRibbonCount];
|
|
|
|
int count = GetRibbonResults(args, result);
|
|
|
|
if (count == 0)
|
2017-09-06 03:32:07 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
data.AddLine(GetValid(LRibbonAllValid));
|
2022-08-16 04:04:30 +00:00
|
|
|
return;
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2022-08-12 04:03:08 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
var msg = GetMessage(result[..count]);
|
|
|
|
data.AddLine(GetInvalid(msg));
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Checks if the <see cref="index"/> is not an invalid/missing ribbon in the result parse.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="index">Ribbon Index to check for</param>
|
|
|
|
/// <param name="args">Inputs to analyze</param>
|
|
|
|
/// <returns>True if not present in the flagged result span.</returns>
|
|
|
|
public static bool IsValidExtra(RibbonIndex index, RibbonVerifierArguments args)
|
2022-08-12 17:16:12 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
Span<RibbonResult> result = stackalloc RibbonResult[MaxRibbonCount];
|
|
|
|
int count = GetRibbonResults(args, result);
|
|
|
|
if (count == 0)
|
2022-08-12 17:16:12 +00:00
|
|
|
return true;
|
2022-06-18 18:04:24 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
var span = result[..count];
|
|
|
|
foreach (var x in span)
|
2017-09-04 02:51:29 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
if (x.Equals(index))
|
|
|
|
return false;
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2022-08-16 04:04:30 +00:00
|
|
|
return true;
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Uses the input <see cref="args"/> and stores results in the <see cref="result"/> span.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="args">Inputs to analyze</param>
|
|
|
|
/// <param name="result">Result storage</param>
|
|
|
|
/// <returns>Count of elements filled in the <see cref="result"/> span.</returns>
|
|
|
|
public static int GetRibbonResults(RibbonVerifierArguments args, Span<RibbonResult> result)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
var list = new RibbonResultList(result);
|
|
|
|
return GetRibbonResults(args, ref list);
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static int GetRibbonResults(RibbonVerifierArguments args, ref RibbonResultList list)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
if (!args.Entity.IsEgg)
|
|
|
|
Parse(args, ref list);
|
2022-06-18 18:04:24 +00:00
|
|
|
else
|
2022-08-16 04:04:30 +00:00
|
|
|
ParseEgg(args, ref list);
|
2020-06-20 02:51:57 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
return list.Count;
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static string GetMessage(ReadOnlySpan<RibbonResult> result)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
var total = result.Length;
|
|
|
|
int missing = GetCountMissing(result);
|
|
|
|
int invalid = total - missing;
|
|
|
|
var sb = new StringBuilder(total * 20);
|
|
|
|
if (missing != 0)
|
|
|
|
AppendAll(result, sb, LRibbonFMissing_0, true);
|
|
|
|
if (invalid != 0)
|
2017-09-04 02:51:29 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
if (missing != 0) // need to visually separate the message
|
|
|
|
sb.Append(Environment.NewLine);
|
|
|
|
AppendAll(result, sb, LRibbonFInvalid_0, false);
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2022-08-16 04:04:30 +00:00
|
|
|
return sb.ToString();
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static int GetCountMissing(ReadOnlySpan<RibbonResult> result)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
int count = 0;
|
|
|
|
foreach (var x in result)
|
2017-09-04 02:51:29 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
if (x.IsMissing)
|
|
|
|
count++;
|
2017-09-04 02:51:29 +00:00
|
|
|
}
|
2022-08-16 04:04:30 +00:00
|
|
|
return count;
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private const string MessageSplitNextRibbon = ", ";
|
2021-11-20 02:23:49 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static void AppendAll(ReadOnlySpan<RibbonResult> result, StringBuilder sb, string startText, bool stateMissing)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
int added = 0;
|
|
|
|
sb.Append(startText);
|
|
|
|
foreach (var x in result)
|
2020-09-18 23:23:17 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
if (x.IsMissing != stateMissing)
|
|
|
|
continue;
|
|
|
|
if (added++ != 0)
|
|
|
|
sb.Append(MessageSplitNextRibbon);
|
|
|
|
var localized = RibbonStrings.GetName(x.PropertyName);
|
|
|
|
sb.Append(localized);
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
|
|
|
}
|
2020-09-18 23:23:17 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static void Parse(RibbonVerifierArguments args, ref RibbonResultList list)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
var pk = args.Entity;
|
|
|
|
if (pk is IRibbonSetOnly3 o3)
|
|
|
|
o3.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetEvent3 e3)
|
|
|
|
e3.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetEvent4 e4)
|
|
|
|
e4.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetUnique3 u3)
|
|
|
|
u3.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetCommon3 s3)
|
|
|
|
s3.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetUnique4 u4)
|
|
|
|
u4.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetCommon4 s4)
|
|
|
|
s4.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetCommon6 s6)
|
|
|
|
s6.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetCommon7 s7)
|
|
|
|
s7.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetCommon8 s8)
|
|
|
|
s8.Parse(args, ref list);
|
2022-11-25 01:42:17 +00:00
|
|
|
if (pk is IRibbonSetCommon9 s9)
|
|
|
|
s9.Parse(args, ref list);
|
|
|
|
if (pk is IRibbonSetMark9 m9)
|
|
|
|
m9.Parse(args, ref list);
|
2017-09-04 02:51:29 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
|
2022-08-16 04:04:30 +00:00
|
|
|
private static void ParseEgg(RibbonVerifierArguments args, ref RibbonResultList list)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2022-08-16 04:04:30 +00:00
|
|
|
var pk = args.Entity;
|
|
|
|
if (pk is IRibbonSetOnly3 o3)
|
|
|
|
o3.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetEvent3 e3)
|
|
|
|
e3.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetEvent4 e4)
|
2022-11-25 01:42:17 +00:00
|
|
|
e4.ParseEgg(ref list, args); // Some event eggs can have ribbons!
|
2022-08-16 04:04:30 +00:00
|
|
|
if (pk is IRibbonSetUnique3 u3)
|
|
|
|
u3.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetCommon3 s3)
|
|
|
|
s3.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetUnique4 u4)
|
|
|
|
u4.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetCommon4 s4)
|
|
|
|
s4.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetCommon6 s6)
|
|
|
|
s6.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetCommon7 s7)
|
|
|
|
s7.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetCommon8 s8)
|
|
|
|
s8.ParseEgg(ref list);
|
2022-11-25 01:42:17 +00:00
|
|
|
if (pk is IRibbonSetCommon9 s9)
|
|
|
|
s9.ParseEgg(ref list);
|
|
|
|
if (pk is IRibbonSetMark9 m9)
|
|
|
|
m9.ParseEgg(ref list);
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2017-09-04 02:51:29 +00:00
|
|
|
}
|