using System.Collections.Generic; using System.Linq; namespace PKHeX.Core { /// /// Miscellaneous setup utility for legality checking data sources. /// internal static class EncounterUtil { /// /// Gets the relevant objects that appear in the relevant game. /// /// Table of valid encounters that appear for the game pairing /// Game to filter for /// Array of encounter objects that can be encountered in the input game internal static EncounterStatic[] GetStaticEncounters(IEnumerable source, GameVersion game) { return source.Where(s => s.Version.Contains(game)).ToArray(); } /// /// Direct fetch for data; can also be used to fetch supplementary encounter streams. /// /// Unpacking identification ASCII characters (first two bytes of binary) /// Resource name (will be prefixed with "encounter_" /// Array of encounter areas internal static T[] GetEncounterTables(string ident, string resource) where T : EncounterArea32, new() { byte[] mini = Util.GetBinaryResource($"encounter_{resource}.pkl"); return EncounterArea32.GetArray(BinLinker.Unpack(mini, ident)); } /// /// Direct fetch for data; can also be used to fetch supplementary encounter streams. /// /// Unpacking identification ASCII characters (first two bytes of binary) /// Resource name (will be prefixed with "encounter_") /// Array of encounter areas internal static T[] GetEncounterTables8(string ident, string resource) where T : EncounterAreaSH, new() { byte[] mini = Util.GetBinaryResource($"encounter_{resource}.pkl"); return EncounterAreaSH.GetArray(BinLinker.Unpack(mini, ident)); } /// /// Combines slot arrays with the same . /// /// Input encounter areas to combine /// Combined Array of encounter areas. No duplicate location IDs will be present. internal static T[] AddExtraTableSlots(params T[][] tables) where T : EncounterArea, new() { return tables.SelectMany(s => s).GroupBy(l => l.Location) .Select(t => t.Count() == 1 ? t.First() // only one table, just return the area : new T { Location = t.Key, Slots = t.SelectMany(s => s.Slots).ToArray() }) .ToArray(); } /// /// Marks Encounter Slots for party lead's ability slot influencing. /// /// Magnet Pull attracts Steel type slots, and Static attracts Electric /// Encounter Area array for game /// Personal data for use with a given species' type internal static void MarkEncountersStaticMagnetPull(IEnumerable areas, PersonalTable t) { foreach (EncounterArea area in areas) { foreach (var grp in area.Slots.GroupBy(z => z.Type)) MarkEncountersStaticMagnetPull(grp, t); } } internal static void MarkEncountersStaticMagnetPull(IEnumerable grp, PersonalTable t) { GetStaticMagnet(t, grp, out List s, out List m); for (var i = 0; i < s.Count; i++) { var slot = s[i]; slot.Permissions.StaticIndex = i; slot.Permissions.StaticCount = s.Count; } for (var i = 0; i < m.Count; i++) { var slot = m[i]; slot.Permissions.MagnetPullIndex = i; slot.Permissions.MagnetPullCount = m.Count; } } internal static void MarkEncountersStaticMagnetPullPermutation(IEnumerable grp, PersonalTable t, List permuted) { GetStaticMagnet(t, grp, out List s, out List m); // Apply static/magnet values; if any permutation has a unique slot combination, add it to the slot list. for (int i = 0; i < s.Count; i++) { var slot = s[i]; if (slot.Permissions.StaticIndex >= 0) // already has unique data { if (slot.IsMatchStatic(i, s.Count)) continue; // same values, no permutation if (permuted.Any(z => z.SlotNumber == slot.SlotNumber && z.IsMatchStatic(i, s.Count) && z.Species == slot.Species)) continue; // same values, previously permuted s[i] = slot = slot.Clone(); permuted.Add(slot); } slot.Permissions.StaticIndex = i; slot.Permissions.StaticCount = s.Count; } for (int i = 0; i < m.Count; i++) { var slot = m[i]; if (slot.Permissions.MagnetPullIndex >= 0) // already has unique data { if (slot.IsMatchStatic(i, m.Count)) continue; // same values, no permutation if (permuted.Any(z => z.SlotNumber == slot.SlotNumber && z.IsMatchMagnet(i, m.Count) && z.Species == slot.Species)) continue; // same values, previously permuted m[i] = slot = slot.Clone(); permuted.Add(slot); } slot.Permissions.MagnetPullIndex = i; slot.Permissions.MagnetPullCount = m.Count; } } private static void GetStaticMagnet(PersonalTable t, IEnumerable grp, out List s, out List m) { const int steel = (int)MoveType.Steel; const int electric = (int)MoveType.Electric + 1; // offset by 1 in gen3/4 for the ??? type s = new List(); m = new List(); foreach (EncounterSlot Slot in grp) { var p = t[Slot.Species]; if (p.IsType(steel)) m.Add(Slot); if (p.IsType(electric)) s.Add(Slot); } } /// /// Sets the value, for use in determining split-generation origins. /// /// Only used for Gen 1 & 2, as data is not present. /// In-game encounter data /// Version ID to set internal static void MarkEncountersVersion(IEnumerable areas, GameVersion game) { foreach (EncounterArea area in areas) { foreach (var Slot in area.Slots) Slot.Version = game; } } /// /// Sets the value. /// /// Generation number to set /// In-game encounter data internal static void MarkEncountersGeneration(int generation, params IEnumerable[] encounters) { foreach (var table in encounters) MarkEncountersGeneration(generation, table); } /// /// Sets the value, for use in determining split-generation origins. /// /// Generation number to set /// In-game encounter data internal static void MarkEncountersGeneration(int generation, params IEnumerable[] areas) { foreach (var table in areas) { foreach (var area in table) MarkEncountersGeneration(generation, area.Slots); } } private static void MarkEncountersGeneration(int generation, IEnumerable encounters) { foreach (var enc in encounters) enc.Generation = generation; } /// /// Groups areas by location id, raw data has areas with different slots but the same location id. /// /// Similar to , this method combines a single array. /// Ingame encounter data internal static void ReduceAreasSize(ref T[] Areas) where T : EncounterArea, new() { Areas = Areas.GroupBy(a => a.Location).Select(a => new T { Location = a.Key, Slots = a.SelectMany(m => m.Slots).ToArray() }).ToArray(); } internal static T[] ConcatAll(params IEnumerable[] arr) => arr.SelectMany(z => z).ToArray(); internal static void MarkEncounterAreaArray(params T[][] areas) where T : EncounterArea { foreach (var area in areas) MarkEncounterAreas(area); } private static void MarkEncounterAreas(params T[] areas) where T : EncounterArea { foreach (var area in areas) { foreach (var slot in area.Slots) slot.Area = area; } } internal static EncounterStatic Clone(this EncounterStatic s, int location) { var result = s.Clone(); result.Location = location; return result; } internal static EncounterStatic[] Clone(this EncounterStatic s, int[] locations) { EncounterStatic[] Encounters = new EncounterStatic[locations.Length]; for (int i = 0; i < locations.Length; i++) Encounters[i] = s.Clone(locations[i]); return Encounters; } internal static IEnumerable DreamRadarClone(this EncounterStatic s) { for (int i = 0; i < 8; i++) yield return s.DreamRadarClone((5 * i) + 5); // Level from 5->40 depends on the number of badges } private static EncounterStatic DreamRadarClone(this EncounterStatic s, int level) { var result = s.Clone(level); result.Level = level; result.Location = 30015;// Pokemon Dream Radar result.Gift = true; // Only result.Ball = 25; // Dream Ball return result; } internal static void MarkEncounterTradeStrings(T[] table, string[][] strings) where T : EncounterTrade { int half = strings[1].Length / 2; for (int i = 0; i < half; i++) { var t = table[i]; t.Nicknames = getNames(i, strings); t.TrainerNames = getNames(i + half, strings); } string[] getNames(int i, IEnumerable names) => names.Select(z => z.Length > i ? z[i] : string.Empty).ToArray(); } internal static void MarkEncounterGame(IEnumerable table, GameVersion version) where T: IVersion, IVersionSet { foreach (var t in table.Where(z => ((IVersion)z).Version == GameVersion.Any)) ((IVersionSet)t).Version = version; } } }