using System.Collections.Generic; using System.IO; using System.Linq; using static PKHeX.Core.MessageStrings; namespace PKHeX.Core { /// /// Contains extension methods for use with a . /// public static class BoxUtil { /// /// Dumps a folder of files to the . /// /// that is being dumped from. /// Folder to store files. /// Option to save in child folders with the Box Name as the folder name. /// -1 if aborted, otherwise the amount of files dumped. public static int DumpBoxes(this SaveFile SAV, string path, bool boxFolders = false) { if (!SAV.HasBox) return -1; var boxdata = SAV.BoxData; var ctr = 0; foreach (var pk in boxdata) { if (pk.Species == 0 || !pk.Valid) continue; var boxfolder = path; if (boxFolders) { var boxName = Util.CleanFileName(SAV.GetBoxName(pk.Box - 1)); boxfolder = Path.Combine(path, boxName); Directory.CreateDirectory(boxfolder); } var fileName = Util.CleanFileName(pk.FileName); var fn = Path.Combine(boxfolder, fileName); if (File.Exists(fn)) continue; File.WriteAllBytes(fn, pk.DecryptedBoxData); ctr++; } return ctr; } /// /// Dumps the to a folder with individual decrypted files. /// /// that is being dumped from. /// Folder to store files. /// Box contents to be dumped. /// -1 if aborted, otherwise the amount of files dumped. public static int DumpBox(this SaveFile SAV, string path, int currentBox) { if (!SAV.HasBox) return -1; var boxdata = SAV.BoxData; var ctr = 0; foreach (var pk in boxdata) { if (pk.Species == 0 || !pk.Valid || pk.Box - 1 != currentBox) continue; var fileName = Path.Combine(path, Util.CleanFileName(pk.FileName)); if (File.Exists(fileName)) continue; File.WriteAllBytes(fileName, pk.DecryptedBoxData); ctr++; } return ctr; } /// /// Loads a folder of files to the . /// /// to load folder to. /// Folder to load files from. Files are only loaded from the top directory. /// Result message from the method. /// First box to start loading to. All prior boxes are not modified. /// Instruction to clear boxes after the starting box. /// Overwrite existing full slots. If true, will only overwrite empty slots. /// Bypass option to not modify properties when setting to Save File. /// Enumerate all files even in sub-folders. /// Count of files imported. public static int LoadBoxes(this SaveFile SAV, string path, out string result, int boxStart = 0, bool boxClear = false, bool overwrite = false, PKMImportSetting noSetb = PKMImportSetting.UseDefault, bool all = false) { if (string.IsNullOrWhiteSpace(path) || !Directory.Exists(path)) { result = MsgSaveBoxExportPathInvalid; return -1; } var opt = all ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly; var filepaths = Directory.EnumerateFiles(path, "*.*", opt); return SAV.LoadBoxes(filepaths, out result, boxStart, boxClear, overwrite, noSetb); } /// /// Loads a folder of files to the . /// /// to load folder to. /// Files to load files from. /// Result message from the method. /// First box to start loading to. All prior boxes are not modified. /// Instruction to clear boxes after the starting box. /// Overwrite existing full slots. If true, will only overwrite empty slots. /// Bypass option to not modify properties when setting to Save File. /// Count of files imported. public static int LoadBoxes(this SaveFile SAV, IEnumerable filepaths, out string result, int boxStart = 0, bool boxClear = false, bool overwrite = false, PKMImportSetting noSetb = PKMImportSetting.UseDefault) { var pks = GetPossiblePKMsFromPaths(SAV, filepaths); return SAV.LoadBoxes(pks, out result, boxStart, boxClear, overwrite, noSetb); } /// /// Loads a folder of files to the . /// /// to load folder to. /// Encounters to create files from. /// Result message from the method. /// First box to start loading to. All prior boxes are not modified. /// Instruction to clear boxes after the starting box. /// Overwrite existing full slots. If true, will only overwrite empty slots. /// Bypass option to not modify properties when setting to Save File. /// Count of files imported. public static int LoadBoxes(this SaveFile SAV, IEnumerable encounters, out string result, int boxStart = 0, bool boxClear = false, bool overwrite = false, PKMImportSetting noSetb = PKMImportSetting.UseDefault) { var pks = encounters.Select(z => z.ConvertToPKM(SAV)); return SAV.LoadBoxes(pks, out result, boxStart, boxClear, overwrite, noSetb); } /// /// Loads a folder of files to the . /// /// to load folder to. /// Unconverted objects to load. /// Result message from the method. /// First box to start loading to. All prior boxes are not modified. /// Instruction to clear boxes after the starting box. /// Overwrite existing full slots. If true, will only overwrite empty slots. /// Bypass option to not modify properties when setting to Save File. /// True if any files are imported. public static int LoadBoxes(this SaveFile SAV, IEnumerable pks, out string result, int boxStart = 0, bool boxClear = false, bool overwrite = false, PKMImportSetting noSetb = PKMImportSetting.UseDefault) { if (!SAV.HasBox) { result = MsgSaveBoxFailNone; return -1; } var compat = SAV.GetCompatible(pks); if (boxClear) SAV.ClearBoxes(boxStart); int ctr = SAV.ImportPKMs(compat, overwrite, boxStart, noSetb); if (ctr <= 0) { result = MsgSaveBoxImportNoFiles; return -1; } result = string.Format(MsgSaveBoxImportSuccess, ctr); return ctr; } public static IEnumerable GetPKMsFromPaths(IEnumerable filepaths, int generation) { var result = filepaths .Where(file => PKX.IsPKM(new FileInfo(file).Length)) .Select(File.ReadAllBytes) .Select(data => PKMConverter.GetPKMfromBytes(data, prefer: generation)); foreach (var pkm in result) { if (pkm != null) yield return pkm; } } private static IEnumerable GetPossiblePKMsFromPaths(SaveFile sav, IEnumerable filepaths) { foreach (var f in filepaths) { var obj = FileUtil.GetSupportedFile(f, sav); switch (obj) { case PKM pk: yield return pk; break; case MysteryGift g when g.IsPokémon: yield return g.ConvertToPKM(sav); break; case GP1 g when g.Species != 0: yield return g.ConvertToPB7(sav); break; } } } } }