using System; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Drawing; using System.Globalization; using System.IO; using System.Linq; using System.Windows.Forms; using PKHeX.Core; using System.Reflection; namespace PKHeX.WinForms { public partial class BatchEditor : Form { private readonly SaveFile SAV; public BatchEditor(PKM pk, SaveFile sav) { pkmref = pk; SAV = sav; InitializeComponent(); DragDrop += TabMain_DragDrop; DragEnter += TabMain_DragEnter; CB_Format.Items.Clear(); CB_Format.Items.Add("Any"); foreach (Type t in types) CB_Format.Items.Add(t.Name.ToLower()); CB_Format.Items.Add("All"); CB_Format.SelectedIndex = CB_Require.SelectedIndex = 0; new ToolTip().SetToolTip(CB_Property, "Property of a given PKM to modify."); new ToolTip().SetToolTip(L_PropType, "PropertyType of the currently loaded PKM in the main window."); new ToolTip().SetToolTip(L_PropValue, "PropertyValue of the currently loaded PKM in the main window."); } private static string[][] GetPropArray() { var p = new string[types.Length][]; for (int i = 0; i < p.Length; i++) p[i] = ReflectFrameworkUtil.GetPropertiesCanWritePublicDeclared(types[i]).Concat(CustomProperties).OrderBy(a => a).ToArray(); // Properties for any PKM var any = ReflectFrameworkUtil.GetPropertiesCanWritePublic(typeof(PK1)).Union(p.SelectMany(a => a)).OrderBy(a => a).ToArray(); // Properties shared by all PKM var all = p.Aggregate(new HashSet(p.First()), (h, e) => { h.IntersectWith(e); return h; }).OrderBy(a => a).ToArray(); var p1 = new string[types.Length + 2][]; Array.Copy(p, 0, p1, 1, p.Length); p1[0] = any; p1[p1.Length - 1] = all; return p1; } private readonly PKM pkmref; private const string CONST_RAND = "$rand"; private const string CONST_SHINY = "$shiny"; private const string CONST_SUGGEST = "$suggest"; private const string CONST_BYTES = "$[]"; private const string PROP_LEGAL = "Legal"; private static readonly string[] CustomProperties = {PROP_LEGAL}; private int currentFormat = -1; private static readonly Type[] types = { typeof (PK7), typeof (PK6), typeof (PK5), typeof (PK4), typeof(BK4), typeof (PK3), typeof (XK3), typeof (CK3), typeof (PK2), typeof (PK1), }; private static readonly string[][] properties = GetPropArray(); // GUI Methods private void B_Open_Click(object sender, EventArgs e) { if (!B_Go.Enabled) return; var fbd = new FolderBrowserDialog(); if (fbd.ShowDialog() != DialogResult.OK) return; TB_Folder.Text = fbd.SelectedPath; TB_Folder.Visible = true; } private void B_SAV_Click(object sender, EventArgs e) { TB_Folder.Text = ""; TB_Folder.Visible = false; } private void B_Go_Click(object sender, EventArgs e) { RunBackgroundWorker(); } private void B_Add_Click(object sender, EventArgs e) { if (CB_Property.SelectedIndex < 0) { WinFormsUtil.Alert("Invalid property selected."); return; } char[] prefix = { '.', '=', '!' }; string s = prefix[CB_Require.SelectedIndex] + CB_Property.Items[CB_Property.SelectedIndex].ToString() + "="; if (RTB_Instructions.Lines.Length != 0 && RTB_Instructions.Lines.Last().Length > 0) s = Environment.NewLine + s; RTB_Instructions.AppendText(s); } private void CB_Format_SelectedIndexChanged(object sender, EventArgs e) { if (currentFormat == CB_Format.SelectedIndex) return; int format = CB_Format.SelectedIndex; CB_Property.Items.Clear(); CB_Property.Items.AddRange(properties[format]); CB_Property.SelectedIndex = 0; currentFormat = format; } private void CB_Property_SelectedIndexChanged(object sender, EventArgs e) { L_PropType.Text = GetPropertyType(CB_Property.Text); if (pkmref.GetType().HasProperty(CB_Property.Text)) { L_PropValue.Text = ReflectFrameworkUtil.GetValue(pkmref, CB_Property.Text).ToString(); L_PropType.ForeColor = L_PropValue.ForeColor; // reset color } else // no property, flag { L_PropValue.Text = string.Empty; L_PropType.ForeColor = Color.Red; } } private static void TabMain_DragEnter(object sender, DragEventArgs e) { if (e.Data.GetDataPresent(DataFormats.FileDrop)) e.Effect = DragDropEffects.Copy; } private void TabMain_DragDrop(object sender, DragEventArgs e) { string[] files = (string[])e.Data.GetData(DataFormats.FileDrop); if (!Directory.Exists(files[0])) return; TB_Folder.Text = files[0]; TB_Folder.Visible = true; RB_SAV.Checked = false; RB_Path.Checked = true; } private BackgroundWorker b; private void RunBackgroundWorker() { if (RTB_Instructions.Lines.Any(line => line.Length == 0)) { WinFormsUtil.Error("Line length error in instruction list."); return; } var sets = StringInstructionSet.GetBatchSets(RTB_Instructions.Lines).ToArray(); if (sets.Any(s => s.Filters.Any(z => string.IsNullOrWhiteSpace(z.PropertyValue)))) { WinFormsUtil.Error("Empty Filter Value detected."); return; } if (sets.Any(z => !z.Instructions.Any())) { WinFormsUtil.Error("No instructions defined for a modification set."); return; } var emptyVal = sets.SelectMany(s => s.Instructions.Where(z => string.IsNullOrWhiteSpace(z.PropertyValue))).ToArray(); if (emptyVal.Any()) { string props = string.Join(", ", emptyVal.Select(z => z.PropertyName)); string invalid = $"Empty Property Value{(emptyVal.Length > 1 ? "s" : "")} detected:" + Environment.NewLine + props; if (DialogResult.Yes != WinFormsUtil.Prompt(MessageBoxButtons.YesNo, invalid, "Continue?")) return; } string destPath = null; if (RB_Path.Checked) { WinFormsUtil.Alert("Please select the folder where the files will be saved to.", "This can be the same folder as the source of PKM files."); var fbd = new FolderBrowserDialog(); var dr = fbd.ShowDialog(); if (dr != DialogResult.OK) return; destPath = fbd.SelectedPath; } FLP_RB.Enabled = RTB_Instructions.Enabled = B_Go.Enabled = false; foreach (var set in sets) { ScreenStrings(set.Filters); ScreenStrings(set.Instructions); } RunBatchEdit(sets, TB_Folder.Text, destPath); } private void RunBatchEdit(StringInstructionSet[] sets, string source, string destination) { len = err = ctr = 0; b = new BackgroundWorker { WorkerReportsProgress = true }; b.DoWork += (sender, e) => { if (RB_SAV.Checked) RunBatchEditSaveFile(sets); else RunBatchEditFolder(sets, source, destination); }; b.ProgressChanged += (sender, e) => SetProgressBar(e.ProgressPercentage); b.RunWorkerCompleted += (sender, e) => { ctr /= sets.Length; len /= sets.Length; string maybe = sets.Length == 1 ? string.Empty : "~"; string result = $"Modified {maybe}{ctr}/{len} files."; if (err > 0) result += Environment.NewLine + $"{maybe}{err} files ignored due to an internal error."; WinFormsUtil.Alert(result); FLP_RB.Enabled = RTB_Instructions.Enabled = B_Go.Enabled = true; SetupProgressBar(0); }; b.RunWorkerAsync(); } private void RunBatchEditFolder(IList sets, string source, string destination) { var files = Directory.GetFiles(source, "*", SearchOption.AllDirectories); SetupProgressBar(files.Length * sets.Count); foreach (var set in sets) ProcessFolder(files, set.Filters, set.Instructions, destination); } private void RunBatchEditSaveFile(IList sets) { IList data; if (SAV.HasParty && process(data = SAV.PartyData)) SAV.PartyData = data; if (SAV.HasBox && process(data = SAV.BoxData)) SAV.BoxData = data; bool process(IList d) { SetupProgressBar(d.Count * sets.Count); foreach (var set in sets) ProcessSAV(d, set.Filters, set.Instructions); return d.Count != 0; } } // Progress Bar private void SetupProgressBar(int count) { MethodInvoker mi = () => { PB_Show.Minimum = 0; PB_Show.Step = 1; PB_Show.Value = 0; PB_Show.Maximum = count; }; if (PB_Show.InvokeRequired) PB_Show.Invoke(mi); else mi.Invoke(); } private void SetProgressBar(int i) { if (PB_Show.InvokeRequired) PB_Show.Invoke((MethodInvoker)(() => PB_Show.Value = i)); else { PB_Show.Value = i; } } // Mass Editing private int ctr, len, err; private void ProcessSAV(IList data, IList Filters, IList Instructions) { for (int i = 0; i < data.Count; i++) { ProcessPKM(data[i], Filters, Instructions); b.ReportProgress(i); } } private void ProcessFolder(IReadOnlyList files, IList Filters, IList Instructions, string destPath) { for (int i = 0; i < files.Count; i++) { string file = files[i]; var fi = new FileInfo(file); if (!PKX.IsPKM(fi.Length)) { b.ReportProgress(i); continue; } int format = fi.Extension.Length > 0 ? (fi.Extension.Last() - '0') & 0xF : SAV.Generation; byte[] data = File.ReadAllBytes(file); var pkm = PKMConverter.GetPKMfromBytes(data, prefer: format); if (ProcessPKM(pkm, Filters, Instructions)) File.WriteAllBytes(Path.Combine(destPath, Path.GetFileName(file)), pkm.DecryptedBoxData); b.ReportProgress(i); } } private bool ProcessPKM(PKM pkm, IEnumerable Filters, IEnumerable Instructions) { if (!pkm.Valid || pkm.Locked) { len++; Debug.WriteLine("Skipped a pkm due to disallowed input: " + (pkm.Locked ? "Locked." : "Not Valid.")); return false; } ModifyResult r = TryModifyPKM(pkm, Filters, Instructions); if (r != ModifyResult.Invalid) len++; if (r == ModifyResult.Error) err++; if (r != ModifyResult.Modified) return false; if (pkm.Species <= 0) return false; pkm.RefreshChecksum(); ctr++; return true; } private string GetPropertyType(string propertyName) { if (CustomProperties.Contains(propertyName)) return "Custom"; int typeIndex = CB_Format.SelectedIndex; if (typeIndex == properties.Length - 1) // All return types[0].GetProperty(propertyName).PropertyType.Name; if (typeIndex == 0) // Any foreach (var p in types.Select(t => t.GetProperty(propertyName)).Where(p => p != null)) return p.PropertyType.Name; return types[typeIndex - 1].GetProperty(propertyName).PropertyType.Name; } // Utility Methods private enum ModifyResult { Invalid, Error, Filtered, Modified, } public class StringInstructionSet { public IList Filters { get; private set; } public IList Instructions { get; private set; } private const string SetSeparator = ";"; public static IEnumerable GetBatchSets(string[] lines) { int start = 0; while (start < lines.Length) { var list = lines.Skip(start).TakeWhile(z => !lines[start++].StartsWith(SetSeparator)).ToList(); yield return GetBatchSet(list); } } private static IEnumerable GetBatchSets(IEnumerable> sets) { return sets.Select(set => GetBatchSet(set.ToList())); } private static StringInstructionSet GetBatchSet(IList set) { return new StringInstructionSet { Filters = StringInstruction.GetFilters(set).ToList(), Instructions = StringInstruction.GetInstructions(set).ToList(), }; } } public class StringInstruction { public string PropertyName { get; set; } public string PropertyValue { get; set; } public bool Evaluator { get; set; } public void SetScreenedValue(string[] arr) { int index = Array.IndexOf(arr, PropertyValue); PropertyValue = index > -1 ? index.ToString() : PropertyValue; } // Extra Functionality private int Min, Max; public bool Random { get; private set; } public int RandomValue => Util.Rand.Next(Min, Max + 1); public void SetRandRange(string pv) { string str = pv.Substring(1); var split = str.Split(','); int.TryParse(split[0], out Min); int.TryParse(split[1], out Max); if (Min == Max) { PropertyValue = Min.ToString(); Debug.WriteLine(PropertyName + " randomization range Min/Max same?"); } else Random = true; } public static IEnumerable GetFilters(IEnumerable lines) { var raw = GetRelevantStrings(lines, '!', '='); return from line in raw let eval = line[0] == '=' let split = line.Substring(1).Split('=') where split.Length == 2 && !string.IsNullOrWhiteSpace(split[0]) select new StringInstruction { PropertyName = split[0], PropertyValue = split[1], Evaluator = eval }; } public static IEnumerable GetInstructions(IEnumerable lines) { var raw = GetRelevantStrings(lines, '.').Select(line => line.Substring(1)); return from line in raw select line.Split('=') into split where split.Length == 2 select new StringInstruction { PropertyName = split[0], PropertyValue = split[1] }; } private static IEnumerable GetRelevantStrings(IEnumerable lines, params char[] pieces) { return lines .Where(line => !string.IsNullOrEmpty(line)) .Where(line => pieces.Any(z => z == line[0])); } } private sealed class PKMInfo { internal PKM pkm { get; } internal PKMInfo(PKM pk) { pkm = pk; } private LegalityAnalysis la; private LegalityAnalysis Legality => la ?? (la = new LegalityAnalysis(pkm)); internal bool Legal => Legality.Valid; internal int[] SuggestedRelearn => Legality.GetSuggestedRelearn(); internal int[] SuggestedMoves => Legality.GetSuggestedMoves(tm: true, tutor: true, reminder: false); internal EncounterStatic SuggestedEncounter => Legality.GetSuggestedMetInfo(); } public static void ScreenStrings(IEnumerable il) { foreach (var i in il.Where(i => !i.PropertyValue.All(char.IsDigit))) { string pv = i.PropertyValue; if (pv.StartsWith("$") && !pv.StartsWith(CONST_BYTES) && pv.Contains(',')) i.SetRandRange(pv); SetInstructionScreenedValue(i); } } private static void SetInstructionScreenedValue(StringInstruction i) { switch (i.PropertyName) { case nameof(PKM.Species): i.SetScreenedValue(GameInfo.Strings.specieslist); return; case nameof(PKM.HeldItem): i.SetScreenedValue(GameInfo.Strings.itemlist); return; case nameof(PKM.Ability): i.SetScreenedValue(GameInfo.Strings.abilitylist); return; case nameof(PKM.Nature): i.SetScreenedValue(GameInfo.Strings.natures); return; case nameof(PKM.Ball): i.SetScreenedValue(GameInfo.Strings.balllist); return; case nameof(PKM.Move1): case nameof(PKM.Move2): case nameof(PKM.Move3): case nameof(PKM.Move4): case nameof(PKM.RelearnMove1): case nameof(PKM.RelearnMove2): case nameof(PKM.RelearnMove3): case nameof(PKM.RelearnMove4): i.SetScreenedValue(GameInfo.Strings.movelist); return; } } private static ModifyResult TryModifyPKM(PKM PKM, IEnumerable Filters, IEnumerable Instructions) { if (!PKM.ChecksumValid || PKM.Species == 0) return ModifyResult.Invalid; Type pkm = PKM.GetType(); PKMInfo info = new PKMInfo(PKM); ModifyResult result = ModifyResult.Error; foreach (var cmd in Filters) { try { if (IsPKMFiltered(pkm, cmd, info, out result)) return result; // why it was filtered out } catch { Debug.WriteLine($"Unable to compare {cmd.PropertyName} to {cmd.PropertyValue}."); } } foreach (var cmd in Instructions) { try { result = SetPKMProperty(PKM, info, cmd); } catch { Debug.WriteLine($"Unable to set {cmd.PropertyName} to {cmd.PropertyValue}."); } } return result; } private static ModifyResult SetPKMProperty(PKM PKM, PKMInfo info, StringInstruction cmd) { if (cmd.PropertyValue.StartsWith(CONST_BYTES)) return SetByteArrayProperty(PKM, cmd) ? ModifyResult.Modified : ModifyResult.Error; if (cmd.PropertyValue == CONST_SUGGEST) return SetSuggestedPKMProperty(PKM, cmd, info) ? ModifyResult.Modified : ModifyResult.Error; SetProperty(PKM, cmd); return ModifyResult.Modified; } private static bool IsPKMFiltered(Type pkm, StringInstruction cmd, PKMInfo info, out ModifyResult result) { result = ModifyResult.Error; if (cmd.PropertyName == PROP_LEGAL) { if (!bool.TryParse(cmd.PropertyValue, out bool legal)) return true; if (legal == info.Legal == cmd.Evaluator) return false; result = ModifyResult.Filtered; return true; } if (!pkm.HasPropertyAll(cmd.PropertyName) || pkm.IsValueEqual(info.pkm, cmd.PropertyName, cmd.PropertyValue) != cmd.Evaluator) { result = ModifyResult.Filtered; return true; } return false; } private static bool SetSuggestedPKMProperty(PKM PKM, StringInstruction cmd, PKMInfo info) { switch (cmd.PropertyName) { case nameof(PKM.HyperTrainFlags): PKM.HyperTrainFlags = GetSuggestedHyperTrainingStatus(PKM); return true; case nameof(PKM.RelearnMoves): PKM.RelearnMoves = info.SuggestedRelearn; return true; case nameof(PKM.Met_Location): var encounter = info.SuggestedEncounter; if (encounter == null) return false; int level = encounter.Level; int location = encounter.Location; int minlvl = Legal.GetLowestLevel(PKM, encounter.LevelMin); PKM.Met_Level = level; PKM.Met_Location = location; PKM.CurrentLevel = Math.Max(minlvl, level); return true; case nameof(PKM.Moves): var moves = info.SuggestedMoves; Util.Shuffle(moves); PKM.SetMoves(moves); return true; default: return false; } } private static int GetSuggestedHyperTrainingStatus(PKM pkm) { if (pkm.Format < 7 || pkm.CurrentLevel != 100) return 0; int val = 0; if (pkm.IV_HP != 31) val |= 1 << 0; if (pkm.IV_ATK < 31 && pkm.IV_ATK > 1) val |= 1 << 1; if (pkm.IV_DEF != 31) val |= 1 << 2; if (pkm.IV_SPE < 31 && pkm.IV_SPE > 1) val |= 1 << 3; if (pkm.IV_SPA != 31) val |= 1 << 4; if (pkm.IV_SPD != 31) val |= 1 << 5; return val; } private static bool SetByteArrayProperty(PKM PKM, StringInstruction cmd) { switch (cmd.PropertyName) { case nameof(PKM.Nickname_Trash): PKM.Nickname_Trash = string2arr(cmd.PropertyValue); return true; case nameof(PKM.OT_Trash): PKM.OT_Trash = string2arr(cmd.PropertyValue); return true; default: return false; } byte[] string2arr(string str) => str.Substring(CONST_BYTES.Length).Split(',').Select(z => Convert.ToByte(z.Trim(), 16)).ToArray(); } private static void SetProperty(PKM PKM, StringInstruction cmd) { if (cmd.PropertyName == nameof(PKM.MetDate)) PKM.MetDate = DateTime.ParseExact(cmd.PropertyValue, "yyyyMMdd", CultureInfo.InvariantCulture, DateTimeStyles.None); else if (cmd.PropertyName == nameof(PKM.EggMetDate)) PKM.EggMetDate = DateTime.ParseExact(cmd.PropertyValue, "yyyyMMdd", CultureInfo.InvariantCulture, DateTimeStyles.None); else if (cmd.PropertyName == nameof(PKM.EncryptionConstant) && cmd.PropertyValue == CONST_RAND) ReflectFrameworkUtil.SetValue(PKM, cmd.PropertyName, Util.Rand32().ToString()); else if ((cmd.PropertyName == nameof(PKM.Ability) || cmd.PropertyName == nameof(PKM.AbilityNumber)) && cmd.PropertyValue.StartsWith("$")) PKM.RefreshAbility(Convert.ToInt16(cmd.PropertyValue[1]) - 0x30); else if (cmd.PropertyName == nameof(PKM.PID) && cmd.PropertyValue == CONST_RAND) PKM.SetPIDGender(PKM.Gender); else if (cmd.PropertyName == nameof(PKM.EncryptionConstant) && cmd.PropertyValue == nameof(PKM.PID)) PKM.EncryptionConstant = PKM.PID; else if (cmd.PropertyName == nameof(PKM.PID) && cmd.PropertyValue == CONST_SHINY) PKM.SetShinyPID(); else if (cmd.PropertyName == nameof(PKM.Species) && cmd.PropertyValue == "0") PKM.Data = new byte[PKM.Data.Length]; else if (cmd.PropertyName.StartsWith("IV") && cmd.PropertyValue == CONST_RAND) SetRandomIVs(PKM, cmd); else if (cmd.Random) ReflectFrameworkUtil.SetValue(PKM, cmd.PropertyName, cmd.RandomValue); else if (cmd.PropertyName == nameof(PKM.IsNicknamed) && cmd.PropertyValue.ToLower() == "false") { PKM.IsNicknamed = false; PKM.Nickname = PKX.GetSpeciesNameGeneration(PKM.Species, PKM.Language, PKM.Format); } else ReflectFrameworkUtil.SetValue(PKM, cmd.PropertyName, cmd.PropertyValue); } private static void SetRandomIVs(PKM PKM, StringInstruction cmd) { if (cmd.PropertyName == nameof(PKM.IVs)) { PKM.SetRandomIVs(); return; } ReflectFrameworkUtil.SetValue(PKM, cmd.PropertyName, Util.Rand32() & PKM.MaxIV); } } public static class ReflectFrameworkUtil { public static bool IsValueEqual(this Type t, object obj, string propertyName, object value) { PropertyInfo pi = t.GetProperty(propertyName, BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance); var v = pi.GetValue(obj, null); var c = ConvertValue(value, pi.PropertyType); return v.Equals(c); } public static void SetValue(object obj, string propertyName, object value) { PropertyInfo pi = obj.GetType().GetProperty(propertyName); pi.SetValue(obj, ConvertValue(value, pi.PropertyType), null); } public static object GetValue(object obj, string propertyName) { PropertyInfo pi = obj.GetType().GetProperty(propertyName); return pi.GetValue(obj, null); } public static IEnumerable GetPropertiesStartWithPrefix(Type type, string prefix, BindingFlags flags = BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public) { return type.GetProperties(flags) .Where(p => p.Name.StartsWith(prefix, StringComparison.Ordinal)) .Select(p => p.Name); } public static IEnumerable GetPropertiesCanWritePublic(Type type, BindingFlags flags = BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public) { return type.GetProperties(flags) .Where(p => p.CanWrite && p.GetSetMethod(nonPublic: true).IsPublic) .Select(p => p.Name); } public static IEnumerable GetPropertiesCanWritePublicDeclared(Type type) { return GetPropertiesCanWritePublic(type, BindingFlags.Public | BindingFlags.Instance | BindingFlags.DeclaredOnly); } public static bool HasProperty(this Type type, string name) { return type.GetProperty(name, BindingFlags.Public | BindingFlags.Instance) != null; } public static bool HasPropertyAll(this Type type, string name) { return type.GetProperty(name, BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance) != null; } private static object ConvertValue(object value, Type type) { if (type == typeof(DateTime?)) // Used for PKM.MetDate and other similar properties { if (DateTime.TryParseExact(value.ToString(), "yyyyMMdd", CultureInfo.InvariantCulture, DateTimeStyles.None, out DateTime dateValue)) return dateValue; return null; } // Convert.ChangeType is suitable for most things return Convert.ChangeType(value, type); } } }