using System; using System.Collections.Generic; using System.IO; using System.Linq; using PKHeX.Core; namespace PKHeX.WinForms { public static class PathUtilWindows { /// /// Gets the 3DS's root folder, usually from an inserted SD card. /// /// Optional parameter to skip the first drive. /// The first drive is usually the system hard drive, or can be a floppy disk drive (slower to check, never has expected data). /// Folder path pointing to the Nintendo 3DS folder. public static string Get3DSLocation(bool skipFirstDrive = true) => FindConsoleRootFolder("Nintendo 3DS", skipFirstDrive); /// /// Gets the Switch's root folder, usually from an inserted SD card. /// /// Optional parameter to skip the first drive. /// The first drive is usually the system hard drive, or can be a floppy disk drive (slower to check, never has expected data). /// Folder path pointing to the Nintendo folder. public static string GetSwitchLocation(bool skipFirstDrive = true) => FindConsoleRootFolder("Nintendo", skipFirstDrive); private static string FindConsoleRootFolder(string path, bool skipFirstDrive) { try { // Skip first drive (some users still have floppy drives and would chew up time!) IEnumerable DriveList = Environment.GetLogicalDrives(); if (skipFirstDrive) DriveList = DriveList.Skip(1); return DriveList .Select(drive => Path.Combine(drive, path)) .FirstOrDefault(Directory.Exists); } catch { } return null; } /// /// Gets a list of 3DS save backup paths for the storage device. /// /// Root location of device /// List of possible 3DS save backup paths. public static IEnumerable Get3DSBackupPaths(string root) { yield return Path.Combine(root, "saveDataBackup"); yield return Path.Combine(root, "filer", "UserSaveData"); yield return Path.Combine(root, "JKSV", "Saves"); yield return Path.Combine(root, "TWLSaveTool"); yield return Path.Combine(root, "fbi", "save"); yield return Path.Combine(root, "gm9", "out"); yield return Path.Combine(root, "3ds", "Checkpoint", "saves"); } /// /// Gets a list of Switch save backup paths for the storage device. /// /// Root location of device /// List of possible 3DS save backup paths. public static IEnumerable GetSwitchBackupPaths(string root) { yield return Path.Combine(root, "switch", "Checkpoint", "saves"); } /// /// Finds a compatible save file that was most recently saved (by file write time). /// /// If this function returns true, full path of a save file or null if no path could be found. If this function returns false, this parameter will be set to the error message. /// Paths to check in addition to the default paths /// A boolean indicating whether or not a file was detected public static bool DetectSaveFile(out string path, params string[] extra) { var foldersToCheck = extra.Where(f => f?.Length > 0); string path3DS = Path.GetPathRoot(Get3DSLocation()); if (path3DS != null) // check for Homebrew/CFW backups foldersToCheck = foldersToCheck.Concat(Get3DSBackupPaths(path3DS)); string pathNX = Path.GetPathRoot(GetSwitchLocation()); if (pathNX != null) // check for Homebrew/CFW backups foldersToCheck = foldersToCheck.Concat(GetSwitchBackupPaths(pathNX)); path = null; List possiblePaths = new List(); foreach (var folder in foldersToCheck) { if (!SaveUtil.GetSavesFromFolder(folder, true, out IEnumerable files)) { if (files != null) // can be null if folder doesn't exist { path = string.Join(Environment.NewLine, files); // `files` contains the error message return false; } } if (files != null) possiblePaths.AddRange(files); } // return newest save file path that is valid foreach (var file in possiblePaths.OrderByDescending(f => new FileInfo(f).LastWriteTime)) { try { var data = File.ReadAllBytes(file); var sav = SaveUtil.GetVariantSAV(data); if (sav?.ChecksumsValid != true) continue; path = file; return true; } catch (Exception e) { path = e.Message + Environment.NewLine + file; return false; } } return true; } } }