using System; using System.Linq; namespace PKHeX.Core { public abstract class Zukan { protected SaveFile SAV { get; set; } protected int PokeDex { get; set; } protected int PokeDexLanguageFlags { get; set; } protected Zukan(SaveFile sav, int dex, int langflag) { SAV = sav; PokeDex = dex; PokeDexLanguageFlags = langflag; if (langflag > dex) throw new ArgumentException(nameof(langflag)); } protected abstract int OFS_SEEN { get; } protected abstract int OFS_CAUGHT { get; } protected abstract int BitSeenSize { get; } protected abstract int DexLangFlagByteCount { get; } protected abstract int DexLangIDCount { get; } protected abstract int GetDexLangFlag(int lang); public Func DexFormIndexFetcher { get; protected set; } protected abstract bool GetSaneFormsToIterate(int species, out int formStart, out int formEnd, int formIn); protected virtual void SetSpindaDexData(PKM pkm, bool alreadySeen) { } protected abstract void SetAllDexFlagsLanguage(int bit, int lang, bool value = true); protected abstract void SetAllDexSeenFlags(int baseBit, int altform, int gender, bool isShiny, bool value = true); protected bool GetFlag(int ofs, int bitIndex) => SAV.GetFlag(PokeDex + ofs + (bitIndex >> 3), bitIndex); protected void SetFlag(int ofs, int bitIndex, bool value = true) => SAV.SetFlag(PokeDex + ofs + (bitIndex >> 3), bitIndex, value); public virtual bool GetCaught(int species) => GetFlag(OFS_CAUGHT, species - 1); public virtual void SetCaught(int species, bool value = true) => SetFlag(OFS_CAUGHT, species - 1, value); public int SeenCount => Enumerable.Range(1, SAV.MaxSpeciesID).Count(GetSeen); public int CaughtCount => Enumerable.Range(1, SAV.MaxSpeciesID).Count(GetCaught); public decimal PercentSeen => (decimal)SeenCount / SAV.MaxSpeciesID; public decimal PercentCaught => (decimal)CaughtCount / SAV.MaxSpeciesID; public virtual bool GetSeen(int species) { // check all 4 seen flags (gender/shiny) for (int i = 0; i < 4; i++) { if (GetSeen(species, i)) return true; } return false; } public bool GetSeen(int species, int i) => GetFlag(OFS_SEEN + (i * BitSeenSize), species - 1); public void SetSeen(int species, int i, bool value) => SetFlag(OFS_SEEN + (i * BitSeenSize), species - 1, value); public bool GetDisplayed(int bit, int i) => GetFlag(OFS_SEEN + ((i + 4) * BitSeenSize), bit); public void SetDisplayed(int bit, int i, bool value) => SetFlag(OFS_SEEN + ((i + 4) * BitSeenSize), bit, value); public bool GetLanguageFlag(int bit, int lang) => GetFlag(PokeDexLanguageFlags, (bit * DexLangIDCount) + lang); public void SetLanguageFlag(int bit, int lang, bool value) => SetFlag(PokeDexLanguageFlags, (bit * DexLangIDCount) + lang, value); public virtual void SetSeen(int species, bool value = true) { if (!value) { ClearSeen(species); return; } // check all 4 seen flags (gender/shiny) for (int i = 0; i < 4; i++) { if (GetFlag(OFS_SEEN + (i * BitSeenSize), species - 1)) return; } var gender = SAV.Personal[species].RandomGender() & 1; SetAllDexSeenFlags(species - 1, 0, gender, false); } private void ClearSeen(int species) { SetCaught(species, false); for (int i = 0; i < 4; i++) SetFlag(OFS_SEEN + (i * BitSeenSize), species - 1, false); } public virtual void SetDex(PKM pkm) { if (PokeDex < 0 || SAV.Version == GameVersion.Invalid) // sanity return; if (pkm.Species == 0 || pkm.Species > SAV.MaxSpeciesID) // out of range return; if (pkm.IsEgg) // do not add return; int species = pkm.Species; if (species == 327) // Spinda SetSpindaDexData(pkm, GetSeen(species)); int bit = pkm.Species - 1; int form = pkm.AltForm; int gender = pkm.Gender & 1; bool shiny = pkm.IsShiny; int lang = pkm.Language; SetDex(species, bit, form, gender, shiny, lang); } protected virtual void SetDex(int species, int bit, int form, int gender, bool shiny, int lang) { SetCaught(species); // Set the Owned Flag SetAllDexSeenFlags(bit, form, gender, shiny); // genderless -> male SetAllDexFlagsLanguage(bit, lang); } protected void SetDexFlags(int baseBit, int formBit, int gender, int shiny, bool value = true) { int shift = gender | (shiny << 1); // Set the [Species/Gender/Shiny] Seen Flag SetFlag(OFS_SEEN + (shift * BitSeenSize), baseBit, value); // Set the Display flag if none are set bool displayed = GetIsSpeciesFormAnyDisplayed(baseBit, formBit); if (!displayed || !value) SetFlag(OFS_SEEN + ((4 + shift) * BitSeenSize), formBit, value); } private bool GetIsSpeciesFormAnyDisplayed(int baseBit, int formBit) { // Check Displayed Status for base form for (int i = 0; i < 4; i++) { if (GetDisplayed(baseBit, i)) return true; } if (baseBit == formBit) return false; // If form is not base form, check form too for (int i = 0; i < 4; i++) { if (GetDisplayed(formBit, i)) return true; } return false; } // Bulk Manipulation public void SeenNone() => SetDexEntriesAll(false, shinyToo: true); public void CaughtNone() => SetAllCaught(false, true); public void SeenAll(bool shinyToo = false) => SetAllSeen(shinyToo); public void CompleteDex(bool shinyToo = false) => SetDexEntriesAll(shinyToo: shinyToo); public void CaughtAll(bool shinyToo = false) { SetAllSeen(true, shinyToo); SetAllCaught(true, shinyToo); } public void SetAllCaught(bool value = true, bool shinyToo = false) { for (int i = 0; i < SAV.MaxSpeciesID; i++) { int species = i + 1; SetCaught(species, value); // Set the Owned Flag SetSeenSingle(i + 1, value, shinyToo); } } public void SetAllSeen(bool value = true, bool shinyToo = false) { for (int i = 0; i < SAV.MaxSpeciesID; i++) SetSeenSingle(i + 1, value, shinyToo); } public void SetDexEntriesAll(bool value = true, int max = -1, bool shinyToo = false) { if (max <= 0) max = SAV.MaxSpeciesID; for (int i = 1; i <= max; i++) { SetSeenSingle(i, value, shinyToo); SetCaughtSingle(i, value); } } public void SetCaughtSingle(int species, bool value = true) { SetCaught(species, value); int baseBit = species - 1; SetAllDexFlagsLanguage(baseBit, value); } public void SetSeenSingle(int species, bool seen = true, bool shinyToo = false) { SetSeen(species, seen); var entry = SAV.Personal[species]; int baseBit = species - 1; int fc = entry.FormeCount; for (int f = 0; f < fc; f++) { if (!entry.OnlyFemale) { SetAllDexSeenFlags(baseBit, f, 0, false, seen); if (shinyToo) SetAllDexSeenFlags(baseBit, f, 0, true, seen); } if (!entry.OnlyMale && !entry.Genderless) { SetAllDexSeenFlags(baseBit, f, 1, false, seen); if (shinyToo) SetAllDexSeenFlags(baseBit, f, 1, true, seen); } } } protected void SetAllDexFlagsLanguage(int bit, bool value = true) { for (int i = 1; i <= DexLangIDCount + 1; i++) SetAllDexFlagsLanguage(bit, i, value); } } }