mirror of
https://github.com/kwsch/PKHeX
synced 2025-01-23 01:35:04 +00:00
9166d0eb64
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
325 lines
10 KiB
C#
325 lines
10 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Drawing;
|
|
using System.Linq;
|
|
using System.Windows.Forms;
|
|
using PKHeX.Core;
|
|
using static PKHeX.Core.MessageStrings;
|
|
|
|
namespace PKHeX.WinForms;
|
|
|
|
public sealed partial class SAV_EventFlags2 : Form
|
|
{
|
|
private readonly EventWorkspace<SAV2, byte> Editor;
|
|
private readonly Dictionary<int, NumericUpDown> WorkDict = new();
|
|
private readonly Dictionary<int, int> FlagDict = new();
|
|
|
|
private bool editing;
|
|
|
|
public SAV_EventFlags2(SAV2 sav)
|
|
{
|
|
InitializeComponent();
|
|
WinFormsUtil.TranslateInterface(this, Main.CurrentLanguage);
|
|
|
|
var editor = Editor = new EventWorkspace<SAV2, byte>(sav);
|
|
DragEnter += Main_DragEnter;
|
|
DragDrop += Main_DragDrop;
|
|
|
|
editing = true;
|
|
CB_Stats.Items.Clear();
|
|
for (int i = 0; i < editor.Values.Length; i++)
|
|
CB_Stats.Items.Add(i.ToString());
|
|
|
|
dgv.SuspendLayout();
|
|
TLP_Const.SuspendLayout();
|
|
TLP_Const.Scroll += WinFormsUtil.PanelScroll;
|
|
TLP_Const.Controls.Clear();
|
|
AddFlagList(editor.Labels, editor.Flags);
|
|
AddConstList(editor.Labels, editor.Values);
|
|
|
|
dgv.ResumeLayout();
|
|
TLP_Const.ResumeLayout();
|
|
|
|
Text = $"{Text} ({sav.Version})";
|
|
|
|
if (CB_Stats.Items.Count > 0)
|
|
{
|
|
CB_Stats.SelectedIndex = 0;
|
|
}
|
|
else
|
|
{
|
|
L_Stats.Visible = CB_Stats.Visible = MT_Stat.Visible = false;
|
|
tabControl1.TabPages.Remove(GB_Constants);
|
|
}
|
|
NUD_Flag.Maximum = editor.Flags.Length - 1;
|
|
NUD_Flag.Value = 0;
|
|
c_CustomFlag.Checked = editor.Flags[0];
|
|
editing = false;
|
|
}
|
|
|
|
private void B_Cancel_Click(object sender, EventArgs e)
|
|
{
|
|
Close();
|
|
}
|
|
|
|
private void B_Save_Click(object sender, EventArgs e)
|
|
{
|
|
Editor.Save();
|
|
Close();
|
|
}
|
|
|
|
private void AddFlagList(EventLabelCollection list, bool[] values)
|
|
{
|
|
var labels = list.Flag;
|
|
if (labels.Count == 0)
|
|
{
|
|
dgv.Visible = false;
|
|
var research = new Label { Text = MsgResearchRequired, Name = "TLP_Flags_Research", ForeColor = Color.Red, AutoSize = true, Location = new Point(20, 20) };
|
|
GB_Flags.Controls.Add(research);
|
|
return;
|
|
}
|
|
|
|
var cFlag = new DataGridViewCheckBoxColumn
|
|
{
|
|
DisplayIndex = 0,
|
|
Width = 20,
|
|
SortMode = DataGridViewColumnSortMode.NotSortable,
|
|
};
|
|
|
|
var cLabel = new DataGridViewTextBoxColumn
|
|
{
|
|
DisplayIndex = 1,
|
|
AutoSizeMode = DataGridViewAutoSizeColumnMode.Fill,
|
|
ReadOnly = true,
|
|
SortMode = DataGridViewColumnSortMode.NotSortable,
|
|
};
|
|
|
|
cFlag.DefaultCellStyle.Alignment = DataGridViewContentAlignment.MiddleCenter;
|
|
cLabel.DefaultCellStyle.Alignment = DataGridViewContentAlignment.MiddleLeft;
|
|
|
|
dgv.Columns.Add(cFlag);
|
|
dgv.Columns.Add(cLabel);
|
|
|
|
var hideBelow = Main.Settings.Advanced.HideEventTypeBelow;
|
|
labels = labels.Where(z => z.Type >= hideBelow).OrderByDescending(z => z.Type).ToList();
|
|
dgv.Rows.Add(labels.Count);
|
|
|
|
for (int i = 0; i < labels.Count; i++)
|
|
FlagDict[labels[i].Index] = i;
|
|
|
|
for (int i = 0; i < labels.Count; i++)
|
|
{
|
|
var (name, index, _) = labels[i];
|
|
var cells = dgv.Rows[i].Cells;
|
|
cells[0].Value = values[index];
|
|
cells[1].Value = name;
|
|
}
|
|
dgv.CellValueChanged += (s, e) =>
|
|
{
|
|
if (e.ColumnIndex != 0 || e.RowIndex == -1)
|
|
return;
|
|
|
|
bool chk = (bool)dgv.Rows[e.RowIndex].Cells[0].Value;
|
|
var index = labels[e.RowIndex].Index;
|
|
values[index] = chk;
|
|
if (NUD_Flag.Value == index)
|
|
c_CustomFlag.Checked = chk;
|
|
};
|
|
dgv.CellMouseUp += (s, e) =>
|
|
{
|
|
if (e.RowIndex == -1)
|
|
return;
|
|
|
|
if (e.ColumnIndex == 0)
|
|
{
|
|
dgv.EndEdit();
|
|
return;
|
|
}
|
|
|
|
if (e.ColumnIndex != 1)
|
|
return;
|
|
|
|
bool chk = (bool)dgv.Rows[e.RowIndex].Cells[0].Value;
|
|
dgv.Rows[e.RowIndex].Cells[0].Value = !chk;
|
|
var index = labels[e.RowIndex].Index;
|
|
values[index] = !chk;
|
|
if (NUD_Flag.Value == index)
|
|
c_CustomFlag.Checked = !chk;
|
|
};
|
|
}
|
|
|
|
private void AddConstList(EventLabelCollection list, ReadOnlySpan<byte> values)
|
|
{
|
|
var labels = list.Work;
|
|
if (labels.Count == 0)
|
|
{
|
|
TLP_Const.Controls.Add(new Label { Text = MsgResearchRequired, Name = "TLP_Const_Research", ForeColor = Color.Red, AutoSize = true }, 0, 0);
|
|
return;
|
|
}
|
|
|
|
var hide = Main.Settings.Advanced.HideEventTypeBelow;
|
|
labels = labels.OrderByDescending(z => z.Type).ToList();
|
|
for (var i = 0; i < labels.Count; i++)
|
|
{
|
|
var entry = labels[i];
|
|
if (entry.Type < hide)
|
|
break;
|
|
var lbl = new Label { Text = entry.Name, Margin = Padding.Empty, AutoSize = true };
|
|
var mtb = new NumericUpDown
|
|
{
|
|
Maximum = ushort.MaxValue,
|
|
Minimum = ushort.MinValue,
|
|
Margin = Padding.Empty,
|
|
Width = 50,
|
|
};
|
|
|
|
var map = entry.PredefinedValues.Select(z => new ComboItem(z.Name, z.Value)).ToList();
|
|
var cb = new ComboBox
|
|
{
|
|
Margin = Padding.Empty,
|
|
Width = 150,
|
|
DropDownStyle = ComboBoxStyle.DropDownList,
|
|
BindingContext = BindingContext,
|
|
DropDownWidth = Width + 100,
|
|
};
|
|
cb.InitializeBinding();
|
|
cb.DataSource = map;
|
|
|
|
lbl.Click += (sender, e) => mtb.Value = 0;
|
|
bool updating = false;
|
|
mtb.ValueChanged += ChangeConstValue;
|
|
void ChangeConstValue(object? sender, EventArgs e)
|
|
{
|
|
if (updating)
|
|
return;
|
|
|
|
updating = true;
|
|
var value = (byte)mtb.Value;
|
|
var (_, valueID) = map.Find(z => z.Value == value) ?? map[0];
|
|
if (WinFormsUtil.GetIndex(cb) != valueID)
|
|
cb.SelectedValue = valueID;
|
|
|
|
Editor.Values[entry.Index] = value;
|
|
if (CB_Stats.SelectedIndex == entry.Index)
|
|
MT_Stat.Text = ((int)mtb.Value).ToString();
|
|
updating = false;
|
|
}
|
|
cb.SelectedValueChanged += (o, args) =>
|
|
{
|
|
if (editing || updating)
|
|
return;
|
|
var value = WinFormsUtil.GetIndex(cb);
|
|
mtb.Value = value == NamedEventConst.CustomMagicValue ? 0 : value;
|
|
};
|
|
|
|
mtb.Value = values[entry.Index];
|
|
if (mtb.Value == 0)
|
|
ChangeConstValue(this, EventArgs.Empty);
|
|
|
|
TLP_Const.Controls.Add(lbl, 0, i);
|
|
TLP_Const.Controls.Add(cb, 1, i);
|
|
TLP_Const.Controls.Add(mtb, 2, i);
|
|
|
|
WorkDict.Add(entry.Index, mtb);
|
|
}
|
|
}
|
|
|
|
private void ChangeCustomBool(object sender, EventArgs e)
|
|
{
|
|
if (editing)
|
|
return;
|
|
editing = true;
|
|
var index = (int) NUD_Flag.Value;
|
|
Editor.Flags[index] = c_CustomFlag.Checked;
|
|
if (FlagDict.TryGetValue(index, out var rowIndex))
|
|
dgv.Rows[rowIndex].Cells[0].Value = c_CustomFlag.Checked;
|
|
editing = false;
|
|
}
|
|
|
|
private void ChangeCustomFlag(object sender, EventArgs e)
|
|
{
|
|
int flag = (int)NUD_Flag.Value;
|
|
c_CustomFlag.Checked = Editor.Flags[flag];
|
|
}
|
|
|
|
private void ChangeCustomFlag(object sender, KeyEventArgs e) => ChangeCustomFlag(sender, (EventArgs)e);
|
|
|
|
private void ChangeConstantIndex(object sender, EventArgs e)
|
|
{
|
|
var constants = Editor.Values;
|
|
var index = CB_Stats.SelectedIndex;
|
|
MT_Stat.Text = constants[index].ToString();
|
|
}
|
|
|
|
private void ChangeCustomConst(object sender, EventArgs e)
|
|
{
|
|
if (editing)
|
|
return;
|
|
editing = true;
|
|
var index = CB_Stats.SelectedIndex;
|
|
var parse = byte.TryParse(MT_Stat.Text, out var value) ? value : (byte)0;
|
|
Editor.Values[index] = parse;
|
|
if (WorkDict.TryGetValue(index, out var mtb))
|
|
mtb.Value = parse;
|
|
editing = false;
|
|
}
|
|
|
|
private void ChangeSAV(object sender, EventArgs e)
|
|
{
|
|
if (TB_NewSAV.Text.Length > 0 && TB_OldSAV.Text.Length > 0)
|
|
DiffSaves();
|
|
}
|
|
|
|
private void OpenSAV(object sender, EventArgs e)
|
|
{
|
|
using var ofd = new OpenFileDialog();
|
|
if (ofd.ShowDialog() == DialogResult.OK)
|
|
LoadSAV(sender, ofd.FileName);
|
|
}
|
|
|
|
private void LoadSAV(object sender, string path)
|
|
{
|
|
var dest = sender == B_LoadOld ? TB_OldSAV : TB_NewSAV;
|
|
dest.Text = path;
|
|
}
|
|
|
|
private void DiffSaves()
|
|
{
|
|
var diff = new EventBlockDiff<SAV2, byte>(TB_OldSAV.Text, TB_NewSAV.Text);
|
|
if (diff.Message != EventWorkDiffCompatibility.Valid)
|
|
{
|
|
WinFormsUtil.Alert(diff.Message.GetMessage());
|
|
return;
|
|
}
|
|
|
|
TB_IsSet.Text = string.Join(", ", diff.SetFlags.Select(z => $"{z:0000}"));
|
|
TB_UnSet.Text = string.Join(", ", diff.ClearedFlags.Select(z => $"{z:0000}"));
|
|
|
|
if (diff.WorkDiff.Count == 0)
|
|
{
|
|
WinFormsUtil.Alert("No Event Constant diff found.");
|
|
return;
|
|
}
|
|
|
|
var promptCopy = WinFormsUtil.Prompt(MessageBoxButtons.YesNo, "Copy Event Constant diff to clipboard?");
|
|
if (promptCopy == DialogResult.Yes)
|
|
WinFormsUtil.SetClipboardText(string.Join(Environment.NewLine, diff.WorkDiff));
|
|
}
|
|
|
|
private static void Main_DragEnter(object? sender, DragEventArgs? e)
|
|
{
|
|
if (e?.Data is null)
|
|
return;
|
|
if (e.Data.GetDataPresent(DataFormats.FileDrop))
|
|
e.Effect = DragDropEffects.Copy;
|
|
}
|
|
|
|
private void Main_DragDrop(object? sender, DragEventArgs? e)
|
|
{
|
|
if (e?.Data?.GetData(DataFormats.FileDrop) is not string[] { Length: not 0 } files)
|
|
return;
|
|
var dr = WinFormsUtil.Prompt(MessageBoxButtons.YesNo, Name, "Yes: Old Save" + Environment.NewLine + "No: New Save");
|
|
var button = dr == DialogResult.Yes ? B_LoadOld : B_LoadNew;
|
|
LoadSAV(button, files[0]);
|
|
}
|
|
}
|