mirror of
https://github.com/kwsch/PKHeX
synced 2024-11-22 03:53:08 +00:00
9fcd3a2769
Speed comes before SPA/SPD so I just have it reassign the pm6stat.
3816 lines
172 KiB
C#
3816 lines
172 KiB
C#
using System;
|
||
using System.Collections.Generic;
|
||
using System.ComponentModel;
|
||
using System.Drawing;
|
||
using System.IO;
|
||
using System.Linq;
|
||
using System.Security.Cryptography;
|
||
using System.Text;
|
||
using System.Threading;
|
||
using System.Windows.Forms;
|
||
|
||
namespace PKHeX
|
||
{
|
||
public partial class Form1 : Form
|
||
{
|
||
public Form1()
|
||
{
|
||
#region Pop open a splash screen while we load up.
|
||
Thread SplashSCR = new Thread(() => new SplashScreen(this).ShowDialog());
|
||
SplashSCR.Start();
|
||
#endregion
|
||
#region Initialize Form
|
||
InitializeComponent();
|
||
defaultControlWhite = CB_Species.BackColor;
|
||
defaultControlText = Label_Species.ForeColor;
|
||
CB_ExtraBytes.SelectedIndex = 0;
|
||
|
||
// Resize Main Window to PKX Editing Mode
|
||
largeWidth = Width;
|
||
shortWidth = (Width * (30500 / 620)) / 100 + 1;
|
||
Width = shortWidth;
|
||
|
||
// Initialize Boxes
|
||
byte[] ezeros = PKX.encryptArray(new byte[232]);
|
||
for (int i = 0; i < 30 * 31; i++)
|
||
Array.Copy(ezeros, 0, savefile, SaveGame.Box + i * 0xE8, 0xE8);
|
||
#endregion
|
||
#region Language Detection before loading
|
||
// Set up Language Selection
|
||
foreach (var cbItem in main_langlist)
|
||
CB_MainLanguage.Items.Add(cbItem);
|
||
|
||
// Try and detect the language
|
||
int[] main_langnum = {1, 2, 3, 4, 5, 7, 8, 9};
|
||
main_langnum = main_langnum.Concat(Enumerable.Range(10, lang_val.Length).Select(i => i).ToArray()).ToArray();
|
||
string filename = Path.GetFileNameWithoutExtension(System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName);
|
||
string lastTwoChars = filename.Substring(filename.Length - 2);
|
||
int lang = Array.IndexOf(lang_val, lastTwoChars);
|
||
CB_MainLanguage.SelectedIndex = lang >= 0 ? main_langnum[lang] - 1 : ((lastTwoChars == "jp") ? 1 : 0);
|
||
|
||
#region HaX
|
||
HaX = (filename.IndexOf("HaX", StringComparison.Ordinal) >= 0);
|
||
{
|
||
CHK_HackedStats.Enabled = CHK_HackedStats.Visible =
|
||
DEV_Ability.Enabled = DEV_Ability.Visible =
|
||
MT_Level.Enabled = MT_Level.Visible =
|
||
TB_AbilityNumber.Visible =
|
||
MT_Form.Enabled = MT_Form.Visible = HaX;
|
||
|
||
TB_Level.Visible =
|
||
CB_Ability.Visible = !HaX;
|
||
}
|
||
#endregion
|
||
Status = "Language set up";
|
||
#endregion
|
||
#region Localize & Populate
|
||
InitializeFields(); Status = "Fields set up";
|
||
CB_Language.SelectedIndex = (lang >= 0 && lang < 7) ? main_langnum[lang] : 1;
|
||
#endregion
|
||
#region Add ContextMenus to the PictureBoxes (PKX slots)
|
||
|
||
ContextMenuStrip mnu = new ContextMenuStrip();
|
||
ToolStripMenuItem mnuView = new ToolStripMenuItem("View");
|
||
ToolStripMenuItem mnuSet = new ToolStripMenuItem("Set");
|
||
ToolStripMenuItem mnuDelete = new ToolStripMenuItem("Delete");
|
||
// Assign event handlers
|
||
mnuView.Click += clickView;
|
||
mnuSet.Click += clickSet;
|
||
mnuDelete.Click += clickDelete;
|
||
// Add to main context menu
|
||
mnu.Items.AddRange(new ToolStripItem[] { mnuView, mnuSet, mnuDelete });
|
||
|
||
// Assign to datagridview for Box Pokemon and Party Pokemon
|
||
foreach (PictureBox pb in PAN_Box.Controls)
|
||
pb.ContextMenuStrip = mnu;
|
||
foreach (PictureBox pb in PAN_Party.Controls)
|
||
pb.ContextMenuStrip = mnu;
|
||
|
||
// Add ContextMenus to the PictureBoxes that are read only
|
||
PictureBox[] pba2 = {
|
||
bbpkx1,bbpkx2,bbpkx3,bbpkx4,bbpkx5,bbpkx6,
|
||
|
||
dcpkx1, dcpkx2, gtspkx, fusedpkx, subepkx1, subepkx2, subepkx3
|
||
};
|
||
ContextMenuStrip mnu2 = new ContextMenuStrip();
|
||
ToolStripMenuItem mnu2View = new ToolStripMenuItem("View");
|
||
|
||
// Assign event handlers
|
||
mnu2View.Click += clickView;
|
||
|
||
// Add to main context menu
|
||
mnu2.Items.AddRange(new ToolStripItem[] { mnu2View });
|
||
|
||
// Assign to datagridview
|
||
foreach (PictureBox p in pba2)
|
||
p.ContextMenuStrip = mnu2;
|
||
|
||
#endregion
|
||
#region Enable Drag and Drop on the form & tab control.
|
||
AllowDrop = true;
|
||
DragEnter += tabMain_DragEnter;
|
||
DragDrop += tabMain_DragDrop;
|
||
|
||
// Enable Drag and Drop on each tab.
|
||
tabMain.AllowDrop = true;
|
||
tabMain.DragEnter += tabMain_DragEnter;
|
||
tabMain.DragDrop += tabMain_DragDrop;
|
||
|
||
foreach (TabPage tab in tabMain.Controls)
|
||
{
|
||
tab.AllowDrop = true;
|
||
tab.DragEnter += tabMain_DragEnter;
|
||
tab.DragDrop += tabMain_DragDrop;
|
||
}
|
||
|
||
// ToolTips for Drag&Drop
|
||
ToolTip dragoutTip1 = new ToolTip();
|
||
dragoutTip1.SetToolTip(dragout, "PK6 QuickSave");
|
||
|
||
// Box Drag & Drop
|
||
foreach (PictureBox pb in PAN_Box.Controls)
|
||
pb.AllowDrop = true;
|
||
|
||
// Box to Tabs D&D
|
||
dragout.AllowDrop = true;
|
||
#endregion
|
||
#region Finish Up
|
||
// Load the arguments
|
||
Status = "Checking load args.";
|
||
string[] args = Environment.GetCommandLineArgs();
|
||
pathSDF = Util.GetSDFLocation();
|
||
path3DS = Util.get3DSLocation();
|
||
if (args.Length > 1)
|
||
openQuick(args[1]);
|
||
else if (pathSDF != null)
|
||
openQuick(Path.Combine(pathSDF, "main"));
|
||
else if (File.Exists(Util.NormalizePath(Path.Combine(Util.GetTempFolder(), "root" + Path.DirectorySeparatorChar + "main"))))
|
||
openQuick(Util.NormalizePath(Path.Combine(Util.GetTempFolder(), "root" + Path.DirectorySeparatorChar + "main")));
|
||
|
||
GB_OT.Click += clickGT;
|
||
GB_nOT.Click += clickGT;
|
||
GB_Daycare.Click += switchDaycare;
|
||
GB_RelearnMoves.Click += clickMoves;
|
||
|
||
Status = "Setting game font.";
|
||
TB_Nickname.Font = PKX.getPKXFont(11);
|
||
TB_OT.Font = (Font)TB_Nickname.Font.Clone();
|
||
TB_OTt2.Font = (Font)TB_Nickname.Font.Clone();
|
||
Status = "Initialized!";
|
||
CB_Species.SelectedIndex = 1;
|
||
|
||
init = true;
|
||
|
||
// Splash Screen closes on its own.
|
||
BringToFront();
|
||
WindowState = FormWindowState.Minimized;
|
||
Show();
|
||
WindowState = FormWindowState.Normal;
|
||
if (HaX) Util.Alert("Illegal mode activated.", "Please behave.");
|
||
#endregion
|
||
}
|
||
|
||
#region Global Variables: Always Visible!
|
||
public byte[] buff = new byte[260]; // Tab Pokemon Data Storage
|
||
public byte[] savefile = new byte[0x100000];
|
||
public byte[] cyberSAV = new byte[0x65600];
|
||
public static byte[] ramsav;
|
||
public static bool ramsavloaded;
|
||
public bool savegame_oras = true;
|
||
public bool cybergadget;
|
||
public bool savLoaded;
|
||
public int savindex;
|
||
public bool savedited;
|
||
public string pathSDF;
|
||
public string path3DS;
|
||
|
||
public static string Status = "Starting up PKHeX...";
|
||
public static bool HaX;
|
||
public static bool specialChars; // Open Form Tracking
|
||
public static Color defaultControlWhite;
|
||
public static Color defaultControlText;
|
||
public static int colorizedbox = 32;
|
||
public static Image mixedHighlight = Util.LayerImage(Properties.Resources.slotSet, Properties.Resources.slotView, 0, 0, 0.5);
|
||
public static Image colorizedcolor;
|
||
public static int colorizedslot;
|
||
public static int largeWidth, shortWidth;
|
||
public static string eggname = "";
|
||
public static string[] lang_val = { "en", "ja", "fr", "it", "de", "es", "ko", "zh", "pt" };
|
||
public static string[] main_langlist =
|
||
{
|
||
"English", // ENG
|
||
"日本語", // JPN
|
||
"Français", // FRE
|
||
"Italiano", // ITA
|
||
"Deutsch", // GER
|
||
"Español", // SPA
|
||
"한국어", // KOR
|
||
"中文", // CHN
|
||
"Português", // Portuguese
|
||
};
|
||
public static string[] gendersymbols = { "♂", "♀", "-" };
|
||
public static string[] specieslist = { };
|
||
public static string[] movelist = { };
|
||
public static string[] itemlist = { };
|
||
public static string[] abilitylist = { };
|
||
public static string[] types = { };
|
||
public static string[] natures = { };
|
||
public static string[] characteristics = { };
|
||
public static string[] memories = { };
|
||
public static string[] genloc = { };
|
||
public static string[] forms = { };
|
||
public static string[] metHGSS_00000 = { };
|
||
public static string[] metHGSS_02000 = { };
|
||
public static string[] metHGSS_03000 = { };
|
||
public static string[] metBW2_00000 = { };
|
||
public static string[] metBW2_30000 = { };
|
||
public static string[] metBW2_40000 = { };
|
||
public static string[] metBW2_60000 = { };
|
||
public static string[] metXY_00000 = { };
|
||
public static string[] metXY_30000 = { };
|
||
public static string[] metXY_40000 = { };
|
||
public static string[] metXY_60000 = { };
|
||
public static string[] trainingbags = { };
|
||
public static string[] trainingstage = { };
|
||
public static string[] wallpapernames = { };
|
||
public static string[] encountertypelist = { };
|
||
public static string[] gamelanguages = { };
|
||
public static string[] consoleregions = { };
|
||
public static string[] balllist = { };
|
||
public static string[] gamelist = { };
|
||
public static string[] puffs = { };
|
||
public static string[] itempouch = { };
|
||
public static string origintrack;
|
||
public static string curlanguage = "en";
|
||
public volatile bool init;
|
||
public static bool unicode;
|
||
public ToolTip Tip1 = new ToolTip();
|
||
public ToolTip Tip2 = new ToolTip();
|
||
public ToolTip Tip3 = new ToolTip();
|
||
public ToolTip NatureTip = new ToolTip();
|
||
public PKX.Structures.SaveGame SaveGame = new PKX.Structures.SaveGame("ORAS");
|
||
#endregion
|
||
|
||
#region //// MAIN MENU FUNCTIONS ////
|
||
// Main Menu Strip UI Functions
|
||
private void mainMenuOpen(object sender, EventArgs e)
|
||
{
|
||
string cyberpath = Util.GetTempFolder();
|
||
pathSDF = Util.GetSDFLocation();
|
||
path3DS = Util.get3DSLocation();
|
||
if (pathSDF != null)
|
||
{
|
||
OpenPKX.InitialDirectory = pathSDF;
|
||
OpenPKX.RestoreDirectory = true;
|
||
OpenPKX.FilterIndex = 4;
|
||
}
|
||
else if (path3DS != null)
|
||
{
|
||
OpenPKX.InitialDirectory = path3DS;
|
||
OpenPKX.RestoreDirectory = true;
|
||
OpenPKX.FilterIndex = 4;
|
||
}
|
||
else if (Directory.Exists(Path.Combine(cyberpath, "root")))
|
||
{
|
||
OpenPKX.InitialDirectory = Path.Combine(cyberpath, "root");
|
||
OpenPKX.RestoreDirectory = true;
|
||
OpenPKX.FilterIndex = 4;
|
||
}
|
||
else if (Directory.Exists(cyberpath))
|
||
{
|
||
OpenPKX.InitialDirectory = cyberpath;
|
||
OpenPKX.RestoreDirectory = true;
|
||
OpenPKX.FilterIndex = 4;
|
||
}
|
||
|
||
DialogResult result = OpenPKX.ShowDialog();
|
||
if (result != DialogResult.OK) return;
|
||
|
||
string path = OpenPKX.FileName;
|
||
openQuick(path);
|
||
}
|
||
private void mainMenuSave(object sender, EventArgs e)
|
||
{
|
||
if (!verifiedPKX()) { return; }
|
||
SavePKX.FileName = TB_Nickname.Text + " - " + TB_PID.Text;
|
||
DialogResult result = SavePKX.ShowDialog();
|
||
if (result != DialogResult.OK) return;
|
||
string path = SavePKX.FileName;
|
||
// Injection Dummy Override
|
||
if (path.Contains("pokemon.ekx")) path = Path.GetDirectoryName(path) + Path.DirectorySeparatorChar + "pokemon.ekx";
|
||
string ext = Path.GetExtension(path);
|
||
|
||
if (File.Exists(path) && !path.Contains("pokemon.ekx"))
|
||
{
|
||
// File already exists, save a .bak
|
||
byte[] backupfile = File.ReadAllBytes(path);
|
||
File.WriteAllBytes(path + ".bak", backupfile);
|
||
}
|
||
byte[] pkx = preparepkx();
|
||
|
||
if ((ext == ".ekx") || (ext == ".bin") || (ext == ".pkx") || (ext == ".ek6") || (ext == ".pk6"))
|
||
{
|
||
if ((ext == ".ekx") || (ext == ".bin") || (ext == ".ek6")) // User Requested Encrypted File
|
||
pkx = PKX.encryptArray(pkx);
|
||
File.WriteAllBytes(path, pkx.ToArray());
|
||
}
|
||
else
|
||
{
|
||
Util.Error(String.Format("Foreign File Extension: {0}", ext), "Exporting as encrypted.");
|
||
pkx = PKX.encryptArray(pkx);
|
||
File.WriteAllBytes(path, pkx);
|
||
}
|
||
}
|
||
private void mainMenuExit(object sender, EventArgs e)
|
||
{
|
||
if (ModifierKeys == (Keys.Control | Keys.E)) // Hotkey Triggered
|
||
if (DialogResult.Yes != Util.Prompt(MessageBoxButtons.YesNo, "Quit PKHeX?")) return;
|
||
Close();
|
||
}
|
||
private void mainMenuAbout(object sender, EventArgs e)
|
||
{
|
||
// Open a new form with the About details.
|
||
new About().ShowDialog();
|
||
}
|
||
private void mainMenuWiden(object sender, EventArgs e)
|
||
{
|
||
int newwidth;
|
||
if (Width < Height)
|
||
{
|
||
newwidth = largeWidth;
|
||
tabBoxMulti.SelectedIndex = 0;
|
||
}
|
||
else
|
||
newwidth = shortWidth;
|
||
|
||
Width = newwidth;
|
||
}
|
||
private void mainMenuCodeGen(object sender, EventArgs e)
|
||
{
|
||
// Open Code Generator
|
||
byte[] formdata = null;
|
||
if (verifiedPKX()) formdata = preparepkx();
|
||
CodeGenerator CodeGen = new CodeGenerator(this, formdata);
|
||
CodeGen.ShowDialog();
|
||
byte[] data = CodeGen.returnArray;
|
||
if (data == null) return;
|
||
byte[] decdata = PKX.decryptArray(data);
|
||
Array.Copy(decdata, buff, 232);
|
||
try { populateFields(buff); }
|
||
catch
|
||
{
|
||
Array.Copy(new byte[232], buff, 232);
|
||
populateFields(buff);
|
||
Util.Error("Imported code did not decrypt properly", "Please verify that what you imported was correct.");
|
||
}
|
||
}
|
||
private void mainMenuBoxReport(object sender, EventArgs e)
|
||
{
|
||
frmReport ReportForm = new frmReport();
|
||
int offset = 0x27A00; if (savegame_oras) offset = 0x33000 + 0x5400;
|
||
ReportForm.Show();
|
||
ReportForm.PopulateData(savefile, savindex, offset);
|
||
}
|
||
private void mainMenuUnicode(object sender, EventArgs e)
|
||
{
|
||
unicode = (gendersymbols[0] == "♂");
|
||
if (unicode)
|
||
{
|
||
gendersymbols = new[] { "M", "F", "-" };
|
||
BTN_Shinytize.Text = "*";
|
||
TB_Nickname.Font = TB_OT.Font = TB_OTt2.Font = Label_TID.Font;
|
||
}
|
||
else
|
||
{
|
||
gendersymbols = new[] { "♂", "♀", "-" };
|
||
BTN_Shinytize.Text = "☆";
|
||
TB_Nickname.Font = TB_OT.Font = TB_OTt2.Font = PKX.getPKXFont(11);
|
||
}
|
||
// Switch active gender labels to new if they are active.
|
||
if (PKX.getGender(Label_Gender.Text) < 2)
|
||
Label_Gender.Text = gendersymbols[PKX.getGender(Label_Gender.Text)];
|
||
if (PKX.getGender(Label_OTGender.Text) < 2)
|
||
Label_OTGender.Text = gendersymbols[PKX.getGender(Label_OTGender.Text)];
|
||
if (PKX.getGender(Label_CTGender.Text) < 2)
|
||
Label_CTGender.Text = gendersymbols[PKX.getGender(Label_CTGender.Text)];
|
||
}
|
||
|
||
// Main Menu Subfunctions
|
||
private void openQuick(string path)
|
||
{
|
||
// detect if it is a folder (load into boxes or not)
|
||
if (Directory.Exists(path))
|
||
{ loadBoxesFromDB(path); return; }
|
||
|
||
string ext = Path.GetExtension(path);
|
||
FileInfo fi = new FileInfo(path);
|
||
if (fi.Length > 0x10009C)
|
||
Util.Error("Input file is too large.", path);
|
||
else
|
||
{
|
||
|
||
byte[] input; try { input = File.ReadAllBytes(path); }
|
||
catch { Util.Error("File is in use by another program!", path); return; }
|
||
|
||
try { openFile(input, path, ext); }
|
||
catch
|
||
{
|
||
try
|
||
{
|
||
byte[] blank = PKX.encryptArray(new byte[260]);
|
||
|
||
for (int i = 0; i < 232; i++)
|
||
blank[i] ^= input[i];
|
||
|
||
openFile(blank, path, ext);
|
||
}
|
||
catch { openFile(input, path, ext); }
|
||
}
|
||
}
|
||
}
|
||
private void openFile(byte[] input, string path, string ext)
|
||
{
|
||
#region Powersaves Read-Only Conversion
|
||
if (input.Length == 0x10009C) // Resize to 1MB
|
||
{
|
||
Array.Copy(input, 0x9C, input, 0, 0x100000);
|
||
Array.Resize(ref input, 0x100000);
|
||
}
|
||
#endregion
|
||
|
||
#region Saves
|
||
if ((input.Length == 0x76000) && BitConverter.ToUInt32(input, 0x75E10) == 0x42454546) // ORAS
|
||
openMAIN(input, path, "ORAS", true);
|
||
else if ((input.Length == 0x65600) && BitConverter.ToUInt32(input, 0x65410) == 0x42454546) // XY
|
||
openMAIN(input, path, "XY", false);
|
||
// Verify the Data Input Size is Proper
|
||
else if (input.Length == 0x100000)
|
||
{
|
||
if (openXOR(input, path)) // Check if we can load the save via xorpad
|
||
return; // only if a save is loaded we abort
|
||
if (BitConverter.ToUInt64(input, 0x10) != 0) // encrypted save
|
||
{ Util.Error("PKHeX only edits decrypted save files.", "This save file is not decrypted."); return; }
|
||
|
||
string GameType = "XY"; // Default Game Type to load.
|
||
if (BitConverter.ToUInt32(input, 0x7B210) == 0x42454546) GameType = "ORAS"; // BEEF magic in checksum block
|
||
if ((BitConverter.ToUInt32(input, 0x100) != 0x41534944) && (BitConverter.ToUInt32(input, 0x5234) != 0x6E69616D))
|
||
{
|
||
DialogResult dialogResult = Util.Prompt(MessageBoxButtons.YesNo, "Save file is not decrypted.", "Press Yes to ignore this warning and continue loading the save file.");
|
||
if (dialogResult != DialogResult.Yes) return;
|
||
|
||
DialogResult sdr = Util.Prompt(MessageBoxButtons.YesNoCancel, "Press Yes to load the sav at 0x3000", "Press No for the one at 0x82000");
|
||
if (sdr == DialogResult.Cancel) return;
|
||
|
||
savindex = (sdr == DialogResult.Yes) ? 0 : 1;
|
||
B_SwitchSAV.Enabled = true;
|
||
open1MB(input, path, GameType, false);
|
||
}
|
||
else if (PKX.detectSAVIndex(input, ref savindex) == 2)
|
||
{
|
||
DialogResult dialogResult = Util.Prompt(MessageBoxButtons.YesNo, "Hash verification failed.", "Press Yes to ignore this warning and continue loading the save file.");
|
||
if (dialogResult != DialogResult.Yes) return;
|
||
|
||
DialogResult sdr = Util.Prompt(MessageBoxButtons.YesNoCancel, "Press Yes to load the sav at 0x3000", "Press No for the one at 0x82000");
|
||
if (sdr == DialogResult.Cancel)
|
||
{
|
||
savindex = 0;
|
||
return; // abort load
|
||
}
|
||
savindex = (sdr == DialogResult.Yes) ? 0 : 1;
|
||
B_SwitchSAV.Enabled = true;
|
||
open1MB(input, path, GameType, false);
|
||
}
|
||
else
|
||
{
|
||
B_ExportSAV.Enabled = true;
|
||
B_SwitchSAV.Enabled = true;
|
||
PKX.detectSAVIndex(input, ref savindex);
|
||
open1MB(input, path, GameType, false);
|
||
}
|
||
}
|
||
#endregion
|
||
#region PK6/EK6
|
||
else if ((input.Length == 260) || (input.Length == 232))
|
||
{
|
||
// Check if Input is PKX
|
||
if ((ext == ".pk6") || (ext == ".ek6") || (ext == ".pkx") || (ext == ".ekx") || (ext == ".bin") || (ext == ""))
|
||
{
|
||
// Check if Encrypted before Loading
|
||
buff = (BitConverter.ToUInt16(input, 0xC8) == 0 && BitConverter.ToUInt16(input, 0x58) == 0) ? input : PKX.decryptArray(input);
|
||
populateFields(buff);
|
||
}
|
||
else
|
||
Util.Error("Unable to recognize file." + Environment.NewLine + "Only valid .pk* .ek* .bin supported.", String.Format("File Loaded:{0}{1}", Environment.NewLine, path));
|
||
}
|
||
#endregion
|
||
#region PK3/PK4/PK5
|
||
else if ((input.Length == 136) || (input.Length == 220) || (input.Length == 236) || (input.Length == 100) || (input.Length == 80)) // to convert g5pkm
|
||
{
|
||
var Converter = new pk2pk();
|
||
if (!PKX.verifychk(input)) Util.Error("Invalid File (Checksum Error)");
|
||
try // to convert g5pkm
|
||
{
|
||
byte[] data = Converter.ConvertPKM(input, savefile, savindex);
|
||
Array.Copy(data, buff, 232);
|
||
populateFields(buff);
|
||
}
|
||
catch
|
||
{
|
||
Array.Copy(new byte[232], buff, 232);
|
||
populateFields(buff);
|
||
Util.Error("Attempted to load previous generation PKM.", "Conversion failed.");
|
||
}
|
||
}
|
||
#endregion
|
||
#region Trade Packets
|
||
else if (input.Length == 363 && BitConverter.ToUInt16(input, 0x6B) == 0)
|
||
{
|
||
// EAD Packet of 363 length
|
||
byte[] c = new byte[260];
|
||
Array.Copy(input, 0x67, c, 0, 260);
|
||
}
|
||
else if (input.Length == 407 && BitConverter.ToUInt16(input, 0x98) == 0)
|
||
{
|
||
// EAD Packet of 407 length
|
||
byte[] c = new byte[260];
|
||
Array.Copy(input, 0x93, c, 0, 260);
|
||
}
|
||
#endregion
|
||
#region Box Data
|
||
else if ((input.Length == 0xE8 * 30 || input.Length == 0xE8 * 30 * 31) && BitConverter.ToUInt16(input, 4) == 0 && BitConverter.ToUInt32(input , 8) > 0)
|
||
{
|
||
Array.Copy(input, 0, savefile, SaveGame.Box + 0xE8 * 30 * ((input.Length == 0xE8*30) ? CB_BoxSelect.SelectedIndex : 0), input.Length);
|
||
setPKXBoxes();
|
||
Width = largeWidth;
|
||
Util.Alert("Box Binary loaded."); }
|
||
#endregion
|
||
#region injectiondebug
|
||
else if (input.Length == 0x10000)
|
||
{
|
||
int offset = -1; // Seek to find data start
|
||
for (int i = 0; i < 0x800; i++)
|
||
{
|
||
byte[] data = PKX.decryptArray(input.Skip(i).Take(0xE8).ToArray());
|
||
if (PKX.getCHK(data) != BitConverter.ToUInt16(data, 6)) continue;
|
||
offset = i; break;
|
||
}
|
||
if (offset < 0) { Util.Alert(path, "Unable to read the input file; not an expected injectiondebug.bin."); return; }
|
||
CB_BoxSelect.SelectedIndex = 0;
|
||
Array.Copy(input, offset, savefile, SaveGame.Box + 0xE8 * 30 * CB_BoxSelect.SelectedIndex, 9 * 30 * 0xE8);
|
||
setPKXBoxes();
|
||
Width = largeWidth;
|
||
Util.Alert("Injection Binary loaded."); }
|
||
#endregion
|
||
#region RAMSAV
|
||
else if (( /*XY*/ input.Length == 0x70000 || /*ORAS*/ input.Length == 0x80000) && Path.GetFileName(path).Contains("ram"))
|
||
{
|
||
if (input.Length == 0x80000)
|
||
// Scan for FEEB in XY location, 3DS only overwrites data if file already exists.
|
||
for (int i = 0x60000; i < 0x64000; i+=4)
|
||
if (BitConverter.ToUInt32(input, i) == 0x42454546) { Array.Resize(ref input, 0x70000); break; }
|
||
|
||
bool o = (input.Length == 0x80000);
|
||
try { openMAIN(ram2sav.getMAIN(input), path, (o) ? "ORAS" : "XY", o, true); } catch { return; }
|
||
ramsav = (byte[])input.Clone();
|
||
}
|
||
#endregion
|
||
#region Battle Video
|
||
else if (input.Length == 0x2E60 && BitConverter.ToUInt64(input, 0xE18) != 0 && BitConverter.ToUInt16(input, 0xE12) == 0)
|
||
{
|
||
if (Util.Prompt(MessageBoxButtons.YesNo, "Load Batte Video Pokémon data to " + CB_BoxSelect.Text + "?", "The first 24 slots will be overwritten.") != DialogResult.Yes) return;
|
||
for (int i = 0; i < 24; i++)
|
||
Array.Copy(input, 0xE18 + 260 * i + (i / 6) * 8, savefile, SaveGame.Box + i * 0xE8 + CB_BoxSelect.SelectedIndex * 30 * 0xE8, 0xE8);
|
||
setPKXBoxes();
|
||
}
|
||
#endregion
|
||
#region Wondercard
|
||
else if (input.Length == 0x108 && ext == ".wc6")
|
||
new SAV_Wondercard(this, input).Show();
|
||
#endregion
|
||
else
|
||
Util.Error("Attempted to load an unsupported file type/size.", "File Loaded:" + Environment.NewLine + path, "File Size:" + Environment.NewLine + new FileInfo(path).Length.ToString("X8"));
|
||
}
|
||
private void openMAIN(byte[] input, string path, string GameType, bool oras, bool ram = false)
|
||
{
|
||
ramsavloaded = ram;
|
||
L_Save.Text = "SAV: " + Path.GetFileName(path);
|
||
SaveGame = new PKX.Structures.SaveGame(GameType);
|
||
|
||
// Load CyberGadget
|
||
savindex = 0;
|
||
savefile = new byte[0x100000];
|
||
cyberSAV = input;
|
||
cybergadget = true;
|
||
B_ExportSAV.Enabled = true;
|
||
B_SwitchSAV.Enabled = false;
|
||
Array.Copy(input, 0, savefile, 0x5400, input.Length);
|
||
|
||
openSave(oras);
|
||
}
|
||
private void open1MB(byte[] input, string path, string GameType, bool oras)
|
||
{
|
||
ramsavloaded = false;
|
||
L_Save.Text = "SAV: " + Path.GetFileName(path);
|
||
SaveGame = new PKX.Structures.SaveGame(GameType);
|
||
savegame_oras = oras;
|
||
|
||
savefile = input;
|
||
cybergadget = false;
|
||
|
||
// Logic to allow unlocking of Switch SAV
|
||
// Setup SHA
|
||
SHA256 mySHA256 = SHA256.Create();
|
||
|
||
// Check both IVFC Hashes
|
||
byte[] zeroarray = new byte[0x200];
|
||
Array.Copy(savefile, 0x2000 + 0 * 0x7F000, zeroarray, 0, 0x20);
|
||
byte[] hashValue1 = mySHA256.ComputeHash(zeroarray);
|
||
Array.Copy(savefile, 0x2000 + 1 * 0x7F000, zeroarray, 0, 0x20);
|
||
byte[] hashValue2 = mySHA256.ComputeHash(zeroarray);
|
||
|
||
byte[] realHash1 = new byte[0x20];
|
||
byte[] realHash2 = new byte[0x20];
|
||
|
||
Array.Copy(savefile, 0x43C - 0 * 0x130, realHash1, 0, 0x20);
|
||
Array.Copy(savefile, 0x43C - 1 * 0x130, realHash2, 0, 0x20);
|
||
|
||
B_SwitchSAV.Enabled = (hashValue1.SequenceEqual(realHash1) && hashValue2.SequenceEqual(realHash2));
|
||
getSAVOffsets(ref oras); // to detect if we are ORAS or not
|
||
Array.Copy(savefile, 0x5400 + 0x7F000 * savindex, cyberSAV, 0, cyberSAV.Length);
|
||
|
||
openSave(oras);
|
||
}
|
||
private bool openXOR(byte[] input, string path)
|
||
{
|
||
// Detection of stored Decryption XORpads:
|
||
if (ModifierKeys == Keys.Control) return false; // no xorpad compatible
|
||
byte[] savID = new byte[0x10]; Array.Copy(input, 0x10, savID, 0, 0x10);
|
||
string exepath = Application.StartupPath;
|
||
string xorpath = exepath.Clone().ToString();
|
||
string[] XORpads = Directory.GetFiles(xorpath);
|
||
|
||
check:
|
||
foreach (byte[] data in from file in XORpads let fi = new FileInfo(file) where (fi.Name.ToLower().Contains("xorpad") || fi.Name.ToLower().Contains("key")) && (fi.Length == 0x10009C || fi.Length == 0x100000) select File.ReadAllBytes(file))
|
||
{
|
||
// Fix xorpad alignment
|
||
byte[] xorpad = data;
|
||
if (xorpad.Length == 0x10009C)
|
||
{
|
||
Array.Copy(xorpad, 0x9C, xorpad, 0, 0x100000);
|
||
Array.Resize(ref xorpad, 0x100000);
|
||
}
|
||
byte[] xorID = new byte[0x10]; Array.Copy(xorpad, 0x10, xorID, 0, 0x10);
|
||
if (!xorID.SequenceEqual(savID)) continue;
|
||
|
||
// Set up Decrypted File
|
||
byte[] decryptedPS = new byte[0x76000];
|
||
Array.Copy(input, 0x5400, decryptedPS, 0, 0x76000);
|
||
|
||
// xor through and decrypt
|
||
for (int z = 0; z < 0x76000; z++)
|
||
decryptedPS[z] ^= xorpad[0x5400 + z];
|
||
|
||
// Weakly check the validity of the decrypted content
|
||
if (BitConverter.ToUInt32(decryptedPS, 0x76000 - 0x1F0) != 0x42454546) // Not OR/AS
|
||
if (BitConverter.ToUInt32(decryptedPS, 0x65600 - 0x1F0) != 0x42454546)
|
||
continue; // Not X/Y, so continue.
|
||
else
|
||
Array.Resize(ref decryptedPS, 0x65600); // set to X/Y size
|
||
else Array.Resize(ref decryptedPS, 0x76000); // set to ORAS size just in case
|
||
|
||
// Save file is now decrypted! Reset the loading variables.
|
||
bool oras = (decryptedPS.Length == 0x76000);
|
||
string GameType = oras ? "ORAS" : "XY";
|
||
|
||
// Trigger Loading of the decrypted save file.
|
||
openMAIN(decryptedPS, path, GameType, oras);
|
||
|
||
// Abort the opening of a non-cyber file.
|
||
return true;
|
||
}
|
||
// End file check loop, check the input path for xorpads too if it isn't the same as the EXE (quite common).
|
||
if (xorpath != exepath) return false; // no xorpad compatible
|
||
xorpath = Path.GetDirectoryName(path); goto check;
|
||
}
|
||
private void openSave(bool oras)
|
||
{
|
||
savegame_oras = oras;
|
||
// Enable Secondary Tools
|
||
GB_SAVtools.Enabled =
|
||
B_JPEG.Enabled = true;
|
||
|
||
savedited = false;
|
||
Menu_ToggleBoxUI.Visible = false;
|
||
int startBox = savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] & 0x1F;
|
||
|
||
B_VerifySHA.Enabled = !cybergadget;
|
||
B_VerifyCHK.Enabled = !ramsavloaded;
|
||
|
||
setBoxNames(); // Display the Box Names
|
||
setPKXBoxes(); // Reload all of the PKX Windows
|
||
setSAVLabel(); // Reload the label indicating current save
|
||
|
||
// Version Exclusive Editors
|
||
GB_SUBE.Visible = !oras;
|
||
B_OpenSecretBase.Visible = oras;
|
||
|
||
if (startBox > 30) {tabBoxMulti.SelectedIndex = 1; }
|
||
else { tabBoxMulti.SelectedIndex = 0; CB_BoxSelect.SelectedIndex = startBox; }
|
||
|
||
Width = largeWidth;
|
||
savLoaded = true;
|
||
// Indicate audibly the save is loaded
|
||
System.Media.SystemSounds.Beep.Play();
|
||
}
|
||
|
||
// Language Translation
|
||
private void changeMainLanguage(object sender, EventArgs e)
|
||
{
|
||
if (init) buff = preparepkx(); // get data currently in form
|
||
|
||
Menu_Options.DropDown.Close();
|
||
InitializeStrings();
|
||
InitializeLanguage();
|
||
Util.TranslateInterface(this, lang_val[CB_MainLanguage.SelectedIndex], menuStrip1); // Translate the UI to language.
|
||
populateFields(buff); // put data back in form
|
||
}
|
||
private void InitializeStrings()
|
||
{
|
||
if (CB_MainLanguage.SelectedIndex < 8)
|
||
curlanguage = lang_val[CB_MainLanguage.SelectedIndex];
|
||
|
||
string l = curlanguage;
|
||
natures = Util.getStringList("Natures", l);
|
||
types = Util.getStringList("Types", l);
|
||
abilitylist = Util.getStringList("Abilities", l);
|
||
movelist = Util.getStringList("Moves", l);
|
||
itemlist = Util.getStringList("Items", l);
|
||
characteristics = Util.getStringList("Character", l);
|
||
specieslist = Util.getStringList("Species", l);
|
||
wallpapernames = Util.getStringList("Wallpaper", l);
|
||
itempouch = Util.getStringList("ItemPouch", l);
|
||
encountertypelist = Util.getStringList("EncounterType", l);
|
||
gamelist = Util.getStringList("Games", l);
|
||
gamelanguages = Util.getNulledStringArray(Util.getSimpleStringList("languages"));
|
||
consoleregions = Util.getNulledStringArray(Util.getSimpleStringList("regions3ds"));
|
||
|
||
balllist = new string[Legal.Items_Ball.Length];
|
||
for (int i = 0; i < balllist.Length; i++)
|
||
balllist[i] = itemlist[Legal.Items_Ball[i]];
|
||
|
||
if ((l != "zh") || (l == "zh" && !init)) // load initial binaries
|
||
{
|
||
forms = Util.getStringList("Forms", l);
|
||
memories = Util.getStringList("Memories", l);
|
||
genloc = Util.getStringList("GenLoc", l);
|
||
trainingbags = Util.getStringList("TrainingBag", l);
|
||
trainingstage = Util.getStringList("SuperTraining", l);
|
||
puffs = Util.getStringList("Puff", l);
|
||
}
|
||
|
||
// Fix Item Names (Duplicate entries)
|
||
itemlist[456] += " (OLD)"; // S.S. Ticket
|
||
itemlist[463] += " (OLD)"; // Storage Key
|
||
itemlist[478] += " (OLD)"; // Basement Key
|
||
itemlist[626] += " (2)"; // Xtransceiver
|
||
itemlist[629] += " (2)"; // DNA Splicers
|
||
itemlist[637] += " (2)"; // Dropped Item
|
||
itemlist[707] += " (2)"; // Travel Trunk
|
||
itemlist[713] += " (2)"; // Alt Bike
|
||
itemlist[714] += " (2)"; // Holo Caster
|
||
itemlist[729] += " (1)"; // Meteorite
|
||
itemlist[740] += " (2)"; // Contest Costume
|
||
itemlist[751] += " (2)"; // Meteorite
|
||
itemlist[771] += " (3)"; // Meteorite
|
||
itemlist[772] += " (4)"; // Meteorite
|
||
|
||
// Get the Egg Name and then replace it with --- for the comboboxes.
|
||
eggname = specieslist[0];
|
||
specieslist[0] = "---";
|
||
|
||
// Get the met locations... for all of the games...
|
||
metHGSS_00000 = Util.getStringList("hgss_00000", l);
|
||
metHGSS_02000 = Util.getStringList("hgss_02000", l);
|
||
metHGSS_03000 = Util.getStringList("hgss_03000", l);
|
||
|
||
metBW2_00000 = Util.getStringList("bw2_00000", l);
|
||
metBW2_30000 = Util.getStringList("bw2_30000", l);
|
||
metBW2_40000 = Util.getStringList("bw2_40000", l);
|
||
metBW2_60000 = Util.getStringList("bw2_60000", l);
|
||
|
||
metXY_00000 = Util.getStringList("xy_00000", l);
|
||
metXY_30000 = Util.getStringList("xy_30000", l);
|
||
metXY_40000 = Util.getStringList("xy_40000", l);
|
||
metXY_60000 = Util.getStringList("xy_60000", l);
|
||
|
||
// Fix up some of the Location strings to make them more descriptive:
|
||
metHGSS_02000[1] += " (NPC)"; // Anything from an NPC
|
||
metHGSS_02000[2] += " (" + eggname + ")"; // Egg From Link Trade
|
||
metBW2_00000[36] = metBW2_00000[84] + "/" + metBW2_00000[36]; // Cold Storage in BW = PWT in BW2
|
||
|
||
// BW2 Entries from 76 to 105 are for Entralink in BW
|
||
for (int i = 76; i < 106; i++)
|
||
metBW2_00000[i] = metBW2_00000[i] + "●";
|
||
|
||
// Localize the Poketransfer to the language (30001)
|
||
string[] ptransp = { "Poké Transfer", "ポケシフター", "Poké Fret", "Pokétrasporto", "Poképorter", "Pokétransfer", "포케시프터", "ポケシフター" };
|
||
metBW2_30000[1 - 1] = ptransp[Array.IndexOf(lang_val, curlanguage)];
|
||
metBW2_30000[2 - 1] += " (NPC)"; // Anything from an NPC
|
||
metBW2_30000[3 - 1] += " (" + eggname + ")"; // Egg From Link Trade
|
||
|
||
// Zorua/Zoroark events
|
||
metBW2_30000[10 - 1] = specieslist[251] + " (" + specieslist[570] + " 1)"; // Celebi's Zorua Event
|
||
metBW2_30000[11 - 1] = specieslist[251] + " (" + specieslist[570] + " 2)"; // Celebi's Zorua Event
|
||
metBW2_30000[12 - 1] = specieslist[571] + " (" + "1)"; // Zoroark
|
||
metBW2_30000[13 - 1] = specieslist[571] + " (" + "2)"; // Zoroark
|
||
|
||
metBW2_60000[3 - 1] += " (" + eggname + ")"; // Egg Treasure Hunter/Breeder, whatever...
|
||
|
||
metXY_00000[104] += " (X/Y)"; // Victory Road
|
||
metXY_00000[298] += " (OR/AS)"; // Victory Road
|
||
metXY_30000[0] += " (NPC)"; // Anything from an NPC
|
||
metXY_30000[1] += " (" + eggname + ")"; // Egg From Link Trade
|
||
|
||
// Set the first entry of a met location to "" (nothing)
|
||
// Fix (None) tags
|
||
abilitylist[0] = itemlist[0] = movelist[0] = metXY_00000[0] = metBW2_00000[0] = metHGSS_00000[0] = "(" + itemlist[0] + ")";
|
||
|
||
// Force an update to the met locations
|
||
origintrack = "";
|
||
|
||
if (init)
|
||
updateIVs(null, null); // Prompt an update for the characteristics
|
||
}
|
||
#endregion
|
||
|
||
#region //// PKX WINDOW FUNCTIONS ////
|
||
private void InitializeFields()
|
||
{
|
||
// Now that the ComboBoxes are ready, load the data.
|
||
populateFields(buff);
|
||
{
|
||
TB_OT.Text = "PKHeX";
|
||
TB_TID.Text = 12345.ToString();
|
||
TB_SID.Text = 54321.ToString();
|
||
CB_Species.SelectedIndex = 1;
|
||
CB_GameOrigin.SelectedIndex = 0;
|
||
CB_Language.SelectedIndex = 0;
|
||
CB_BoxSelect.SelectedIndex = 0;
|
||
CB_GameOrigin.SelectedIndex = 0;
|
||
CB_PPu1.SelectedIndex = CB_PPu2.SelectedIndex = CB_PPu3.SelectedIndex = CB_PPu4.SelectedIndex = 0;
|
||
CB_Ball.SelectedIndex = 0;
|
||
CB_Country.SelectedIndex = 0;
|
||
setAbilityList(TB_AbilityNumber, Util.getIndex(CB_Species), CB_Ability, CB_Form);
|
||
}
|
||
}
|
||
private void InitializeLanguage()
|
||
{
|
||
// Set the Display
|
||
CB_Country.DisplayMember =
|
||
CB_SubRegion.DisplayMember =
|
||
CB_3DSReg.DisplayMember =
|
||
CB_Language.DisplayMember =
|
||
CB_Ball.DisplayMember =
|
||
CB_HeldItem.DisplayMember =
|
||
CB_Species.DisplayMember =
|
||
DEV_Ability.DisplayMember =
|
||
CB_Nature.DisplayMember =
|
||
CB_EncounterType.DisplayMember =
|
||
CB_GameOrigin.DisplayMember =
|
||
CB_HPType.DisplayMember = "Text";
|
||
|
||
// Set the Value
|
||
CB_Country.ValueMember =
|
||
CB_SubRegion.ValueMember =
|
||
CB_3DSReg.ValueMember =
|
||
CB_Language.ValueMember =
|
||
CB_Ball.ValueMember =
|
||
CB_HeldItem.ValueMember =
|
||
CB_Species.ValueMember =
|
||
DEV_Ability.ValueMember =
|
||
CB_Nature.ValueMember =
|
||
CB_EncounterType.ValueMember =
|
||
CB_GameOrigin.ValueMember =
|
||
CB_HPType.ValueMember = "Value";
|
||
|
||
// Set the various ComboBox DataSources up with their allowed entries
|
||
setCountrySubRegion(CB_Country, "countries");
|
||
CB_3DSReg.DataSource = Util.getUnsortedCBList("regions3ds");
|
||
CB_Language.DataSource = Util.getUnsortedCBList("languages");
|
||
int[] ball_nums = { 7, 576, 13, 492, 497, 14, 495, 493, 496, 494, 11, 498, 8, 6, 12, 15, 9, 5, 499, 10, 1, 16 };
|
||
int[] ball_vals = { 7, 25, 13, 17, 22, 14, 20, 18, 21, 19, 11, 23, 8, 6, 12, 15, 9, 5, 24, 10, 1, 16 };
|
||
CB_Ball.DataSource = Util.getVariedCBList(Util.getCBList(itemlist, new[] { 4 }, new[] { 3 }, new[] { 2 }, new[] { 1 }), itemlist, ball_nums, ball_vals);
|
||
CB_HeldItem.DataSource = Util.getCBList(itemlist, (DEV_Ability.Enabled) ? null : Legal.Items_Held);
|
||
CB_Species.DataSource = Util.getCBList(specieslist, null);
|
||
DEV_Ability.DataSource = Util.getCBList(abilitylist, null);
|
||
CB_Nature.DataSource = Util.getCBList(natures, null);
|
||
CB_EncounterType.DataSource = Util.getCBList(encountertypelist, new[] { 0 }, Legal.Gen4EncounterTypes);
|
||
CB_GameOrigin.DataSource = Util.getCBList(gamelist, Legal.Games_6oras, Legal.Games_6xy, Legal.Games_5, Legal.Games_4, Legal.Games_4e, Legal.Games_4r, Legal.Games_3, Legal.Games_3e, Legal.Games_3r, Legal.Games_3s);
|
||
|
||
string[] hptypes = new string[types.Length - 2]; Array.Copy(types, 1, hptypes, 0, hptypes.Length);
|
||
CB_HPType.DataSource = Util.getCBList(hptypes, null);
|
||
|
||
// Set the Move ComboBoxes too..
|
||
{
|
||
var moves = Util.getCBList(movelist, null);
|
||
foreach (ComboBox cb in new[] { CB_Move1, CB_Move2, CB_Move3, CB_Move4, CB_RelearnMove1, CB_RelearnMove2, CB_RelearnMove3, CB_RelearnMove4 })
|
||
{
|
||
cb.DisplayMember = "Text"; cb.ValueMember = "Value";
|
||
cb.DataSource = new BindingSource(moves, null);
|
||
}
|
||
}
|
||
}
|
||
private void populateFields(byte[] data)
|
||
{
|
||
// Store all loaded data in a persistent buffer for easy access.
|
||
Array.Resize(ref buff, data.Length);
|
||
Array.Copy(data, buff, data.Length);
|
||
|
||
init = false;
|
||
CAL_EggDate.Value = new DateTime(2000, 01, 01);
|
||
Tab_Main.Focus();
|
||
// Encryption Constant
|
||
TB_EC.Text = BitConverter.ToUInt32(buff, 0).ToString("X8");
|
||
|
||
// Block A
|
||
int species = BitConverter.ToInt16(buff, 0x08);
|
||
int helditem = BitConverter.ToUInt16(buff, 0x0A);
|
||
uint TID = BitConverter.ToUInt16(buff, 0x0C);
|
||
uint SID = BitConverter.ToUInt16(buff, 0x0E);
|
||
uint exp = BitConverter.ToUInt32(buff, 0x10);
|
||
int ability = buff[0x14];
|
||
int abilitynum = buff[0x15];
|
||
// 0x16, 0x17 - Training Bags handled by the Ribbon Editor
|
||
uint PID = BitConverter.ToUInt32(buff, 0x18);
|
||
int nature = buff[0x1C];
|
||
int feflag = buff[0x1D] % 2;
|
||
int genderflag = (buff[0x1D] >> 1) & 0x3;
|
||
int altforms = (buff[0x1D] >> 3);
|
||
int HP_EV = buff[0x1E];
|
||
int ATK_EV = buff[0x1F];
|
||
int DEF_EV = buff[0x20];
|
||
int SPA_EV = buff[0x22];
|
||
int SPD_EV = buff[0x23];
|
||
int SPE_EV = buff[0x21];
|
||
int cnt_cool = buff[0x24];
|
||
int cnt_beauty = buff[0x25];
|
||
int cnt_cute = buff[0x26];
|
||
int cnt_smart = buff[0x27];
|
||
int cnt_tough = buff[0x28];
|
||
int cnt_sheen = buff[0x29];
|
||
int markings = buff[0x2A];
|
||
int PKRS_Strain = buff[0x2B] >> 4;
|
||
int PKRS_Duration = buff[0x2B] % 0x10;
|
||
|
||
// Medals and Ribbons, passed with buff to new form
|
||
// 0x2C, 0x2D, 0x2E, 0x2F
|
||
// 0x33, 0x34, 0x35, 0x36
|
||
// 0x34, 0x35, 0x36, 0x37
|
||
// 0x38, 0x39, 0x3A
|
||
|
||
// 0x3B, 0x3C, 0x3D, 0x3E, 0x3F - Unused/Unknown
|
||
|
||
// Block B
|
||
string nicknamestr = Util.TrimFromZero(Encoding.Unicode.GetString(buff, 0x40, 24));
|
||
// 0x58, 0x59 - unused
|
||
int move1 = BitConverter.ToInt16(buff, 0x5A);
|
||
int move2 = BitConverter.ToInt16(buff, 0x5C);
|
||
int move3 = BitConverter.ToInt16(buff, 0x5E);
|
||
int move4 = BitConverter.ToInt16(buff, 0x60);
|
||
int move1_pp = buff[0x62];
|
||
int move2_pp = buff[0x63];
|
||
int move3_pp = buff[0x64];
|
||
int move4_pp = buff[0x65];
|
||
int move1_ppu = buff[0x66];
|
||
int move2_ppu = buff[0x67];
|
||
int move3_ppu = buff[0x68];
|
||
int move4_ppu = buff[0x69];
|
||
int eggmove1 = BitConverter.ToInt16(buff, 0x6A);
|
||
int eggmove2 = BitConverter.ToInt16(buff, 0x6C);
|
||
int eggmove3 = BitConverter.ToInt16(buff, 0x6E);
|
||
int eggmove4 = BitConverter.ToInt16(buff, 0x70);
|
||
|
||
// 0x72 - Super Training Flag - Passed with buff to new form
|
||
|
||
// 0x73 - Unused/Unknown
|
||
uint IV32 = BitConverter.ToUInt32(buff, 0x74);
|
||
uint HP_IV = IV32 & 0x1F;
|
||
uint ATK_IV = (IV32 >> 5) & 0x1F;
|
||
uint DEF_IV = (IV32 >> 10) & 0x1F;
|
||
uint SPE_IV = (IV32 >> 15) & 0x1F;
|
||
uint SPA_IV = (IV32 >> 20) & 0x1F;
|
||
uint SPD_IV = (IV32 >> 25) & 0x1F;
|
||
uint isegg = (IV32 >> 30) & 1;
|
||
uint isnick = (IV32 >> 31);
|
||
|
||
// Block C
|
||
string notOT = Util.TrimFromZero(Encoding.Unicode.GetString(buff, 0x78, 24));
|
||
bool notOTG = Convert.ToBoolean(buff[0x92]);
|
||
// Memory Editor edits everything else with buff in a new form
|
||
|
||
// Block D
|
||
string ot = Util.TrimFromZero(Encoding.Unicode.GetString(buff, 0xB0, 24));
|
||
// 0xC8, 0xC9 - unused
|
||
int OTfriendship = buff[0xCA];
|
||
// int OTaffection = buff[0xCB]; // Handled by Memory Editor
|
||
// 0xCC, 0xCD, 0xCE, 0xCF, 0xD0
|
||
int egg_year = buff[0xD1];
|
||
int egg_month = buff[0xD2];
|
||
int egg_day = buff[0xD3];
|
||
int met_year = buff[0xD4];
|
||
int met_month = buff[0xD5];
|
||
int met_day = buff[0xD6];
|
||
// 0xD7 - unused
|
||
int eggloc = BitConverter.ToUInt16(buff, 0xD8);
|
||
int metloc = BitConverter.ToUInt16(buff, 0xDA);
|
||
int ball = buff[0xDC];
|
||
int metlevel = buff[0xDD] & 0x7F;
|
||
int otgender = (buff[0xDD]) >> 7;
|
||
int encountertype = buff[0xDE];
|
||
int gamevers = buff[0xDF];
|
||
int countryID = buff[0xE0];
|
||
int regionID = buff[0xE1];
|
||
int dsregID = buff[0xE2];
|
||
int otlang = buff[0xE3];
|
||
// 0xE4, 0xE5, 0xE6, 0xE7 - unused
|
||
|
||
//
|
||
// Populate Fields
|
||
//
|
||
|
||
CHK_Fateful.Checked = Convert.ToBoolean(feflag);
|
||
CHK_IsEgg.Checked = Convert.ToBoolean(isegg);
|
||
CHK_Nicknamed.Checked = Convert.ToBoolean(isnick);
|
||
Label_OTGender.Text = gendersymbols[otgender];
|
||
|
||
// Private Use Character Fixing Text
|
||
{
|
||
nicknamestr = nicknamestr.Replace("\uE08F", "\u2640");
|
||
nicknamestr = nicknamestr.Replace("\uE08E", "\u2642");
|
||
}
|
||
|
||
// Set Markings
|
||
CHK_Circle.Checked = ((markings >> 0) & 1) == 1;
|
||
CHK_Triangle.Checked = ((markings >> 1) & 1) == 1;
|
||
CHK_Square.Checked = ((markings >> 2) & 1) == 1;
|
||
CHK_Heart.Checked = ((markings >> 3) & 1) == 1;
|
||
CHK_Star.Checked = ((markings >> 4) & 1) == 1;
|
||
CHK_Diamond.Checked = ((markings >> 5) & 1) == 1;
|
||
|
||
// Set Generic Identification Data
|
||
TB_PID.Text = PID.ToString("X8");
|
||
CB_Species.SelectedValue = species;
|
||
CB_HeldItem.SelectedValue = helditem;
|
||
setAbilityList(TB_AbilityNumber, species, CB_Ability, CB_Form);
|
||
TB_AbilityNumber.Text = abilitynum.ToString();
|
||
CB_Ability.SelectedIndex = (abilitynum < 6) ? abilitynum >> 1 : 0; // with some simple error handling
|
||
CB_Nature.SelectedValue = nature;
|
||
|
||
TB_EXP.Text = exp.ToString();
|
||
TB_TID.Text = TID.ToString("00000");
|
||
TB_SID.Text = SID.ToString("00000");
|
||
|
||
TB_OT.Text = ot;
|
||
TB_Nickname.Text = nicknamestr;
|
||
TB_OTt2.Text = notOT;
|
||
|
||
if (buff[0x93] == 1) // = 1
|
||
{
|
||
TB_Friendship.Text = buff[0xA2].ToString();
|
||
GB_nOT.BackgroundImage = mixedHighlight;
|
||
GB_OT.BackgroundImage = null;
|
||
}
|
||
else // = 0
|
||
{
|
||
TB_Friendship.Text = OTfriendship.ToString();
|
||
GB_OT.BackgroundImage = mixedHighlight;
|
||
GB_nOT.BackgroundImage = null;
|
||
}
|
||
|
||
CB_Language.SelectedValue = otlang;
|
||
CB_Country.SelectedValue = countryID;
|
||
CB_SubRegion.SelectedValue = regionID;
|
||
CB_3DSReg.SelectedValue = dsregID;
|
||
CB_GameOrigin.SelectedValue = gamevers;
|
||
CB_EncounterType.SelectedValue = encountertype;
|
||
CB_Ball.SelectedValue = ball;
|
||
|
||
if (met_month == 0) { met_month = 1; }
|
||
if (met_day == 0) { met_day = 1; }
|
||
try { CAL_MetDate.Value = new DateTime(met_year + 2000, met_month, met_day); }
|
||
catch { CAL_MetDate.Value = new DateTime(2000, 1, 1); }
|
||
|
||
if (eggloc != 0)
|
||
{
|
||
// Was obtained initially as an egg.
|
||
CHK_AsEgg.Checked = true;
|
||
GB_EggConditions.Enabled = true;
|
||
|
||
CB_EggLocation.SelectedValue = eggloc;
|
||
try { CAL_EggDate.Value = new DateTime(egg_year + 2000, egg_month, egg_day); }
|
||
catch { CAL_MetDate.Value = new DateTime(2000, 1, 1); }
|
||
}
|
||
else { CHK_AsEgg.Checked = GB_EggConditions.Enabled = false; CB_EggLocation.SelectedValue = 0; }
|
||
|
||
CB_MetLocation.SelectedValue = metloc;
|
||
|
||
// Set CT Gender to None if no CT, else set to gender symbol.
|
||
Label_CTGender.Text = ((TB_OTt2.Text == "") || (notOT == "")) ? "" : (notOTG) ? gendersymbols[1] : gendersymbols[0];
|
||
|
||
TB_MetLevel.Text = metlevel.ToString();
|
||
|
||
// Reset Label and ComboBox visibility, as well as non-data checked status.
|
||
Label_PKRS.Visible = CB_PKRSStrain.Visible = CHK_Infected.Checked = PKRS_Strain != 0;
|
||
Label_PKRSdays.Visible = CB_PKRSDays.Visible = PKRS_Duration != 0;
|
||
|
||
// Set SelectedIndexes for PKRS
|
||
CB_PKRSStrain.SelectedIndex = PKRS_Strain;
|
||
CHK_Cured.Checked = (PKRS_Strain > 0 && PKRS_Duration == 0);
|
||
CB_PKRSDays.SelectedIndex = Math.Min(CB_PKRSDays.Items.Count - 1, PKRS_Duration); // to strip out bad hacked 'rus
|
||
|
||
TB_Cool.Text = cnt_cool.ToString();
|
||
TB_Beauty.Text = cnt_beauty.ToString();
|
||
TB_Cute.Text = cnt_cute.ToString();
|
||
TB_Smart.Text = cnt_smart.ToString();
|
||
TB_Tough.Text = cnt_tough.ToString();
|
||
TB_Sheen.Text = cnt_sheen.ToString();
|
||
|
||
TB_HPIV.Text = HP_IV.ToString();
|
||
TB_ATKIV.Text = ATK_IV.ToString();
|
||
TB_DEFIV.Text = DEF_IV.ToString();
|
||
TB_SPAIV.Text = SPA_IV.ToString();
|
||
TB_SPDIV.Text = SPD_IV.ToString();
|
||
TB_SPEIV.Text = SPE_IV.ToString();
|
||
|
||
TB_HPEV.Text = HP_EV.ToString();
|
||
TB_ATKEV.Text = ATK_EV.ToString();
|
||
TB_DEFEV.Text = DEF_EV.ToString();
|
||
TB_SPAEV.Text = SPA_EV.ToString();
|
||
TB_SPDEV.Text = SPD_EV.ToString();
|
||
TB_SPEEV.Text = SPE_EV.ToString();
|
||
|
||
CB_Move1.SelectedValue = move1;
|
||
CB_Move2.SelectedValue = move2;
|
||
CB_Move3.SelectedValue = move3;
|
||
CB_Move4.SelectedValue = move4;
|
||
CB_RelearnMove1.SelectedValue = eggmove1;
|
||
CB_RelearnMove2.SelectedValue = eggmove2;
|
||
CB_RelearnMove3.SelectedValue = eggmove3;
|
||
CB_RelearnMove4.SelectedValue = eggmove4;
|
||
CB_PPu1.SelectedIndex = move1_ppu;
|
||
CB_PPu2.SelectedIndex = move2_ppu;
|
||
CB_PPu3.SelectedIndex = move3_ppu;
|
||
CB_PPu4.SelectedIndex = move4_ppu;
|
||
TB_PP1.Text = move1_pp.ToString();
|
||
TB_PP2.Text = move2_pp.ToString();
|
||
TB_PP3.Text = move3_pp.ToString();
|
||
TB_PP4.Text = move4_pp.ToString();
|
||
|
||
int level = PKX.getLevel(species, ref exp);
|
||
TB_Level.Text = level.ToString();
|
||
|
||
// Set Form if count is enough, else if count is more than 1 set equal to max else zero.
|
||
CB_Form.SelectedIndex = (CB_Form.Items.Count > altforms) ? altforms : (CB_Form.Items.Count > 1) ? CB_Form.Items.Count - 1 : 0;
|
||
|
||
// Load Extrabyte Value
|
||
TB_ExtraByte.Text = buff[Convert.ToInt32(CB_ExtraBytes.Text, 16)].ToString();
|
||
|
||
// Load Gender Flag
|
||
Label_Gender.Text = gendersymbols[genderflag];
|
||
updateStats();
|
||
setIsShiny();
|
||
|
||
CB_EncounterType.Visible = Label_EncounterType.Visible = !(gamevers > 12 || gamevers < 7);
|
||
if (gamevers > 12 || gamevers < 7)
|
||
CB_EncounterType.SelectedValue = 0;
|
||
|
||
init = true;
|
||
updatePKRSInfected(null, null);
|
||
updatePKRSCured(null, null);
|
||
|
||
if (HaX) // DEV Illegality
|
||
{
|
||
DEV_Ability.SelectedValue = ability;
|
||
MT_Level.Text = level.ToString();
|
||
MT_Form.Text = altforms.ToString();
|
||
}
|
||
|
||
// Set the Preview Box
|
||
getQuickFiller(dragout);
|
||
|
||
// Highlight the Current Handler
|
||
clickGT((buff[0x93] == 1) ? GB_nOT : GB_OT, null);
|
||
|
||
if (!PKX.verifychk(buff))
|
||
Util.Alert("PKX File has an invalid checksum.");
|
||
}
|
||
// General Use Functions shared by other Forms //
|
||
public void setCountrySubRegion(ComboBox CB, string type)
|
||
{
|
||
int index = CB.SelectedIndex;
|
||
// fix for Korean / Chinese being swapped
|
||
string cl = curlanguage + "";
|
||
cl = (cl == "zh") ? "ko" : (cl == "ko") ? "zh" : cl;
|
||
|
||
CB.DataSource = Util.getCBList(type, cl);
|
||
|
||
if (index > 0 && index < CB.Items.Count && init)
|
||
CB.SelectedIndex = index;
|
||
}
|
||
public void setForms(int species, ComboBox cb, Label l = null)
|
||
{
|
||
// Form Tables
|
||
cb.DisplayMember = "Text";
|
||
cb.ValueMember = "Value";
|
||
PKX.PersonalParser.Personal MonData = PKX.PersonalGetter.GetPersonal(species);
|
||
bool hasForms = !(MonData.AltFormCount == 0 && species != 664 && species != 665); // If no forms & not Scatterbug / Spewpa...
|
||
cb.Enabled = cb.Visible = hasForms;
|
||
if (l != null) l.Visible = hasForms;
|
||
|
||
cb.DataSource = PKX.getFormList(species, types, forms, gendersymbols).ToList();
|
||
}
|
||
public void setAbilityList(MaskedTextBox tb_abil, int species, ComboBox cb_abil, ComboBox cb_forme)
|
||
{
|
||
if (!init && tb_abil.Text == "")
|
||
return;
|
||
int newabil = Convert.ToInt16(tb_abil.Text) >> 1;
|
||
|
||
int form = cb_forme.SelectedIndex;
|
||
byte[] abils = PKX.getAbilities(species, form);
|
||
|
||
// Build Ability List
|
||
List<string> ability_list = new List<string>
|
||
{
|
||
abilitylist[abils[0]] + " (1)",
|
||
abilitylist[abils[1]] + " (2)",
|
||
abilitylist[abils[2]] + " (H)"
|
||
};
|
||
cb_abil.DataSource = ability_list;
|
||
|
||
cb_abil.SelectedIndex = newabil < 3 ? newabil : 0;
|
||
}
|
||
// PKX Data Calculation Functions //
|
||
private void setIsShiny()
|
||
{
|
||
bool isShiny = PKX.getIsShiny(Util.getHEXval(TB_PID), Util.ToUInt32(TB_TID.Text), Util.ToUInt32(TB_SID.Text));
|
||
|
||
// Set the Controls
|
||
BTN_Shinytize.Visible = BTN_Shinytize.Enabled = !isShiny;
|
||
Label_IsShiny.Visible = isShiny;
|
||
|
||
// Refresh Markings (for Shiny Star if applicable)
|
||
setMarkings();
|
||
}
|
||
private void setMarkings()
|
||
{
|
||
PictureBox[] pba = { PB_Mark1, PB_Mark2, PB_Mark3, PB_Mark4, PB_Mark5, PB_Mark6 };
|
||
CheckBox[] cba = { CHK_Circle, CHK_Triangle, CHK_Square, CHK_Heart, CHK_Star, CHK_Diamond };
|
||
for (int i = 0; i < 6; i++)
|
||
pba[i].Image = Util.ChangeOpacity(pba[i].InitialImage, (Convert.ToUInt16(cba[i].Checked)) * 0.9 + 0.1);
|
||
|
||
PB_MarkShiny.Image = Util.ChangeOpacity(PB_MarkShiny.InitialImage, (Convert.ToUInt16(!BTN_Shinytize.Enabled)) * 0.9 + 0.1);
|
||
PB_MarkCured.Image = Util.ChangeOpacity(PB_MarkCured.InitialImage, (Convert.ToUInt16(CHK_Cured.Checked)) * 0.9 + 0.1);
|
||
int gameindex = Util.getIndex(CB_GameOrigin);
|
||
PB_MarkPentagon.Image = Util.ChangeOpacity(PB_MarkPentagon.InitialImage, (Convert.ToUInt16(gameindex == 24 || gameindex == 25 || gameindex == 26 || gameindex == 27)) * 0.9 + 0.1);
|
||
}
|
||
// Clicked Label Shortcuts //
|
||
private void clickQR(object sender, EventArgs e)
|
||
{
|
||
if (ModifierKeys == Keys.Alt)
|
||
{
|
||
// Fetch data from QR code...
|
||
byte[] ekx = Util.getQRData();
|
||
|
||
if (ekx == null) return;
|
||
|
||
if (ekx.Length != 232) { Util.Alert("Decoded data not 232 bytes.", String.Format("QR Data Size: {0}", ekx.Length)); }
|
||
else try
|
||
{
|
||
byte[] pkx = PKX.decryptArray(ekx);
|
||
if (PKX.verifychk(pkx)) { Array.Copy(pkx, buff, 0xE8); populateFields(buff); }
|
||
else Util.Alert("Invalid checksum in QR data.");
|
||
}
|
||
catch { Util.Alert("Error loading decrypted data."); }
|
||
}
|
||
else
|
||
{
|
||
if (!verifiedPKX()) return;
|
||
byte[] pkx = preparepkx();
|
||
byte[] ekx = PKX.encryptArray(pkx);
|
||
|
||
Array.Resize(ref ekx, 232);
|
||
const string server = "http://loadcode.projectpokemon.org/b1s1.html#"; // Rehosted with permission from LC/MS -- massive thanks!
|
||
Image qr = Util.getQRImage(ekx, server);
|
||
|
||
if (qr == null) return;
|
||
|
||
PKX data = new PKX(pkx, "Tabs");
|
||
string[] r = PKX.getPKXSummary(data);
|
||
new QR(qr, dragout.Image, r[0], r[1], r[2], "PKHeX @ ProjectPokemon.org").ShowDialog();
|
||
}
|
||
}
|
||
private void clickFriendship(object sender, EventArgs e)
|
||
{
|
||
if (ModifierKeys == Keys.Control) // prompt to reset
|
||
TB_Friendship.Text = buff[0x93] == 0 ? buff[0xCA].ToString() : buff[0xA2].ToString();
|
||
else
|
||
TB_Friendship.Text = TB_Friendship.Text == "255" ? PKX.getBaseFriendship(Util.getIndex(CB_Species)).ToString() : "255";
|
||
}
|
||
private void clickGender(object sender, EventArgs e)
|
||
{
|
||
// Get Gender Threshold
|
||
PKX.PersonalParser.Personal MonData = PKX.PersonalGetter.GetPersonal(Util.getIndex(CB_Species));
|
||
int gt = MonData.GenderRatio;
|
||
|
||
if (gt == 255 || gt == 0 || gt == 254) // Single gender/genderless
|
||
return;
|
||
|
||
if (gt >= 255) return;
|
||
// If not a single gender(less) species: (should be <254 but whatever, 255 never happens^)
|
||
Label_Gender.Text = gendersymbols[PKX.getGender(Label_Gender.Text) ^ 1];
|
||
|
||
if (PKX.getGender(CB_Form.Text) < 2) // Gendered Forms
|
||
CB_Form.SelectedIndex = PKX.getGender(Label_Gender.Text);
|
||
|
||
getQuickFiller(dragout);
|
||
}
|
||
private void clickPPUps(object sender, EventArgs e)
|
||
{
|
||
CB_PPu1.SelectedIndex = (ModifierKeys != Keys.Control && Util.getIndex(CB_Move1) > 0) ? 3 : 0;
|
||
CB_PPu2.SelectedIndex = (ModifierKeys != Keys.Control && Util.getIndex(CB_Move2) > 0) ? 3 : 0;
|
||
CB_PPu3.SelectedIndex = (ModifierKeys != Keys.Control && Util.getIndex(CB_Move3) > 0) ? 3 : 0;
|
||
CB_PPu4.SelectedIndex = (ModifierKeys != Keys.Control && Util.getIndex(CB_Move4) > 0) ? 3 : 0;
|
||
}
|
||
private void clickMarking(object sender, EventArgs e)
|
||
{
|
||
PictureBox[] pba = { PB_Mark1, PB_Mark2, PB_Mark3, PB_Mark4, PB_Mark5, PB_Mark6 };
|
||
CheckBox[] cba = { CHK_Circle, CHK_Triangle, CHK_Square, CHK_Heart, CHK_Star, CHK_Diamond };
|
||
|
||
CheckBox cb = cba[Array.IndexOf(pba, sender as PictureBox)];
|
||
cb.Checked = !cb.Checked;
|
||
setMarkings();
|
||
}
|
||
private void clickOT(object sender, EventArgs e)
|
||
{
|
||
string OT = Util.TrimFromZero(Encoding.Unicode.GetString(savefile, SaveGame.TrainerCard + 0x48 + savindex * 0x7F000, 0x1A));
|
||
if (OT.Length <= 0) return;
|
||
|
||
TB_OT.Text = OT;
|
||
int savshift = 0x7F000 * savindex;
|
||
// Set Gender Label
|
||
int g6trgend = savefile[SaveGame.TrainerCard + 0x5 + savshift];
|
||
Label_OTGender.Text = g6trgend == 1 ? gendersymbols[1] : gendersymbols[0];
|
||
|
||
// Get TID/SID
|
||
TB_TID.Text = BitConverter.ToUInt16(savefile, SaveGame.TrainerCard + 0 + savshift).ToString();
|
||
TB_SID.Text = BitConverter.ToUInt16(savefile, SaveGame.TrainerCard + 2 + savshift).ToString();
|
||
int game = savefile[SaveGame.TrainerCard + 0x4 + savshift];
|
||
int subreg = savefile[SaveGame.TrainerCard + 0x26 + savshift];
|
||
int country = savefile[SaveGame.TrainerCard + 0x27 + savshift];
|
||
int _3DSreg = savefile[SaveGame.TrainerCard + 0x2C + savshift];
|
||
int lang = savefile[SaveGame.TrainerCard + 0x2D + savshift];
|
||
|
||
// CB_GameOrigin.SelectedValue = game;
|
||
|
||
CB_GameOrigin.SelectedValue = game;
|
||
CB_SubRegion.SelectedValue = subreg;
|
||
CB_Country.SelectedValue = country;
|
||
CB_3DSReg.SelectedValue = _3DSreg;
|
||
CB_Language.SelectedValue = lang;
|
||
updateNickname(null, null);
|
||
}
|
||
private void clickCT(object sender, EventArgs e)
|
||
{
|
||
if (TB_OTt2.Text.Length > 0)
|
||
Label_CTGender.Text = gendersymbols[savefile[0x19405 + savindex * 0x7F000]];
|
||
}
|
||
private void clickGT(object sender, EventArgs e)
|
||
{
|
||
if (sender as GroupBox == GB_OT)
|
||
{
|
||
buff[0x93] = 0;
|
||
TB_Friendship.Text = buff[0xCA].ToString();
|
||
GB_OT.BackgroundImage = mixedHighlight;
|
||
GB_nOT.BackgroundImage = null;
|
||
}
|
||
else if (TB_OTt2.Text.Length > 0)
|
||
{
|
||
buff[0x93] = 1;
|
||
TB_Friendship.Text = buff[0xA2].ToString();
|
||
GB_OT.BackgroundImage = null;
|
||
GB_nOT.BackgroundImage = mixedHighlight;
|
||
}
|
||
}
|
||
private void clickTRGender(object sender, EventArgs e)
|
||
{
|
||
Label lbl = sender as Label;
|
||
if (lbl.Text != "") // set gender label (toggle M/F)
|
||
lbl.Text = (PKX.getGender(lbl.Text) == 0) ? gendersymbols[1] : gendersymbols[0];
|
||
}
|
||
private void clickMoves(object sender, EventArgs e)
|
||
{
|
||
if (DialogResult.Yes != Util.Prompt(MessageBoxButtons.YesNo, "Copy current moves to Relearn Moves?"))
|
||
return;
|
||
|
||
CB_RelearnMove1.SelectedIndex = CB_Move1.SelectedIndex > 0 ? CB_Move1.SelectedIndex : 0;
|
||
CB_RelearnMove2.SelectedIndex = CB_Move2.SelectedIndex > 0 ? CB_Move2.SelectedIndex : 0;
|
||
CB_RelearnMove3.SelectedIndex = CB_Move3.SelectedIndex > 0 ? CB_Move3.SelectedIndex : 0;
|
||
CB_RelearnMove4.SelectedIndex = CB_Move4.SelectedIndex > 0 ? CB_Move4.SelectedIndex : 0;
|
||
}
|
||
// Prompted Updates of PKX Functions //
|
||
private bool changingFields;
|
||
private void updateEXPLevel(object sender, EventArgs e)
|
||
{
|
||
if (changingFields) return;
|
||
|
||
if (sender as MaskedTextBox == TB_EXP)
|
||
{
|
||
changingFields = true;
|
||
// Change the Level
|
||
uint exp = Util.ToUInt32(TB_EXP);
|
||
int level = (Util.ToInt32(TB_EXP.Text) == 0) ? 1 : PKX.getLevel(Util.getIndex(CB_Species), ref exp);
|
||
|
||
TB_Level.Text = level.ToString();
|
||
if (!MT_Level.Visible)
|
||
TB_EXP.Text = exp.ToString();
|
||
else
|
||
MT_Level.Text = level.ToString();
|
||
}
|
||
else
|
||
{
|
||
changingFields = true;
|
||
// Change the XP
|
||
int level = Util.ToInt32((MT_Level.Focused ? MT_Level : TB_Level).Text);
|
||
if (level > 100) TB_Level.Text = "100";
|
||
if (level > 255) MT_Level.Text = "255";
|
||
|
||
TB_EXP.Text = PKX.getEXP(level, Util.getIndex(CB_Species)).ToString();
|
||
}
|
||
changingFields = false;
|
||
updateStats();
|
||
}
|
||
private void updateHPType(object sender, EventArgs e)
|
||
{
|
||
if (changingFields) return;
|
||
changingFields = true;
|
||
int[] ivs =
|
||
{
|
||
Util.ToInt32(TB_HPIV.Text), Util.ToInt32(TB_ATKIV.Text), Util.ToInt32(TB_DEFIV.Text),
|
||
Util.ToInt32(TB_SPAIV.Text), Util.ToInt32(TB_SPDIV.Text), Util.ToInt32(TB_SPEIV.Text)
|
||
};
|
||
|
||
// Change IVs to match the new Hidden Power
|
||
int[] newIVs = PKX.setHPIVs(Util.getIndex(CB_HPType), ivs);
|
||
TB_HPIV.Text = newIVs[0].ToString();
|
||
TB_ATKIV.Text = newIVs[1].ToString();
|
||
TB_DEFIV.Text = newIVs[2].ToString();
|
||
|
||
TB_SPAIV.Text = newIVs[3].ToString();
|
||
TB_SPDIV.Text = newIVs[4].ToString();
|
||
TB_SPEIV.Text = newIVs[5].ToString();
|
||
|
||
// Refresh View
|
||
changingFields = false;
|
||
updateIVs(null, null);
|
||
}
|
||
private void updateIVs(object sender, EventArgs e)
|
||
{
|
||
if (changingFields) return;
|
||
if (sender != null)
|
||
if (Util.ToInt32((sender as MaskedTextBox).Text) > 31)
|
||
(sender as MaskedTextBox).Text = "31";
|
||
|
||
int[] ivs =
|
||
{
|
||
Util.ToInt32(TB_HPIV.Text), Util.ToInt32(TB_ATKIV.Text), Util.ToInt32(TB_DEFIV.Text),
|
||
Util.ToInt32(TB_SPAIV.Text), Util.ToInt32(TB_SPDIV.Text), Util.ToInt32(TB_SPEIV.Text)
|
||
};
|
||
|
||
changingFields = true;
|
||
CB_HPType.SelectedValue = PKX.getHPType(ivs);
|
||
changingFields = false;
|
||
|
||
int ivtotal = ivs.Sum();
|
||
TB_IVTotal.Text = ivtotal.ToString();
|
||
|
||
// Potential Reading
|
||
if (!unicode)
|
||
{
|
||
if (ivtotal <= 90)
|
||
L_Potential.Text = "★☆☆☆";
|
||
else if (ivtotal <= 120)
|
||
L_Potential.Text = "★★☆☆";
|
||
else if (ivtotal <= 150)
|
||
L_Potential.Text = "★★★☆";
|
||
else
|
||
L_Potential.Text = "★★★★";
|
||
}
|
||
else
|
||
{
|
||
if (ivtotal <= 90)
|
||
L_Potential.Text = "+";
|
||
else if (ivtotal <= 120)
|
||
L_Potential.Text = "++";
|
||
else if (ivtotal <= 150)
|
||
L_Potential.Text = "+++";
|
||
else
|
||
L_Potential.Text = "++++";
|
||
}
|
||
|
||
// Characteristic with EC%6
|
||
int pm6 = (int)(Util.getHEXval(TB_EC) % 6); // EC MOD 6
|
||
int maxIV = ivs.Max();
|
||
int pm6stat = 0;
|
||
|
||
for (int i = 0; i < 6; i++)
|
||
{
|
||
pm6stat = (pm6 + i) % 6;
|
||
if (ivs[pm6stat] == maxIV)
|
||
break; // P%6 is this stat
|
||
}
|
||
|
||
L_Characteristic.Text = characteristics[new[]{0, 1, 2, 4, 5, 3}[pm6stat] * 5 + maxIV % 5];
|
||
updateStats();
|
||
}
|
||
private void updateEVs(object sender, EventArgs e)
|
||
{
|
||
if (sender != null)
|
||
if (Util.ToInt32((sender as MaskedTextBox).Text) > 252)
|
||
(sender as MaskedTextBox).Text = "252";
|
||
|
||
int HP_EV = Util.ToInt32(TB_HPEV.Text);
|
||
int ATK_EV = Util.ToInt32(TB_ATKEV.Text);
|
||
int DEF_EV = Util.ToInt32(TB_DEFEV.Text);
|
||
int SPA_EV = Util.ToInt32(TB_SPAEV.Text);
|
||
int SPD_EV = Util.ToInt32(TB_SPDEV.Text);
|
||
int SPE_EV = Util.ToInt32(TB_SPEEV.Text);
|
||
|
||
int evtotal = HP_EV + ATK_EV + DEF_EV + SPA_EV + SPD_EV + SPE_EV;
|
||
|
||
if (evtotal > 510) // Background turns Red
|
||
TB_EVTotal.BackColor = Color.Red;
|
||
else if (evtotal == 510) // Maximum EVs
|
||
TB_EVTotal.BackColor = Color.Honeydew;
|
||
else TB_EVTotal.BackColor = Color.WhiteSmoke;
|
||
|
||
TB_EVTotal.Text = evtotal.ToString();
|
||
updateStats();
|
||
}
|
||
private void updateRandomIVs(object sender, EventArgs e)
|
||
{
|
||
if (ModifierKeys == Keys.Control || ModifierKeys == Keys.Shift)
|
||
{
|
||
// Max IVs
|
||
TB_HPIV.Text = 31.ToString();
|
||
TB_ATKIV.Text = 31.ToString();
|
||
TB_DEFIV.Text = 31.ToString();
|
||
TB_SPAIV.Text = 31.ToString();
|
||
TB_SPDIV.Text = 31.ToString();
|
||
TB_SPEIV.Text = 31.ToString();
|
||
}
|
||
else
|
||
{
|
||
TB_HPIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
TB_ATKIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
TB_DEFIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
TB_SPAIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
TB_SPDIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
TB_SPEIV.Text = (Util.rnd32() & 0x1F).ToString();
|
||
}
|
||
}
|
||
private void updateRandomEVs(object sender, EventArgs e)
|
||
{
|
||
if (ModifierKeys == Keys.Control || ModifierKeys == Keys.Shift)
|
||
{
|
||
// Max IVs
|
||
TB_HPEV.Text = 0.ToString();
|
||
TB_ATKEV.Text = 0.ToString();
|
||
TB_DEFEV.Text = 0.ToString();
|
||
TB_SPAEV.Text = 0.ToString();
|
||
TB_SPDEV.Text = 0.ToString();
|
||
TB_SPEEV.Text = 0.ToString();
|
||
}
|
||
else
|
||
{
|
||
byte[] evs = PKX.getRandomEVs();
|
||
TB_HPEV.Text = evs[0].ToString();
|
||
TB_ATKEV.Text = evs[1].ToString();
|
||
TB_DEFEV.Text = evs[2].ToString();
|
||
TB_SPAEV.Text = evs[3].ToString();
|
||
TB_SPDEV.Text = evs[4].ToString();
|
||
TB_SPEEV.Text = evs[5].ToString();
|
||
}
|
||
}
|
||
private void updateRandomPID(object sender, EventArgs e)
|
||
{
|
||
TB_PID.Text = PKX.getRandomPID(Util.getIndex(CB_Species), PKX.getGender(Label_Gender.Text)).ToString("X8");
|
||
getQuickFiller(dragout);
|
||
}
|
||
private void updateRandomEC(object sender, EventArgs e)
|
||
{
|
||
TB_EC.Text = Util.rnd32().ToString("X8");
|
||
}
|
||
private void updateHackedStats(object sender, EventArgs e)
|
||
{
|
||
Stat_HP.Enabled =
|
||
Stat_ATK.Enabled =
|
||
Stat_DEF.Enabled =
|
||
Stat_SPA.Enabled =
|
||
Stat_SPD.Enabled =
|
||
Stat_SPE.Enabled = CHK_HackedStats.Checked;
|
||
}
|
||
private void update255_MTB(object sender, EventArgs e)
|
||
{
|
||
MaskedTextBox mtb = sender as MaskedTextBox;
|
||
try
|
||
{
|
||
if (Util.ToInt32((sender as MaskedTextBox).Text) > 255)
|
||
(sender as MaskedTextBox).Text = "255";
|
||
}
|
||
catch { mtb.Text = "0"; }
|
||
}
|
||
private void update255_TB(object sender, EventArgs e)
|
||
{
|
||
TextBox tb = sender as TextBox;
|
||
try
|
||
{
|
||
if (Util.ToInt32((sender as TextBox).Text) > 255)
|
||
(sender as TextBox).Text = "255";
|
||
}
|
||
catch { tb.Text = "0"; }
|
||
}
|
||
private void updateForm(object sender, EventArgs e)
|
||
{
|
||
updateStats();
|
||
// Repopulate Abilities if Species Form has different abilities
|
||
setAbilityList(TB_AbilityNumber, Util.getIndex(CB_Species), CB_Ability, CB_Form);
|
||
|
||
// Gender Forms
|
||
if (PKX.getGender(CB_Form.Text) < 2 && Util.getIndex(CB_Species) != 201) // don't do this for Unown
|
||
Label_Gender.Text = CB_Form.Text;
|
||
}
|
||
private void updatePP(object sender, EventArgs e)
|
||
{
|
||
TB_PP1.Text = (PKX.getMovePP(Util.getIndex(CB_Move1), CB_PPu1.SelectedIndex)).ToString();
|
||
TB_PP2.Text = (PKX.getMovePP(Util.getIndex(CB_Move2), CB_PPu2.SelectedIndex)).ToString();
|
||
TB_PP3.Text = (PKX.getMovePP(Util.getIndex(CB_Move3), CB_PPu3.SelectedIndex)).ToString();
|
||
TB_PP4.Text = (PKX.getMovePP(Util.getIndex(CB_Move4), CB_PPu4.SelectedIndex)).ToString();
|
||
}
|
||
private void updatePKRSstrain(object sender, EventArgs e)
|
||
{
|
||
// Change the PKRS Days to the legal bounds.
|
||
int currentDuration = CB_PKRSDays.SelectedIndex;
|
||
CB_PKRSDays.Items.Clear();
|
||
int[] days = Enumerable.Range(0, CB_PKRSStrain.SelectedIndex % 4 + 2).Select(i => i).ToArray();
|
||
foreach (int day in days) CB_PKRSDays.Items.Add(day);
|
||
|
||
// Set the days back if they're legal, else set it to 1. (0 always passes).
|
||
CB_PKRSDays.SelectedIndex = (currentDuration < CB_PKRSDays.Items.Count) ? currentDuration : 1;
|
||
|
||
if (CB_PKRSStrain.SelectedIndex != 0) return;
|
||
|
||
// Never Infected
|
||
CB_PKRSDays.SelectedIndex = 0;
|
||
CHK_Cured.Checked = false;
|
||
CHK_Infected.Checked = false;
|
||
}
|
||
private void updatePKRSdays(object sender, EventArgs e)
|
||
{
|
||
if (CB_PKRSDays.SelectedIndex != 0) return;
|
||
|
||
// If no days are selected
|
||
if (CB_PKRSStrain.SelectedIndex == 0)
|
||
{
|
||
// Never Infected
|
||
CHK_Cured.Checked = false;
|
||
CHK_Infected.Checked = false;
|
||
}
|
||
else CHK_Cured.Checked = true;
|
||
}
|
||
private void updatePKRSCured(object sender, EventArgs e)
|
||
{
|
||
if (!init) return;
|
||
// Cured PokeRus is toggled
|
||
if (CHK_Cured.Checked)
|
||
{
|
||
// Has Had PokeRus
|
||
Label_PKRSdays.Visible = CB_PKRSDays.Visible = false;
|
||
CB_PKRSDays.SelectedIndex = 0;
|
||
|
||
Label_PKRS.Visible = CB_PKRSStrain.Visible = true;
|
||
CHK_Infected.Checked = true;
|
||
|
||
// If we're cured we have to have a strain infection.
|
||
if (CB_PKRSStrain.SelectedIndex == 0)
|
||
CB_PKRSStrain.SelectedIndex = 1;
|
||
}
|
||
else if (!CHK_Infected.Checked)
|
||
{
|
||
// Not Infected, Disable the other
|
||
Label_PKRS.Visible = CB_PKRSStrain.Visible = false;
|
||
CB_PKRSStrain.SelectedIndex = 0;
|
||
}
|
||
else
|
||
{
|
||
// Still Infected for a duration
|
||
Label_PKRSdays.Visible = CB_PKRSDays.Visible = true;
|
||
CB_PKRSDays.SelectedValue = 1;
|
||
}
|
||
// if not cured yet, days > 0
|
||
if (!CHK_Cured.Checked && CHK_Infected.Checked && CB_PKRSDays.SelectedIndex == 0)
|
||
CB_PKRSDays.SelectedIndex++;
|
||
|
||
setMarkings();
|
||
}
|
||
private void updatePKRSInfected(object sender, EventArgs e)
|
||
{
|
||
if (!init) return;
|
||
if (CHK_Cured.Checked && !CHK_Infected.Checked) { CHK_Cured.Checked = false; return; }
|
||
if (CHK_Cured.Checked) return;
|
||
Label_PKRS.Visible = CB_PKRSStrain.Visible = CHK_Infected.Checked;
|
||
if (!CHK_Infected.Checked) { CB_PKRSStrain.SelectedIndex = 0; CB_PKRSDays.SelectedIndex = 0; Label_PKRSdays.Visible = CB_PKRSDays.Visible = false; }
|
||
else if (CB_PKRSStrain.SelectedIndex == 0) { CB_PKRSStrain.SelectedIndex = 1; Label_PKRSdays.Visible = CB_PKRSDays.Visible = true; updatePKRSCured(sender, e);}
|
||
|
||
// if not cured yet, days > 0
|
||
if (CHK_Infected.Checked && CB_PKRSDays.SelectedIndex == 0) CB_PKRSDays.SelectedIndex++;
|
||
}
|
||
private void updateCountry(object sender, EventArgs e)
|
||
{
|
||
if (Util.getIndex(sender as ComboBox) > 0)
|
||
setCountrySubRegion(CB_SubRegion, "sr_" + Util.getIndex(sender as ComboBox).ToString("000"));
|
||
}
|
||
private void updateSpecies(object sender, EventArgs e)
|
||
{
|
||
// Change Species Prompted
|
||
int species = Util.getIndex(CB_Species);
|
||
int level = Util.ToInt32(TB_Level.Text);
|
||
if (MT_Level.Visible) level = Util.ToInt32(MT_Level.Text);
|
||
|
||
// Get Forms for Given Species
|
||
setForms(species, CB_Form, Label_Form);
|
||
|
||
// Recalculate EXP for Given Level
|
||
uint exp = PKX.getEXP(level, species);
|
||
TB_EXP.Text = exp.ToString();
|
||
|
||
// Check for Gender Changes
|
||
// Get Gender Threshold
|
||
species = Util.getIndex(CB_Species);
|
||
PKX.PersonalParser.Personal MonData = PKX.PersonalGetter.GetPersonal(species);
|
||
int gt = MonData.GenderRatio;
|
||
int genderflag;
|
||
|
||
if (gt == 255) // Genderless
|
||
genderflag = 2;
|
||
else if (gt == 254) // Female Only
|
||
genderflag = 1;
|
||
else if (gt == 0) // Male Only
|
||
genderflag = 0;
|
||
else // get gender from old PID correlation
|
||
genderflag = ((Util.getHEXval(TB_PID) & 0xFF) <= gt) ? 1 : 0;
|
||
|
||
Label_Gender.Text = gendersymbols[genderflag];
|
||
setAbilityList(TB_AbilityNumber, Util.getIndex(CB_Species), CB_Ability, CB_Form);
|
||
updateForm(null, null);
|
||
|
||
// If species changes and no nickname, set the new name == speciesName.
|
||
if (!CHK_Nicknamed.Checked)
|
||
updateNickname(sender, e);
|
||
}
|
||
private void updateOriginGame(object sender, EventArgs e)
|
||
{
|
||
int gameorigin = Util.getIndex(CB_GameOrigin);
|
||
|
||
if (gameorigin < 24 && origintrack != "Past")
|
||
{
|
||
// Load Past Gen Locations
|
||
#region B2W2 Met Locations
|
||
{
|
||
// Build up our met list
|
||
var met_list = Util.getCBList(metBW2_00000, new[] { 0 });
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_60000, 60001, new[] { 60002 });
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_30000, 30001, new[] { 30003 });
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_00000, 00000, Legal.Met_BW2_0);
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_30000, 30001, Legal.Met_BW2_3);
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_40000, 40001, Legal.Met_BW2_4);
|
||
met_list = Util.getOffsetCBList(met_list, metBW2_60000, 60001, Legal.Met_BW2_6);
|
||
CB_MetLocation.DisplayMember = "Text";
|
||
CB_MetLocation.ValueMember = "Value";
|
||
CB_MetLocation.DataSource = met_list;
|
||
CB_EggLocation.DisplayMember = "Text";
|
||
CB_EggLocation.ValueMember = "Value";
|
||
CB_EggLocation.DataSource = new BindingSource(met_list, null);
|
||
CB_EggLocation.SelectedValue = 0;
|
||
CB_MetLocation.SelectedValue = gameorigin < 20 ? 30001 : 60001;
|
||
origintrack = "Past";
|
||
}
|
||
#endregion
|
||
}
|
||
else if (gameorigin > 23 && (origintrack != "XY"))
|
||
{
|
||
// Load X/Y/OR/AS locations
|
||
#region ORAS Met Locations
|
||
{
|
||
// Build up our met list
|
||
var met_list = Util.getCBList(metXY_00000, new[] { 0 });
|
||
met_list = Util.getOffsetCBList(met_list, metXY_60000, 60001, new[] { 60002 });
|
||
met_list = Util.getOffsetCBList(met_list, metXY_30000, 30001, new[] { 30002 });
|
||
met_list = Util.getOffsetCBList(met_list, metXY_00000, 00000, Legal.Met_XY_0);
|
||
met_list = Util.getOffsetCBList(met_list, metXY_30000, 30001, Legal.Met_XY_3);
|
||
met_list = Util.getOffsetCBList(met_list, metXY_40000, 40001, Legal.Met_XY_4);
|
||
met_list = Util.getOffsetCBList(met_list, metXY_60000, 60001, Legal.Met_XY_6);
|
||
CB_MetLocation.DisplayMember = "Text";
|
||
CB_MetLocation.ValueMember = "Value";
|
||
CB_MetLocation.DataSource = met_list;
|
||
CB_EggLocation.DisplayMember = "Text";
|
||
CB_EggLocation.ValueMember = "Value";
|
||
CB_EggLocation.DataSource = new BindingSource(met_list, null);
|
||
CB_EggLocation.SelectedValue = 0;
|
||
CB_MetLocation.SelectedValue = 0;
|
||
origintrack = "XY";
|
||
}
|
||
#endregion
|
||
}
|
||
if (gameorigin < 0x10 && origintrack != "Gen4")
|
||
{
|
||
// Load Gen 4 egg locations if Gen 4 Origin.
|
||
#region HGSS Met Locations
|
||
var met_list = Util.getCBList(metHGSS_00000, new[] { 0 });
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_02000, 2000, new[] { 2000 });
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_02000, 2000, new[] { 2002 });
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_03000, 3000, new[] { 3001 });
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_00000, 0000, Legal.Met_HGSS_0);
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_02000, 2000, Legal.Met_HGSS_2);
|
||
met_list = Util.getOffsetCBList(met_list, metHGSS_03000, 3000, Legal.Met_HGSS_3);
|
||
|
||
CB_EggLocation.DisplayMember = "Text";
|
||
CB_EggLocation.ValueMember = "Value";
|
||
CB_EggLocation.DataSource = met_list;
|
||
CB_EggLocation.SelectedValue = 0;
|
||
origintrack = "Gen4";
|
||
#endregion
|
||
}
|
||
|
||
// Visibility logic for Gen 4 encounter type; only show for Gen 4 Pokemon.
|
||
CB_EncounterType.Visible = Label_EncounterType.Visible = !(gameorigin > 12 || gameorigin < 7);
|
||
// If not Gen 4, set Encounter Type to 0 after it set !visible.
|
||
if (gameorigin > 12 || gameorigin < 7)
|
||
CB_EncounterType.SelectedValue = 0;
|
||
|
||
setMarkings(); // Set/Remove KB marking
|
||
}
|
||
private void updateExtraByteValue(object sender, EventArgs e)
|
||
{
|
||
// Changed Extra Byte's Value
|
||
if (Util.ToInt32((sender as MaskedTextBox).Text) > 255)
|
||
(sender as MaskedTextBox).Text = "255";
|
||
|
||
int value = Util.ToInt32(TB_ExtraByte.Text);
|
||
int offset = Convert.ToInt32(CB_ExtraBytes.Text, 16);
|
||
buff[offset] = (byte)value;
|
||
}
|
||
private void updateExtraByteIndex(object sender, EventArgs e)
|
||
{
|
||
// Byte changed, need to refresh the Text box for the byte's value.
|
||
TB_ExtraByte.Text = buff[Convert.ToInt32(CB_ExtraBytes.Text, 16)].ToString();
|
||
}
|
||
private void updateNickname(object sender, EventArgs e)
|
||
{
|
||
if (init && ModifierKeys == Keys.Control) { getShowdownSet(); return; }
|
||
if (!init || (CHK_Nicknamed.Checked)) return;
|
||
|
||
// Fetch Current Species and set it as Nickname Text
|
||
int species = Util.getIndex(CB_Species);
|
||
if (species == 0 || species > 721)
|
||
TB_Nickname.Text = "";
|
||
else
|
||
{
|
||
// get language
|
||
int lang = Util.getIndex(CB_Language);
|
||
|
||
string l;
|
||
switch (lang)
|
||
{
|
||
case 1: l = "ja"; break;
|
||
case 2: l = "en"; break;
|
||
case 3: l = "fr"; break;
|
||
case 4: l = "it"; break;
|
||
case 5: l = "de"; break;
|
||
case 7: l = "es"; break;
|
||
case 8: l = "ko"; break;
|
||
default: l = curlanguage; break;
|
||
}
|
||
if (CHK_IsEgg.Checked) species = 0; // Set species to 0 to get the egg name.
|
||
TB_Nickname.Text = Util.getStringList("Species", l)[(CHK_IsEgg.Checked) ? 0 : species];
|
||
}
|
||
}
|
||
private void updateNicknameClick(object sender, MouseEventArgs e)
|
||
{
|
||
TextBox tb = (!(sender is TextBox)) ? TB_Nickname : (sender as TextBox);
|
||
// Special Character Form
|
||
if (ModifierKeys == Keys.Control && !specialChars)
|
||
(new f2_Text(tb)).Show();
|
||
}
|
||
private void updateNotOT(object sender, EventArgs e)
|
||
{
|
||
if (TB_OTt2.Text == "")
|
||
{
|
||
clickGT(GB_OT, null); // Switch CT over to OT.
|
||
Label_CTGender.Text = "";
|
||
TB_Friendship.Text = buff[0xCA].ToString();
|
||
}
|
||
else if (Label_CTGender.Text == "")
|
||
Label_CTGender.Text = gendersymbols[0];
|
||
}
|
||
private void updateIsEgg(object sender, EventArgs e)
|
||
{
|
||
if (CHK_IsEgg.Checked)
|
||
{
|
||
CHK_Nicknamed.Checked = false;
|
||
TB_Friendship.Text = "1";
|
||
|
||
// If we are an egg, it won't have a met location.
|
||
CHK_AsEgg.Checked = true;
|
||
GB_EggConditions.Enabled = true;
|
||
|
||
CAL_MetDate.Value = new DateTime(2000, 01, 01);
|
||
CB_MetLocation.SelectedIndex = 2;
|
||
}
|
||
else // Not Egg
|
||
{
|
||
if (!CHK_Nicknamed.Checked)
|
||
updateNickname(null, null);
|
||
|
||
TB_Friendship.Text = PKX.getBaseFriendship(Util.getIndex(CB_Species)).ToString();
|
||
|
||
if (CB_EggLocation.SelectedIndex == 0)
|
||
{
|
||
CAL_EggDate.Value = new DateTime(2000, 01, 01);
|
||
CHK_AsEgg.Checked = false;
|
||
GB_EggConditions.Enabled = false;
|
||
}
|
||
}
|
||
// Display hatch counter if it is an egg, Display Friendship if it is not.
|
||
Label_HatchCounter.Visible = CHK_IsEgg.Checked;
|
||
Label_Friendship.Visible = !CHK_IsEgg.Checked;
|
||
|
||
// Update image to (not) show egg.
|
||
if (!init) return;
|
||
updateNickname(null, null);
|
||
getQuickFiller(dragout);
|
||
}
|
||
private void updateMetAsEgg(object sender, EventArgs e)
|
||
{
|
||
GB_EggConditions.Enabled = CHK_AsEgg.Checked;
|
||
if (CHK_AsEgg.Checked) return;
|
||
// Remove egg met data
|
||
CHK_IsEgg.Checked = false;
|
||
CAL_EggDate.Value = new DateTime(2000, 01, 01);
|
||
CB_EggLocation.SelectedValue = 0;
|
||
}
|
||
private void updateShinyPID(object sender, EventArgs e)
|
||
{
|
||
uint PID = Util.getHEXval(TB_PID);
|
||
uint UID = (PID >> 16);
|
||
uint LID = (PID & 0xFFFF);
|
||
uint PSV = UID ^ LID;
|
||
uint TID = Util.ToUInt32(TB_TID);
|
||
uint SID = Util.ToUInt32(TB_SID);
|
||
uint TSV = TID ^ SID;
|
||
uint XOR = TSV ^ PSV;
|
||
|
||
// Preserve Gen5 Origin Ability bit just in case
|
||
XOR &= 0xFFFE; XOR |= UID & 1;
|
||
|
||
// New XOR should be 0 or 1.
|
||
if (XOR > 16)
|
||
TB_PID.Text = (((UID ^ XOR) << 16) + LID).ToString("X8");
|
||
|
||
setIsShiny();
|
||
getQuickFiller(dragout);
|
||
}
|
||
private void update_ID(object sender, EventArgs e)
|
||
{
|
||
// Trim out nonhex characters
|
||
TB_PID.Text = Util.getHEXval(TB_PID).ToString("X8");
|
||
TB_EC.Text = Util.getHEXval(TB_EC).ToString("X8");
|
||
|
||
// Max TID/SID is 65535
|
||
if (Util.ToUInt32(TB_TID.Text) > 65535) TB_TID.Text = "65535";
|
||
if (Util.ToUInt32(TB_SID.Text) > 65535) TB_SID.Text = "65535";
|
||
|
||
setIsShiny();
|
||
getQuickFiller(dragout);
|
||
updateIVs(null, null); // If the PID is changed, PID%6 (Characteristic) might be changed.
|
||
TB_PID.Select(60, 0); // position cursor at end of field
|
||
}
|
||
private void validateComboBox(object sender, CancelEventArgs e)
|
||
{
|
||
if (!(sender is ComboBox)) { return; }
|
||
|
||
ComboBox cb = sender as ComboBox;
|
||
cb.SelectionLength = 0;
|
||
|
||
cb.BackColor = cb.SelectedValue == null ? Color.DarkSalmon : defaultControlWhite;
|
||
|
||
if (init)
|
||
{ getQuickFiller(dragout); }
|
||
}
|
||
private void validateComboBox2(object sender, EventArgs e)
|
||
{
|
||
ComboBox cb = sender as ComboBox;
|
||
validateComboBox(sender, e as CancelEventArgs);
|
||
if (cb == CB_Ability)
|
||
TB_AbilityNumber.Text = (1 << CB_Ability.SelectedIndex).ToString();
|
||
else if ((cb == CB_Move1) || (cb == CB_Move2) || (cb == CB_Move3) || (cb == CB_Move4))
|
||
updatePP(sender, e);
|
||
getNatureModification(sender, null);
|
||
updateIVs(null, null); // updating Nature will trigger stats to update as well
|
||
}
|
||
private void removedropCB(object sender, KeyEventArgs e)
|
||
{
|
||
((ComboBox)sender).DroppedDown = false;
|
||
}
|
||
private void updateStats()
|
||
{
|
||
// Gather the needed information.
|
||
int species = Util.getIndex(CB_Species);
|
||
int level = Util.ToInt32((MT_Level.Enabled) ? MT_Level.Text : TB_Level.Text);
|
||
if (level == 0) { level = 1; }
|
||
int form = CB_Form.SelectedIndex;
|
||
int HP_IV = Util.ToInt32(TB_HPIV.Text);
|
||
int ATK_IV = Util.ToInt32(TB_ATKIV.Text);
|
||
int DEF_IV = Util.ToInt32(TB_DEFIV.Text);
|
||
int SPA_IV = Util.ToInt32(TB_SPAIV.Text);
|
||
int SPD_IV = Util.ToInt32(TB_SPDIV.Text);
|
||
int SPE_IV = Util.ToInt32(TB_SPEIV.Text);
|
||
|
||
int HP_EV = Util.ToInt32(TB_HPEV.Text);
|
||
int ATK_EV = Util.ToInt32(TB_ATKEV.Text);
|
||
int DEF_EV = Util.ToInt32(TB_DEFEV.Text);
|
||
int SPA_EV = Util.ToInt32(TB_SPAEV.Text);
|
||
int SPD_EV = Util.ToInt32(TB_SPDEV.Text);
|
||
int SPE_EV = Util.ToInt32(TB_SPEEV.Text);
|
||
|
||
int nature = Util.getIndex(CB_Nature);
|
||
|
||
// Generate the stats.
|
||
ushort[] stats = PKX.getStats(species, level, nature, form,
|
||
HP_EV, ATK_EV, DEF_EV, SPA_EV, SPD_EV, SPE_EV,
|
||
HP_IV, ATK_IV, DEF_IV, SPA_IV, SPD_IV, SPE_IV);
|
||
|
||
Stat_HP.Text = stats[0].ToString();
|
||
Stat_ATK.Text = stats[1].ToString();
|
||
Stat_DEF.Text = stats[2].ToString();
|
||
Stat_SPA.Text = stats[4].ToString();
|
||
Stat_SPD.Text = stats[5].ToString();
|
||
Stat_SPE.Text = stats[3].ToString();
|
||
|
||
// Recolor the Stat Labels based on boosted stats.
|
||
{
|
||
int incr = nature / 5;
|
||
int decr = nature % 5;
|
||
|
||
Label[] labarray = { Label_ATK, Label_DEF, Label_SPE, Label_SPA, Label_SPD };
|
||
// Reset Label Colors
|
||
foreach (Label label in labarray)
|
||
label.ForeColor = defaultControlText;
|
||
|
||
// Set Colored StatLabels only if Nature isn't Neutral
|
||
if (incr == decr) return;
|
||
labarray[incr].ForeColor = Color.Red;
|
||
labarray[decr].ForeColor = Color.Blue;
|
||
}
|
||
}
|
||
private void getShowdownSet()
|
||
{
|
||
if (!Clipboard.ContainsText()) return;
|
||
|
||
// Get Simulator Data
|
||
PKX.Simulator.Set Set = new PKX.Simulator.Set(
|
||
Clipboard.GetText(), // Input Set
|
||
Util.getStringList("Species", "en"),
|
||
Util.getStringList("Items", "en"),
|
||
Util.getStringList("Natures", "en"),
|
||
Util.getStringList("Moves", "en"),
|
||
Util.getStringList("Abilities", "en"));
|
||
|
||
if (Set.Species < 0) return;
|
||
if (DialogResult.Yes != Util.Prompt(MessageBoxButtons.YesNo, "Import this set?", Clipboard.GetText())) return;
|
||
|
||
// Set Species & Nickname
|
||
CB_Species.SelectedValue = Set.Species;
|
||
CHK_Nicknamed.Checked = (Set.Nickname != null);
|
||
if (Set.Nickname != null)
|
||
TB_Nickname.Text = Set.Nickname;
|
||
if (Set.Gender != null && PKX.getGender(Label_Gender.Text) != 2 && PKX.getGender(Set.Gender) != 2)
|
||
Label_Gender.Text = Set.Gender;
|
||
|
||
// Set Form
|
||
string[] formStrings = PKX.getFormList(Set.Species,
|
||
Util.getStringList("Types", "en"),
|
||
Util.getStringList("Forms", "en"), gendersymbols);
|
||
int form = 0;
|
||
for (int i = 0; i < formStrings.Length; i++)
|
||
if (formStrings[i].Contains(Set.Form ?? ""))
|
||
{ form = i; break; }
|
||
CB_Form.SelectedIndex = form;
|
||
|
||
// Set Ability
|
||
byte[] abilities = PKX.getAbilities(Set.Species, form);
|
||
int ability = Array.IndexOf(abilities, (byte)Set.Ability);
|
||
if (ability < 0) ability = 0;
|
||
CB_Ability.SelectedIndex = ability;
|
||
ComboBox[] m = { CB_Move1, CB_Move2, CB_Move3, CB_Move4, };
|
||
for (int i = 0; i < 4; i++) m[i].SelectedValue = Set.Moves[i];
|
||
|
||
// Set Item and Nature
|
||
CB_HeldItem.SelectedValue = (Set.Item < 0) ? 0 : Set.Item;
|
||
CB_Nature.SelectedValue = (Set.Nature < 0) ? 0 : Set.Nature;
|
||
|
||
// Set IVs
|
||
TB_HPIV.Text = Set.IVs[0].ToString();
|
||
TB_ATKIV.Text = Set.IVs[1].ToString();
|
||
TB_DEFIV.Text = Set.IVs[2].ToString();
|
||
TB_SPAIV.Text = Set.IVs[3].ToString();
|
||
TB_SPDIV.Text = Set.IVs[4].ToString();
|
||
TB_SPEIV.Text = Set.IVs[5].ToString();
|
||
|
||
// Set EVs
|
||
TB_HPEV.Text = Set.EVs[0].ToString();
|
||
TB_ATKEV.Text = Set.EVs[1].ToString();
|
||
TB_DEFEV.Text = Set.EVs[2].ToString();
|
||
TB_SPAEV.Text = Set.EVs[3].ToString();
|
||
TB_SPDEV.Text = Set.EVs[4].ToString();
|
||
TB_SPEEV.Text = Set.EVs[5].ToString();
|
||
|
||
// Set Level and Friendship
|
||
TB_Level.Text = Set.Level.ToString();
|
||
TB_Friendship.Text = Set.Friendship.ToString();
|
||
|
||
// Reset IV/EVs
|
||
BTN_RerollPID.PerformClick();
|
||
BTN_RerollEC.PerformClick();
|
||
if (Set.Shiny) BTN_Shinytize.PerformClick();
|
||
}
|
||
// Secondary Windows for Ribbons/Amie/Memories
|
||
private void openRibbons(object sender, EventArgs e)
|
||
{
|
||
new RibbMedal(this).ShowDialog();
|
||
}
|
||
private void openHistory(object sender, EventArgs e)
|
||
{
|
||
new MemoryAmie(this).ShowDialog();
|
||
}
|
||
// Open/Save Array Manipulation //
|
||
private bool verifiedPKX()
|
||
{
|
||
if (ModifierKeys == (Keys.Control | Keys.Shift | Keys.Alt))
|
||
return true; // Override
|
||
// Make sure the PKX Fields are filled out properly (color check)
|
||
#region ComboBoxes to verify they are set.
|
||
ComboBox[] cba = {
|
||
CB_Species, CB_Nature, CB_HeldItem, CB_Ability, // Main Tab
|
||
CB_MetLocation, CB_EggLocation, CB_Ball, // Met Tab
|
||
CB_Move1, CB_Move2, CB_Move3, CB_Move4, // Moves
|
||
CB_RelearnMove1, CB_RelearnMove2, CB_RelearnMove3, CB_RelearnMove4 // Moves
|
||
};
|
||
for (int i = 0; i < cba.Length; i++)
|
||
{
|
||
int back = cba[i].BackColor.ToArgb();
|
||
if (back == SystemColors.Control.ToArgb() || back == 0 ||
|
||
!(back != -1 & back != defaultControlWhite.ToArgb())) continue;
|
||
if (i < 6) // Main Tab
|
||
tabMain.SelectedIndex = 0;
|
||
else if (i < 9) // Met Tab
|
||
tabMain.SelectedIndex = 1;
|
||
else // Moves
|
||
tabMain.SelectedIndex = 3;
|
||
goto invalid;
|
||
}
|
||
#endregion
|
||
// Further logic checking
|
||
if (Convert.ToUInt32(TB_EVTotal.Text) > 510 && !CHK_HackedStats.Checked)
|
||
{ tabMain.SelectedIndex = 2; goto invalid; }
|
||
// If no errors detected...
|
||
if (Util.getIndex(CB_Species) != 0) return true;
|
||
// Else
|
||
tabMain.SelectedIndex = 0;
|
||
|
||
// else...
|
||
invalid:
|
||
{ System.Media.SystemSounds.Exclamation.Play(); return false; }
|
||
}
|
||
private byte[] preparepkx(bool click = true)
|
||
{
|
||
if (click)
|
||
tabMain.Select(); // hack to make sure comboboxes are set (users scrolling through and immediately setting causes this)
|
||
// Stuff the global byte array with our PKX form data
|
||
// Create a new storage so we don't muck up things with the original
|
||
if (buff.Length == 232) Array.Resize(ref buff, 260);
|
||
byte[] pkx = new byte[0x104];
|
||
Array.Copy(buff, pkx, 0x104);
|
||
// Repopulate PKX with Edited Stuff
|
||
if (Util.getIndex(CB_GameOrigin) < 24)
|
||
{
|
||
uint EC = Util.getHEXval(TB_EC);
|
||
uint PID = Util.getHEXval(TB_PID);
|
||
uint SID = Util.ToUInt32(TB_TID.Text);
|
||
uint TID = Util.ToUInt32(TB_TID.Text);
|
||
uint LID = PID & 0xFFFF;
|
||
uint HID = PID >> 16;
|
||
uint XOR = (TID ^ LID ^ SID ^ HID);
|
||
|
||
// Ensure we don't have a shiny.
|
||
if (XOR >> 3 == 1) // Illegal, fix. (not 16<XOR>=8)
|
||
{
|
||
// Keep as shiny, so we have to mod the PID
|
||
PID ^= XOR;
|
||
TB_PID.Text = PID.ToString("X8");
|
||
TB_EC.Text = PID.ToString("X8");
|
||
}
|
||
else if ((XOR ^ 0x8000) >> 3 == 1 && PID != EC)
|
||
TB_EC.Text = (PID ^ 0x80000000).ToString("X8");
|
||
else // Not Illegal, no fix.
|
||
TB_EC.Text = PID.ToString("X8");
|
||
}
|
||
|
||
Array.Copy(BitConverter.GetBytes(Util.getHEXval(TB_EC)), 0, pkx, 0, 4); // EC
|
||
Array.Copy(BitConverter.GetBytes(0), 0, pkx, 0x4, 4); // 0 CHK for now
|
||
|
||
// Block A
|
||
int species = Util.getIndex(CB_Species);
|
||
Array.Copy(BitConverter.GetBytes(species), 0, pkx, 0x08, 2); // Species
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_HeldItem)), 0, pkx, 0x0A, 2); // Held Item
|
||
Array.Copy(BitConverter.GetBytes(Util.ToUInt32(TB_TID.Text)), 0, pkx, 0x0C, 2); // TID
|
||
Array.Copy(BitConverter.GetBytes(Util.ToUInt32(TB_SID.Text)), 0, pkx, 0x0E, 2); // SID
|
||
Array.Copy(BitConverter.GetBytes(Convert.ToUInt32(TB_EXP.Text)), 0, pkx, 0x10, 4); // EXP
|
||
pkx[0x14] = (byte)Array.IndexOf(abilitylist, (CB_Ability.Text).Remove((CB_Ability.Text).Length - 4)); // Ability
|
||
pkx[0x15] = (byte)(Util.ToInt32((TB_AbilityNumber.Text))); // Number
|
||
// pkx[0x16], pkx[0x17] are handled by the Medals UI (Hits & Training Bag)
|
||
Array.Copy(BitConverter.GetBytes(Util.getHEXval(TB_PID)), 0, pkx, 0x18, 4); // PID
|
||
pkx[0x1C] = (byte)((Util.getIndex(CB_Nature))); // Nature
|
||
int fegform = Convert.ToInt32(CHK_Fateful.Checked); // Fateful
|
||
fegform |= (PKX.getGender(Label_Gender.Text) << 1); // Gender
|
||
fegform |= (Math.Min((MT_Form.Enabled) ? Convert.ToInt32(MT_Form.Text) : CB_Form.SelectedIndex, 32) << 3); // Form
|
||
pkx[0x1D] = (byte)fegform;
|
||
pkx[0x1E] = (byte)Util.ToInt32(TB_HPEV.Text); // EVs
|
||
pkx[0x1F] = (byte)Util.ToInt32(TB_ATKEV.Text);
|
||
pkx[0x20] = (byte)Util.ToInt32(TB_DEFEV.Text);
|
||
pkx[0x21] = (byte)Util.ToInt32(TB_SPEEV.Text);
|
||
pkx[0x22] = (byte)Util.ToInt32(TB_SPAEV.Text);
|
||
pkx[0x23] = (byte)Util.ToInt32(TB_SPDEV.Text);
|
||
|
||
pkx[0x24] = (byte)Util.ToInt32(TB_Cool.Text); // CNT
|
||
pkx[0x25] = (byte)Util.ToInt32(TB_Beauty.Text);
|
||
pkx[0x26] = (byte)Util.ToInt32(TB_Cute.Text);
|
||
pkx[0x27] = (byte)Util.ToInt32(TB_Smart.Text);
|
||
pkx[0x28] = (byte)Util.ToInt32(TB_Tough.Text);
|
||
pkx[0x29] = (byte)Util.ToInt32(TB_Sheen.Text);
|
||
|
||
int markings = CHK_Circle.Checked ? (1 << 0) : 0;
|
||
markings |= CHK_Triangle.Checked ? (1 << 1) : 0;
|
||
markings |= CHK_Square.Checked ? (1 << 2) : 0;
|
||
markings |= CHK_Heart.Checked ? (1 << 3) : 0;
|
||
markings |= CHK_Star.Checked ? (1 << 4) : 0;
|
||
markings |= CHK_Diamond.Checked ? (1 << 5) : 0;
|
||
pkx[0x2A] = (byte)markings;
|
||
pkx[0x2B] = (byte)(CB_PKRSStrain.SelectedIndex << 4 | CB_PKRSDays.SelectedIndex);
|
||
|
||
// Already in buff (then transferred to new pkx)
|
||
// 0x2C, 0x2D, 0x2E, 0x2F
|
||
// 0x30, 0x31, 0x32, 0x33
|
||
// 0x34, 0x35, 0x36, 0x37
|
||
// 0x38, 0x39
|
||
|
||
// Unused
|
||
// 0x3A, 0x3B
|
||
// 0x3C, 0x3D, 0x3E, 0x3F
|
||
|
||
// Block B
|
||
// Convert Nickname field back to bytes
|
||
string nicknamestr = TB_Nickname.Text;
|
||
{
|
||
nicknamestr = nicknamestr.Replace("\u2640", "\uE08F");
|
||
nicknamestr = nicknamestr.Replace("\u2642", "\uE08E");
|
||
nicknamestr = nicknamestr.Replace("\u0027", "\u2019"); // ' to ’
|
||
}
|
||
byte[] nicknamebytes = Encoding.Unicode.GetBytes(nicknamestr);
|
||
Array.Resize(ref nicknamebytes, nicknamebytes.Length + 2); // pad with zero byte terminator
|
||
Array.Copy(nicknamebytes, 0, pkx, 0x40, nicknamebytes.Length);
|
||
|
||
// 0x58, 0x59 unused
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_Move1)), 0, pkx, 0x5A, 2); // Move 1
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_Move2)), 0, pkx, 0x5C, 2); // Move 2
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_Move3)), 0, pkx, 0x5E, 2); // Move 3
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_Move4)), 0, pkx, 0x60, 2); // Move 4
|
||
|
||
pkx[0x62] = (byte)(Util.getIndex(CB_Move1) > 0 ? Util.ToInt32(TB_PP1.Text) : 0); // Max PP
|
||
pkx[0x63] = (byte)(Util.getIndex(CB_Move2) > 0 ? Util.ToInt32(TB_PP2.Text) : 0);
|
||
pkx[0x64] = (byte)(Util.getIndex(CB_Move3) > 0 ? Util.ToInt32(TB_PP3.Text) : 0);
|
||
pkx[0x65] = (byte)(Util.getIndex(CB_Move4) > 0 ? Util.ToInt32(TB_PP4.Text) : 0);
|
||
|
||
pkx[0x66] = (byte)(Util.getIndex(CB_Move1) > 0 ? CB_PPu1.SelectedIndex : 0); // PP Ups
|
||
pkx[0x67] = (byte)(Util.getIndex(CB_Move2) > 0 ? CB_PPu2.SelectedIndex : 0);
|
||
pkx[0x68] = (byte)(Util.getIndex(CB_Move3) > 0 ? CB_PPu3.SelectedIndex : 0);
|
||
pkx[0x69] = (byte)(Util.getIndex(CB_Move4) > 0 ? CB_PPu4.SelectedIndex : 0);
|
||
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_RelearnMove1)), 0, pkx, 0x6A, 2); // EggMove 1
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_RelearnMove2)), 0, pkx, 0x6C, 2); // EggMove 2
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_RelearnMove3)), 0, pkx, 0x6E, 2); // EggMove 3
|
||
Array.Copy(BitConverter.GetBytes(Util.getIndex(CB_RelearnMove4)), 0, pkx, 0x70, 2); // EggMove 4
|
||
|
||
// 0x72 - Ribbon editor sets this flag (Secret Super Training)
|
||
// 0x73
|
||
|
||
uint IV32 = Util.ToUInt32(TB_HPIV.Text) & 0x1F;
|
||
IV32 |= ((Util.ToUInt32(TB_ATKIV.Text) & 0x1F) << 5);
|
||
IV32 |= ((Util.ToUInt32(TB_DEFIV.Text) & 0x1F) << 10);
|
||
IV32 |= ((Util.ToUInt32(TB_SPEIV.Text) & 0x1F) << 15);
|
||
IV32 |= ((Util.ToUInt32(TB_SPAIV.Text) & 0x1F) << 20);
|
||
IV32 |= ((Util.ToUInt32(TB_SPDIV.Text) & 0x1F) << 25);
|
||
IV32 |= (Convert.ToUInt32(CHK_IsEgg.Checked) << 30);
|
||
IV32 |= (Convert.ToUInt32(CHK_Nicknamed.Checked) << 31);
|
||
|
||
Array.Copy(BitConverter.GetBytes(IV32), 0, pkx, 0x74, 4); // Copy in IVs
|
||
|
||
// Block C
|
||
// Convert Latest OT field back to bytes
|
||
byte[] OT2 = Encoding.Unicode.GetBytes(Util.TrimFromZero(TB_OTt2.Text).Replace("\u0027", "\u2019"));
|
||
Array.Resize(ref OT2, OT2.Length + 2); // Allow Trash
|
||
Array.Copy(OT2, 0, pkx, 0x78, OT2.Length);
|
||
|
||
// 0x90-0xAF
|
||
pkx[0x92] = Convert.ToByte(PKX.getGender(Label_CTGender.Text) == 1);
|
||
// Plus more, set by MemoryAmie (already in buff)
|
||
|
||
// Block D
|
||
// Convert OT field back to bytes
|
||
byte[] OT = Encoding.Unicode.GetBytes(TB_OT.Text.Replace("\u0027", "\u2019"));
|
||
Array.Resize(ref OT, OT.Length + 2); // Pad with \0000 to terminate
|
||
Array.Copy(OT, 0, pkx, 0xB0, OT.Length);
|
||
|
||
if (pkx[0x93] == 0)
|
||
pkx[0xCA] = (byte)(Util.ToInt32(TB_Friendship.Text) & 0xFF);
|
||
else // 1
|
||
pkx[0xA2] = (byte)(Util.ToInt32(TB_Friendship.Text) & 0xFF);
|
||
|
||
int egg_year = 2000; // Dates
|
||
int egg_month = 0;
|
||
int egg_day = 0;
|
||
int egg_location = 0;
|
||
if (CHK_AsEgg.Checked) // If encountered as an egg, load the Egg Met data from fields.
|
||
{
|
||
egg_year = CAL_EggDate.Value.Year;
|
||
egg_month = CAL_EggDate.Value.Month;
|
||
egg_day = CAL_EggDate.Value.Day;
|
||
egg_location = Util.getIndex(CB_EggLocation);
|
||
}
|
||
// Egg Met Data
|
||
pkx[0xD1] = (byte)(egg_year - 2000);
|
||
pkx[0xD2] = (byte)egg_month;
|
||
pkx[0xD3] = (byte)egg_day;
|
||
// Met Data
|
||
pkx[0xD4] = (byte)(CAL_MetDate.Value.Year - 2000);
|
||
pkx[0xD5] = (byte)CAL_MetDate.Value.Month;
|
||
pkx[0xD6] = (byte)CAL_MetDate.Value.Day;
|
||
|
||
if (CHK_IsEgg.Checked && CB_MetLocation.SelectedIndex == 0) // If still an egg, it has no hatch location/date. Zero it!
|
||
{
|
||
pkx[0xD4] = 0;
|
||
pkx[0xD5] = 0;
|
||
pkx[0xD6] = 0;
|
||
}
|
||
|
||
// 0xD7 Unknown
|
||
int met_location = Util.getIndex(CB_MetLocation); // Locations
|
||
Array.Copy(BitConverter.GetBytes(egg_location), 0, pkx, 0xD8, 2); // Egg Location
|
||
Array.Copy(BitConverter.GetBytes(met_location), 0, pkx, 0xDA, 2); // Met Location
|
||
|
||
pkx[0xDC] = (byte)Util.getIndex(CB_Ball);
|
||
pkx[0xDD] = (byte)(((Util.ToInt32(TB_MetLevel.Text) & 0x7F) | (Convert.ToInt32(PKX.getGender(Label_OTGender.Text) == 1) << 7)));
|
||
pkx[0xDE] = (byte)(Util.ToInt32(CB_EncounterType.SelectedValue.ToString()));
|
||
pkx[0xDF] = (byte)Util.getIndex(CB_GameOrigin);
|
||
pkx[0xE0] = (byte)Util.getIndex(CB_Country);
|
||
pkx[0xE1] = (byte)Util.getIndex(CB_SubRegion);
|
||
pkx[0xE2] = (byte)Util.getIndex(CB_3DSReg);
|
||
pkx[0xE3] = (byte)Util.getIndex(CB_Language);
|
||
// 0xE4-0xE7
|
||
|
||
Array.Resize(ref pkx, 260);
|
||
// Party Stats
|
||
pkx[0xE8] = 0; pkx[0xE9] = 0;
|
||
pkx[0xEA] = 0; pkx[0xEB] = 0;
|
||
pkx[0xEC] = (byte)Util.ToInt32(TB_Level.Text); // Level
|
||
pkx[0xED] = 0; pkx[0xEE] = 0; pkx[0xEF] = 0;
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_HP.Text), 65535)), 0, pkx, 0xF0, 2); // Current HP
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_HP.Text), 65535)), 0, pkx, 0xF2, 2); // Max HP
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_ATK.Text), 65535)), 0, pkx, 0xF4, 2); // ATK
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_DEF.Text), 65535)), 0, pkx, 0xF6, 2); // DEF
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_SPE.Text), 65535)), 0, pkx, 0xF8, 2); // SPE
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_SPA.Text), 65535)), 0, pkx, 0xFA, 2); // SPA
|
||
Array.Copy(BitConverter.GetBytes(Math.Min(Util.ToInt32(Stat_SPD.Text), 65535)), 0, pkx, 0xFC, 2); // SPD
|
||
pkx[0xFE] = 0; pkx[0xFF] = 0;
|
||
pkx[0x100] = 0; pkx[0x101] = 0; pkx[0x102] = 0; pkx[0x103] = 0;
|
||
|
||
// Hax Illegality
|
||
if (HaX)
|
||
{
|
||
pkx[0x14] = (byte)Util.getIndex(DEV_Ability); // Ability
|
||
pkx[0xEC] = (byte)Math.Min(Convert.ToInt32(MT_Level.Text), 255); // Level
|
||
}
|
||
|
||
// Fix Moves if a slot is empty
|
||
for (int i = 0; i < 3; i++)
|
||
{
|
||
if (BitConverter.ToUInt16(pkx, 0x5A + 2*i) != 0) continue;
|
||
Array.Copy(pkx, 0x5C + 2 * i, pkx, 0x5A + 2 * i, 2); // Shift moves down
|
||
Array.Copy(new byte[2], 0, pkx, 0x5C + 2 * i, 2); // Clear next move (error shifted down)
|
||
|
||
// Move PP and PP Ups down one byte.
|
||
pkx[0x62 + i] = pkx[0x63 + i]; pkx[0x63 + i] = 0; // PP
|
||
pkx[0x66 + i] = pkx[0x67 + i]; pkx[0x67 + i] = 0; // PP Ups
|
||
}
|
||
|
||
// Fix Relearn moves if a slot is empty
|
||
for (int i = 0; i < 3; i++)
|
||
{
|
||
if (BitConverter.ToUInt16(pkx, 0x6A + 2*i) != 0) continue;
|
||
Array.Copy(pkx, 0x6C + 2 * i, pkx, 0x6A + 2 * i, 2); // Shift moves down
|
||
Array.Copy(new byte[2], 0, pkx, 0x6C + 2 * i, 2); // Clear next move (error shifted down)
|
||
}
|
||
|
||
// No foreign memories for Pokemon without a foreign trainer
|
||
if (BitConverter.ToUInt16(pkx, 0x78) == 0)
|
||
{
|
||
pkx[0xA2] = pkx[0xA3] = // No Friendship/Affection
|
||
pkx[0xA8] = pkx[0xA9] = // No Memory Var
|
||
pkx[0xA4] = pkx[0xA5] = pkx[0xA6] = 0; // No Memory Types
|
||
}
|
||
|
||
// Now we fix the checksum!
|
||
ushort chk = 0;
|
||
for (int i = 8; i < 232; i += 2) // Loop through the entire PKX
|
||
chk += BitConverter.ToUInt16(pkx, i);
|
||
|
||
// Apply New Checksum
|
||
Array.Copy(BitConverter.GetBytes(chk), 0, pkx, 6, 2);
|
||
|
||
// PKX is now filled
|
||
return pkx;
|
||
}
|
||
// Drag & Drop Events
|
||
private 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);
|
||
string path = files[0]; // open first D&D
|
||
openQuick(path);
|
||
}
|
||
// Decrypted Export
|
||
private void dragout_MouseDown(object sender, MouseEventArgs e)
|
||
{
|
||
if (!verifiedPKX()) { return; }
|
||
{
|
||
// Create Temp File to Drag
|
||
string basepath = Application.StartupPath;
|
||
Cursor.Current = Cursors.Hand;
|
||
|
||
// Make a new file name
|
||
byte[] dragdata = preparepkx();
|
||
PKX pkx = new PKX(dragdata, "Tabs");
|
||
string filename = pkx.Nickname;
|
||
if (filename != pkx.Species)
|
||
filename += " (" + pkx.Species + ")";
|
||
filename += " - " + pkx.PID;
|
||
|
||
filename += (e.Button == MouseButtons.Right) ? ".ek6" : ".pk6";
|
||
dragdata = (e.Button == MouseButtons.Right) ? PKX.encryptArray(preparepkx()) : preparepkx();
|
||
// Strip out party stats (if they are there)
|
||
Array.Resize(ref dragdata, 232);
|
||
// Make file
|
||
string newfile = Path.Combine(basepath, Util.CleanFileName(filename));
|
||
try
|
||
{
|
||
File.WriteAllBytes(newfile, dragdata);
|
||
|
||
string[] filesToDrag = { newfile };
|
||
dragout.DoDragDrop(new DataObject(DataFormats.FileDrop, filesToDrag), DragDropEffects.Move);
|
||
File.Delete(newfile);
|
||
}
|
||
catch (Exception x)
|
||
{ Util.Error("Drag & Drop Error", x.ToString()); }
|
||
File.Delete(newfile);
|
||
}
|
||
}
|
||
private void dragout_DragOver(object sender, DragEventArgs e)
|
||
{
|
||
e.Effect = DragDropEffects.Move;
|
||
}
|
||
// Dragout Display
|
||
private void dragoutHover(object sender, EventArgs e)
|
||
{
|
||
dragout.BackgroundImage = (Util.getIndex(CB_Species) > 0) ? Properties.Resources.slotSet : Properties.Resources.slotDel;
|
||
}
|
||
private void dragoutLeave(object sender, EventArgs e)
|
||
{
|
||
dragout.BackgroundImage = Properties.Resources.slotTrans;
|
||
}
|
||
private void dragoutDrop(object sender, DragEventArgs e)
|
||
{
|
||
openQuick(((string[])e.Data.GetData(DataFormats.FileDrop))[0]);
|
||
}
|
||
#endregion
|
||
|
||
#region //// SAVE FILE FUNCTIONS ////
|
||
// Integrity Checks //
|
||
private void clickVerifyCHK(object sender, EventArgs e)
|
||
{
|
||
if (savedited) { Util.Alert("Save has been edited. Cannot integrity check."); return; }
|
||
|
||
int[] ctr = {0, 0};
|
||
RTB_S.Text += PKX.verifyG6CHK(savefile, savegame_oras, 0, ref ctr);
|
||
if (cybergadget) return;
|
||
RTB_S.Text += PKX.verifyG6CHK(savefile, savegame_oras, 1, ref ctr);
|
||
|
||
if (ctr[0] == ctr[1]) RTB_S.Text = "No checksums are valid.";
|
||
}
|
||
private void clickVerifySHA(object sender, EventArgs e)
|
||
{
|
||
if (savedited) { Util.Alert("Save has been edited. Cannot integrity check."); return; }
|
||
|
||
RTB_S.Text += PKX.verifyG6SHA(savefile, savegame_oras);
|
||
}
|
||
private void clickExportSAV(object sender, EventArgs e)
|
||
{
|
||
// Create another version of the save file.
|
||
byte[] editedsav = new byte[0x100000];
|
||
Array.Copy(savefile, editedsav, savefile.Length);
|
||
// Since we only edited one of the save files, we only have to fix half of the chk/hashes!
|
||
|
||
// Fix Checksums
|
||
editedsav = PKX.writeG6CHK(editedsav, savegame_oras, savindex);
|
||
if (!cybergadget) // Fix Hashes
|
||
editedsav = PKX.writeG6SHA(editedsav, savegame_oras, savindex);
|
||
// Write the active save index
|
||
editedsav[0x168] = (byte)(savindex ^ 1);
|
||
// File Integrity has been restored as well as it can. Export!
|
||
|
||
if (cybergadget)
|
||
#region Saving CyberGadget/RAMSAV
|
||
{
|
||
byte[] cybersav = new byte[savegame_oras ? 0x76000 : 0x65600];
|
||
Array.Copy(editedsav, 0x5400, cybersav, 0, cybersav.Length);
|
||
if (ramsav == null)
|
||
{
|
||
// Chunk Error Checking
|
||
byte[] FFFF = new byte[0x200];
|
||
byte[] section = new byte[0x200];
|
||
for (int i = 0; i < 0x200; i++)
|
||
FFFF[i] = 0xFF;
|
||
|
||
for (int i = 0; i < cybersav.Length / 0x200; i++)
|
||
{
|
||
Array.Copy(cybersav, i * 0x200, section, 0, 0x200);
|
||
if (!section.SequenceEqual(FFFF)) continue;
|
||
string problem = String.Format("0x200 chunk @ 0x{0} is FF'd.", (i * 0x200).ToString("X5"))
|
||
+ Environment.NewLine + "Cyber will screw up (as of August 31st)." + Environment.NewLine + Environment.NewLine;
|
||
|
||
// Check to see if it is in the Pokedex
|
||
if (i * 0x200 > 0x14E00 && i * 0x200 < 0x15700)
|
||
{
|
||
problem += "Problem lies in the Pokedex. ";
|
||
if (i * 0x200 == 0x15400)
|
||
problem += "Remove a language flag for a species ~ ex " + specieslist[548];
|
||
}
|
||
|
||
if (Util.Prompt(MessageBoxButtons.YesNo, problem, "Continue saving?") != DialogResult.Yes)
|
||
return;
|
||
}
|
||
}
|
||
SaveFileDialog cySAV = new SaveFileDialog();
|
||
|
||
// Try for file path
|
||
string cyberpath = Util.GetTempFolder();
|
||
if (ramsav != null && Directory.Exists(path3DS))
|
||
{
|
||
cySAV.InitialDirectory = path3DS;
|
||
cySAV.RestoreDirectory = true;
|
||
}
|
||
else if (pathSDF != null && Directory.Exists(pathSDF))
|
||
{
|
||
cySAV.InitialDirectory = pathSDF;
|
||
cySAV.RestoreDirectory = true;
|
||
}
|
||
else if (Directory.Exists(Path.Combine(cyberpath, "root")))
|
||
{
|
||
cySAV.InitialDirectory = Path.Combine(cyberpath, "root");
|
||
cySAV.RestoreDirectory = true;
|
||
}
|
||
else if (Directory.Exists(cyberpath))
|
||
{
|
||
cySAV.InitialDirectory = cyberpath;
|
||
cySAV.RestoreDirectory = true;
|
||
}
|
||
if (ramsavloaded && ModifierKeys == Keys.Shift) // Export RAM SAV to another.
|
||
{
|
||
Util.Alert("Please specify the target cart/console-RAM save.");
|
||
OpenFileDialog ofd = new OpenFileDialog();
|
||
if (ofd.ShowDialog() != DialogResult.OK) return;
|
||
string target = ofd.FileName;
|
||
byte[] targetRAM = File.ReadAllBytes(target);
|
||
byte[] newRAM = ram2sav.getRAM(targetRAM, cybersav);
|
||
|
||
cySAV.Filter = "ramsav|*.bin";
|
||
cySAV.FileName = "ramsav.bin";
|
||
DialogResult sdr = cySAV.ShowDialog();
|
||
if (sdr != DialogResult.OK) return;
|
||
string path = cySAV.FileName;
|
||
File.WriteAllBytes(path, newRAM);
|
||
Util.Alert("Saved RAM SAV to:" + Environment.NewLine + path, "Target RAM:" + Environment.NewLine + target);
|
||
}
|
||
else if (ramsavloaded && ModifierKeys != Keys.Control) // Export RAM SAV if it is the currently loaded one.
|
||
{
|
||
cySAV.Filter = "ramsav|*.bin";
|
||
cySAV.FileName = "ramsav.bin";
|
||
DialogResult sdr = cySAV.ShowDialog();
|
||
if (sdr != DialogResult.OK) return;
|
||
string path = cySAV.FileName;
|
||
File.WriteAllBytes(path, ram2sav.getRAM(ramsav, cybersav));
|
||
Util.Alert("Saved RAM SAV to:", path);
|
||
}
|
||
else
|
||
{
|
||
cySAV.Filter = "Cyber SAV|*.*";
|
||
cySAV.FileName = L_Save.Text.Split(new[] { ": " }, StringSplitOptions.None)[1];
|
||
DialogResult sdr = cySAV.ShowDialog();
|
||
if (sdr != DialogResult.OK) return;
|
||
string path = cySAV.FileName;
|
||
File.WriteAllBytes(path, cybersav);
|
||
Util.Alert("Saved Cyber SAV to:", path);
|
||
}
|
||
}
|
||
#endregion
|
||
else
|
||
#region Saving Full Save File
|
||
{
|
||
// Save Full Save File
|
||
SaveFileDialog savesav = new SaveFileDialog
|
||
{
|
||
Filter = "SAV|*.bin;*.sav",
|
||
FileName = L_Save.Text.Split(new[] {": "}, StringSplitOptions.None)[1]
|
||
};
|
||
DialogResult result = savesav.ShowDialog();
|
||
if (result != DialogResult.OK) return;
|
||
string path = savesav.FileName;
|
||
|
||
if (File.Exists(path))
|
||
{
|
||
// File already exists, save a .bak
|
||
byte[] backupfile = File.ReadAllBytes(path);
|
||
File.WriteAllBytes(path + ".bak", backupfile);
|
||
}
|
||
File.WriteAllBytes(path, editedsav);
|
||
Util.Alert("Saved 1MB SAV to:", path);
|
||
}
|
||
#endregion
|
||
}
|
||
// Box/SAV Functions //
|
||
private void switchSAVTab(object sender, EventArgs e)
|
||
{
|
||
if (tabBoxMulti.SelectedIndex == 0)
|
||
savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] = (byte)CB_BoxSelect.SelectedIndex;
|
||
if (tabBoxMulti.SelectedIndex == 1)
|
||
savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] = 31;
|
||
}
|
||
private void clickBoxRight(object sender, EventArgs e)
|
||
{
|
||
CB_BoxSelect.SelectedIndex = (CB_BoxSelect.SelectedIndex + 1) % 31;
|
||
savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] = (byte)CB_BoxSelect.SelectedIndex;
|
||
}
|
||
private void clickBoxLeft(object sender, EventArgs e)
|
||
{
|
||
CB_BoxSelect.SelectedIndex = (CB_BoxSelect.SelectedIndex + 30) % 31;
|
||
savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] = (byte)CB_BoxSelect.SelectedIndex;
|
||
}
|
||
private void clickSlot(object sender, EventArgs e)
|
||
{
|
||
switch (ModifierKeys)
|
||
{
|
||
case (Keys.Control | Keys.Alt): clickClone(sender, e); break;
|
||
case Keys.Control: clickView(sender, e); break;
|
||
case Keys.Shift: clickSet(sender, e); break;
|
||
case Keys.Alt: clickDelete(sender, e); break;
|
||
}
|
||
}
|
||
private void clickView(object sender, EventArgs e)
|
||
{
|
||
int slot = getSlot(sender);
|
||
int offset = getPKXOffset(slot);
|
||
|
||
PictureBox[] pba = {
|
||
bpkx1, bpkx2, bpkx3, bpkx4, bpkx5, bpkx6,
|
||
bpkx7, bpkx8, bpkx9, bpkx10,bpkx11,bpkx12,
|
||
bpkx13,bpkx14,bpkx15,bpkx16,bpkx17,bpkx18,
|
||
bpkx19,bpkx20,bpkx21,bpkx22,bpkx23,bpkx24,
|
||
bpkx25,bpkx26,bpkx27,bpkx28,bpkx29,bpkx30,
|
||
|
||
ppkx1, ppkx2, ppkx3, ppkx4, ppkx5, ppkx6,
|
||
bbpkx1,bbpkx2,bbpkx3,bbpkx4,bbpkx5,bbpkx6,
|
||
|
||
dcpkx1, dcpkx2, gtspkx, fusedpkx,subepkx1,subepkx2,subepkx3,
|
||
};
|
||
|
||
PictureBox picturebox = pba[slot];
|
||
if (picturebox.Image == null)
|
||
{ System.Media.SystemSounds.Exclamation.Play(); return; }
|
||
|
||
// Load the PKX file
|
||
if (BitConverter.ToUInt64(savefile, offset + 8) != 0)
|
||
{
|
||
byte[] ekxdata = new byte[0xE8];
|
||
Array.Copy(savefile, offset, ekxdata, 0, 0xE8);
|
||
byte[] pkxdata = PKX.decryptArray(ekxdata);
|
||
int species = BitConverter.ToInt16(pkxdata, 0x08); // Get Species
|
||
if (species == 0)
|
||
{
|
||
System.Media.SystemSounds.Exclamation.Play();
|
||
return;
|
||
}
|
||
try
|
||
{
|
||
Array.Copy(pkxdata, buff, 0xE8);
|
||
populateFields(buff);
|
||
}
|
||
catch // If it fails, try XORing encrypted zeroes
|
||
{
|
||
try
|
||
{
|
||
byte[] blank = PKX.encryptArray(new byte[0xE8]);
|
||
|
||
for (int i = 0; i < 0xE8; i++)
|
||
blank[i] = (byte)(buff[i] ^ blank[i]);
|
||
|
||
populateFields(blank);
|
||
}
|
||
catch // Still fails, just let the original errors occur.
|
||
{ populateFields(buff); }
|
||
}
|
||
// Visual to display what slot is currently loaded.
|
||
getSlotColor(slot, Properties.Resources.slotView);
|
||
}
|
||
else
|
||
System.Media.SystemSounds.Exclamation.Play();
|
||
}
|
||
private void clickSet(object sender, EventArgs e)
|
||
{
|
||
if (!verifiedPKX()) { return; }
|
||
int slot = getSlot(sender);
|
||
if (slot == 30 && (CB_Species.SelectedIndex == 0 || CHK_IsEgg.Checked)) { Util.Alert("Can't have empty/egg first slot."); return; }
|
||
int offset = getPKXOffset(slot);
|
||
|
||
byte[] pkxdata = preparepkx();
|
||
byte[] ekxdata = PKX.encryptArray(pkxdata);
|
||
|
||
if (!savegame_oras)
|
||
{
|
||
// User Protection
|
||
bool move1 = BitConverter.ToInt16(pkxdata, 0x5A) > 617;
|
||
bool move2 = BitConverter.ToInt16(pkxdata, 0x5C) > 617;
|
||
bool move3 = BitConverter.ToInt16(pkxdata, 0x5E) > 617;
|
||
bool move4 = BitConverter.ToInt16(pkxdata, 0x60) > 617;
|
||
bool ability = pkxdata[0x14] > 188;
|
||
bool item = BitConverter.ToInt16(pkxdata, 0x0A) > 717;
|
||
string err = "";
|
||
|
||
if (move1 || move2 || move3 || move4)
|
||
err = "Move does not exist in X/Y.";
|
||
else if (ability)
|
||
err = "Ability does not exist in X/Y.";
|
||
else if (item)
|
||
err = "Item does not exist in X/Y.";
|
||
|
||
if ((err != "") && Util.Prompt(MessageBoxButtons.YesNo, err, "Continue?") != DialogResult.Yes)
|
||
return;
|
||
}
|
||
if (slot >= 30 && slot < 36) // Party
|
||
Array.Copy(ekxdata, 0, savefile, offset, 0x104);
|
||
else if (slot < 30 || (slot >= 36 && slot < 42 && DEV_Ability.Enabled))
|
||
Array.Copy(ekxdata, 0, savefile, offset, 0xE8);
|
||
else return;
|
||
|
||
setParty();
|
||
setPokedex(pkxdata);
|
||
getQuickFiller(getPictureBox(slot), pkxdata);
|
||
savedited = true;
|
||
|
||
getSlotColor(slot, Properties.Resources.slotSet);
|
||
}
|
||
private void clickDelete(object sender, EventArgs e)
|
||
{
|
||
int slot = getSlot(sender);
|
||
if (slot == 30 && setParty() == 1 && !DEV_Ability.Enabled) { Util.Alert("Can't delete first slot."); return; }
|
||
int offset = getPKXOffset(slot);
|
||
|
||
byte[] pkxdata = new byte[0x104];
|
||
byte[] ekxdata = PKX.encryptArray(pkxdata);
|
||
|
||
savedited = true;
|
||
|
||
if (slot >= 30 && slot < 36) // Party
|
||
{ Array.Copy(ekxdata, 0, savefile, offset, 0x104); setParty(); return; }
|
||
if (slot < 30 || (slot >= 36 && slot < 42 && DEV_Ability.Enabled))
|
||
{ Array.Copy(ekxdata, 0, savefile, offset, 0xE8); }
|
||
else return;
|
||
|
||
getQuickFiller(getPictureBox(slot), pkxdata);
|
||
getSlotColor(slot, Properties.Resources.slotDel);
|
||
}
|
||
private void clickClone(object sender, EventArgs e)
|
||
{
|
||
if (getSlot(sender) > 30) return; // only perform action if cloning to boxes
|
||
if (!verifiedPKX()) { return; } // don't copy garbage to the box
|
||
|
||
byte[] pkxdata;
|
||
int box = CB_BoxSelect.SelectedIndex + 1; // get box we're cloning to
|
||
{
|
||
if (Util.Prompt(MessageBoxButtons.YesNo, String.Format("Clone Pokemon from Editing Tabs to all slots in Box {0}?", box)) == DialogResult.Yes)
|
||
{
|
||
pkxdata = preparepkx();
|
||
setPokedex(pkxdata);
|
||
}
|
||
else if (Util.Prompt(MessageBoxButtons.YesNo, String.Format("Delete Pokemon from all slots in Box {0}?", box)) == DialogResult.Yes)
|
||
pkxdata = new Byte[0xE8];
|
||
else
|
||
return; // abort clone/delete
|
||
}
|
||
|
||
byte[] ekxdata = PKX.encryptArray(pkxdata);
|
||
for (int i = 0; i < 30; i++) // write encrypted array to all box slots
|
||
Array.Copy(ekxdata, 0, savefile, getPKXOffset(i), 0xE8);
|
||
|
||
for (int i = 0; i < 30; i++)
|
||
getQuickFiller(getPictureBox(i), pkxdata);
|
||
|
||
savedited = true;
|
||
}
|
||
// Generic Subfunctions //
|
||
private void setPokedex(byte[] pkxdata)
|
||
{
|
||
if (savindex > 1) return;
|
||
int species = BitConverter.ToUInt16(pkxdata, 0x8); // Species
|
||
int lang = pkxdata[0xE3] - 1; if (lang > 5) lang--; // 0-6 language vals
|
||
int origin = pkxdata[0xDF]; // Native / Non Native
|
||
int gender = (pkxdata[0x1D] & 2) >> 1; // Gender
|
||
uint pid = BitConverter.ToUInt32(pkxdata, 0x18);
|
||
ushort TID = BitConverter.ToUInt16(pkxdata, 0xC);
|
||
ushort SID = BitConverter.ToUInt16(pkxdata, 0xE);
|
||
int shiny = (PKX.getPSV(pid) ^ PKX.getTSV(TID, SID)) >> 4 == 0 ? 1 : 0;
|
||
int dexoff = savindex * 0x7F000 + SaveGame.PokeDex; // Same offset for XY-ORAS
|
||
int langoff = 0x3C8; if (savegame_oras) langoff = 0x400; // Not the same offset for language bools
|
||
int shiftoff = (shiny * 0x60 * 2) + (gender * 0x60) + 0x60;
|
||
|
||
// Set the [Species/Gender/Shiny] Owned Flag
|
||
savefile[dexoff + shiftoff + (species - 1) / 8 + 0x8] |= (byte)(1 << ((species - 1) % 8));
|
||
|
||
// Owned quality flag
|
||
if (origin < 0x18 && species < 650 && !savegame_oras) // Pre 650 for X/Y, and not for ORAS; Set the Foreign Owned Flag
|
||
savefile[0x1AA4C + 0x7F000 * savindex + (species - 1) / 8] |= (byte)(1 << ((species - 1) % 8));
|
||
else if (origin >= 0x18 || savegame_oras) // Set Native Owned Flag (should always happen)
|
||
savefile[dexoff + (species - 1) / 8 + 0x8] |= (byte)(1 << ((species - 1) % 8));
|
||
|
||
// Set the Language
|
||
if (lang < 0) lang = 1;
|
||
savefile[dexoff + langoff + ((species - 1) * 7 + lang) / 8] |= (byte)(1 << ((((species - 1) * 7) + lang) % 8));
|
||
}
|
||
private byte setParty()
|
||
{
|
||
byte partymembers = 0; // start off with a ctr of 0
|
||
int offset = SaveGame.Party + 0x7F000 * savindex;
|
||
for (int i = 0; i < 6; i++)
|
||
{
|
||
// Gather all the species
|
||
byte[] data = new byte[0x104];
|
||
Array.Copy(savefile, offset + i * 0x104, data, 0, 0x104);
|
||
byte[] decdata = PKX.decryptArray(data);
|
||
int species = BitConverter.ToInt16(decdata, 8);
|
||
if ((species != 0) && (species < 722))
|
||
Array.Copy(data, 0, savefile, offset + (partymembers++) * 0x104, 0x104);
|
||
}
|
||
|
||
// Write in the current party count
|
||
savefile[offset + 6 * 0x104 + savindex * 0x7F000] = partymembers;
|
||
// Zero out the party slots that are empty.
|
||
for (int i = 0; i < 6; i++)
|
||
if (i >= partymembers)
|
||
Array.Copy(PKX.encryptArray(new byte[0x104]), 0, savefile, offset + (i * 0x104), 0x104);
|
||
|
||
// Repeat for Battle Box.
|
||
byte battlemem = 0;
|
||
int offset2 = SaveGame.BattleBox + 0x7F000 * savindex;
|
||
for (int i = 0; i < 6; i++)
|
||
{
|
||
// Gather all the species
|
||
byte[] data = new byte[0x104];
|
||
Array.Copy(savefile, offset2 + i * 0xE8, data, 0, 0xE8);
|
||
byte[] decdata = PKX.decryptArray(data);
|
||
int species = BitConverter.ToInt16(decdata, 8);
|
||
if ((species != 0) && (species < 722))
|
||
Array.Copy(data, 0, savefile, offset2 + (battlemem++) * 0xE8, 0xE8);
|
||
}
|
||
|
||
// Zero out the party slots that are empty.
|
||
for (int i = 0; i < 6; i++)
|
||
if (i >= battlemem)
|
||
Array.Copy(PKX.encryptArray(new byte[0x104]), 0, savefile, offset2 + (i * 0xE8), 0xE8);
|
||
|
||
if (battlemem == 0)
|
||
savefile[offset2 + 6 * 0xE8 + savindex * 0x7F000] = 0;
|
||
|
||
// Refresh slots
|
||
for (int i = 0; i < 6; i++)
|
||
{
|
||
getQuickFiller(getPictureBox(i + 30), PKX.decryptArray(savefile.Skip(SaveGame.Party + 0x7F000 * savindex + 260 * i).Take(232).ToArray()));
|
||
getQuickFiller(getPictureBox(i + 36), PKX.decryptArray(savefile.Skip(SaveGame.BattleBox + 0x7F000 * savindex + 232 * i).Take(232).ToArray()));
|
||
}
|
||
|
||
return partymembers;
|
||
}
|
||
private int getPKXOffset(int slot)
|
||
{
|
||
int offset = SaveGame.Box + CB_BoxSelect.SelectedIndex * (0xE8 * 30) + slot * 0xE8;
|
||
|
||
if (slot > 29) // Not a party
|
||
{
|
||
if (slot < 36) // Party Slot
|
||
offset = SaveGame.Party + (slot - 30) * 0x104;
|
||
else if (slot < 42) // Battle Box Slot
|
||
offset = SaveGame.BattleBox + (slot - 36) * 0xE8;
|
||
else if (slot < 44) // Daycare
|
||
offset = SaveGame.Daycare + 8 + (slot - 42) * 0xF0;
|
||
else if (slot < 45) // GTS
|
||
offset = SaveGame.GTS;
|
||
else if (slot < 46) // Fused
|
||
offset = SaveGame.Fused;
|
||
else // SUBE
|
||
offset = SaveGame.SUBE + (slot - 46) * 0xEC;
|
||
}
|
||
offset += 0x7F000 * savindex;
|
||
return offset;
|
||
}
|
||
private int getSlot(object sender)
|
||
{
|
||
string name = (sender is ToolStripItem)
|
||
?
|
||
((sender as ToolStripItem).Owner as ContextMenuStrip).SourceControl.Name
|
||
:
|
||
(sender as PictureBox).Name;
|
||
|
||
string[] pba = {
|
||
"bpkx1", "bpkx2", "bpkx3", "bpkx4", "bpkx5", "bpkx6",
|
||
"bpkx7", "bpkx8", "bpkx9", "bpkx10","bpkx11","bpkx12",
|
||
"bpkx13","bpkx14","bpkx15","bpkx16","bpkx17","bpkx18",
|
||
"bpkx19","bpkx20","bpkx21","bpkx22","bpkx23","bpkx24",
|
||
"bpkx25","bpkx26","bpkx27","bpkx28","bpkx29","bpkx30",
|
||
|
||
"ppkx1", "ppkx2", "ppkx3", "ppkx4", "ppkx5", "ppkx6",
|
||
"bbpkx1","bbpkx2","bbpkx3","bbpkx4","bbpkx5","bbpkx6",
|
||
|
||
"dcpkx1", "dcpkx2", "gtspkx", "fusedpkx","subepkx1","subepkx2","subepkx3",
|
||
};
|
||
int slot = Array.IndexOf(pba, name);
|
||
return slot;
|
||
}
|
||
private void setPKXBoxes()
|
||
{
|
||
int boxoffset = SaveGame.Box + 0x7F000 * savindex + CB_BoxSelect.SelectedIndex * (0xE8 * 30);
|
||
|
||
int boxbgofst = (0x7F000 * savindex) + 0x9C1E + CB_BoxSelect.SelectedIndex;
|
||
int boxbgval = 1 + savefile[boxbgofst];
|
||
string imagename = "box_wp" + boxbgval.ToString("00"); if (savegame_oras && boxbgval > 16) imagename += "o";
|
||
PAN_Box.BackgroundImage = (Image)Properties.Resources.ResourceManager.GetObject(imagename);
|
||
|
||
PictureBox[] pba = {
|
||
bpkx1, bpkx2, bpkx3, bpkx4, bpkx5, bpkx6,
|
||
bpkx7, bpkx8, bpkx9, bpkx10,bpkx11,bpkx12,
|
||
bpkx13,bpkx14,bpkx15,bpkx16,bpkx17,bpkx18,
|
||
bpkx19,bpkx20,bpkx21,bpkx22,bpkx23,bpkx24,
|
||
bpkx25,bpkx26,bpkx27,bpkx28,bpkx29,bpkx30,
|
||
|
||
ppkx1, ppkx2, ppkx3, ppkx4, ppkx5, ppkx6,
|
||
bbpkx1,bbpkx2,bbpkx3,bbpkx4,bbpkx5,bbpkx6,
|
||
|
||
dcpkx1, dcpkx2, gtspkx, fusedpkx,subepkx1,subepkx2,subepkx3,
|
||
};
|
||
for (int i = 0; i < 30; i++)
|
||
getSlotFiller(boxoffset + 0xE8 * i, pba[i]);
|
||
|
||
// Reload Party
|
||
for (int i = 0; i < 6; i++)
|
||
getSlotFiller(SaveGame.Party + (0x7F000 * savindex) + 0x104 * i, pba[i + 30]);
|
||
|
||
// Reload Battle Box
|
||
for (int i = 0; i < 6; i++)
|
||
getSlotFiller(SaveGame.BattleBox + (0x7F000 * savindex) + 0xE8 * i, pba[i + 36]);
|
||
|
||
// Reload Daycare
|
||
Label[] dclabela = { L_DC1, L_DC2, };
|
||
TextBox[] dctexta = { TB_Daycare1XP, TB_Daycare2XP };
|
||
|
||
for (int i = 0; i < 2; i++)
|
||
{
|
||
getSlotFiller(SaveGame.Daycare + (0x7F000 * savindex) + 0xE8 * i + 8 * (i + 1), pba[i + 42]);
|
||
dctexta[i].Text = BitConverter.ToUInt32(savefile, SaveGame.Daycare + (0x7F000 * savindex) + 0xF0 * i + 4).ToString();
|
||
if (Convert.ToBoolean(savefile[SaveGame.Daycare + (0x7F000 * savindex) + 0xF0 * i])) // If Occupied
|
||
dclabela[i].Text = (i + 1) + ": ✓";
|
||
else
|
||
{
|
||
dclabela[i].Text = (i + 1) + ": ✘";
|
||
pba[i + 42].Image = Util.ChangeOpacity(pba[i + 42].Image, 0.6);
|
||
}
|
||
}
|
||
DayCare_HasEgg.Checked = Convert.ToBoolean(savefile[SaveGame.Daycare + (0x7F000 * savindex) + 0x1E0]);
|
||
TB_RNGSeed.Text = BitConverter.ToUInt64(savefile, SaveGame.Daycare + (0x7F000 * savindex) + 0x1E8).ToString("X16");
|
||
|
||
// GTS
|
||
getSlotFiller(SaveGame.GTS + (0x7F000 * savindex), pba[44]);
|
||
|
||
// Fused
|
||
getSlotFiller(SaveGame.Fused + (0x7F000 * savindex), pba[45]);
|
||
|
||
// SUBE
|
||
for (int i = 0; i < 3; i++)
|
||
{
|
||
int offset = 0x22C90 + i * 0xEC + (0x7F000 * savindex);
|
||
if (BitConverter.ToUInt64(savefile, offset) != 0)
|
||
getSlotFiller(offset, pba[46 + i]);
|
||
else pba[46 + i].Image = null;
|
||
}
|
||
|
||
// Recoloring of a storage box slot (to not show for other storage boxes)
|
||
if (colorizedslot < 32)
|
||
pba[colorizedslot].BackgroundImage = (colorizedbox == CB_BoxSelect.SelectedIndex) ? colorizedcolor : null;
|
||
}
|
||
private void setBoxNames()
|
||
{
|
||
int selectedbox = CB_BoxSelect.SelectedIndex; // precache selected box
|
||
// Build ComboBox Dropdown Items
|
||
try
|
||
{
|
||
CB_BoxSelect.Items.Clear();
|
||
for (int i = 0; i < 31; i++)
|
||
CB_BoxSelect.Items.Add(Encoding.Unicode.GetString(savefile, SaveGame.PCLayout + (0x7F000 * savindex) + 0x22 * i, 0x22));
|
||
}
|
||
catch
|
||
{
|
||
CB_BoxSelect.Items.Clear();
|
||
for (int i = 1; i < 32; i++)
|
||
CB_BoxSelect.Items.Add("Box " + i);
|
||
}
|
||
CB_BoxSelect.SelectedIndex = selectedbox; // restore selected box
|
||
}
|
||
private void setSAVLabel()
|
||
{
|
||
L_SAVINDEX.Text = (savindex + 1).ToString();
|
||
RTB_S.AppendText("Loaded Save File " + (savindex + 1) + Environment.NewLine);
|
||
}
|
||
private void getSAVOffsets(ref bool oras)
|
||
{
|
||
// Get the save file offsets for the input game
|
||
bool enableInterface = false;
|
||
if (BitConverter.ToUInt32(savefile, 0x6A810 + 0x7F000 * savindex) == 0x42454546)
|
||
{
|
||
enableInterface = true;
|
||
SaveGame = new PKX.Structures.SaveGame("XY");
|
||
}
|
||
else if (BitConverter.ToUInt32(savefile, 0x7B210 + 0x7F000 * savindex) == 0x42454546)
|
||
{
|
||
enableInterface = true;
|
||
SaveGame = new PKX.Structures.SaveGame("ORAS");
|
||
oras = true;
|
||
}
|
||
else
|
||
{
|
||
Util.Error("Unrecognized Save File loaded.");
|
||
SaveGame = new PKX.Structures.SaveGame("Error");
|
||
}
|
||
|
||
// Enable Buttons
|
||
GB_SAVtools.Enabled = B_JPEG.Enabled = B_VerifyCHK.Enabled = B_VerifySHA.Enabled = B_SwitchSAV.Enabled
|
||
= enableInterface;
|
||
}
|
||
private PictureBox getPictureBox(int slot)
|
||
{
|
||
PictureBox[] pba = {
|
||
bpkx1, bpkx2, bpkx3, bpkx4, bpkx5, bpkx6,
|
||
bpkx7, bpkx8, bpkx9, bpkx10,bpkx11,bpkx12,
|
||
bpkx13,bpkx14,bpkx15,bpkx16,bpkx17,bpkx18,
|
||
bpkx19,bpkx20,bpkx21,bpkx22,bpkx23,bpkx24,
|
||
bpkx25,bpkx26,bpkx27,bpkx28,bpkx29,bpkx30,
|
||
|
||
ppkx1, ppkx2, ppkx3, ppkx4, ppkx5, ppkx6,
|
||
bbpkx1,bbpkx2,bbpkx3,bbpkx4,bbpkx5,bbpkx6,
|
||
|
||
dcpkx1, dcpkx2, gtspkx, fusedpkx,subepkx1,subepkx2,subepkx3,
|
||
};
|
||
return pba[slot];
|
||
}
|
||
private void getQuickFiller(PictureBox pb, byte[] dslotdata = null)
|
||
{
|
||
if (!init) return;
|
||
dslotdata = dslotdata ?? preparepkx(false); // don't perform control loss click
|
||
|
||
if (pb == dragout) L_QR.Visible = (BitConverter.ToInt16(dslotdata, 0x08) != 0); // Species
|
||
pb.Image = PKX.getSprite(dslotdata);
|
||
}
|
||
private void getSlotFiller(int offset, PictureBox pb)
|
||
{
|
||
byte[] slotdata = new byte[0xE8];
|
||
Array.Copy(savefile, offset, slotdata, 0, 0xE8); // Fill Our EKX Slot
|
||
byte[] dslotdata = PKX.decryptArray(slotdata);
|
||
|
||
if (PKX.getCHK(dslotdata) != BitConverter.ToUInt16(dslotdata, 6) // Invalid Checksum
|
||
&& (!savLoaded && !slotdata.SequenceEqual(new byte[0xE8]))) // And Save Loaded
|
||
{
|
||
// Bad Egg present in slot.
|
||
pb.Image = null;
|
||
pb.BackColor = Color.Red;
|
||
return;
|
||
}
|
||
pb.BackColor = Color.Transparent;
|
||
pb.Image = PKX.getSprite(dslotdata);
|
||
}
|
||
private void getSlotColor(int slot, Image color)
|
||
{
|
||
PictureBox[] pba = {
|
||
bpkx1, bpkx2, bpkx3, bpkx4, bpkx5, bpkx6,
|
||
bpkx7, bpkx8, bpkx9, bpkx10,bpkx11,bpkx12,
|
||
bpkx13,bpkx14,bpkx15,bpkx16,bpkx17,bpkx18,
|
||
bpkx19,bpkx20,bpkx21,bpkx22,bpkx23,bpkx24,
|
||
bpkx25,bpkx26,bpkx27,bpkx28,bpkx29,bpkx30,
|
||
|
||
ppkx1, ppkx2, ppkx3, ppkx4, ppkx5, ppkx6,
|
||
bbpkx1,bbpkx2,bbpkx3,bbpkx4,bbpkx5,bbpkx6,
|
||
|
||
dcpkx1, dcpkx2, gtspkx, fusedpkx,subepkx1,subepkx2,subepkx3,
|
||
};
|
||
|
||
foreach (PictureBox t in pba)
|
||
t.BackgroundImage = null;
|
||
|
||
if (slot < 32)
|
||
colorizedbox = CB_BoxSelect.SelectedIndex;
|
||
|
||
pba[slot].BackgroundImage = color;
|
||
colorizedcolor = color;
|
||
colorizedslot = slot;
|
||
}
|
||
private void getBox(object sender, EventArgs e)
|
||
{
|
||
setPKXBoxes();
|
||
savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] = (byte)CB_BoxSelect.SelectedIndex;
|
||
}
|
||
private void getTSV(object sender, EventArgs e)
|
||
{
|
||
uint tsv = PKX.getTSV(Util.ToUInt32(TB_TID.Text), Util.ToUInt32(TB_SID.Text));
|
||
Tip1.SetToolTip(TB_TID, "TSV: " + tsv.ToString("0000"));
|
||
Tip2.SetToolTip(TB_SID, "TSV: " + tsv.ToString("0000"));
|
||
|
||
uint psv = PKX.getPSV(Util.getHEXval(TB_PID));
|
||
Tip3.SetToolTip(TB_PID, "PSV: " + psv.ToString("0000"));
|
||
}
|
||
|
||
private void getNatureModification(object sender, EventArgs e)
|
||
{
|
||
if (sender is ComboBox && (sender as ComboBox).Name != CB_Nature.Name) return;
|
||
int nature = Util.getIndex(CB_Nature);
|
||
int incr = nature / 5;
|
||
int decr = nature % 5;
|
||
|
||
Label[] labarray = { Label_ATK, Label_DEF, Label_SPE, Label_SPA, Label_SPD };
|
||
// Reset Label Colors
|
||
foreach (Label label in labarray)
|
||
label.ForeColor = defaultControlText;
|
||
|
||
// Set Colored StatLabels only if Nature isn't Neutral
|
||
NatureTip.SetToolTip(CB_Nature,
|
||
incr != decr
|
||
? String.Format("+{0} / -{1}", labarray[incr].Text, labarray[decr].Text).Replace(":", "")
|
||
: "-/-");
|
||
}
|
||
private void switchDaycare(object sender, EventArgs e)
|
||
{
|
||
if (!savegame_oras) return;
|
||
if (DialogResult.Yes == Util.Prompt(MessageBoxButtons.YesNo, "Would you like to switch the view to the other Daycare?", String.Format("Currently viewing daycare {0}.", SaveGame.Daycare / 0x211F0 + 1)))
|
||
// If ORAS, alter the daycare offset via toggle.
|
||
SaveGame.Daycare = (SaveGame.Daycare == 0x21000) ? 0x211F0 : 0x21000;
|
||
|
||
// Refresh Boxes
|
||
setPKXBoxes();
|
||
}
|
||
private void mainMenuBoxDumpLoad(object sender, EventArgs e)
|
||
{
|
||
DialogResult dr = Util.Prompt(MessageBoxButtons.YesNoCancel, "Press Yes to Import All from Folder." + Environment.NewLine + "Press No to Dump All to Folder.", "Press Cancel to Abort.");
|
||
if (dr == DialogResult.Cancel) return;
|
||
string exepath = Application.StartupPath;
|
||
string path = "";
|
||
{
|
||
int offset = SaveGame.Box;
|
||
const int size = 232;
|
||
if (dr == DialogResult.Yes) // Import
|
||
{
|
||
if (Directory.Exists(Path.Combine(exepath, "db")))
|
||
{
|
||
DialogResult ld = Util.Prompt(MessageBoxButtons.YesNo, "Load from PKHeX's database?");
|
||
if (ld == DialogResult.Yes)
|
||
path = Path.Combine(exepath, "db");
|
||
else if (ld == DialogResult.No)
|
||
{
|
||
// open folder dialog
|
||
FolderBrowserDialog fbd = new FolderBrowserDialog();
|
||
if (fbd.ShowDialog() == DialogResult.OK)
|
||
path = fbd.SelectedPath;
|
||
}
|
||
else return;
|
||
}
|
||
else
|
||
{
|
||
// open folder dialog
|
||
FolderBrowserDialog fbd = new FolderBrowserDialog();
|
||
if (fbd.ShowDialog() == DialogResult.OK)
|
||
path = fbd.SelectedPath;
|
||
}
|
||
loadBoxesFromDB(path);
|
||
}
|
||
else if (dr == DialogResult.No)
|
||
{
|
||
// Dump all of box content to files.
|
||
{
|
||
DialogResult ld = Util.Prompt(MessageBoxButtons.YesNo, "Save to PKHeX's database?");
|
||
if (ld == DialogResult.Yes)
|
||
{
|
||
path = Path.Combine(exepath, "db");
|
||
if (!Directory.Exists(path))
|
||
Directory.CreateDirectory(path);
|
||
}
|
||
else if (ld == DialogResult.No)
|
||
{
|
||
// open folder dialog
|
||
FolderBrowserDialog fbd = new FolderBrowserDialog();
|
||
if (fbd.ShowDialog() == DialogResult.OK)
|
||
path = fbd.SelectedPath;
|
||
}
|
||
else return;
|
||
}
|
||
for (int i = 0; i < 31 * 30 * size; i += size)
|
||
{
|
||
byte[] ekxdata = new byte[size];
|
||
Array.Copy(savefile, offset + i, ekxdata, 0, size);
|
||
byte[] pkxdata = PKX.decryptArray(ekxdata);
|
||
|
||
|
||
int species = BitConverter.ToInt16(pkxdata, 0x08);
|
||
if (species == 0) continue;
|
||
uint chk = BitConverter.ToUInt16(pkxdata, 0x06);
|
||
uint EC = BitConverter.ToUInt32(pkxdata, 0);
|
||
uint IV32 = BitConverter.ToUInt32(pkxdata, 0x74);
|
||
|
||
string nick;
|
||
if (Convert.ToBoolean((IV32 >> 31) & 1))
|
||
nick = Util.TrimFromZero(Encoding.Unicode.GetString(pkxdata, 0x40, 24)) + " (" + specieslist[species] + ")";
|
||
else
|
||
nick = specieslist[species];
|
||
if (Convert.ToBoolean((IV32 >> 30) & 1))
|
||
nick += " (" + eggname + ")";
|
||
|
||
string isshiny = "";
|
||
|
||
// Is Shiny
|
||
if (PKX.getIsShiny(BitConverter.ToUInt32(pkxdata, 0x18), Util.ToUInt32(TB_TID.Text), Util.ToUInt32(TB_SID.Text)))
|
||
isshiny = " ★";
|
||
|
||
string savedname =
|
||
species.ToString("000") + isshiny + " - "
|
||
+ nick + " - "
|
||
+ chk.ToString("X4") + EC.ToString("X8")
|
||
+ ".pk6";
|
||
Array.Resize(ref pkxdata, 232);
|
||
if (!File.Exists(Path.Combine(path, savedname)))
|
||
File.WriteAllBytes(Path.Combine(path, Util.CleanFileName(savedname)), pkxdata);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
private void loadBoxesFromDB(string path)
|
||
{
|
||
if (path == "") return;
|
||
int offset = SaveGame.Box;
|
||
int ctr = CB_BoxSelect.SelectedIndex * 30;
|
||
int pastctr = 0;
|
||
|
||
// Clear out the box data array.
|
||
// Array.Clear(savefile, offset, size * 30 * 31);
|
||
DialogResult dr = Util.Prompt(MessageBoxButtons.YesNoCancel, "Clear subsequent boxes when importing data?", "If you only want to overwrite for new data, press no.");
|
||
if (dr == DialogResult.Cancel) return;
|
||
if (dr == DialogResult.Yes)
|
||
{
|
||
byte[] ezeros = PKX.encryptArray(new byte[232]);
|
||
for (int i = ctr; i < 30 * 31; i++)
|
||
Array.Copy(ezeros, 0, savefile, offset + i * 232, 232);
|
||
}
|
||
string[] filepaths = Directory.GetFiles(path, "*.*", SearchOption.TopDirectoryOnly);
|
||
var Converter = new pk2pk();
|
||
|
||
foreach (string t in filepaths)
|
||
{
|
||
long len = new FileInfo(t).Length;
|
||
if (len > 260)
|
||
continue;
|
||
if (
|
||
len != 232 && len != 260 // 6th Gen
|
||
&& len != 136 && len != 220 && len != 236 // 5th Gen
|
||
&& len != 100 && len != 80) // 4th Gen
|
||
continue;
|
||
byte[] data = new byte[232];
|
||
switch (Path.GetExtension(t)) // Filter all files by extension
|
||
{
|
||
case ".pk5":
|
||
case ".pk4":
|
||
case ".pk3":
|
||
case ".3gpkm":
|
||
case ".pkm":
|
||
{
|
||
// Verify PKM (decrypted)
|
||
byte[] input = File.ReadAllBytes(t);
|
||
if (!PKX.verifychk(input)) continue;
|
||
{
|
||
try // to convert g5pkm
|
||
{ data = PKX.encryptArray(Converter.ConvertPKM(input, savefile, savindex)); pastctr++; }
|
||
catch { continue; }
|
||
}
|
||
}
|
||
break;
|
||
case ".pk6":
|
||
case ".pkx":
|
||
{
|
||
byte[] input = File.ReadAllBytes(t);
|
||
if ((BitConverter.ToUInt16(input, 0xC8) == 0) && (BitConverter.ToUInt16(input, 0x58) == 0))
|
||
{
|
||
if (BitConverter.ToUInt16(input, 0x8) == 0) // if species = 0
|
||
continue;
|
||
Array.Resize(ref input, 232);
|
||
|
||
if (PKX.getCHK(input) != BitConverter.ToUInt16(input, 0x6)) continue;
|
||
data = PKX.encryptArray(input);
|
||
}
|
||
}
|
||
break;
|
||
case ".ek6":
|
||
case ".ekx":
|
||
{
|
||
byte[] input = File.ReadAllBytes(t);
|
||
Array.Resize(ref input, 232);
|
||
Array.Copy(input, data, 232);
|
||
// check if it is good data
|
||
byte[] decrypteddata = PKX.decryptArray(input);
|
||
|
||
if (BitConverter.ToUInt16(decrypteddata, 0xC8) != 0 && BitConverter.ToUInt16(decrypteddata, 0x58) != 0)
|
||
continue; // don't allow improperly encrypted files. they must be encrypted properly.
|
||
//else if (BitConverter.ToUInt16(decrypteddata, 0x8) == 0) // if species = 0
|
||
// continue;
|
||
// We'll allow blank ekx files for those wanting to see the decrypted data.
|
||
if (PKX.getCHK(input) != BitConverter.ToUInt16(decrypteddata, 0x6)) continue;
|
||
}
|
||
break;
|
||
default:
|
||
continue;
|
||
}
|
||
Array.Copy(data, 0, savefile, offset + ctr++ * 232, 232);
|
||
setPokedex(PKX.decryptArray(data)); // Set the Pokedex data
|
||
if (ctr == 30 * 31) break; // break out if we have written all 31 boxes
|
||
}
|
||
if (ctr <= 0) return;
|
||
// if we've written at least one pk6 in, go ahead and make sure the window is stretched.
|
||
if (Width < Height) // expand if boxes aren't visible
|
||
{
|
||
Width = largeWidth;
|
||
tabBoxMulti.SelectedIndex = 0;
|
||
}
|
||
setPKXBoxes();
|
||
string result = String.Format("Loaded {0} files to boxes.", ctr);
|
||
if (pastctr > 0)
|
||
Util.Alert(result, String.Format("Conversion successful for {0} past generation files.", pastctr));
|
||
else
|
||
Util.Alert(result);
|
||
}
|
||
private void B_SaveBoxBin_Click(object sender, EventArgs e)
|
||
{
|
||
DialogResult dr = Util.Prompt(MessageBoxButtons.YesNoCancel, "Yes: Export All Boxes" + Environment.NewLine + String.Format("No: Export {1} (Box {0})", CB_BoxSelect.SelectedIndex + 1, CB_BoxSelect.Text) + Environment.NewLine + "Cancel: Abort");
|
||
|
||
if (dr == DialogResult.Yes)
|
||
{
|
||
SaveFileDialog sfd = new SaveFileDialog {Filter = "Box Data|*.bin", FileName = "pcdata.bin"};
|
||
if (sfd.ShowDialog() == DialogResult.OK)
|
||
File.WriteAllBytes(sfd.FileName, savefile.Skip(SaveGame.Box).Take(0xE8 * 30 * 31).ToArray());
|
||
}
|
||
else if (dr == DialogResult.No)
|
||
{
|
||
SaveFileDialog sfd = new SaveFileDialog {Filter = "Box Data|*.bin", FileName = "boxdata.bin"};
|
||
if (sfd.ShowDialog() == DialogResult.OK)
|
||
File.WriteAllBytes(sfd.FileName, savefile.Skip(SaveGame.Box + 0xE8 * 30 * CB_BoxSelect.SelectedIndex).Take(0xE8 * 30).ToArray());
|
||
}
|
||
}
|
||
// Subfunction Save Buttons //
|
||
private void B_OpenWondercards_Click(object sender, EventArgs e)
|
||
{
|
||
// Open Wondercard Menu
|
||
new SAV_Wondercard(this).ShowDialog();
|
||
}
|
||
private void B_OpenBoxLayout_Click(object sender, EventArgs e)
|
||
{
|
||
// Open Box Layout Menu
|
||
new SAV_BoxLayout(this).ShowDialog();
|
||
setBoxNames(); // fix box names
|
||
setPKXBoxes(); // refresh box background
|
||
}
|
||
private void B_OpenTrainerInfo_Click(object sender, EventArgs e)
|
||
{
|
||
new SAV_Trainer(this).ShowDialog();
|
||
}
|
||
private void B_OpenPokepuffs_Click(object sender, EventArgs e)
|
||
{
|
||
new SAV_Pokepuff(this).ShowDialog();
|
||
}
|
||
private void B_OpenItemPouch_Click(object sender, EventArgs e)
|
||
{
|
||
new SAV_Inventory(this).ShowDialog();
|
||
}
|
||
private void B_OpenBerryField_Click(object sender, EventArgs e)
|
||
{
|
||
if (savegame_oras)
|
||
{
|
||
DialogResult dr = Util.Prompt(MessageBoxButtons.YesNo, "No editing support for ORAS :(", "Repopulate all with random berries?");
|
||
if (dr != DialogResult.Yes) return; // abort
|
||
// Randomize the trees.
|
||
int offset = 0x1C400 + 0x5400 + savindex * 0x7F000;
|
||
byte[] ready = { 0x05, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x80, 0x40, 0x01, 0x00, 0x00, 0x00, };
|
||
int[] berrylist =
|
||
{
|
||
0,149,150,151,152,153,154,155,156,157,158,159,160,161,162,
|
||
163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,
|
||
178,179,180,181,182,183,184,185,186,187,188,189,190,191,192,
|
||
193,194,195,196,197,198,199,200,201,202,203,204,205,206,207,
|
||
208,209,210,211,212,686,687,688,
|
||
};
|
||
for (int i = 0; i < 90; i++)
|
||
{
|
||
Array.Copy(ready, 0, savefile, offset + 0x10 * i, 0x10); // prep the berry template tree (which we replace offset 0x6 for the Tree Item)
|
||
int randberry = (int)(Util.rnd32() % berrylist.Length); // generate a random berry that will go into the tree
|
||
int index = berrylist[randberry]; // get berry item ID from list
|
||
Array.Copy(BitConverter.GetBytes(index), 0, savefile, offset + 0x10 * i + 6, 2); // put berry into tree.
|
||
}
|
||
}
|
||
else
|
||
new SAV_BerryField(this, SaveGame.BerryField).ShowDialog();
|
||
}
|
||
private void B_OpenEventFlags_Click(object sender, EventArgs e)
|
||
{
|
||
// Open Flag Menu
|
||
if (savegame_oras)
|
||
new SAV_EventFlagsORAS(this).ShowDialog();
|
||
else
|
||
new SAV_EventFlagsXY(this).ShowDialog();
|
||
}
|
||
private void B_OpenSuperTraining_Click(object sender, EventArgs e)
|
||
{
|
||
// Open ST Menu
|
||
new SAV_SuperTrain(this).ShowDialog();
|
||
}
|
||
private void B_OpenOPowers_Click(object sender, EventArgs e)
|
||
{
|
||
// Open O-Power Menu
|
||
if (savegame_oras)
|
||
{
|
||
DialogResult dr = Util.Prompt(MessageBoxButtons.YesNo, "No editing support for ORAS :(", "Max O-Powers with a working code?");
|
||
if (dr != DialogResult.Yes) return;
|
||
byte[] maxoras =
|
||
{
|
||
0x00, 0x01, 0x01, 0x01,
|
||
0x01, 0x00, 0x01, 0x01,
|
||
0x01, 0x01, 0x00, 0x01,
|
||
0x01, 0x01, 0x01, 0x00,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x00, 0x01, 0x01, 0x01,
|
||
0x01, 0x00, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x00, 0x01,
|
||
0x01, 0x01, 0x01, 0x00,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x01, 0x01, 0x01,
|
||
0x01, 0x00, 0x00, 0x00,
|
||
};
|
||
Array.Copy(maxoras, 0, savefile, 0x17400 + 0x5400 + 0x7F000 * savindex, 0x44);
|
||
}
|
||
else
|
||
new SAV_OPower(this).ShowDialog();
|
||
}
|
||
private void B_OpenPokedex_Click(object sender, EventArgs e)
|
||
{
|
||
// Open Pokedex Menu
|
||
if (savegame_oras)
|
||
new SAV_PokedexORAS(this).ShowDialog();
|
||
else
|
||
new SAV_PokedexXY(this).ShowDialog();
|
||
}
|
||
private void B_OUTPasserby_Click(object sender, EventArgs e)
|
||
{
|
||
string result = "";
|
||
result += "PSS List" + Environment.NewLine;
|
||
string[] headers = {
|
||
"PSS Data - Friends",
|
||
"PSS Data - Acquaintances",
|
||
"PSS Data - Passerby",
|
||
};
|
||
int offset = savindex * 0x7F000 + SaveGame.PSS;
|
||
for (int g = 0; g < 3; g++)
|
||
{
|
||
result += Environment.NewLine
|
||
+ "----" + Environment.NewLine
|
||
+ headers[g] + Environment.NewLine
|
||
+ "----" + Environment.NewLine;
|
||
// uint count = BitConverter.ToUInt32(savefile, offset + 0x4E20);
|
||
int r_offset = offset;
|
||
|
||
for (int i = 0; i < 100; i++)
|
||
{
|
||
ulong unkn = BitConverter.ToUInt64(savefile, r_offset);
|
||
if (unkn == 0) break; // No data present here
|
||
if (i > 0) result += Environment.NewLine + Environment.NewLine;
|
||
|
||
string otname = Util.TrimFromZero(Encoding.Unicode.GetString(savefile, r_offset + 8, 0x1A));
|
||
string message = Util.TrimFromZero(Encoding.Unicode.GetString(savefile, r_offset + 0x22, 0x22));
|
||
|
||
// Trim terminated
|
||
|
||
// uint unk1 = BitConverter.ToUInt32(savefile, r_offset + 0x44);
|
||
// ulong unk2 = BitConverter.ToUInt64(savefile, r_offset + 0x48);
|
||
// uint unk3 = BitConverter.ToUInt32(savefile, r_offset + 0x50);
|
||
// uint unk4 = BitConverter.ToUInt16(savefile, r_offset + 0x54);
|
||
byte region = savefile[r_offset + 0x56];
|
||
byte country = savefile[r_offset + 0x57];
|
||
byte game = savefile[r_offset + 0x5A];
|
||
// ulong outfit = BitConverter.ToUInt64(savefile, r_offset + 0x5C);
|
||
int favpkm = BitConverter.ToUInt16(savefile, r_offset + 0x9C) & 0x7FF;
|
||
string gamename;
|
||
try { gamename = gamelist[game]; }
|
||
catch { gamename = "UNKNOWN GAME"; }
|
||
|
||
string[] cr = PKX.getCountryRegionText(country, region, curlanguage);
|
||
result +=
|
||
"OT: " + otname + Environment.NewLine +
|
||
"Message: " + message + Environment.NewLine +
|
||
"Game: " + gamename + Environment.NewLine +
|
||
"Country: " + cr[0] + Environment.NewLine +
|
||
"Region: " + cr[1] + Environment.NewLine +
|
||
"Favorite: " + specieslist[favpkm];
|
||
|
||
r_offset += 0xC8; // Advance to next entry
|
||
}
|
||
offset += 0x5000; // Advance to next block
|
||
}
|
||
RTB_T.Text = result;
|
||
RTB_T.Font = new Font("Courier New", 8);
|
||
tabBoxMulti.SelectedIndex = 3;
|
||
}
|
||
private void B_OUTHallofFame_Click(object sender, EventArgs e)
|
||
{
|
||
// Open HoF Menu
|
||
new SAV_HallOfFame(this).ShowDialog();
|
||
}
|
||
private void B_OpenSecretBase_Click(object sender, EventArgs e)
|
||
{
|
||
// Open Secret Base Menu
|
||
new SAV_SecretBase(this).ShowDialog();
|
||
}
|
||
private void B_JPEG_Click(object sender, EventArgs e)
|
||
{
|
||
int offset = 0x7F000 * savindex + SaveGame.JPEG;
|
||
|
||
string filename = Encoding.Unicode.GetString(savefile, offset + 0, 0x1A).Replace("\0", string.Empty);
|
||
filename += "'s picture";
|
||
offset += 0x54;
|
||
if (savefile[offset] != 0xFF)
|
||
{
|
||
Util.Alert("No PGL picture data found in the save file!");
|
||
return;
|
||
}
|
||
const int length = 0xE004;
|
||
|
||
byte[] jpeg = new byte[length];
|
||
Array.Copy(savefile, offset, jpeg, 0, length);
|
||
SaveFileDialog savejpeg = new SaveFileDialog {FileName = filename, Filter = "JPEG|*.jpeg"};
|
||
if (savejpeg.ShowDialog() != DialogResult.OK) return;
|
||
string path = savejpeg.FileName;
|
||
if (File.Exists(path))
|
||
{
|
||
// File already exists, save a .bak
|
||
byte[] backupfile = File.ReadAllBytes(path);
|
||
File.WriteAllBytes(path + ".bak", backupfile);
|
||
}
|
||
File.WriteAllBytes(path, jpeg);
|
||
}
|
||
// Save Folder Related
|
||
private void clickSaveFileName(object sender, EventArgs e)
|
||
{
|
||
// Get latest SaveDataFiler save location
|
||
pathSDF = Util.GetSDFLocation();
|
||
string path = null;
|
||
|
||
if (pathSDF != null && ModifierKeys != Keys.Control) // if we have a result
|
||
path = Path.Combine(pathSDF, "main");
|
||
else if (File.Exists(Util.NormalizePath(Path.Combine(Util.GetTempFolder(), "root" + Path.DirectorySeparatorChar + "main")))) // else if cgse exists
|
||
path = Util.NormalizePath(Path.Combine(Util.GetTempFolder(), "root" + Path.DirectorySeparatorChar + "main"));
|
||
|
||
if (path == null) return;
|
||
if (Util.Prompt(MessageBoxButtons.YesNo, "Open save file from the following location?", path) == DialogResult.Yes)
|
||
openQuick(path); // load save
|
||
}
|
||
private void clickOpenTempFolder(object sender, EventArgs e)
|
||
{
|
||
string path;
|
||
switch (ModifierKeys)
|
||
{
|
||
case Keys.Control: // Cache
|
||
path = Util.GetCacheFolder();
|
||
if (Directory.Exists(path)) System.Diagnostics.Process.Start("explorer.exe", @path); else Util.Alert("Can't find the cache folder.");
|
||
break;
|
||
case Keys.Alt: // SaveDataFiler
|
||
path = Util.GetSDFLocation();
|
||
if (Directory.Exists(path)) System.Diagnostics.Process.Start("explorer.exe", @path); else Util.Alert("Can't find the SDF folder.");
|
||
break;
|
||
default: // Root
|
||
path = Util.GetTempFolder();
|
||
if (Directory.Exists(Path.Combine(path, "root"))) System.Diagnostics.Process.Start("explorer.exe", @Path.Combine(path, "root"));
|
||
else if (Directory.Exists(path)) System.Diagnostics.Process.Start("explorer.exe", @path);
|
||
else { Util.Error("Can't find the temporary file.", "Make sure the Cyber Gadget software is paused."); }
|
||
break;
|
||
}
|
||
}
|
||
|
||
private void clickSwitchSAV(object sender, EventArgs e)
|
||
{
|
||
DialogResult switchsav = Util.Prompt(MessageBoxButtons.YesNo, String.Format("Current Savefile is Save {0}.", (savindex + 1)), String.Format("Would you like to switch to Save {0}?", ((savindex + 1) % 2 + 1)));
|
||
if (switchsav != DialogResult.Yes) return;
|
||
savindex = (savindex + 1) % 2;
|
||
setBoxNames();
|
||
setPKXBoxes();
|
||
setSAVLabel();
|
||
CB_BoxSelect.SelectedIndex = savefile[SaveGame.PCLayout + savindex * 0x7FFFF + 0x43F] & 0x1F;
|
||
}
|
||
|
||
// Drag & Drop within Box
|
||
private void pbBoxSlot_MouseDown(object sender, MouseEventArgs e)
|
||
{
|
||
if (ModifierKeys == Keys.Control || ModifierKeys == Keys.Alt || ModifierKeys == Keys.Shift || ModifierKeys == (Keys.Control | Keys.Alt))
|
||
{ clickSlot(sender, e); return; }
|
||
PictureBox pb = (PictureBox)(sender);
|
||
if (pb.Image == null)
|
||
return;
|
||
|
||
pkm_from_slot = getSlot(sender);
|
||
int offset = getPKXOffset(pkm_from_slot);
|
||
if (e.Button != MouseButtons.Left || e.Clicks != 1) return;
|
||
// Create Temp File to Drag
|
||
string basepath = Application.StartupPath;
|
||
Cursor.Current = Cursors.Hand;
|
||
|
||
// Prepare Data
|
||
Array.Copy(savefile, offset, pkm_from, 0, 0xE8);
|
||
pkm_from_offset = offset;
|
||
|
||
// Make a new file name based off the PID
|
||
byte[] dragdata = PKX.decryptArray(pkm_from);
|
||
Array.Resize(ref dragdata, 0xE8);
|
||
PKX pkx = new PKX(dragdata, "Boxes");
|
||
string filename = pkx.Nickname;
|
||
if (filename != pkx.Species)
|
||
filename += " (" + pkx.Species + ")";
|
||
filename += " - " + pkx.PID + ".pk6";
|
||
|
||
// Make File
|
||
string newfile = Path.Combine(basepath, Util.CleanFileName(filename));
|
||
try
|
||
{
|
||
File.WriteAllBytes(newfile, dragdata);
|
||
|
||
string[] filesToDrag = { newfile };
|
||
DoDragDrop(new DataObject(DataFormats.FileDrop, filesToDrag), DragDropEffects.Move);
|
||
File.Delete(newfile); // after drop, delete the temporary file
|
||
}
|
||
catch (ArgumentException x)
|
||
{ Util.Error("Drag & Drop Error:", x.ToString()); }
|
||
File.Delete(newfile);
|
||
pkm_from_offset = 0;
|
||
}
|
||
private void pbBoxSlot_DragDrop(object sender, DragEventArgs e)
|
||
{
|
||
int slot = getSlot(sender);
|
||
int offset = getPKXOffset(slot);
|
||
|
||
// Check for In-Dropped files (PKX,SAV,ETC)
|
||
string[] files = (string[])e.Data.GetData(DataFormats.FileDrop);
|
||
if (Directory.Exists(files[0])) { loadBoxesFromDB(files[0]); return; }
|
||
if (pkm_from_offset == 0)
|
||
{
|
||
if (files.Length > 0)
|
||
{
|
||
FileInfo fi = new FileInfo(files[0]);
|
||
|
||
// Detect if PKM/PKX
|
||
if ((fi.Length == 136) || (fi.Length == 220) || (fi.Length == 236) || (fi.Length == 100) || (fi.Length == 80))
|
||
{
|
||
byte[] input = File.ReadAllBytes(files[0]);
|
||
var Converter = new pk2pk();
|
||
if (!PKX.verifychk(input)) Util.Alert("Invalid File Loaded.", "Checksum is not valid.");
|
||
try // to convert past gen pkm
|
||
{
|
||
byte[] data = Converter.ConvertPKM(input, savefile, savindex);
|
||
Array.Copy(PKX.encryptArray(data), 0, savefile, offset, 0xE8);
|
||
}
|
||
catch
|
||
{ Util.Error("Attempted to load previous generation PKM.", "Conversion failed."); }
|
||
}
|
||
else if (fi.Length == 232 || fi.Length == 260)
|
||
{
|
||
byte[] data = File.ReadAllBytes(files[0]);
|
||
if (fi.Extension == ".pkx" || fi.Extension == ".pk6")
|
||
data = PKX.encryptArray(data);
|
||
else if (fi.Extension != ".ekx" && fi.Extension != ".ek6")
|
||
{ openQuick(files[0]); return; } // lazy way of aborting
|
||
|
||
byte[] decdata = PKX.decryptArray(data);
|
||
if (!PKX.verifychk(decdata))
|
||
Util.Alert("Attempted to load Invalid File.", "Checksum is not valid.");
|
||
else
|
||
{
|
||
Array.Copy(data, 0, savefile, offset, 0xE8);
|
||
setPokedex(decdata);
|
||
getQuickFiller(getPictureBox(slot), decdata);
|
||
getSlotColor(slot, Properties.Resources.slotSet);
|
||
}
|
||
}
|
||
else // not PKX/EKX, so load with the general function
|
||
{ openQuick(files[0]); }
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if (ModifierKeys == Keys.Alt && slot > -1) // overwrite delete old slot
|
||
{
|
||
byte[] cleardata = new Byte[0xE8];
|
||
|
||
// Clear from slot picture
|
||
getQuickFiller(getPictureBox(pkm_from_slot), cleardata);
|
||
|
||
// Clear from slot data
|
||
Array.Copy(PKX.encryptArray(cleardata), 0, savefile, pkm_from_offset, 0xE8);
|
||
}
|
||
else if (ModifierKeys != Keys.Control && slot > -1)
|
||
{
|
||
// Load data from destination
|
||
byte[] swapdata = new Byte[0xE8];
|
||
Array.Copy(savefile, offset, swapdata, 0, 0xE8);
|
||
|
||
// Swap slot picture
|
||
getQuickFiller(getPictureBox(pkm_from_slot), PKX.decryptArray(swapdata));
|
||
|
||
// Swap slot data to source
|
||
Array.Copy(swapdata, 0, savefile, pkm_from_offset, 0xE8);
|
||
}
|
||
// Copy from temp slot to new.
|
||
Array.Copy(pkm_from, 0, savefile, offset, 0xE8);
|
||
getQuickFiller(getPictureBox(slot), PKX.decryptArray(pkm_from));
|
||
|
||
pkm_from_offset = 0; // Clear offset value
|
||
}
|
||
|
||
savedited = true;
|
||
}
|
||
private void pbBoxSlot_DragEnter(object sender, DragEventArgs e)
|
||
{
|
||
if (e.Data != null)
|
||
e.Effect = DragDropEffects.Move;
|
||
}
|
||
private byte[] pkm_from = PKX.encryptArray(new byte[0xE8]);
|
||
private int pkm_from_offset;
|
||
private int pkm_from_slot = -1;
|
||
#endregion
|
||
}
|
||
}
|