using PKHeX.Core; using System; using System.Collections.Generic; using System.Drawing; using System.IO; using System.Linq; using System.Reflection; using System.Windows.Forms; using static PKHeX.Core.MessageStrings; namespace PKHeX.WinForms { public static class WinFormsUtil { internal static void TranslateInterface(Control form, string lang) => form.TranslateInterface(lang); internal static void CenterToForm(this Control child, Control parent) { int x = parent.Location.X + ((parent.Width - child.Width) / 2); int y = parent.Location.Y + ((parent.Height - child.Height) / 2); child.Location = new Point(Math.Max(x, 0), Math.Max(y, 0)); } public static Form FirstFormOfType(this Form f) => Array.Find(f.OwnedForms, form => form is T); public static T FindFirstControlOfType(Control aParent) where T : class { while (true) { var t = aParent as T; if (t != null) return t; if (aParent.Parent != null) aParent = aParent.Parent; else return null; } } public static Control GetUnderlyingControl(object sender) => ((sender as ToolStripItem)?.Owner as ContextMenuStrip)?.SourceControl ?? sender as PictureBox; #region Message Displays /// /// Displays a dialog showing the details of an error. /// /// User-friendly message about the error. /// Instance of the error's . /// The associated with the dialog. internal static DialogResult Error(string friendlyMessage, Exception exception) { System.Media.SystemSounds.Exclamation.Play(); return ErrorWindow.ShowErrorDialog(friendlyMessage, exception, true); } /// /// Displays a dialog showing the details of an error. /// /// User-friendly message about the error. /// The associated with the dialog. internal static DialogResult Error(params string[] lines) { System.Media.SystemSounds.Exclamation.Play(); string msg = string.Join(Environment.NewLine + Environment.NewLine, lines); return MessageBox.Show(msg, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error); } internal static DialogResult Alert(params string[] lines) { System.Media.SystemSounds.Asterisk.Play(); string msg = string.Join(Environment.NewLine + Environment.NewLine, lines); return MessageBox.Show(msg, "Alert", MessageBoxButtons.OK, MessageBoxIcon.Information); } internal static DialogResult Prompt(MessageBoxButtons btn, params string[] lines) { System.Media.SystemSounds.Question.Play(); string msg = string.Join(Environment.NewLine + Environment.NewLine, lines); return MessageBox.Show(msg, "Prompt", btn, MessageBoxIcon.Asterisk); } #endregion /// /// Gets the selected value of the input . If no value is selected, will return 0. /// /// ComboBox to retrieve value for. internal static int GetIndex(ComboBox cb) { return (int)(cb?.SelectedValue ?? 0); } public static void PanelScroll(object sender, ScrollEventArgs e) { if (!(sender is Panel p) || e.NewValue < 0) return; switch (e.ScrollOrientation) { case ScrollOrientation.HorizontalScroll: p.HorizontalScroll.Value = Clamp(e.NewValue, p.HorizontalScroll); break; case ScrollOrientation.VerticalScroll: p.VerticalScroll.Value = Clamp(e.NewValue, p.VerticalScroll); break; } int Clamp(int value, ScrollProperties prop) => Math.Max(prop.Minimum, Math.Min(prop.Maximum, value)); } public static void DoubleBuffered(this DataGridView dgv, bool setting) { Type dgvType = dgv.GetType(); PropertyInfo pi = dgvType.GetProperty("DoubleBuffered", BindingFlags.Instance | BindingFlags.NonPublic); pi.SetValue(dgv, setting, null); } /// /// Initializes the to be bound to a provided list. /// /// Control to initialize binding public static void InitializeBinding(this ListControl control) { control.DisplayMember = nameof(ComboItem.Text); control.ValueMember = nameof(ComboItem.Value); } public static void RemoveDropCB(object sender, KeyEventArgs e) => ((ComboBox)sender).DroppedDown = false; /// /// Iterates the Control's child controls recursively to obtain all controls of the specified type. /// /// Type of control /// /// All children and subchildren contained by . public static IEnumerable GetAllControlsOfType(Control control) where T : Control { foreach (var c in control.Controls.Cast()) { if (c is T match) yield return match; foreach (var sub in GetAllControlsOfType(c)) yield return sub; } } /// /// Reads in custom extension types that allow the program to open more extensions. /// /// Extensions to add public static void AddSaveFileExtensions(IEnumerable exts) => CustomSaveExtensions.AddRange(exts.Except(CustomSaveExtensions)); private static readonly List CustomSaveExtensions = new List { // THESE ARE SAVE FILE EXTENSION TYPES. SAVE STATE (RAM SNAPSHOT) EXTENSIONS DO NOT GO HERE. "sav", // standard "dat", // VC data "gci", // Dolphin GameCubeImage "dsv", // DeSmuME "srm", // RetroArch save files "fla", // flashcard "SaveRAM", // BizHawk }; private static string ExtraSaveExtensions => ";" + string.Join(";", CustomSaveExtensions.Select(z => $"*.{z}")); /// /// Opens a dialog to open a , file, or any other supported file. /// /// Misc extensions of files supported by the SAV. /// Output result path /// Result of whether or not a file is to be loaded from the output path. public static bool OpenSAVPKMDialog(IEnumerable Extensions, out string path) { string supported = string.Join(";", Extensions.Select(s => $"*.{s}").Concat(new[] { "*.pkm" })); OpenFileDialog ofd = new OpenFileDialog { Filter = "All Files|*.*" + $"|Supported Files (*.*)|main;*.bin;{supported};*.bak" + ExtraSaveExtensions + "|Save Files (*.sav)|main" + ExtraSaveExtensions + "|Decrypted PKM File (*.pkm)|" + supported + "|Binary File|*.bin" + "|Backup File|*.bak" }; // Detect main string msg = null; var sav = SaveDetection.DetectSaveFile(Environment.GetLogicalDrives(), ref msg); if (sav == null && !string.IsNullOrWhiteSpace(msg)) Error(msg); if (sav != null) ofd.FileName = sav.FileName; if (ofd.ShowDialog() != DialogResult.OK) { path = null; return false; } path = ofd.FileName; return true; } /// /// Opens a dialog to save a file. /// /// file to be saved. /// Result of whether or not the file was saved. public static bool SavePKMDialog(PKM pk) { string pkx = pk.Extension; bool allowEncrypted = pk.Format > 3 || pk is PK3; SaveFileDialog sfd = new SaveFileDialog { Filter = $"Decrypted PKM File|*.{pkx}" + (allowEncrypted ? $"|Encrypted PKM File|*.e{pkx.Substring(1)}" : "") + "|Binary File|*.bin" + "|All Files|*.*", DefaultExt = pkx, FileName = Util.CleanFileName(pk.FileName) }; if (sfd.ShowDialog() != DialogResult.OK) return false; SavePKM(pk, sfd.FileName, pkx); return true; } private static void SavePKM(PKM pk, string path, string pkx) { SaveBackup(path); string ext = Path.GetExtension(path); var data = $".{pkx}" == ext ? pk.DecryptedBoxData : pk.EncryptedPartyData; File.WriteAllBytes(path, data); } private static void SaveBackup(string path) { if (!File.Exists(path)) return; // File already exists, save a .bak string bakpath = $"{path}.bak"; if (!File.Exists(bakpath)) File.Move(path, bakpath); } /// /// Opens a dialog to save a file. /// /// to be saved. /// Box the player will be greeted with when accessing the PC ingame. /// Result of whether or not the file was saved. public static bool SaveSAVDialog(SaveFile SAV, int CurrentBox = 0) { // Chunk Error Checking string err = SAV.MiscSaveChecks(); if (err.Length > 0 && Prompt(MessageBoxButtons.YesNo, err, MsgSaveExportContinue) != DialogResult.Yes) return false; SaveFileDialog main = new SaveFileDialog { Filter = SAV.Filter, FileName = SAV.FileName, FilterIndex = 1000, // default to last, All Files RestoreDirectory = true }; if (Directory.Exists(SAV.FileFolder)) main.InitialDirectory = SAV.FileFolder; // Export if (main.ShowDialog() != DialogResult.OK) return false; if (SAV.HasBox) SAV.CurrentBox = CurrentBox; var ext = Path.GetExtension(main.FileName)?.ToLower(); bool dsv = ext == ".dsv"; bool gci = ext == ".gci"; try { File.WriteAllBytes(main.FileName, SAV.Write(dsv, gci)); SAV.Edited = false; Alert(MsgSaveExportSuccessPath, main.FileName); } catch (Exception x) { if (x is UnauthorizedAccessException || x is FileNotFoundException || x is IOException) Error(MsgFileWriteFail + Environment.NewLine + x.Message, MsgFileWriteProtectedAdvice); else throw; } return true; } /// /// Opens a dialog to save a file. /// /// to be saved. /// Game the gift originates from /// Result of whether or not the file was saved. public static bool SaveMGDialog(MysteryGift gift, GameVersion origin) { SaveFileDialog output = new SaveFileDialog { Filter = GetMysterGiftFilter(gift.Format, origin), FileName = Util.CleanFileName(gift.FileName) }; if (output.ShowDialog() != DialogResult.OK) return false; string path = output.FileName; if (File.Exists(path)) { // File already exists, save a .bak string bakpath = $"{path}.bak"; if (!File.Exists(bakpath)) File.Move(path, bakpath); } File.WriteAllBytes(path, gift.Data); return true; } /// /// Gets the File Dialog filter for a Mystery Gift I/O operation. /// /// Format specifier for the /// Game the format originated from/to public static string GetMysterGiftFilter(int Format, GameVersion origin) { const string all = "|All Files|*.*"; switch (Format) { case 4: return "Gen4 Mystery Gift|*.pgt;*.pcd;*.wc4" + all; case 5: return "Gen5 Mystery Gift|*.pgf" + all; case 6: return "Gen6 Mystery Gift|*.wc6;*.wc6full" + all; case 7: return GameVersion.GG.Contains(origin) ? "Beluga Gift Record|*.wr7" + all : "Gen7 Mystery Gift|*.wc7;*.wc7full" + all; default: return string.Empty; } } } }