using System; using System.Drawing; using System.Windows.Forms; using PKHeX.Core; using PKHeX.Drawing; using PKHeX.Drawing.Misc; using PKHeX.Drawing.PokeSprite; namespace PKHeX.WinForms.Controls; public partial class StatEditor : UserControl { public StatEditor() { InitializeComponent(); MT_IVs = new[] { TB_IVHP, TB_IVATK, TB_IVDEF, TB_IVSPE, TB_IVSPA, TB_IVSPD }; MT_EVs = new[] { TB_EVHP, TB_EVATK, TB_EVDEF, TB_EVSPE, TB_EVSPA, TB_EVSPD }; MT_AVs = new[] { TB_AVHP, TB_AVATK, TB_AVDEF, TB_AVSPE, TB_AVSPA, TB_AVSPD }; MT_GVs = new[] { TB_GVHP, TB_GVATK, TB_GVDEF, TB_GVSPE, TB_GVSPA, TB_GVSPD }; MT_Stats = new[] { Stat_HP, Stat_ATK, Stat_DEF, Stat_SPE, Stat_SPA, Stat_SPD }; L_Stats = new[] { Label_HP, Label_ATK, Label_DEF, Label_SPE, Label_SPA, Label_SPD }; MT_Base = new[] { TB_BaseHP, TB_BaseATK, TB_BaseDEF, TB_BaseSPE, TB_BaseSPA, TB_BaseSPD }; TB_BST.ResetForeColor(); TB_IVTotal.ForeColor = TB_EVTotal.ForeColor = MT_EVs[0].ForeColor; } public Color EVsInvalid { get; set; } = Color.Red; public Color EVsMaxed { get; set; } = Color.Honeydew; public Color EVsFishy { get; set; } = Color.LightYellow; public Color StatIncreased { get; set; } = Color.Red; public Color StatDecreased { get; set; } = Color.Blue; public Color StatHyperTrained { get; set; } = Color.LightGreen; public IMainEditor MainEditor { private get; set; } = null!; public bool HaX { get => CHK_HackedStats.Enabled; set => CHK_HackedStats.Enabled = CHK_HackedStats.Visible = value; } private readonly ToolTip EVTip = new(); public bool Valid { get { if (Entity.Format < 3) return true; if (CHK_HackedStats.Checked) return true; if (Entity is IAwakened a) return a.AwakeningAllValid(); return Convert.ToUInt32(TB_EVTotal.Text) <= 510; } } private readonly Label[] L_Stats; private readonly MaskedTextBox[] MT_EVs, MT_IVs, MT_AVs, MT_GVs, MT_Stats, MT_Base; private PKM Entity => MainEditor.Entity; private bool ChangingFields { get => MainEditor.ChangingFields; set => MainEditor.ChangingFields = value; } private void ClickIV(object sender, EventArgs e) { if (sender is not MaskedTextBox t) return; switch (ModifierKeys) { case Keys.Alt: // Min t.Text = 0.ToString(); break; case Keys.Control: // Max { var index = Array.IndexOf(MT_IVs, t); t.Text = Entity.GetMaximumIV(index, true).ToString(); break; } case Keys.Shift when Entity is IHyperTrain h: // HT { var index = Array.IndexOf(MT_IVs, t); bool flag = h.HyperTrainInvert(index); UpdateHyperTrainingFlag(index, flag); UpdateStats(); break; } } } private void ClickEV(object sender, EventArgs e) { if (sender is not MaskedTextBox t) return; if ((ModifierKeys & Keys.Control) != 0) // Max { int index = Array.IndexOf(MT_EVs, t); int newEV = Entity.GetMaximumEV(index); t.Text = newEV.ToString(); } else if ((ModifierKeys & Keys.Alt) != 0) // Min { t.Text = 0.ToString(); } } private void ClickAV(object sender, EventArgs e) { if (sender is not MaskedTextBox t) return; if ((ModifierKeys & Keys.Control) != 0) // Max { var max = AwakeningUtil.AwakeningMax.ToString(); t.Text = t.Text == max ? 0.ToString() : max; } else if ((ModifierKeys & Keys.Alt) != 0) // Min { t.Text = 0.ToString(); } } private void ClickGV(object sender, EventArgs e) { if (sender is not MaskedTextBox t || Entity is not IGanbaru g) return; if ((ModifierKeys & Keys.Control) != 0) // Max { int index = Array.IndexOf(MT_GVs, t); var max = g.GetMax(Entity, index).ToString(); t.Text = t.Text == max ? 0.ToString() : max; } else if ((ModifierKeys & Keys.Alt) != 0) // Min { t.Text = 0.ToString(); } } public void UpdateIVs(object sender, EventArgs e) { if (sender is MaskedTextBox m) { int value = Util.ToInt32(m.Text); if (value > Entity.MaxIV) { m.Text = Entity.MaxIV.ToString(); return; // recursive on text set } int index = Array.IndexOf(MT_IVs, m); Entity.SetIV(index, value); if (Entity is IGanbaru g) RefreshGanbaru(Entity, g, index); } RefreshDerivedValues(e); UpdateStats(); } private void RefreshDerivedValues(object _) { if (Entity.Format < 3) { TB_IVHP.Text = Entity.IV_HP.ToString(); TB_IVSPD.Text = Entity.IV_SPD.ToString(); MainEditor.UpdateIVsGB(false); } if (!ChangingFields) { ChangingFields = true; CB_HPType.SelectedValue = Entity.HPType; Label_HiddenPowerPower.Text = Entity.HPPower.ToString(); ChangingFields = false; } // Potential Reading L_Potential.Text = Entity.GetPotentialString(MainEditor.Unicode); TB_IVTotal.Text = Entity.IVTotal.ToString(); UpdateCharacteristic(Entity.Characteristic); } private void UpdateEVs(object sender, EventArgs e) { if (sender is MaskedTextBox m) { int value = Util.ToInt32(m.Text); if (value > Entity.MaxEV) { m.Text = Entity.MaxEV.ToString(); return; // recursive on text set } int index = Array.IndexOf(MT_EVs, m); Entity.SetEV(index, value); } UpdateEVTotals(); if (Entity.Format < 3) { ChangingFields = true; TB_EVSPD.Text = TB_EVSPA.Text; ChangingFields = false; } UpdateStats(); } private void UpdateAVs(object sender, EventArgs e) { if (Entity is not IAwakened a) return; if (sender is MaskedTextBox m) { var value = (byte)Math.Min(byte.MaxValue, Util.ToInt32(m.Text)); if (value > AwakeningUtil.AwakeningMax) { m.Text = AwakeningUtil.AwakeningMax.ToString(); return; // recursive on text set } int index = Array.IndexOf(MT_AVs, m); a.SetAV(index, value); } UpdateAVTotals(); UpdateStats(); } private void UpdateGVs(object sender, EventArgs e) { if (Entity is not IGanbaru g) return; if (sender is MaskedTextBox m) { int value = Util.ToInt32(m.Text); if (value > GanbaruExtensions.TrueMax) { m.Text = GanbaruExtensions.TrueMax.ToString(); return; // recursive on text set } int index = Array.IndexOf(MT_GVs, m); g.SetGV(index, (byte)value); RefreshGanbaru(Entity, g, index); } UpdateStats(); } private void UpdateRandomEVs(object sender, EventArgs e) { Span values = stackalloc int[6]; switch (ModifierKeys) { case Keys.Control: EffortValues.SetMax(values, Entity); break; case Keys.Alt: EffortValues.Clear(values); break; default: EffortValues.SetRandom(values, Entity.Format); break; } LoadEVs(values); UpdateEVs(sender, EventArgs.Empty); } private void UpdateHackedStats(object sender, EventArgs e) { foreach (var s in MT_Stats) s.Enabled = CHK_HackedStats.Checked; if (!CHK_HackedStats.Checked) UpdateStats(); } private void UpdateHackedStatText(object sender, EventArgs e) { if (!CHK_HackedStats.Checked || sender is not TextBox tb) return; string text = tb.Text; if (string.IsNullOrWhiteSpace(text)) tb.Text = "0"; else if (Convert.ToUInt32(text) > ushort.MaxValue) tb.Text = "65535"; } private void UpdateHyperTrainingFlag(int index, bool value) { var tb = MT_IVs[index]; if (value) tb.BackColor = StatHyperTrained; else tb.ResetBackColor(); } private void UpdateHPType(object sender, EventArgs e) { if (ChangingFields) return; // Change IVs to match the new Hidden Power Span ivs = stackalloc int[6]; Entity.GetIVs(ivs); int hpower = WinFormsUtil.GetIndex(CB_HPType); if (Main.Settings.EntityEditor.HiddenPowerOnChangeMaxPower) ivs.Fill(Entity.MaxIV); HiddenPower.SetIVs(hpower, ivs, Entity.Context); LoadIVs(ivs); } private void ClickStatLabel(object sender, MouseEventArgs e) { if (Entity.Format < 3) return; if (ModifierKeys == Keys.None) return; int index = Array.IndexOf(L_Stats, sender as Label) - 1; if (index < 0) return; var request = ModifierKeys switch { Keys.Control => NatureAmpRequest.Neutral, Keys.Alt => NatureAmpRequest.Decrease, _ => NatureAmpRequest.Increase, }; var newNature = request.GetNewNature(index, Entity.StatNature); if (newNature == -1) return; MainEditor.ChangeNature(newNature); } private void LoadHyperTraining() { if (Entity is not IHyperTrain h) { foreach (var iv in MT_IVs) iv.ResetBackColor(); return; } for (int i = 0; i < MT_IVs.Length; i++) UpdateHyperTrainingFlag(i, h.IsHyperTrained(i)); } private void UpdateAVTotals() { if (Entity is not IAwakened a) return; var total = a.AwakeningSum(); TB_AVTotal.Text = total.ToString(); } private void UpdateEVTotals() { var evtotal = Entity.EVTotal; TB_EVTotal.BackColor = GetEVTotalColor(evtotal, TB_IVTotal.BackColor); TB_EVTotal.Text = evtotal.ToString(); EVTip.SetToolTip(TB_EVTotal, $"Remaining: {510 - evtotal}"); } private Color GetEVTotalColor(int evtotal, Color defaultColor) => evtotal switch { > 510 => EVsInvalid, // Background turns Red 510 => EVsMaxed, // Maximum EVs 508 => EVsFishy, // Fishy EVs _ => defaultColor, }; public void UpdateStats() { // Generate the stats. // Some entity formats don't store stat values regardless of Box/Party/Etc format. // If its attack stat is zero, we need to generate party stats. // PK1 format stores Current HP in the compact format, so we have to use attack stat! if (!CHK_HackedStats.Checked || Entity.Stat_ATK == 0) { var pt = MainEditor.RequestSaveFile.Personal; var pi = pt.GetFormEntry(Entity.Species, Entity.Form); Span stats = stackalloc ushort[6]; Entity.LoadStats(pi, stats); Entity.SetStats(stats); LoadBST(pi); LoadPartyStats(Entity); } if (Entity is ITeraType) { var pi = Entity.PersonalInfo; PB_TeraType1.SetType(pi.Type1); PB_TeraType2.SetType(pi.Type2); } } private void LoadBST(IBaseStat pi) { int bst = 0; for (int index = 0; index < 6; index++) { var value = pi.GetBaseStatValue(index); var s = MT_Base[index]; s.Text = value.ToString("000"); s.BackColor = ColorUtil.ColorBaseStat(value); bst += value; } TB_BST.Text = bst.ToString("000"); TB_BST.BackColor = ColorUtil.ColorBaseStatTotal(bst); } public void UpdateRandomIVs(object sender, EventArgs e) { Span ivs = stackalloc int[6]; if (ModifierKeys == Keys.Control) ivs.Fill(Entity.MaxIV); else if (ModifierKeys == Keys.Alt) ivs.Clear(); else Entity.SetRandomIVs(ivs); LoadIVs(ivs); if (Entity is IGanbaru g) { Entity.SetIVs(ivs); if (ModifierKeys == Keys.Control) g.SetSuggestedGanbaruValues(Entity); else if (ModifierKeys == Keys.Alt) g.ClearGanbaruValues(); LoadGVs(g); } } private void UpdateRandomAVs(object sender, EventArgs e) { if (Entity is not IAwakened a) return; switch (ModifierKeys) { case Keys.Control: a.SetSuggestedAwakenedValues(Entity); break; case Keys.Alt: a.AwakeningMinimum(); // will still set AVs by level gain break; default: a.AwakeningSetRandom(); break; } LoadAVs(a); } public void UpdateCharacteristic() => UpdateCharacteristic(Entity.Characteristic); private void UpdateCharacteristic(int characteristic) { L_Characteristic.Visible = Label_CharacteristicPrefix.Visible = characteristic > -1; if (characteristic > -1) L_Characteristic.Text = GameInfo.Strings.characteristics[characteristic]; } public string UpdateNatureModification(int nature) { // Reset Label Colors foreach (var l in L_Stats) l.ResetForeColor(); // Set Colored StatLabels only if Nature isn't Neutral var (up, dn) = NatureAmp.GetNatureModification(nature); if (NatureAmp.IsNeutralOrInvalid(nature, up, dn)) return "-/-"; var incr = L_Stats[up + 1]; var decr = L_Stats[dn + 1]; incr.ForeColor = StatIncreased; decr.ForeColor = StatDecreased; return $"+{incr.Text} / -{decr.Text}".Replace(":", ""); } public void SetATKIVGender(int gender) { Entity.SetAttackIVFromGender(gender); TB_IVATK.Text = Entity.IV_ATK.ToString(); } public void LoadPartyStats(PKM pk) { Stat_HP.Text = pk.Stat_HPCurrent.ToString(); Stat_ATK.Text = pk.Stat_ATK.ToString(); Stat_DEF.Text = pk.Stat_DEF.ToString(); Stat_SPA.Text = pk.Stat_SPA.ToString(); Stat_SPD.Text = pk.Stat_SPD.ToString(); Stat_SPE.Text = pk.Stat_SPE.ToString(); } public void SavePartyStats(PKM pk) { pk.Stat_HPCurrent = Util.ToInt32(Stat_HP.Text); pk.Stat_HPMax = Util.ToInt32(Stat_HP.Text); pk.Stat_ATK = Util.ToInt32(Stat_ATK.Text); pk.Stat_DEF = Util.ToInt32(Stat_DEF.Text); pk.Stat_SPE = Util.ToInt32(Stat_SPE.Text); pk.Stat_SPA = Util.ToInt32(Stat_SPA.Text); pk.Stat_SPD = Util.ToInt32(Stat_SPD.Text); if (!HaX) pk.Stat_Level = pk.CurrentLevel; } public void LoadEVs(ReadOnlySpan EVs) { ChangingFields = true; TB_EVHP.Text = EVs[0].ToString(); TB_EVATK.Text = EVs[1].ToString(); TB_EVDEF.Text = EVs[2].ToString(); TB_EVSPE.Text = EVs[3].ToString(); TB_EVSPA.Text = EVs[4].ToString(); TB_EVSPD.Text = EVs[5].ToString(); ChangingFields = false; UpdateStats(); } public void LoadIVs(ReadOnlySpan IVs) { ChangingFields = true; TB_IVHP.Text = IVs[0].ToString(); TB_IVATK.Text = IVs[1].ToString(); TB_IVDEF.Text = IVs[2].ToString(); TB_IVSPE.Text = IVs[3].ToString(); TB_IVSPA.Text = IVs[4].ToString(); TB_IVSPD.Text = IVs[5].ToString(); ChangingFields = false; LoadHyperTraining(); RefreshDerivedValues(TB_IVSPD); UpdateStats(); } public void LoadAVs(IAwakened a) { ChangingFields = true; TB_AVHP.Text = a.AV_HP.ToString(); TB_AVATK.Text = a.AV_ATK.ToString(); TB_AVDEF.Text = a.AV_DEF.ToString(); TB_AVSPE.Text = a.AV_SPE.ToString(); TB_AVSPA.Text = a.AV_SPA.ToString(); TB_AVSPD.Text = a.AV_SPD.ToString(); ChangingFields = false; UpdateStats(); } public void LoadGVs(IGanbaru a) { ChangingFields = true; TB_GVHP.Text = a.GV_HP.ToString(); TB_GVATK.Text = a.GV_ATK.ToString(); TB_GVDEF.Text = a.GV_DEF.ToString(); TB_GVSPE.Text = a.GV_SPE.ToString(); TB_GVSPA.Text = a.GV_SPA.ToString(); TB_GVSPD.Text = a.GV_SPD.ToString(); ChangingFields = false; for (int i = 0; i < 6; i++) RefreshGanbaru(Entity, a, i); UpdateStats(); } private void L_DynamaxLevel_Click(object sender, EventArgs e) { var cb = CB_DynamaxLevel; bool isMin = cb.SelectedIndex == 0; cb.SelectedIndex = isMin ? cb.Items.Count - 1 : 0; } private void RefreshGanbaru(PKM entity, IGanbaru ganbaru, int i) { int current = ganbaru.GetGV(i); var max = ganbaru.GetMax(entity, i); var tb = MT_GVs[i]; if (current > max) tb.BackColor = EVsInvalid; else if (current == max) tb.BackColor = StatHyperTrained; else tb.ResetBackColor(); } public void ToggleInterface(PKM pk, int gen) { FLP_StatsTotal.Visible = gen >= 3; FLP_Characteristic.Visible = gen >= 3; FLP_HPType.Visible = gen <= 7 || pk is PB8; FLP_TeraType.Visible = FLP_TeraInner.Visible = pk is ITeraType; Label_HiddenPowerPower.Visible = gen <= 5; FLP_DynamaxLevel.Visible = gen == 8; FLP_AlphaNoble.Visible = pk is PA8; switch (gen) { case 1: FLP_SpD.Visible = false; Label_SPA.Visible = false; Label_SPC.Visible = true; TB_IVHP.Enabled = false; SetEVMaskSize(Stat_HP.Size, "00000", MT_EVs); break; case 2: FLP_SpD.Visible = true; Label_SPA.Visible = true; Label_SPC.Visible = false; TB_IVHP.Enabled = false; SetEVMaskSize(Stat_HP.Size, "00000", MT_EVs); TB_EVSPD.Enabled = TB_IVSPD.Enabled = false; break; default: FLP_SpD.Visible = true; Label_SPA.Visible = true; Label_SPC.Visible = false; TB_IVHP.Enabled = true; SetEVMaskSize(TB_EVTotal.Size, "000", MT_EVs); TB_EVSPD.Enabled = TB_IVSPD.Enabled = true; break; } var showAV = pk is IAwakened; Label_AVs.Visible = TB_AVTotal.Visible = BTN_RandomAVs.Visible = showAV; foreach (var mtb in MT_AVs) mtb.Visible = showAV; Label_EVs.Visible = TB_EVTotal.Visible = BTN_RandomEVs.Visible = !showAV; foreach (var mtb in MT_EVs) mtb.Visible = !showAV; var showGV = pk is IGanbaru; Label_GVs.Visible = showGV; foreach (var mtb in MT_GVs) mtb.Visible = showGV; static void SetEVMaskSize(Size s, string Mask, MaskedTextBox[] arr) { foreach (var ctrl in arr) { ctrl.Size = s; ctrl.Mask = Mask; } } } private const string TeraOverrideNone = "---"; private const byte TeraOverrideNoneValue = TeraTypeUtil.OverrideNone; private void L_TeraTypeOriginal_Click(object sender, EventArgs e) { var pi = Entity.PersonalInfo; var current = WinFormsUtil.GetIndex(CB_TeraTypeOriginal); var update = pi.Type1 == current ? pi.Type2 : pi.Type1; SetOriginalTeraType(update); } private void SetOriginalTeraType(byte value) { CB_TeraTypeOriginal.SelectedValue = (int)value; CB_TeraTypeOverride.SelectedValue = (int)TeraOverrideNoneValue; } private void PB_TeraType1_Click(object sender, EventArgs e) => SetOriginalTeraType(Entity.PersonalInfo.Type1); private void PB_TeraType2_Click(object sender, EventArgs e) => SetOriginalTeraType(Entity.PersonalInfo.Type2); public void InitializeDataSources() { ChangingFields = true; CB_HPType.InitializeBinding(); CB_TeraTypeOriginal.InitializeBinding(); CB_TeraTypeOverride.InitializeBinding(); var types = GameInfo.Strings.types; CB_HPType.DataSource = Util.GetCBList(types.AsSpan(1, 16)); var tera = Util.GetCBList(types); tera.Insert(0, new(TeraOverrideNone, TeraOverrideNoneValue)); CB_TeraTypeOriginal.DataSource = new BindingSource(tera, null); CB_TeraTypeOverride.DataSource = new BindingSource(tera, null); ChangingFields = false; } private void CHK_Gigantamax_CheckedChanged(object sender, EventArgs e) { if (!ChangingFields) ((PKMEditor)MainEditor).UpdateSprite(); } private void CHK_IsAlpha_CheckedChanged(object sender, EventArgs e) { if (!ChangingFields) ((PKMEditor)MainEditor).UpdateSprite(); } private void L_TeraTypeOverride_Click(object sender, EventArgs e) => CB_TeraTypeOverride.SelectedValue = (int)TeraOverrideNoneValue; private void ChangeTeraType(object sender, EventArgs e) { if (ChangingFields && sender == CB_TeraTypeOriginal) return; var original = (byte)WinFormsUtil.GetIndex(CB_TeraTypeOriginal); var update = (byte)WinFormsUtil.GetIndex(CB_TeraTypeOverride); if (!ChangingFields && Entity is ITeraType t) // Store back { if (sender == CB_TeraTypeOriginal) t.TeraTypeOriginal = (MoveType)original; else if (sender == CB_TeraTypeOverride) t.TeraTypeOverride = (MoveType)update; } var type = update; if (type == TeraOverrideNoneValue) type = original; PB_TeraType.Image = TypeSpriteUtil.GetTypeSpriteGem(type); if (!ChangingFields) ((PKMEditor)MainEditor).UpdateSprite(); } public void CenterSubEditors() { FLP_PKMEditors.HorizontallyCenter(this); } } public sealed class TypePictureBox : PictureBox { private byte Type; public void SetType(byte type) => BackColor = TypeColor.GetTypeSpriteColor(Type = type); private readonly ToolTip Tip = new() { InitialDelay = 500, ReshowDelay = 500, ShowAlways = true }; // Show a tooltip when hovered. protected override void OnMouseHover(EventArgs e) { base.OnMouseHover(e); var name = GameInfo.Strings.types[Type]; Tip.SetToolTip(this, name); } }