using System; using System.Collections.Generic; using System.Linq; using static PKHeX.Core.GameVersion; namespace PKHeX.Core.Searching; /// /// search settings & searcher /// public sealed class SearchSettings { public int Format { get; init; } public int Generation { get; init; } public ushort Species { get; init; } public int Ability { get; init; } = -1; public int Nature { get; init; } = -1; public int Item { get; init; } = -1; public int Version { get; init; } = -1; public int HiddenPowerType { get; init; } = -1; public SearchComparison SearchFormat { get; init; } public SearchComparison SearchLevel { get; init; } public bool? SearchShiny { get; set; } public bool? SearchLegal { get; set; } public bool? SearchEgg { get; set; } public int? ESV { get; set; } public int? Level { get; init; } public int IVType { get; init; } public int EVType { get; init; } public CloneDetectionMethod SearchClones { get; set; } public IList BatchInstructions { get; init; } = Array.Empty(); private StringInstruction[] BatchFilters { get; set; } = Array.Empty(); public readonly List Moves = new(); // ReSharper disable once CollectionNeverUpdated.Global /// /// Extra Filters to be checked after all other filters have been checked. /// /// Collection is iterated right before clones are checked. public List> ExtraFilters { get; } = new(); /// /// Adds a move to the required move list. /// /// public void AddMove(ushort move) { if (move != 0 && !Moves.Contains(move)) Moves.Add(move); } /// /// Searches the input list, filtering out entries as specified by the settings. /// /// List of entries to search /// Search results that match all criteria public IEnumerable Search(IEnumerable list) { BatchFilters = StringInstruction.GetFilters(BatchInstructions).ToArray(); var result = SearchInner(list); if (SearchClones != CloneDetectionMethod.None) { var method = SearchUtil.GetCloneDetectMethod(SearchClones); result = SearchUtil.GetExtraClones(result, method); } return result; } /// /// Searches the input list, filtering out entries as specified by the settings. /// /// List of entries to search /// Search results that match all criteria public IEnumerable Search(IEnumerable list) { BatchFilters = StringInstruction.GetFilters(BatchInstructions).ToArray(); var result = SearchInner(list); if (SearchClones != CloneDetectionMethod.None) { var method = SearchUtil.GetCloneDetectMethod(SearchClones); string GetHash(SlotCache z) => method(z.Entity); result = SearchUtil.GetExtraClones(result, GetHash); } return result; } private IEnumerable SearchInner(IEnumerable list) { foreach (var pk in list) { if (!IsSearchMatch(pk)) continue; yield return pk; } } private IEnumerable SearchInner(IEnumerable list) { foreach (var entry in list) { var pk = entry.Entity; if (!IsSearchMatch(pk)) continue; yield return entry; } } private bool IsSearchMatch(PKM pk) { if (!SearchSimple(pk)) return false; if (!SearchIntermediate(pk)) return false; if (!SearchComplex(pk)) return false; foreach (var filter in ExtraFilters) { if (!filter(pk)) return false; } return true; } private bool SearchSimple(PKM pk) { if (Format > 0 && !SearchUtil.SatisfiesFilterFormat(pk, Format, SearchFormat)) return false; if (Species != 0 && pk.Species != Species) return false; if (Ability > -1 && pk.Ability != Ability) return false; if (Nature > -1 && pk.StatNature != Nature) return false; if (Item > -1 && pk.HeldItem != Item) return false; if (Version > -1 && pk.Version != Version) return false; return true; } private bool SearchIntermediate(PKM pk) { if (Generation > 0 && !SearchUtil.SatisfiesFilterGeneration(pk, Generation)) return false; if (Moves.Count > 0 && !SearchUtil.SatisfiesFilterMoves(pk, Moves)) return false; if (HiddenPowerType > -1 && pk.HPType != HiddenPowerType) return false; if (SearchShiny != null && pk.IsShiny != SearchShiny) return false; if (IVType > 0 && !SearchUtil.SatisfiesFilterIVs(pk, IVType)) return false; if (EVType > 0 && !SearchUtil.SatisfiesFilterEVs(pk, EVType)) return false; return true; } private bool SearchComplex(PKM pk) { if (SearchEgg != null && !FilterResultEgg(pk)) return false; if (Level is { } x and not 0 && !SearchUtil.SatisfiesFilterLevel(pk, SearchLevel, x)) return false; if (SearchLegal != null && new LegalityAnalysis(pk).Valid != SearchLegal) return false; if (BatchFilters.Length != 0 && !SearchUtil.SatisfiesFilterBatchInstruction(pk, BatchFilters)) return false; return true; } private bool FilterResultEgg(PKM pk) { if (SearchEgg == false) return !pk.IsEgg; if (ESV != null) return pk.IsEgg && pk.PSV == ESV; return pk.IsEgg; } public IReadOnlyList GetVersions(SaveFile sav) => GetVersions(sav, GetFallbackVersion(sav)); public IReadOnlyList GetVersions(SaveFile sav, GameVersion fallback) { if (Version > 0) return new[] {(GameVersion) Version}; return Generation switch { 1 when !ParseSettings.AllowGen1Tradeback => new[] {RD, BU, GN, YW}, 2 when sav is SAV2 {Korean: true} => new[] {GD, SI}, 1 or 2 => new[] {RD, BU, GN, YW, /* */ GD, SI, C}, _ when fallback.GetGeneration() == Generation => GameUtil.GetVersionsWithinRange(sav, Generation).ToArray(), _ => GameUtil.GameVersions, }; } private static GameVersion GetFallbackVersion(ITrainerInfo sav) { var parent = GameUtil.GetMetLocationVersionGroup((GameVersion)sav.Game); if (parent == Invalid) parent = GameUtil.GetMetLocationVersionGroup(GameUtil.GetVersion(sav.Generation)); return parent; } }