2022-06-26 06:08:28 +00:00
|
|
|
using System;
|
2018-02-27 05:22:35 +00:00
|
|
|
using System.Drawing;
|
|
|
|
using System.Windows.Forms;
|
|
|
|
using PKHeX.Core;
|
2019-09-29 16:47:06 +00:00
|
|
|
using PKHeX.Drawing;
|
2022-11-25 01:42:17 +00:00
|
|
|
using PKHeX.Drawing.Misc;
|
|
|
|
using PKHeX.Drawing.PokeSprite;
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
namespace PKHeX.WinForms.Controls;
|
|
|
|
|
|
|
|
public partial class StatEditor : UserControl
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
public StatEditor()
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
InitializeComponent();
|
2023-12-04 04:13:20 +00:00
|
|
|
MT_IVs = [TB_IVHP, TB_IVATK, TB_IVDEF, TB_IVSPE, TB_IVSPA, TB_IVSPD];
|
|
|
|
MT_EVs = [TB_EVHP, TB_EVATK, TB_EVDEF, TB_EVSPE, TB_EVSPA, TB_EVSPD];
|
|
|
|
MT_AVs = [TB_AVHP, TB_AVATK, TB_AVDEF, TB_AVSPE, TB_AVSPA, TB_AVSPD];
|
|
|
|
MT_GVs = [TB_GVHP, TB_GVATK, TB_GVDEF, TB_GVSPE, TB_GVSPA, TB_GVSPD];
|
|
|
|
MT_Stats = [Stat_HP, Stat_ATK, Stat_DEF, Stat_SPE, Stat_SPA, Stat_SPD];
|
|
|
|
L_Stats = [Label_HP, Label_ATK, Label_DEF, Label_SPE, Label_SPA, Label_SPD];
|
|
|
|
MT_Base = [TB_BaseHP, TB_BaseATK, TB_BaseDEF, TB_BaseSPE, TB_BaseSPA, TB_BaseSPD];
|
2022-06-18 18:04:24 +00:00
|
|
|
|
|
|
|
TB_BST.ResetForeColor();
|
|
|
|
TB_IVTotal.ForeColor = TB_EVTotal.ForeColor = MT_EVs[0].ForeColor;
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
2018-07-14 20:17:56 +00:00
|
|
|
|
2023-12-04 04:13:20 +00:00
|
|
|
public PKMEditor MainEditor { private get; set; } = null!;
|
2022-06-18 18:04:24 +00:00
|
|
|
public bool HaX { get => CHK_HackedStats.Enabled; set => CHK_HackedStats.Enabled = CHK_HackedStats.Visible = value; }
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2023-01-22 04:02:33 +00:00
|
|
|
private readonly ToolTip EVTip = new();
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public bool Valid
|
|
|
|
{
|
|
|
|
get
|
2018-11-14 03:21:14 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
2018-11-14 03:21:14 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private bool ChangingFields
|
|
|
|
{
|
|
|
|
get => MainEditor.ChangingFields;
|
|
|
|
set => MainEditor.ChangingFields = value;
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void ClickIV(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is not MaskedTextBox t)
|
|
|
|
return;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
switch (ModifierKeys)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
case Keys.Alt: // Min
|
|
|
|
t.Text = 0.ToString();
|
|
|
|
break;
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
case Keys.Control: // Max
|
2023-04-22 03:47:04 +00:00
|
|
|
{
|
|
|
|
var index = Array.IndexOf(MT_IVs, t);
|
|
|
|
t.Text = Entity.GetMaximumIV(index, true).ToString();
|
|
|
|
break;
|
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
|
|
|
|
case Keys.Shift when Entity is IHyperTrain h: // HT
|
2023-04-22 03:47:04 +00:00
|
|
|
{
|
|
|
|
var index = Array.IndexOf(MT_IVs, t);
|
|
|
|
bool flag = h.HyperTrainInvert(index);
|
|
|
|
UpdateHyperTrainingFlag(index, flag);
|
|
|
|
UpdateStats();
|
|
|
|
break;
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void ClickEV(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is not MaskedTextBox t)
|
|
|
|
return;
|
2018-11-14 03:21:14 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
if ((ModifierKeys & Keys.Control) != 0) // Max
|
|
|
|
{
|
|
|
|
int index = Array.IndexOf(MT_EVs, t);
|
|
|
|
int newEV = Entity.GetMaximumEV(index);
|
|
|
|
t.Text = newEV.ToString();
|
2018-11-14 03:21:14 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
else if ((ModifierKeys & Keys.Alt) != 0) // Min
|
2022-02-05 01:36:51 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
t.Text = 0.ToString();
|
|
|
|
}
|
|
|
|
}
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void ClickAV(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is not MaskedTextBox t)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ((ModifierKeys & Keys.Control) != 0) // Max
|
|
|
|
{
|
2023-04-21 04:23:15 +00:00
|
|
|
var max = AwakeningUtil.AwakeningMax.ToString();
|
2022-06-18 18:04:24 +00:00
|
|
|
t.Text = t.Text == max ? 0.ToString() : max;
|
|
|
|
}
|
|
|
|
else if ((ModifierKeys & Keys.Alt) != 0) // Min
|
|
|
|
{
|
|
|
|
t.Text = 0.ToString();
|
2022-02-05 01:36:51 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
|
|
|
private void ClickGV(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is not MaskedTextBox t || Entity is not IGanbaru g)
|
|
|
|
return;
|
2018-11-14 03:21:14 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
if ((ModifierKeys & Keys.Control) != 0) // Max
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
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();
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void UpdateIVs(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is MaskedTextBox m)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
int value = Util.ToInt32(m.Text);
|
|
|
|
if (value > Entity.MaxIV)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
m.Text = Entity.MaxIV.ToString();
|
|
|
|
return; // recursive on text set
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
int index = Array.IndexOf(MT_IVs, m);
|
|
|
|
Entity.SetIV(index, value);
|
|
|
|
if (Entity is IGanbaru g)
|
|
|
|
RefreshGanbaru(Entity, g, index);
|
|
|
|
}
|
|
|
|
RefreshDerivedValues(e);
|
|
|
|
UpdateStats();
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void RefreshDerivedValues(object _)
|
|
|
|
{
|
|
|
|
if (Entity.Format < 3)
|
|
|
|
{
|
|
|
|
TB_IVHP.Text = Entity.IV_HP.ToString();
|
|
|
|
TB_IVSPD.Text = Entity.IV_SPD.ToString();
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
MainEditor.UpdateIVsGB(false);
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
if (!ChangingFields)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
ChangingFields = true;
|
|
|
|
CB_HPType.SelectedValue = Entity.HPType;
|
|
|
|
Label_HiddenPowerPower.Text = Entity.HPPower.ToString();
|
|
|
|
ChangingFields = false;
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
// Potential Reading
|
|
|
|
L_Potential.Text = Entity.GetPotentialString(MainEditor.Unicode);
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
TB_IVTotal.Text = Entity.IVTotal.ToString();
|
|
|
|
UpdateCharacteristic(Entity.Characteristic);
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateEVs(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (sender is MaskedTextBox m)
|
2018-11-14 03:21:14 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
int value = Util.ToInt32(m.Text);
|
|
|
|
if (value > Entity.MaxEV)
|
2018-11-14 03:21:14 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
m.Text = Entity.MaxEV.ToString();
|
|
|
|
return; // recursive on text set
|
2018-11-14 03:21:14 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
int index = Array.IndexOf(MT_EVs, m);
|
|
|
|
Entity.SetEV(index, value);
|
2018-11-14 03:21:14 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
UpdateEVTotals();
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Entity.Format < 3)
|
|
|
|
{
|
|
|
|
ChangingFields = true;
|
|
|
|
TB_EVSPD.Text = TB_EVSPA.Text;
|
|
|
|
ChangingFields = false;
|
2022-02-05 01:36:51 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
UpdateStats();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void UpdateAVs(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (Entity is not IAwakened a)
|
|
|
|
return;
|
|
|
|
if (sender is MaskedTextBox m)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
var value = (byte)Math.Min(byte.MaxValue, Util.ToInt32(m.Text));
|
2023-04-21 04:23:15 +00:00
|
|
|
if (value > AwakeningUtil.AwakeningMax)
|
2021-01-21 06:09:25 +00:00
|
|
|
{
|
2023-04-21 04:23:15 +00:00
|
|
|
m.Text = AwakeningUtil.AwakeningMax.ToString();
|
2022-06-18 18:04:24 +00:00
|
|
|
return; // recursive on text set
|
2021-01-21 06:09:25 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
int index = Array.IndexOf(MT_AVs, m);
|
|
|
|
a.SetAV(index, value);
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
UpdateAVTotals();
|
|
|
|
UpdateStats();
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateGVs(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (Entity is not IGanbaru g)
|
|
|
|
return;
|
|
|
|
if (sender is MaskedTextBox m)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
int value = Util.ToInt32(m.Text);
|
|
|
|
if (value > GanbaruExtensions.TrueMax)
|
|
|
|
{
|
|
|
|
m.Text = GanbaruExtensions.TrueMax.ToString();
|
|
|
|
return; // recursive on text set
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
int index = Array.IndexOf(MT_GVs, m);
|
|
|
|
g.SetGV(index, (byte)value);
|
|
|
|
RefreshGanbaru(Entity, g, index);
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
UpdateStats();
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateRandomEVs(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
Span<int> 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);
|
|
|
|
}
|
2022-01-26 00:45:41 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateHackedStats(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
foreach (var s in MT_Stats)
|
|
|
|
s.Enabled = CHK_HackedStats.Checked;
|
|
|
|
if (!CHK_HackedStats.Checked)
|
|
|
|
UpdateStats();
|
|
|
|
}
|
2022-05-07 03:38:55 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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";
|
|
|
|
}
|
2022-01-26 00:45:41 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateHyperTrainingFlag(int index, bool value)
|
|
|
|
{
|
|
|
|
var tb = MT_IVs[index];
|
|
|
|
if (value)
|
|
|
|
tb.BackColor = StatHyperTrained;
|
|
|
|
else
|
|
|
|
tb.ResetBackColor();
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateHPType(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (ChangingFields)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Change IVs to match the new Hidden Power
|
2022-06-26 06:08:28 +00:00
|
|
|
Span<int> ivs = stackalloc int[6];
|
|
|
|
Entity.GetIVs(ivs);
|
2022-06-18 18:04:24 +00:00
|
|
|
int hpower = WinFormsUtil.GetIndex(CB_HPType);
|
|
|
|
if (Main.Settings.EntityEditor.HiddenPowerOnChangeMaxPower)
|
2022-06-26 06:08:28 +00:00
|
|
|
ivs.Fill(Entity.MaxIV);
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
HiddenPower.SetIVs(hpower, ivs, Entity.Context);
|
2022-06-18 18:04:24 +00:00
|
|
|
LoadIVs(ivs);
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void ClickStatLabel(object sender, MouseEventArgs e)
|
|
|
|
{
|
|
|
|
if (Entity.Format < 3)
|
|
|
|
return;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
if (ModifierKeys == Keys.None)
|
|
|
|
return;
|
2018-11-14 03:21:14 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
int index = Array.IndexOf(L_Stats, sender as Label) - 1;
|
|
|
|
if (index < 0)
|
|
|
|
return;
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
var request = ModifierKeys switch
|
2018-08-26 00:04:01 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
Keys.Control => NatureAmpRequest.Neutral,
|
|
|
|
Keys.Alt => NatureAmpRequest.Decrease,
|
|
|
|
_ => NatureAmpRequest.Increase,
|
2021-01-05 01:31:43 +00:00
|
|
|
};
|
2018-08-26 00:04:01 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
var newNature = request.GetNewNature(index, Entity.StatNature);
|
|
|
|
if (newNature == -1)
|
|
|
|
return;
|
2018-03-18 17:38:29 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
MainEditor.ChangeNature(newNature);
|
|
|
|
}
|
2018-03-18 17:38:29 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void LoadHyperTraining()
|
|
|
|
{
|
|
|
|
if (Entity is not IHyperTrain h)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
foreach (var iv in MT_IVs)
|
|
|
|
iv.ResetBackColor();
|
|
|
|
return;
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
for (int i = 0; i < MT_IVs.Length; i++)
|
|
|
|
UpdateHyperTrainingFlag(i, h.IsHyperTrained(i));
|
|
|
|
}
|
2018-11-16 22:21:29 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateAVTotals()
|
|
|
|
{
|
|
|
|
if (Entity is not IAwakened a)
|
|
|
|
return;
|
|
|
|
var total = a.AwakeningSum();
|
|
|
|
TB_AVTotal.Text = total.ToString();
|
|
|
|
}
|
2018-11-16 22:21:29 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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}");
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2023-09-28 05:15:59 +00:00
|
|
|
private Color GetEVTotalColor(int evtotal, Color defaultColor) => EffortValues.GetGrade(evtotal) switch
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
2023-09-28 05:15:59 +00:00
|
|
|
EffortValueGrade.Illegal => EVsInvalid, // Background turns Red
|
|
|
|
EffortValueGrade.MaxLegal => EVsMaxed, // Maximum EVs
|
|
|
|
EffortValueGrade.MaxEffective => EVsFishy, // Fishy EVs
|
2022-06-18 18:04:24 +00:00
|
|
|
_ => defaultColor,
|
|
|
|
};
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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);
|
2023-01-22 06:01:30 +00:00
|
|
|
Span<ushort> stats = stackalloc ushort[6];
|
|
|
|
Entity.LoadStats(pi, stats);
|
|
|
|
Entity.SetStats(stats);
|
2022-06-18 18:04:24 +00:00
|
|
|
LoadBST(pi);
|
|
|
|
LoadPartyStats(Entity);
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2022-11-25 01:42:17 +00:00
|
|
|
if (Entity is ITeraType)
|
|
|
|
{
|
|
|
|
var pi = Entity.PersonalInfo;
|
|
|
|
PB_TeraType1.SetType(pi.Type1);
|
|
|
|
PB_TeraType2.SetType(pi.Type2);
|
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
private void LoadBST(IBaseStat pi)
|
2022-06-18 18:04:24 +00:00
|
|
|
{
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
int bst = 0;
|
|
|
|
for (int index = 0; index < 6; index++)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
var value = pi.GetBaseStatValue(index);
|
|
|
|
var s = MT_Base[index];
|
|
|
|
s.Text = value.ToString("000");
|
|
|
|
s.BackColor = ColorUtil.ColorBaseStat(value);
|
|
|
|
bst += value;
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
Refactoring: Move Source (Legality) (#3560)
Rewrites a good amount of legality APIs pertaining to:
* Legal moves that can be learned
* Evolution chains & cross-generation paths
* Memory validation with forgotten moves
In generation 8, there are 3 separate contexts an entity can exist in: SW/SH, BD/SP, and LA. Not every entity can cross between them, and not every entity from generation 7 can exist in generation 8 (Gogoat, etc). By creating class models representing the restrictions to cross each boundary, we are able to better track and validate data.
The old implementation of validating moves was greedy: it would iterate for all generations and evolutions, and build a full list of every move that can be learned, storing it on the heap. Now, we check one game group at a time to see if the entity can learn a move that hasn't yet been validated. End result is an algorithm that requires 0 allocation, and a smaller/quicker search space.
The old implementation of storing move parses was inefficient; for each move that was parsed, a new object is created and adjusted depending on the parse. Now, move parse results are `struct` and store the move parse contiguously in memory. End result is faster parsing and 0 memory allocation.
* `PersonalTable` objects have been improved with new API methods to check if a species+form can exist in the game.
* `IEncounterTemplate` objects have been improved to indicate the `EntityContext` they originate in (similar to `Generation`).
* Some APIs have been extended to accept `Span<T>` instead of Array/IEnumerable
2022-08-03 23:15:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
TB_BST.Text = bst.ToString("000");
|
|
|
|
TB_BST.BackColor = ColorUtil.ColorBaseStatTotal(bst);
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void UpdateRandomIVs(object sender, EventArgs e)
|
|
|
|
{
|
2022-06-26 06:08:28 +00:00
|
|
|
Span<int> ivs = stackalloc int[6];
|
|
|
|
if (ModifierKeys == Keys.Control)
|
2023-10-29 05:02:12 +00:00
|
|
|
{
|
2022-06-26 06:08:28 +00:00
|
|
|
ivs.Fill(Entity.MaxIV);
|
2023-10-29 05:02:12 +00:00
|
|
|
}
|
2022-06-26 06:08:28 +00:00
|
|
|
else if (ModifierKeys == Keys.Alt)
|
2023-10-29 05:02:12 +00:00
|
|
|
{
|
2023-01-22 04:02:33 +00:00
|
|
|
ivs.Clear();
|
2023-10-29 05:02:12 +00:00
|
|
|
}
|
2022-06-26 06:08:28 +00:00
|
|
|
else
|
2023-10-29 05:02:12 +00:00
|
|
|
{
|
|
|
|
var pk = Entity;
|
|
|
|
var la = new LegalityAnalysis(pk);
|
|
|
|
var enc = la.EncounterMatch;
|
|
|
|
if (enc is IFlawlessIVCount { FlawlessIVCount: not 0 } fc)
|
|
|
|
pk.SetRandomIVs(ivs, fc.FlawlessIVCount);
|
|
|
|
else if (enc is IFixedIVSet { IVs: { IsSpecified: true } iv })
|
|
|
|
pk.SetRandomIVs(ivs, iv);
|
2023-10-30 03:27:42 +00:00
|
|
|
else if (enc is IFlawlessIVCountConditional c && c.GetFlawlessIVCount(pk) is { Max: not 0 } x)
|
|
|
|
pk.SetRandomIVs(ivs, Util.Rand.Next(x.Min, x.Max + 1));
|
2023-10-29 05:02:12 +00:00
|
|
|
else
|
|
|
|
pk.SetRandomIVs(ivs);
|
|
|
|
}
|
2022-06-26 06:08:28 +00:00
|
|
|
|
|
|
|
LoadIVs(ivs);
|
2022-06-18 18:04:24 +00:00
|
|
|
if (Entity is IGanbaru g)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-26 06:08:28 +00:00
|
|
|
Entity.SetIVs(ivs);
|
2022-06-18 18:04:24 +00:00
|
|
|
if (ModifierKeys == Keys.Control)
|
|
|
|
g.SetSuggestedGanbaruValues(Entity);
|
|
|
|
else if (ModifierKeys == Keys.Alt)
|
|
|
|
g.ClearGanbaruValues();
|
|
|
|
LoadGVs(g);
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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:
|
2023-01-22 04:02:33 +00:00
|
|
|
a.AwakeningMinimum(); // will still set AVs by level gain
|
2022-06-18 18:04:24 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
a.AwakeningSetRandom();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
LoadAVs(a);
|
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void UpdateCharacteristic() => UpdateCharacteristic(Entity.Characteristic);
|
2018-11-14 03:21:14 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void UpdateCharacteristic(int characteristic)
|
|
|
|
{
|
|
|
|
L_Characteristic.Visible = Label_CharacteristicPrefix.Visible = characteristic > -1;
|
|
|
|
if (characteristic > -1)
|
|
|
|
L_Characteristic.Text = GameInfo.Strings.characteristics[characteristic];
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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 "-/-";
|
|
|
|
|
2023-02-01 02:31:40 +00:00
|
|
|
var incr = L_Stats[up + 1];
|
|
|
|
var decr = L_Stats[dn + 1];
|
2022-06-18 18:04:24 +00:00
|
|
|
incr.ForeColor = StatIncreased;
|
|
|
|
decr.ForeColor = StatDecreased;
|
|
|
|
return $"+{incr.Text} / -{decr.Text}".Replace(":", "");
|
|
|
|
}
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void SetATKIVGender(int gender)
|
|
|
|
{
|
|
|
|
Entity.SetAttackIVFromGender(gender);
|
|
|
|
TB_IVATK.Text = Entity.IV_ATK.ToString();
|
|
|
|
}
|
2022-04-25 03:19:50 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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();
|
|
|
|
}
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void LoadEVs(ReadOnlySpan<int> 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();
|
|
|
|
}
|
2018-11-14 03:21:14 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void LoadIVs(ReadOnlySpan<int> 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;
|
2023-04-22 03:47:04 +00:00
|
|
|
TB_GVHP.Text = a.GV_HP.ToString();
|
2022-06-18 18:04:24 +00:00
|
|
|
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();
|
|
|
|
}
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
2022-11-25 01:42:17 +00:00
|
|
|
FLP_TeraType.Visible = FLP_TeraInner.Visible = pk is ITeraType;
|
2022-06-18 18:04:24 +00:00
|
|
|
Label_HiddenPowerPower.Visible = gen <= 5;
|
2022-11-25 01:42:17 +00:00
|
|
|
FLP_DynamaxLevel.Visible = gen == 8;
|
2022-06-18 18:04:24 +00:00
|
|
|
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;
|
|
|
|
|
2023-07-09 02:29:46 +00:00
|
|
|
FLP_PKMEditors.PerformLayout();
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
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)
|
2018-02-27 05:22:35 +00:00
|
|
|
{
|
2022-06-18 18:04:24 +00:00
|
|
|
ctrl.Size = s;
|
|
|
|
ctrl.Mask = Mask;
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
|
|
|
}
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2018-07-27 02:34:27 +00:00
|
|
|
|
2022-11-25 01:42:17 +00:00
|
|
|
private const string TeraOverrideNone = "---";
|
|
|
|
private const byte TeraOverrideNoneValue = TeraTypeUtil.OverrideNone;
|
|
|
|
|
|
|
|
private void L_TeraTypeOriginal_Click(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
var pi = Entity.PersonalInfo;
|
2023-07-06 04:14:09 +00:00
|
|
|
if (!Entity.SV)
|
|
|
|
{
|
|
|
|
var expect = TeraTypeUtil.GetTeraTypeImport(pi.Type1, pi.Type2);
|
|
|
|
SetOriginalTeraType((byte)expect);
|
|
|
|
return;
|
|
|
|
}
|
2022-11-25 01:42:17 +00:00
|
|
|
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);
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
public void InitializeDataSources()
|
|
|
|
{
|
|
|
|
ChangingFields = true;
|
2022-11-25 01:42:17 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
CB_HPType.InitializeBinding();
|
2022-11-25 01:42:17 +00:00
|
|
|
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);
|
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
ChangingFields = false;
|
|
|
|
}
|
2020-10-25 20:47:42 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void CHK_Gigantamax_CheckedChanged(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (!ChangingFields)
|
2023-12-04 04:13:20 +00:00
|
|
|
MainEditor.UpdateSprite();
|
2022-06-18 18:04:24 +00:00
|
|
|
}
|
2022-02-05 01:36:51 +00:00
|
|
|
|
2022-06-18 18:04:24 +00:00
|
|
|
private void CHK_IsAlpha_CheckedChanged(object sender, EventArgs e)
|
|
|
|
{
|
|
|
|
if (!ChangingFields)
|
2023-12-04 04:13:20 +00:00
|
|
|
MainEditor.UpdateSprite();
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|
2022-11-25 01:42:17 +00:00
|
|
|
|
2023-07-06 04:14:09 +00:00
|
|
|
private void L_TeraTypeOverride_Click(object sender, EventArgs e) => CB_TeraTypeOverride.SelectedValue = Entity.SV ? (int)TeraOverrideNoneValue : CB_TeraTypeOriginal.SelectedValue;
|
2022-11-25 01:42:17 +00:00
|
|
|
|
|
|
|
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)
|
2023-12-04 04:13:20 +00:00
|
|
|
MainEditor.UpdateSprite();
|
2022-11-25 01:42:17 +00:00
|
|
|
}
|
2023-02-02 00:55:38 +00:00
|
|
|
|
|
|
|
public void CenterSubEditors()
|
|
|
|
{
|
|
|
|
FLP_PKMEditors.HorizontallyCenter(this);
|
|
|
|
}
|
2022-11-25 01:42:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
2018-02-27 05:22:35 +00:00
|
|
|
}
|