PKHeX/PKHeX.Core/Saves/Util/SaveExtensions.cs

159 lines
7 KiB
C#
Raw Normal View History

2018-07-21 04:32:33 +00:00
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Linq;
using static PKHeX.Core.MessageStrings;
namespace PKHeX.Core
{
public static class SaveExtensions
{
/// <summary>
/// Checks if the <see cref="PKM"/> is compatible with the input <see cref="SaveFile"/>, and makes any necessary modifications to force compatibility.
/// </summary>
/// <remarks>Should only be used when forcing a backwards conversion to sanitize the PKM fields to the target format.
/// If the PKM is compatible, some properties may be forced to sanitized values.</remarks>
/// <param name="SAV">Save File target that the PKM will be injected.</param>
/// <param name="pk">PKM input that is to be injected into the Save File.</param>
/// <returns>Indication whether or not the PKM is compatible.</returns>
public static bool IsPKMCompatibleWithModifications(this SaveFile SAV, PKM pk) => PKMConverter.IsPKMCompatibleWithModifications(pk, SAV);
/// <summary>
/// Sets the details of a path to a <see cref="SaveFile"/> object.
/// </summary>
/// <param name="sav">Save File to set path details to.</param>
/// <param name="path">Full Path of the file</param>
public static void SetFileInfo(this SaveFile sav, string path)
{
if (!sav.Exportable) // Blank save file
{
sav.FileFolder = sav.FilePath = null;
sav.FileName = "Blank Save File";
return;
}
sav.FilePath = path;
sav.FileFolder = Path.GetDirectoryName(path);
2018-07-30 04:54:02 +00:00
sav.FileName = string.Empty;
var bakName = Util.CleanFileName(sav.BAKName);
2018-07-21 04:32:33 +00:00
sav.FileName = Path.GetFileName(path);
2018-07-30 04:54:02 +00:00
if (sav.FileName?.EndsWith(bakName) == true)
sav.FileName = sav.FileName.Substring(0, sav.FileName.Length - bakName.Length);
2018-07-21 04:32:33 +00:00
}
/// <summary>
/// Checks a <see cref="PKM"/> file for compatibility to the <see cref="SaveFile"/>.
/// </summary>
/// <param name="SAV"><see cref="SaveFile"/> that is being checked.</param>
/// <param name="pkm"><see cref="PKM"/> that is being tested for compatibility.</param>
/// <returns></returns>
public static IReadOnlyList<string> IsPKMCompatible(this SaveFile SAV, PKM pkm)
{
IBasicStrings strings = GameInfo.Strings;
return SAV.GetSaveFileErrata(pkm, strings);
}
private static IReadOnlyList<string> GetSaveFileErrata(this SaveFile SAV, PKM pkm, IBasicStrings strings)
{
var errata = new List<string>();
if (SAV.Generation > 1)
{
ushort held = (ushort)pkm.HeldItem;
2018-08-21 14:46:55 +00:00
var itemstr = GameInfo.Strings.GetItemStrings(pkm.Format, (GameVersion) pkm.Version);
if (held > itemstr.Count)
2018-07-21 04:32:33 +00:00
errata.Add($"{MsgIndexItemRange} {held}");
else if (held > SAV.MaxItemID)
2018-08-21 14:46:55 +00:00
errata.Add($"{MsgIndexItemGame} {itemstr[held]}");
2018-07-21 04:32:33 +00:00
else if (!pkm.CanHoldItem(SAV.HeldItems))
2018-08-21 14:46:55 +00:00
errata.Add($"{MsgIndexItemHeld} {itemstr[held]}");
2018-07-21 04:32:33 +00:00
}
if (pkm.Species > strings.Species.Count)
errata.Add($"{MsgIndexSpeciesRange} {pkm.Species}");
else if (SAV.MaxSpeciesID < pkm.Species)
errata.Add($"{MsgIndexSpeciesGame} {strings.Species[pkm.Species]}");
if (!SAV.Personal[pkm.Species].IsFormeWithinRange(pkm.AltForm) && !FormConverter.IsValidOutOfBoundsForme(pkm.Species, pkm.AltForm, pkm.GenNumber))
errata.Add(string.Format(LegalityCheckStrings.LFormInvalidRange, Math.Max(0, SAV.Personal[pkm.Species].FormeCount - 1), pkm.AltForm));
2018-07-21 04:32:33 +00:00
if (pkm.Moves.Any(m => m > strings.Move.Count))
errata.Add($"{MsgIndexMoveRange} {string.Join(", ", pkm.Moves.Where(m => m > strings.Move.Count).Select(m => m.ToString()))}");
else if (pkm.Moves.Any(m => m > SAV.MaxMoveID))
errata.Add($"{MsgIndexMoveGame} {string.Join(", ", pkm.Moves.Where(m => m > SAV.MaxMoveID).Select(m => strings.Move[m]))}");
if (pkm.Ability > strings.Ability.Count)
errata.Add($"{MsgIndexAbilityRange} {pkm.Ability}");
else if (pkm.Ability > SAV.MaxAbilityID)
errata.Add($"{MsgIndexAbilityGame} {strings.Ability[pkm.Ability]}");
return errata;
}
/// <summary>
/// Imports compatible <see cref="PKM"/> data to the <see cref="SAV"/>, starting at the provided box.
/// </summary>
/// <param name="SAV">Save File that will receive the <see cref="compat"/> data.</param>
/// <param name="compat">Compatible <see cref="PKM"/> data that can be set to the <see cref="SAV"/> without conversion.</param>
/// <param name="overwrite">Overwrite existing full slots. If true, will only overwrite empty slots.</param>
/// <param name="boxStart">First box to start loading to. All prior boxes are not modified.</param>
/// <param name="noSetb">Bypass option to not modify <see cref="PKM"/> properties when setting to Save File.</param>
/// <returns>Count of injected <see cref="PKM"/>.</returns>
public static int ImportPKMs(this SaveFile SAV, IEnumerable<PKM> compat, bool overwrite = false, int boxStart = 0, bool? noSetb = null)
2018-07-21 04:32:33 +00:00
{
int startCount = boxStart * SAV.BoxSlotCount;
int maxCount = SAV.SlotCount;
int index = startCount;
2018-07-21 04:32:33 +00:00
foreach (var pk in compat)
{
if (overwrite)
{
while (SAV.IsSlotOverwriteProtected(index))
++index;
}
else
2018-07-21 04:32:33 +00:00
{
index = SAV.NextOpenBoxSlot(index-1);
if (index < 0) // Boxes full!
break;
2018-07-21 04:32:33 +00:00
}
SAV.SetBoxSlotAtIndex(pk, index, noSetb);
2018-07-21 04:32:33 +00:00
if (++index == maxCount) // Boxes full!
2018-07-21 04:32:33 +00:00
break;
}
return index - startCount; // actual imported count
2018-07-21 04:32:33 +00:00
}
public static IEnumerable<PKM> GetCompatible(this SaveFile SAV, IEnumerable<PKM> pks)
{
var savtype = SAV.PKMType;
foreach (var temp in pks)
{
var pk = PKMConverter.ConvertToType(temp, savtype, out string c);
if (pk == null)
{
Debug.WriteLine(c);
continue;
}
if (PKMConverter.IsIncompatibleGB(pk.Format, SAV.Japanese, pk.Japanese))
{
c = PKMConverter.GetIncompatibleGBMessage(pk, SAV.Japanese);
Debug.WriteLine(c);
continue;
}
2018-07-21 04:32:33 +00:00
var compat = SAV.IsPKMCompatible(pk);
if (compat.Count > 0)
continue;
yield return pk;
}
}
}
}