fceux/src/drivers/win/header_editor.cpp

1698 lines
50 KiB
C++

#include <windows.h>
#include "common.h"
#include "fceu.h"
#include "types.h"
#include "x6502.h"
#include "cart.h"
#include "ines.h"
#include "resource.h"
#include "header_editor.h"
// VS System type
char* vsSysList[] = {
"Normal",
"RBI Baseball",
"TKO Boxing",
"Super Xevious",
"Ice Climber",
"Dual Normal",
"Dual Raid on Bungeling Bay",
0
};
// VS PPU type
char* vsPPUList[] = {
"RP2C03B",
"RP2C03G",
"RP2C04-0001",
"RP2C04-0002",
"RP2C04-0003",
"RP2C04-0004",
"RC2C03B",
"RC2C03C",
"RC2C05-01 ($2002 AND $??=$1B)",
"RC2C05-02 ($2002 AND $3F=$3D)",
"RC2C05-03 ($2002 AND $1F=$1C)",
"RC2C05-04 ($2002 AND $1F=$1B)",
"RC2C05-05 ($2002 AND $1F=$??)",
"Reserved",
"Reserved",
"Reserved",
0
};
// Extend console type
char* extConsoleList[] = {
"Normal",
"VS. System",
"Playchoice 10",
"Bit Corp. Creator",
"V.R. Tech. VT01 monochrome",
"V.R. Tech. VT01 red / cyan",
"V.R. Tech. VT02",
"V.R. Tech. VT03",
"V.R. Tech. VT09",
"V.R. Tech. VT32",
"V.R. Tech. VT369",
"Reserved",
"Reserved",
"Reserved",
"Reserved",
"Reserved",
0
};
// Default input device list
char* inputDevList[] = {
"Unspecified",
"Standard NES / Famicom Controllers",
"Four-score (NES)",
"Four-score (Famicom)",
"VS. System",
"VS. System (controllers swapped)",
"VS. Pinball (J)",
"VS. Zapper",
"Zapper",
"Double Zappers",
"Bandai Hyper Shot",
"Power Pad Side A",
"Power Pad Side B",
"Family Trainer Side A",
"Family Trainer Side B",
"Arkanoid Paddle (NES)",
"Arkanoid Paddle (Famicom)",
"Double Arkanoid Paddle",
"Konami Hyper Shot",
"Pachinko",
"Exciting Boxing Punching Bag",
"Jissen Mahjong",
"Party Tap",
"Oeka Kids Tablet",
"Barcode Reader",
"Miracle Piano",
"Pokkun Moguraa",
"Top Rider",
"Double-Fisted",
"Famicom 3D",
"Doremikko Keyboard",
"R.O.B. Gyro Set",
"Famicom Data Recorder",
"ASCII Turbo File",
"IGS Storage Battle Box",
"Family BASIC Keyboard",
"PEC-586 Keyboard",
"Bit Corp. Keyboard",
"Subor Keyboard",
"Subor Keyboard + Mouse A",
"Subor Keyboard + Mouse B",
"SNES Mouse",
"Multicart",
"Double SNES controllers",
"RacerMate Bicycle",
"U-Force",
"R.O.B. Stack-Up",
0
};
char** dropDownList[] = {
vsSysList, vsPPUList, extConsoleList, inputDevList, 0
};
int dropDownIdList[] = {
IDC_VS_SYSTEM_COMBO,
IDC_VS_PPU_COMBO,
IDC_SYSTEM_EXTEND_COMBO,
IDC_INPUT_DEVICE_COMBO,
0
};
HWND hHeadEditor = NULL;
bool LoadHeader(HWND parent, iNES_HEADER* header)
{
int error = 0;
enum errors {
OK,
OPEN_FAILED,
INVALID_HEADER,
FDS_HEADER,
UNIF_HEADER,
NSF_HEADER//,
// NSFE_HEADER,
// NSF2_HEADER
};
FCEUFILE* fp = FCEU_fopen(LoadedRomFName, NULL, "rb", NULL);
if (!GameInfo)
strcpy(LoadedRomFName, fp->fullFilename.c_str());
if (fp)
{
if (FCEU_fread(header, 1, sizeof(iNES_HEADER), fp) == sizeof(iNES_HEADER) && !memcmp(header, "NES\x1A", 4))
header->cleanup();
else if (!memcmp(header, "FDS\x1A", 4))
error = errors::FDS_HEADER;
else if (!memcmp(header, "UNIF", 4))
error = errors::UNIF_HEADER;
else if (!memcmp(header, "NESM", 4))
error = errors::NSF_HEADER;
/* else if (!memcmp(header, "NSFE", 4))
error = errors::NSFE_HEADER;
else if (!memcmp(header, "NESM\2", 4))
error = errors::NSF2_HEADER;
*/ else
error = errors::INVALID_HEADER;
FCEU_fclose(fp);
}
else
error = errors::OPEN_FAILED;
if (error)
{
switch (error)
{
case errors::OPEN_FAILED:
{
char buf[1024];
sprintf(buf, "Error opening %s!", LoadedRomFName);
MessageBox(parent, buf, "iNES Header Editor", MB_OK | MB_ICONERROR);
break;
}
case errors::INVALID_HEADER:
MessageBox(parent, "Invalid iNES header.", "iNES Header Editor", MB_OK | MB_ICONERROR);
break;
case errors::FDS_HEADER:
MessageBox(parent, "Editing header of an FDS file is not supported.", "iNES Header Editor", MB_OK | MB_ICONERROR);
break;
case errors::UNIF_HEADER:
MessageBox(parent, "Editing header of a UNIF file is not supported.", "iNES Header Editor", MB_OK | MB_ICONERROR);
break;
case errors::NSF_HEADER:
// case errors::NSF2_HEADER:
// case errors::NSFE_HEADER:
MessageBox(parent, "Editing header of an NSF file is not supported.", "iNES Header Editor", MB_OK | MB_ICONERROR);
break;
}
return false;
}
return true;
}
HWND InitHeaderEditDialog(HWND hwnd, iNES_HEADER* header)
{
hHeadEditor = hwnd;
// these contols don't bother the standard
EnableWindow(GetDlgItem(hwnd, IDC_INESHEADER_GROUP), TRUE);
// Resotore button
EnableWindow(GetDlgItem(hwnd, IDC_RESTORE_BUTTON), TRUE);
// Save as... button
EnableWindow(GetDlgItem(hwnd, IDSAVE), TRUE);
// Version groupbox
EnableWindow(GetDlgItem(hwnd, IDC_VERSION_GROUP), TRUE);
// Standard
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_VERSION_STANDARD), TRUE);
// iNES 2.0
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_VERSION_INES20), TRUE);
// Mapper groupbox
EnableWindow(GetDlgItem(hwnd, IDC_MAPPER_GROUP), TRUE);
// Mapper#
EnableWindow(GetDlgItem(hwnd, IDC_MAPPER_TEXT), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_MAPPER_COMBO), TRUE);
// PRG groupbox
EnableWindow(GetDlgItem(hwnd, IDC_PRG_GROUP), TRUE);
// PRG ROM
EnableWindow(GetDlgItem(hwnd, IDC_PRGROM_TEXT), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_PRGROM_COMBO), TRUE);
// PRG RAM will be finally determined in ToggleUnofficialPrgRamPresent()
// EnableWindow(GetDlgItem(hwnd, IDC_PRGRAM_TEXT), TRUE);
// EnableWindow(GetDlgItem(hwnd, IDC_PRGRAM_COMBO), TRUE);
// CHR ROM
EnableWindow(GetDlgItem(hwnd, IDC_CHR_GROUP), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_CHRROM_TEXT), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_CHRROM_COMBO), TRUE);
// Mirroring groupbox
EnableWindow(GetDlgItem(hwnd, IDC_MIRRORING_GROUP), TRUE);
// Horizontal
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_MIRR_HORIZONTAL), TRUE);
// Vertical
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_MIRR_VERTICAL), TRUE);
// Four-screen
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_MIRR_4SCREEN), TRUE);
// Region Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_REGION_GROUP), TRUE);
// NTSC
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_NTSC), TRUE);
// PAL
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_PAL), TRUE);
// Multiple will be finally determined in ToggleUnofficialPrgRamPresent()
// EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_DUAL), TRUE);
// Trainer
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_TRAINER), TRUE);
// System Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_SYSTEM_GROUP), TRUE);
// Normal
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_SYSTEM_NORMAL), TRUE);
// VS
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_SYSTEM_VS), TRUE);
// Playchoice-10 will be finally determined in ToggleUnofficialPropertiesEnabled()
// EnableWindow(GetDlgItem(hwnd, IDC_RADIO_SYSTEM_PLAYCHOICE10), TRUE);
// Limit text
// Sub Mapper#
SendDlgItemMessage(hwnd, IDC_SUBMAPPER_EDIT, EM_SETLIMITTEXT, 2, 0);
// Misc. ROM(s)
SendDlgItemMessage(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT, EM_SETLIMITTEXT, 1, 0);
// Assign ID to the sub edit control in these comboboxes
// PRG ROM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_PRGROM_COMBO), GW_CHILD), GWL_ID, IDC_PRGROM_EDIT);
// PRG RAM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_PRGRAM_COMBO), GW_CHILD), GWL_ID, IDC_PRGRAM_EDIT);
// PRG NVRAM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO), GW_CHILD), GWL_ID, IDC_PRGNVRAM_EDIT);
// CHR ROM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_CHRROM_COMBO), GW_CHILD), GWL_ID, IDC_CHRROM_EDIT);
// CHR RAM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_CHRRAM_COMBO), GW_CHILD), GWL_ID, IDC_CHRRAM_EDIT);
// CHR NVRAM
SetWindowLongPtr(GetWindow(GetDlgItem(hwnd, IDC_CHRNVRAM_COMBO), GW_CHILD), GWL_ID, IDC_CHRNVRAM_EDIT);
// Change the default wndproc of these control to limit their text
// PRG ROM
DefaultEditCtrlProc = (WNDPROC)SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_PRGROM_COMBO), IDC_PRGROM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
// PRG RAM
SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_PRGRAM_COMBO), IDC_PRGRAM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
// PRG NVRAM
SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO), IDC_PRGNVRAM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
// CHR ROM
SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_CHRROM_COMBO), IDC_CHRROM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
// CHR RAM
SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_CHRRAM_COMBO), IDC_CHRRAM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
// CHR NVRAM
SetWindowLongPtr(GetDlgItem(GetDlgItem(hwnd, IDC_CHRNVRAM_COMBO), IDC_CHRNVRAM_EDIT), GWLP_WNDPROC, (LONG_PTR)FilterEditCtrlProc);
ToggleINES20(hwnd, IsDlgButtonChecked(hwnd, IDC_RADIO_VERSION_INES20) == BST_CHECKED);
char buf[256];
for (int i = 0; dropDownIdList[i]; ++i)
for (int j = 0; dropDownList[i][j]; ++j)
{
sprintf(buf, dropDownList[i] == inputDevList ? "$%02X %s" : "$%X %s", j, dropDownList[i][j]);
SendDlgItemMessage(hwnd, IDC_MAPPER_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, dropDownIdList[i], CB_ADDSTRING, 0, (LPARAM)buf), j);
}
// Mapper#
extern BMAPPINGLocal bmap[];
for (int i = 0; bmap[i].init; ++i)
{
sprintf(buf, "%d %s", bmap[i].number, bmap[i].name);
SendDlgItemMessage(hwnd, IDC_MAPPER_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_MAPPER_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), bmap[i].number);
}
// add usually used size strings
strcpy(buf, "0B");
SendDlgItemMessage(hwnd, IDC_PRGROM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGROM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
SendDlgItemMessage(hwnd, IDC_CHRROM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRROM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
SendDlgItemMessage(hwnd, IDC_PRGRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
SendDlgItemMessage(hwnd, IDC_CHRRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
SendDlgItemMessage(hwnd, IDC_PRGNVRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGNVRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
SendDlgItemMessage(hwnd, IDC_CHRNVRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRNVRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), 0);
int size = 128;
while (size <= 2048 * 1024)
{
if (size >= 1024 << 3)
{
// The size of CHR ROM must be multiple of 8KB
sprintf(buf, "%dKB", size / 1024);
SendDlgItemMessage(hwnd, IDC_CHRROM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRROM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
// The size of PRG ROM must be multiple of 16KB
if (size >= 1024 * 16)
{
// PRG ROM
sprintf(buf, "%dKB", size / 1024);
SendDlgItemMessage(hwnd, IDC_PRGROM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGROM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
}
}
if (size >= 1024)
sprintf(buf, "%dKB", size / 1024);
else
sprintf(buf, "%dB", size);
SendDlgItemMessage(hwnd, IDC_PRGRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
SendDlgItemMessage(hwnd, IDC_CHRRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
SendDlgItemMessage(hwnd, IDC_PRGNVRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_PRGNVRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
SendDlgItemMessage(hwnd, IDC_CHRNVRAM_COMBO, CB_SETITEMDATA, SendDlgItemMessage(hwnd, IDC_CHRNVRAM_COMBO, CB_ADDSTRING, 0, (LPARAM)buf), size);
size <<= 1;
}
SetHeaderData(hwnd, header);
return hwnd;
}
void ToggleINES20(HWND hwnd, bool ines20)
{
// only ines 2.0 has these values
// these are always have when in 2.0
// Submapper#
EnableWindow(GetDlgItem(hwnd, IDC_SUBMAPPER_TEXT), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_SUBMAPPER_EDIT), ines20);
// PRG NVRAM
EnableWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_TEXT), ines20);
// CHR RAM
EnableWindow(GetDlgItem(hwnd, IDC_CHRRAM_COMBO), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_CHRRAM_TEXT), ines20);
// CHR NVRAM
EnableWindow(GetDlgItem(hwnd, IDC_CHRNVRAM_COMBO), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_CHRNVRAM_TEXT), ines20);
// Dendy in Region Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_DENDY), ines20);
// Multiple in Regtion Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_DUAL), ines20);
// Extend in System Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_SYSTEM_EXTEND), ines20);
// Input Device
EnableWindow(GetDlgItem(hwnd, IDC_INPUT_DEVICE_COMBO), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_INPUT_DEVICE_TEXT), ines20);
// Miscellaneous ROMs
EnableWindow(GetDlgItem(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT), ines20);
EnableWindow(GetDlgItem(hwnd, IDC_MISCELLANEOUS_ROMS_TEXT), ines20);
// enable extend dialog only when ines 2.0 and extend button is checked
ToggleExtendSystemList(hwnd, ines20 && IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_EXTEND) == BST_CHECKED);
// enable vs dialog only when ines 2.0 and vs button is checked
ToggleVSSystemGroup(hwnd, ines20 && IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_VS) == BST_CHECKED);
// hide "Battery / PRG-NVRAM" checkbox in ines 2.0 and show PRG NVRAM combo box, negative on the contary, because ines 1.0 has nowhere to set actural size, it can only determin it exists or not.
ShowWindow(GetDlgItem(hwnd, IDC_CHECK_BATTERYNVRAM), ines20 ? SW_HIDE : SW_SHOW);
ShowWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO), ines20 ? SW_SHOW : SW_HIDE);
ShowWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_TEXT), ines20 ? SW_SHOW : SW_HIDE);
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_BATTERYNVRAM), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_PRGNVRAM_TEXT), TRUE);
// 10th byte for the unofficial ines properties
// Unofficial Properties
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_UNOFFICIAL), !ines20);
ToggleUnofficialPropertiesEnabled(hwnd, ines20, IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL) == BST_CHECKED);
// ines 1.0 doesn't support Dendy region, switch it to PAL
if (!ines20 && IsDlgButtonChecked(hwnd, IDC_RADIO_REGION_DENDY) == BST_CHECKED)
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_PAL);
// ines 1.0 doesn't support extend system, switch it to normal
if (!ines20 && IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_EXTEND) == BST_CHECKED)
CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_NORMAL);
}
void ToggleExtendSystemList(HWND hwnd, bool enable)
{
// Extend combo box only enabled when in iNES 2.0 and Extend System was chosen
// Extend combo box
EnableWindow(GetDlgItem(hwnd, IDC_EXTEND_SYSTEM_GROUP), enable);
EnableWindow(GetDlgItem(hwnd, IDC_SYSTEM_EXTEND_COMBO), enable);
EnableWindow(GetDlgItem(hwnd, IDC_EXTEND_SYSTEM_TEXT), enable);
}
void ToggleVSSystemGroup(HWND hwnd, bool enable)
{
// VS System Groupbox only enabled when in iNES 2.0 and VS System in System groupbox is chosen
// VS System Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_VS_SYSTEM_GROUP), enable);
// System
EnableWindow(GetDlgItem(hwnd, IDC_VS_SYSTEM_COMBO), enable);
EnableWindow(GetDlgItem(hwnd, IDC_VS_SYSTEM_TEXT), enable);
// PPU
EnableWindow(GetDlgItem(hwnd, IDC_VS_PPU_COMBO), enable);
EnableWindow(GetDlgItem(hwnd, IDC_VS_PPU_TEXT), enable);
}
void ToggleUnofficialPropertiesEnabled(HWND hwnd, bool ines20, bool check)
{
bool sub_enable = !ines20 && check;
// Unofficial Properties only available in ines 1.0
EnableWindow(GetDlgItem(hwnd, IDC_UNOFFICIAL_GROUP), sub_enable);
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_UNOFFICIAL_PRGRAM), sub_enable);
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_UNOFFICIAL_EXTRA_REGION), sub_enable);
EnableWindow(GetDlgItem(hwnd, IDC_CHECK_UNOFFICIAL_BUS_CONFLICT), sub_enable);
// when unofficial properties is enabled or in ines 2.0 standard, Playchoice-10 in System groupbox is available
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_SYSTEM_PLAYCHOICE10), ines20 || sub_enable);
ToggleUnofficialPrgRamPresent(hwnd, ines20, check, IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_PRGRAM) == BST_CHECKED);
ToggleUnofficialExtraRegionCode(hwnd, ines20, check, IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_EXTRA_REGION) == BST_CHECKED);
// Playchoice-10 is not available in ines 1.0 and unofficial is not checked, switch back to normal
if (!ines20 && !check && IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_PLAYCHOICE10) == BST_CHECKED)
CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_NORMAL);
}
void ToggleUnofficialExtraRegionCode(HWND hwnd, bool ines20, bool unofficial_check, bool check)
{
// The unofficial byte to determine whether multiple region is valid
// Multiple in Region Groupbox
EnableWindow(GetDlgItem(hwnd, IDC_RADIO_REGION_DUAL), ines20 || unofficial_check && check);
// Dual region is not avalable when in ines 1.0 and extra region in unofficial is not checked, switch it back to NTSC
if (!ines20 && (!unofficial_check || !check) && IsDlgButtonChecked(hwnd, IDC_RADIO_REGION_DUAL) == BST_CHECKED)
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_NTSC);
}
void ToggleUnofficialPrgRamPresent(HWND hwnd, bool ines20, bool unofficial_check, bool check) {
// The unofficial byte to determine wheter PRGRAM exists
// PRG RAM
bool enable = ines20 || !unofficial_check || check;
// When unofficial 10th byte was not set, the PRG RAM is defaultly enabled
EnableWindow(GetDlgItem(hwnd, IDC_PRGRAM_TEXT), enable);
EnableWindow(GetDlgItem(hwnd, IDC_PRGRAM_COMBO), enable);
}
INT_PTR CALLBACK HeaderEditorProc(HWND hDlg, UINT uMsg, WPARAM wP, LPARAM lP)
{
static iNES_HEADER* header;
switch (uMsg) {
case WM_INITDIALOG:
{
header = (iNES_HEADER*)lP;
if (!header)
goto wm_close;
else
InitHeaderEditDialog(hDlg, header);
// Put the window aside the main window when in game.
if (GameInfo)
CalcSubWindowPos(hDlg, NULL);
}
break;
case WM_COMMAND:
switch (HIWORD(wP))
{
case BN_CLICKED:
{
int id = LOWORD(wP);
switch (id)
{
case IDC_RADIO_VERSION_STANDARD:
ToggleINES20(hDlg, false);
break;
case IDC_RADIO_VERSION_INES20:
ToggleINES20(hDlg, true);
break;
case IDC_RADIO_SYSTEM_NORMAL:
case IDC_RADIO_SYSTEM_PLAYCHOICE10:
case IDC_RADIO_SYSTEM_EXTEND:
ToggleExtendSystemList(hDlg, IsDlgButtonChecked(hDlg, IDC_RADIO_SYSTEM_EXTEND) == BST_CHECKED);
case IDC_RADIO_SYSTEM_VS:
// Both ines 1.0 and 2.0 can trigger VS System, but only 2.0 enables the extra detailed properties
ToggleVSSystemGroup(hDlg, IsDlgButtonChecked(hDlg, IDC_RADIO_VERSION_INES20) == BST_CHECKED && IsDlgButtonChecked(hDlg, IDC_RADIO_SYSTEM_VS) == BST_CHECKED);
break;
case IDC_CHECK_UNOFFICIAL:
ToggleUnofficialPropertiesEnabled(hDlg, false, IsDlgButtonChecked(hDlg, IDC_CHECK_UNOFFICIAL) == BST_CHECKED);
break;
case IDC_CHECK_UNOFFICIAL_PRGRAM:
ToggleUnofficialPrgRamPresent(hDlg, false, true, IsDlgButtonChecked(hDlg, IDC_CHECK_UNOFFICIAL_PRGRAM) == BST_CHECKED);
break;
case IDC_CHECK_UNOFFICIAL_EXTRA_REGION:
ToggleUnofficialExtraRegionCode(hDlg, false, true, IsDlgButtonChecked(hDlg, IDC_CHECK_UNOFFICIAL_EXTRA_REGION) == BST_CHECKED);
break;
case IDC_RESTORE_BUTTON:
SetHeaderData(hDlg, header);
break;
case IDSAVE:
{
iNES_HEADER newHeader;
if (WriteHeaderData(hDlg, &newHeader))
{
char path[4096] = { 0 };
if (ShowINESFileBox(hDlg, path, true))
SaveINESFile(hDlg, path, &newHeader);
}
}
break;
case IDCLOSE:
goto wm_close;
}
}
}
break;
case WM_CLOSE:
case WM_QUIT:
wm_close:
if (GameInfo)
DestroyWindow(hDlg);
else
{
EndDialog(hDlg, 0);
LoadedRomFName[0] = 0;
}
break;
case WM_DESTROY:
hHeadEditor = NULL;
free(header);
break;
}
return false;
}
void DoHeadEdit()
{
if (hHeadEditor)
{
ShowWindow(hHeadEditor, SW_SHOWNORMAL);
SetForegroundWindow(hHeadEditor);
}
else
{
iNES_HEADER* header = (iNES_HEADER*)calloc(1, sizeof(iNES_HEADER));
if (GameInfo)
{
if (LoadHeader(hAppWnd, header))
CreateDialogParam(fceu_hInstance, MAKEINTRESOURCE(IDD_EDIT_HEADER), hAppWnd, HeaderEditorProc, (LPARAM)header);
else
free(header);
}
else {
// temporarily borrow LoadedRomFName, when no game is loaded, it is unused.
LoadedRomFName[0] = 0;
if (ShowINESFileBox(hAppWnd) && LoadHeader(hAppWnd, header))
DialogBoxParam(fceu_hInstance, MAKEINTRESOURCE(IDD_EDIT_HEADER), hAppWnd, HeaderEditorProc, (LPARAM)header);
else
free(header);
}
}
}
void SetHeaderData(HWND hwnd, iNES_HEADER* header) {
// Temporary buffers
char buf[256];
bool ines20 = (header->ROM_type2 & 0xC) == 8;
bool unofficial = false;
// Check iNES 2.0
CheckRadioButton(hwnd, IDC_RADIO_VERSION_STANDARD, IDC_RADIO_VERSION_INES20, ines20 ? IDC_RADIO_VERSION_INES20 : IDC_RADIO_VERSION_STANDARD);
// Mapper#
int mapper = header->ROM_type >> 4 | header->ROM_type2 & 0xF0;
if (ines20)
mapper |= (header->ROM_type3 & 0xF0) << 4;
sprintf(buf, "%d ", mapper);
if (SendDlgItemMessage(hwnd, IDC_MAPPER_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_MAPPER_COMBO, buf);
// Sub Mapper
sprintf(buf, "%d", ines20 ? header->ROM_type3 >> 4 : 0);
SetDlgItemText(hwnd, IDC_SUBMAPPER_EDIT, buf);
// PRG ROM
strcpy(buf, "0B");
int prg_rom = header->ROM_size;
if (ines20) {
if ((header->Upper_ROM_VROM_size & 0xF) == 0xF)
prg_rom = pow(2, header->ROM_size >> 2) * ((header->ROM_size & 3) * 2 + 1);
else {
prg_rom |= (header->Upper_ROM_VROM_size & 0xF) << 8;
prg_rom *= 1024 * 16;
}
}
else
prg_rom *= 1024 * 16;
if (prg_rom < 1024 || prg_rom % 1024 != 0)
sprintf(buf, "%dB", prg_rom);
else
sprintf(buf, "%dKB", prg_rom / 1024);
if (SendDlgItemMessage(hwnd, IDC_PRGROM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_PRGROM_COMBO, buf);
// PRG RAM
strcpy(buf, "0B");
if (ines20)
{
int shift = header->RAM_size & 0xF;
if (shift)
{
int prg_ram = 64 << shift;
if (prg_ram >= 1024)
sprintf(buf, "%dKB", prg_ram / 1024);
else sprintf(buf, "%dB", prg_ram);
}
} else
{
if (!(header->RAM_size & 0x10) && header->ROM_type3)
sprintf(buf, "%dKB", header->ROM_type3 ? 1 : header->ROM_type3 * 8);
}
if (SendDlgItemMessage(hwnd, IDC_PRGRAM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_PRGRAM_COMBO, buf);
// PRG NVRAM
strcpy(buf, "0B");
if (ines20)
{
int shift = header->RAM_size >> 4;
if (shift)
{
int prg_nvram = 64 << shift;
if (prg_nvram >= 1024)
sprintf(buf, "%dKB", prg_nvram / 1024);
else sprintf(buf, "%dB", prg_nvram);
}
} else
CheckDlgButton(hwnd, IDC_CHECK_BATTERYNVRAM, header->ROM_type & 0x2 ? BST_CHECKED : BST_UNCHECKED);
if (SendDlgItemMessage(hwnd, IDC_PRGNVRAM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_PRGNVRAM_COMBO, buf);
// CHR ROM
strcpy(buf, "0B");
int chr_rom = header->VROM_size;
if (ines20)
{
if ((header->Upper_ROM_VROM_size & 0xF0) == 0xF0)
chr_rom = pow(2, header->VROM_size >> 2) * (((header->VROM_size & 3) * 2) + 1);
else
{
chr_rom |= (header->Upper_ROM_VROM_size & 0xF0) << 4;
chr_rom *= 8 * 1024;
}
}
else
chr_rom *= 8 * 1024;
if (chr_rom < 1024 || chr_rom % 1024 != 0)
sprintf(buf, "%dB", chr_rom);
else
sprintf(buf, "%dKB", chr_rom / 1024);
if (SendDlgItemMessage(hwnd, IDC_CHRROM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_CHRROM_COMBO, buf);
// CHR RAM
sprintf(buf, "0B");
if (ines20)
{
int shift = header->VRAM_size & 0xF;
if (shift)
{
int chr_ram = 64 << shift;
if (chr_ram >= 1024)
sprintf(buf, "%dKB", chr_ram / 1024);
else sprintf(buf, "%dB", chr_ram);
}
}
if (SendDlgItemMessage(hwnd, IDC_CHRRAM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_CHRRAM_COMBO, buf);
// CHR NVRAM
sprintf(buf, "0B");
if (ines20)
{
int shift = header->VRAM_size >> 4;
if (shift)
{
int chr_nvram = 64 << shift;
if (chr_nvram >= 1024)
sprintf(buf, "%dKB", chr_nvram / 1024);
else sprintf(buf, "%dB", chr_nvram);
}
}
if (SendDlgItemMessage(hwnd, IDC_CHRNVRAM_COMBO, CB_SELECTSTRING, 0, (LPARAM)buf) == CB_ERR)
SetDlgItemText(hwnd, IDC_CHRNVRAM_COMBO, buf);
// Mirroring
CheckRadioButton(hwnd, IDC_RADIO_MIRR_HORIZONTAL, IDC_RADIO_MIRR_4SCREEN, header->ROM_type & 8 ? IDC_RADIO_MIRR_4SCREEN : header->ROM_type & 1 ? IDC_RADIO_MIRR_VERTICAL : IDC_RADIO_MIRR_HORIZONTAL);
// Region
if (ines20)
{
int region = header->TV_system & 3;
switch (region) {
case 0:
// NTSC
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_NTSC);
break;
case 1:
// PAL
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_PAL);
break;
case 2:
// Dual
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_DUAL);
break;
case 3:
// Dendy
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_DENDY);
}
}
else {
// Only the unofficial 10th byte has a dual region, we must check it first.
int region = header->RAM_size & 3;
if (region == 3 || region == 1)
{
// Check the unofficial checkmark and the region code
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, IDC_RADIO_REGION_DUAL);
unofficial = true;
}
else
// When the region in unofficial byte is inconsistent with the official byte, based on the official byte
CheckRadioButton(hwnd, IDC_RADIO_REGION_NTSC, IDC_RADIO_REGION_DENDY, header->Upper_ROM_VROM_size & 1 ? IDC_RADIO_REGION_PAL : IDC_RADIO_REGION_NTSC);
}
// System
int system = header->ROM_type2 & 3;
switch (system)
{
default:
// Normal
case 0:
CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_NORMAL);
break;
// VS. System
case 1:
CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_VS);
break;
// PlayChoice-10
case 2:
CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_PLAYCHOICE10);
// PlayChoice-10 is an unofficial setting for ines 1.0
unofficial = !ines20;
break;
// Extend
case 3:
// The 7th byte is different between ines 1.0 and 2.0, so we need to check it
if (ines20) CheckRadioButton(hwnd, IDC_RADIO_SYSTEM_NORMAL, IDC_RADIO_SYSTEM_EXTEND, IDC_RADIO_SYSTEM_EXTEND);
}
// it's quite ambiguous to put them here, but it's better to have a default selection than leave the dropdown blank, because user might switch to ines 2.0 anytime
// Hardware type
int hardware = header->VS_hardware >> 4;
if (SendDlgItemMessage(hwnd, IDC_VS_SYSTEM_COMBO, CB_SETCURSEL, hardware, 0) == CB_ERR)
{
sprintf(buf, "$%X", hardware);
SetDlgItemText(hwnd, IDC_VS_SYSTEM_COMBO, buf);
}
// PPU type
int ppu = header->VS_hardware & 0xF;
if (SendDlgItemMessage(hwnd, IDC_VS_PPU_COMBO, CB_SETCURSEL, ppu, 0) == CB_ERR)
{
sprintf(buf, "$%X", ppu);
SetDlgItemText(hwnd, IDC_VS_SYSTEM_COMBO, buf);
}
// Extend Console
if (SendDlgItemMessage(hwnd, IDC_SYSTEM_EXTEND_COMBO, CB_SETCURSEL, ppu, 0) == CB_ERR)
{
sprintf(buf, "$%X", ppu);
SetDlgItemText(hwnd, IDC_VS_SYSTEM_COMBO, buf);
}
// Input Device:
int input = header->reserved[1] & 0x1F;
if (SendDlgItemMessage(hwnd, IDC_INPUT_DEVICE_COMBO, CB_SETCURSEL, input, 0) == CB_ERR)
{
sprintf(buf, "$%02X", input);
SetDlgItemText(hwnd, IDC_INPUT_DEVICE_COMBO, buf);
}
// Miscellaneous ROM Area(s)
sprintf(buf, "%d", header->reserved[0] & 3);
SetDlgItemText(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT, buf);
// Trainer
CheckDlgButton(hwnd, IDC_CHECK_TRAINER, header->ROM_type & 4 ? BST_CHECKED : BST_UNCHECKED);
// Unofficial Properties Checkmark
CheckDlgButton(hwnd, IDC_CHECK_UNOFFICIAL, unofficial ? BST_CHECKED : BST_UNCHECKED);
// Switch the UI to the proper version
ToggleINES20(hwnd, ines20);
}
bool WriteHeaderData(HWND hwnd, iNES_HEADER* header)
{
// Temporary buffers
char buf[256];
iNES_HEADER _header;
memset(&_header, 0, sizeof(iNES_HEADER));
// Check iNES 2.0
bool ines20 = IsDlgButtonChecked(hwnd, IDC_RADIO_VERSION_INES20) == BST_CHECKED;
// iNES 1.0 unofficial byte available
bool unofficial = !ines20 && IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL) == BST_CHECKED;
// iNES 2.0 signature
if (ines20)
_header.ROM_type2 |= 8;
// Mapper
int mapper;
if (!GetComboBoxListItemData(hwnd, IDC_MAPPER_COMBO, &mapper, buf))
{
MessageBox(hwnd, "The mapper# you have entered is invalid. Please enter a decimal number or select an item from the dropdown list.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_MAPPER_COMBO));
return false;
}
if (mapper < 4096)
{
if (mapper < 256)
{
_header.ROM_type |= (mapper & 0xF) << 4;
_header.ROM_type2 |= (mapper & 0xF0);
} else
{
if (ines20)
_header.ROM_type3 |= mapper >> 8;
else
{
sprintf(buf, "Mapper# should be less than %d in iNES %d.0 format.", 256, 1);
MessageBox(hwnd, buf, "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_MAPPER_COMBO));
return false;
}
}
}
else {
sprintf(buf, "Mapper# should be less than %d in iNES %d.0 format.", 4096, 2);
MessageBox(hwnd, buf, "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_MAPPER_COMBO));
return false;
}
// Sub mapper
if (ines20) {
GetDlgItemText(hwnd, IDC_SUBMAPPER_EDIT, buf, 256);
int submapper;
if (sscanf(buf, "%d", &submapper) > 0)
{
if (submapper < 16)
_header.ROM_type3 |= submapper << 4;
else
{
MessageBox(hwnd, "The sub mapper# should less than 16 in iNES 2.0 format.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_SUBMAPPER_EDIT));
return false;
}
} else
{
MessageBox(hwnd, "The sub mapper# you have entered is invalid. Please enter a decimal number.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_SUBMAPPER_EDIT));
return false;
}
}
// PRG ROM
int prg_rom;
if (GetComboBoxByteSize(hwnd, IDC_PRGROM_COMBO, &prg_rom) == 0)
{
// max value which a iNES 2.0 header can hold
if (prg_rom < 16 * 1024 * 0xEFF)
{
// try to fit the irregular value with the alternative way
if (prg_rom % (16 * 1024) != 0)
{
if (ines20)
{
// try to calculate the nearest value
bool fit = false;
int result = 0x7FFFFFFF;
for (int multiplier = 0; multiplier < 4; ++multiplier)
{
for (int exponent = 0; exponent < 64; ++exponent)
{
int new_result = pow(2, exponent) * (multiplier * 2 + 1);
if (new_result == prg_rom)
{
_header.Upper_ROM_VROM_size |= 0xF;
_header.ROM_size |= exponent << 2;
_header.ROM_size |= multiplier & 3;
fit = true;
break;
}
if (new_result > prg_rom && result > new_result)
result = new_result;
}
if (fit) break;
}
if (!fit)
{
int result10 = (prg_rom / 16 / 1024 + 1) * 16 * 1024;
if (result10 < result)
result = result10;
char buf2[64];
if (result % 1024 != 0)
sprintf(buf2, "%dB", result);
else
sprintf(buf2, "%dKB", result / 1024);
sprintf(buf, "PRG ROM size you entered is invalid in iNES 2.0, do you want to set to its nearest value %s?", buf2);
if (MessageBox(hwnd, buf, "Error", MB_YESNO | MB_ICONERROR) == IDYES)
SetDlgItemText(hwnd, IDC_PRGROM_COMBO, buf2);
else
{
SetFocus(GetDlgItem(hwnd, IDC_PRGROM_COMBO));
return false;
}
}
}
else {
// ines 1.0 can't handle this kind of value
MessageBox(hwnd, "PRG ROM size must be multiple of 16KB in iNES 1.0", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGROM_COMBO));
return false;
}
}
// it's multiple size of 16KB
else {
// it can be fitted in iNES 1.0
if (prg_rom < 16 * 1024 * 0xFF)
_header.ROM_size |= prg_rom / 16 / 1024 & 0xFF;
else
{
if (ines20)
_header.Upper_ROM_VROM_size |= prg_rom / 16 / 1024 >> 8 & 0xF;
else {
MessageBox(hwnd, "PRG ROM size exceeded the limit of iNES 1.0 (4080KB).", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGROM_COMBO));
return false;
}
}
}
}
// A too large size
else {
MessageBox(hwnd, "PRG ROM size you entered is too large to fit into a cartridge, by the way this is an NES emulator, not for XBOX360 or PlayStation2.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGROM_COMBO));
return false;
}
} else
return false;
// PRG RAM
if (ines20 || IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL) == BST_UNCHECKED || IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_PRGRAM) == BST_CHECKED)
{
int prg_ram;
if (GetComboBoxByteSize(hwnd, IDC_PRGRAM_COMBO, &prg_ram) == 0)
{
if (ines20)
{
if (prg_ram < 64 << 0xF)
{
if (prg_ram % 64 == 0)
_header.RAM_size |= (int)log2(prg_ram / 64);
else
{
MessageBox(hwnd, "Invalid PRG RAM size", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGRAM_COMBO));
return false;
}
}
else {
MessageBox(hwnd, "PRG RAM size exceeded the limit (4096KB)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGRAM_COMBO));
return false;
}
}
else {
if (prg_ram < 8 * 1024 * 255)
{
if (prg_ram % (8 * 1024) == 0)
_header.ROM_type3 |= prg_ram / 8 / 1024;
else {
MessageBox(hwnd, "PRG RAM size must be multiple of 8KB in iNES 1.0", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGRAM_COMBO));
return false;
}
}
else {
MessageBox(hwnd, "PRG RAM size exceeded the limit (2040KB)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGRAM_COMBO));
return false;
}
}
}
else
return false;
}
// PRG NVRAM
if (ines20)
{
// only iNES 2.0 has value for PRG VMRAM
int prg_nvram;
if (GetComboBoxByteSize(hwnd, IDC_PRGNVRAM_COMBO, &prg_nvram) == 0)
{
if (prg_nvram < 64 << 0xF)
{
if (prg_nvram % 64 == 0)
_header.RAM_size |= (int)log2(prg_nvram / 64) << 4;
else
{
MessageBox(hwnd, "Invalid PRG NVRAM size", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO));
return false;
}
}
else
{
MessageBox(hwnd, "PRG NVRAM size exceeded the limit (4096KB)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGNVRAM_COMBO));
return false;
}
if (prg_nvram != 0)
_header.ROM_type |= 2;
}
else
return false;
}
else {
// iNES 1.0 is much simpler, it has only 1 bit for check
if (IsDlgButtonChecked(hwnd, IDC_CHECK_BATTERYNVRAM) == BST_CHECKED)
_header.ROM_type |= 2;
}
// CHR ROM
int chr_rom;
if (GetComboBoxByteSize(hwnd, IDC_CHRROM_COMBO, &chr_rom) == 0)
{
// max value which a iNES 2.0 header can hold
if (chr_rom < 8 * 1024 * 0xEFF)
{
// try to fit the irregular value with the alternative way
if (chr_rom % (8 * 1024) != 0)
{
if (ines20)
{
// try to calculate the nearest value
bool fit = false;
int result = 0;
for (int multiplier = 0; multiplier < 4; ++multiplier)
{
for (int exponent = 0; exponent < 64; ++exponent)
{
int new_result = pow(2, exponent) * (multiplier * 2 + 1);
if (new_result == chr_rom)
{
_header.Upper_ROM_VROM_size |= 0xF0;
_header.VROM_size |= exponent << 2;
_header.VROM_size |= multiplier & 3;
fit = true;
break;
}
if (result > new_result && new_result > chr_rom)
result = new_result;
}
if (fit) break;
}
if (!fit)
{
int result10 = (chr_rom / 1024 / 8 + 1) * 8 * 1024;
if (result10 < result)
result = result10;
char buf2[64];
if (result % 1024 != 0)
sprintf(buf2, "%dB", result);
else
sprintf(buf2, "%dKB", result / 1024);
sprintf(buf, "CHR ROM size you entered is invalid in iNES 2.0, do you want to set to its nearest value %s?", buf2);
if (MessageBox(hwnd, buf, "Error", MB_YESNO | MB_ICONERROR) == IDYES)
SetDlgItemText(hwnd, IDC_CHRROM_COMBO, buf2);
else
{
SetFocus(GetDlgItem(hwnd, IDC_CHRROM_COMBO));
return false;
}
}
}
else {
// ines 1.0 can't handle this kind of value
MessageBox(hwnd, "CHR ROM size must be multiple of 8KB in iNES 1.0", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRROM_COMBO));
return false;
}
}
// it's multiple size of 8KB
else {
// it can be fitted in iNES 1.0
if (chr_rom < 8 * 1024 * 0xFF)
_header.VROM_size |= chr_rom / 8 / 1024 & 0xFF;
else
{
if (ines20)
_header.Upper_ROM_VROM_size |= chr_rom / 8 / 1024 >> 4 & 0xF0;
else
{
MessageBox(hwnd, "CHR ROM size exceeded the limit of iNES 1.0 (2040KB).", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_PRGROM_COMBO));
return false;
}
}
}
}
// A too large size
else {
MessageBox(hwnd, "CHR ROM size you entered cannot be fitted in iNES 2.0.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRROM_COMBO));
return false;
}
}
else
return false;
// CHR RAM
if (ines20)
{
// only iNES 2.0 has value for CHR RAM
int chr_ram;
if (GetComboBoxByteSize(hwnd, IDC_CHRRAM_COMBO, &chr_ram) == 0)
{
if (chr_ram < 64 << 0xF)
{
if (chr_ram % 64 == 0)
_header.VRAM_size |= (int)log2(chr_ram / 64);
else
{
MessageBox(hwnd, "Invalid CHR RAM size", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRRAM_COMBO));
return false;
}
}
else {
MessageBox(hwnd, "CHR RAM size exceeded the limit (4096KB)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRRAM_COMBO));
return false;
}
}
else
return false;
}
// CHR NVRAM
if (ines20)
{
// only iNES 2.0 has value for CHR NVRAM
int chr_nvram;
if (GetComboBoxByteSize(hwnd, IDC_CHRNVRAM_COMBO, &chr_nvram) == 0)
{
if (chr_nvram < 64 << 0xF)
{
if (chr_nvram % 64 == 0)
_header.VRAM_size |= (int)log2(chr_nvram / 64) << 4;
else {
MessageBox(hwnd, "Invalid CHR NVRAM size", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRNVRAM_COMBO));
return false;
}
}
else {
MessageBox(hwnd, "CHR NVRAM size exceeded the limit (4096KB)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_CHRNVRAM_COMBO));
return false;
}
if (chr_nvram != 0)
_header.ROM_type |= 2;
}
else
return false;
}
// Mirroring
if (IsDlgButtonChecked(hwnd, IDC_RADIO_MIRR_4SCREEN) == BST_CHECKED)
_header.ROM_type |= 8;
else if (IsDlgButtonChecked(hwnd, IDC_RADIO_MIRR_VERTICAL) == BST_CHECKED)
_header.ROM_type |= 1;
// Region
if (IsDlgButtonChecked(hwnd, IDC_RADIO_REGION_PAL) == BST_CHECKED)
{
if (ines20)
_header.TV_system |= 1;
else {
_header.Upper_ROM_VROM_size |= 1;
if (IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL) == BST_CHECKED && IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_EXTRA_REGION) == BST_CHECKED)
_header.RAM_size |= 2;
}
}
else if (IsDlgButtonChecked(hwnd, IDC_RADIO_REGION_DUAL) == BST_CHECKED)
{
if (ines20)
_header.TV_system |= 2;
else
_header.RAM_size |= 3;
}
else if (IsDlgButtonChecked(hwnd, IDC_RADIO_REGION_DENDY) == BST_CHECKED)
_header.TV_system |= 3;
// System
if (IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_VS) == BST_CHECKED)
{
_header.ROM_type2 |= 1;
if (ines20) {
// VS System type
int system;
if (GetComboBoxListItemData(hwnd, IDC_VS_SYSTEM_COMBO, &system, buf) && system <= 0xF)
_header.VS_hardware |= (system & 0xF) << 4;
else
{
MessageBox(hwnd, "Invalid VS System hardware type.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_VS_SYSTEM_COMBO));
return false;
}
// VS PPU type
int ppu;
if (GetComboBoxListItemData(hwnd, IDC_VS_PPU_COMBO, &ppu, buf) && system <= 0xF)
_header.VS_hardware |= ppu & 0xF;
else
{
MessageBox(hwnd, "Invalid VS System PPU type.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_VS_PPU_COMBO));
return false;
}
}
}
else if (IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_PLAYCHOICE10) == BST_CHECKED)
_header.ROM_type2 |= 2;
else if (IsDlgButtonChecked(hwnd, IDC_RADIO_SYSTEM_EXTEND) == BST_CHECKED)
{
// Extend System
_header.ROM_type2 |= 3;
int extend;
if (GetComboBoxListItemData(hwnd, IDC_SYSTEM_EXTEND_COMBO, &extend, buf) && extend <= 0x3F)
_header.VS_hardware |= extend & 0x3F;
else
{
MessageBox(hwnd, "Invalid extend system type", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_SYSTEM_EXTEND_COMBO));
return false;
}
}
// Input device
if (ines20)
{
int input;
if (GetComboBoxListItemData(hwnd, IDC_INPUT_DEVICE_COMBO, &input, buf, true) && input <= 0x3F)
_header.reserved[1] |= input & 0x3F;
else
{
MessageBox(hwnd, "Invalid input device.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_INPUT_DEVICE_COMBO));
return false;
}
}
// Miscellanous ROM(s)
if (ines20)
{
GetDlgItemText(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT, buf, 256);
int misc_roms = 0;
if (sscanf(buf, "%d", &misc_roms) < 1)
{
MessageBox(hwnd, "Invalid miscellanous ROM(s) count. If you don't know what value should be, we recommend to set it to 0.", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT));
return false;
}
if (misc_roms > 3)
{
MessageBox(hwnd, "Miscellanous ROM(s) count has exceeded the limit of iNES 2.0 (3)", "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, IDC_MISCELLANEOUS_ROMS_EDIT));
return false;
}
_header.reserved[0] |= misc_roms & 3;
}
// iNES 1.0 unofficial properties
if (!ines20 && IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL) == BST_CHECKED)
{
// bus conflict configure in unoffcial bit of iNES 1.0
if (IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_BUS_CONFLICT) == BST_CHECKED)
_header.RAM_size |= 0x20;
// PRG RAM non exist bit flag
if (IsDlgButtonChecked(hwnd, IDC_CHECK_UNOFFICIAL_PRGRAM) == BST_UNCHECKED)
_header.RAM_size |= 0x10;
}
// Trainer
if (IsDlgButtonChecked(hwnd, IDC_CHECK_TRAINER) == BST_CHECKED)
_header.ROM_type |= 4;
extern BMAPPINGLocal bmap[];
bool fceux_support = false;
for (int i = 0; bmap[i].init; ++i)
{
if (mapper == bmap[i].number)
{
fceux_support = true;
break;
}
}
if (!fceux_support)
{
sprintf(buf, "FCEUX doesn't support iNES Mapper# %d, this is not a serious problem, but the ROM will not be run in FCEUX properly.\nDo you want to continue?", mapper);
if (MessageBox(hwnd, buf, "Error", MB_YESNO | MB_ICONWARNING) == IDNO)
{
SetFocus(GetDlgItem(hwnd, IDC_MAPPER_COMBO));
return false;
}
}
memcpy(_header.ID, "NES\x1A", 4);
if (header)
memcpy(header, &_header, sizeof(iNES_HEADER));
#ifdef _DEBUG
printf("header: ");
printf("%02X ", _header.ID[0]);
printf("%02X ", _header.ID[1]);
printf("%02X ", _header.ID[2]);
printf("%02X ", _header.ID[3]);
printf("%02X ", _header.ROM_size);
printf("%02X ", _header.VROM_size);
printf("%02X ", _header.ROM_type);
printf("%02X ", _header.ROM_type2);
printf("%02X ", _header.ROM_type3);
printf("%02X ", _header.Upper_ROM_VROM_size);
printf("%02X ", _header.RAM_size);
printf("%02X ", _header.VRAM_size);
printf("%02X ", _header.TV_system);
printf("%02X ", _header.VS_hardware);
printf("%02X ", _header.reserved[0]);
printf("%02X\n", _header.reserved[1]);
#endif
return true;
}
int GetComboBoxByteSize(HWND hwnd, UINT id, int* value)
{
char* name = "";
char buf[256];
enum errors {
OK = 0,
FORMAT_ERR,
MINUS_ERR,
UNIT_ERR
};
int err = errors::OK;
switch (id)
{
case IDC_PRGROM_COMBO: name = "PRG ROM"; break;
case IDC_PRGRAM_COMBO: name = "PRG RAM"; break;
case IDC_PRGNVRAM_COMBO: name = "PRG NVRAM"; break;
case IDC_CHRROM_COMBO: name = "CHR ROM"; break;
case IDC_CHRRAM_COMBO: name = "CHR RAM"; break;
case IDC_CHRNVRAM_COMBO: name = "CHR NVRAM"; break;
}
if (!GetComboBoxListItemData(hwnd, id, value, buf, true))
{
char buf2[4];
if (sscanf(buf, "%d%3[KMB]", value, buf2) < 2 || !strcmp(buf2, ""))
err = errors::FORMAT_ERR;
else
{
if (*value < 0)
err = errors::MINUS_ERR;
else if (!strcmp(buf2, "KB") || !strcmp(buf2, "K"))
*value *= 1024;
else if (!strcmp(buf2, "MB") || !strcmp(buf2, "M"))
*value *= 1024 * 1024;
else if (strcmp(buf2, "B"))
err = errors::UNIT_ERR;
}
}
switch (err)
{
case errors::FORMAT_ERR:
sprintf(buf, "%s size you entered is invalid, it should be positive decimal integer followed with unit, e.g. 1024B, 128KB, 4MB", name);
break;
case errors::UNIT_ERR:
sprintf(buf, "The unit of %s size you entered is invalid, it must be B, KB or MB", name);
break;
case errors::MINUS_ERR:
sprintf(buf, "Negative value of %s is not supported by iNES header.", name);
break;
}
if (err)
{
MessageBox(hwnd, buf, "Error", MB_OK | MB_ICONERROR);
SetFocus(GetDlgItem(hwnd, id));
}
return err;
}
bool GetComboBoxListItemData(HWND hwnd, UINT id, int* value, char* buf, bool exact)
{
bool success = true;
*value = SendDlgItemMessage(hwnd, id, CB_GETCURSEL, 0, 0);
if (*value != CB_ERR)
*value = SendDlgItemMessage(hwnd, id, CB_GETITEMDATA, *value, 0);
else {
GetDlgItemText(hwnd, id, buf, 256);
if (exact)
*value = SendDlgItemMessage(hwnd, id, CB_FINDSTRINGEXACT, 0, (LPARAM)buf);
else
*value = SendDlgItemMessage(hwnd, id, CB_SELECTSTRING, 0, (LPARAM)buf);
if (*value != CB_ERR)
*value = SendDlgItemMessage(hwnd, id, CB_GETITEMDATA, *value, 0);
else
{
switch (id)
{
default:
success = false;
break;
case IDC_MAPPER_COMBO:
if (!(success = sscanf(buf, "%d", value) > 0))
success = SearchByString(hwnd, id, value, buf);
else
SetDlgItemText(hwnd, id, buf);
break;
case IDC_VS_SYSTEM_COMBO:
case IDC_VS_PPU_COMBO:
case IDC_SYSTEM_EXTEND_COMBO:
if (!(success = sscanf(buf, "$%X", (unsigned int *)value) > 0))
success = SearchByString(hwnd, id, value, buf);
else
SetDlgItemText(hwnd, id, buf);
break;
case IDC_INPUT_DEVICE_COMBO:
if (success = sscanf(buf, "$%X", (unsigned int *)value) > 0)
{
char buf2[8];
sprintf(buf2, "$%02X", *value);
if (SendDlgItemMessage(hwnd, id, CB_SELECTSTRING, 0, (LPARAM)buf2) == CB_ERR)
SetDlgItemText(hwnd, id, buf);
} else
success = SearchByString(hwnd, id, value, buf);
break;
}
if (!success)
SetDlgItemText(hwnd, id, buf);
}
}
return success;
}
// Warning: when in save mode, the content of buf might be overwritten by the save filename which user changed.
bool ShowINESFileBox(HWND parent, char* buf, bool save)
{
char *filename = NULL, *path = NULL;
bool success = true;
if (save)
{
// When open this dialog for saving prpose, the buf must be a separate buf.
if (buf && buf != LoadedRomFName)
{
extern char* GetRomName(bool force = false);
extern char* GetRomPath(bool force = false);
filename = GetRomName(true);
char* second = strchr(filename, '|');
if (second)
{
char* _filename = (char*)calloc(1, 2048);
strcpy(_filename, second + 1);
char* third = strrchr(filename, '\\');
if (third)
strcpy(_filename, third + 1);
free(filename);
filename = _filename;
}
strcat(filename, " [header modified].nes");
path = GetRomPath(true);
}
else
success = false;
}
else {
if (!buf)
buf = LoadedRomFName;
filename = (char*)calloc(1, 2048);
path = (char*)calloc(1, 2048);
}
if (success)
{
OPENFILENAME ofn;
memset(&ofn, 0, sizeof(OPENFILENAME));
ofn.lStructSize = sizeof(OPENFILENAME);
ofn.lpstrTitle = save ? "Save NES file" : "Open NES file";
ofn.lpstrFilter = "NES ROM file (*.nes)\0*.nes\0All files (*.*)\0*.*\0\0";
ofn.hInstance = fceu_hInstance;
ofn.hwndOwner = parent;
ofn.lpstrFile = filename;
ofn.nMaxFile = 2048;
ofn.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_OVERWRITEPROMPT | OFN_HIDEREADONLY;
ofn.lpstrInitialDir = path;
ofn.lpstrDefExt = "nes";
if (save ? GetSaveFileName(&ofn) : GetOpenFileName(&ofn))
strcpy(buf, filename);
else
success = false;
}
if (filename) free(filename);
if (path) free(path);
return success;
}
bool SaveINESFile(HWND hwnd, char* path, iNES_HEADER* header)
{
char buf[4096];
const char* ext[] = { "nes", 0 };
// Source file
FCEUFILE* source = FCEU_fopen(LoadedRomFName, NULL, "rb", 0, -1, ext);
if (!source)
{
sprintf(buf, "Opening source file %s failed.", LoadedRomFName);
MessageBox(hwnd, buf, "iNES Header Editor", MB_OK | MB_ICONERROR);
return false;
}
// Destination file
FILE* target = FCEUD_UTF8fopen(path, "wb");
if (!target)
{
sprintf(buf, "Creating target file %s failed.", path);
MessageBox(hwnd, buf, "iNES Header Editor", MB_OK | MB_ICONERROR);
return false;
}
memset(buf, 0, sizeof(buf));
// Write the header first
fwrite(header, 1, sizeof(iNES_HEADER), target);
// Skip the original header
FCEU_fseek(source, sizeof(iNES_HEADER), SEEK_SET);
int len;
while (len = FCEU_fread(buf, 1, sizeof(buf), source))
fwrite(buf, len, 1, target);
FCEU_fclose(source);
fclose(target);
return true;
}
bool SearchByString(HWND hwnd, UINT id, int* value, char* buf)
{
if (buf[0] != ' ' && buf[0] != 0)
{
if (id == IDC_MAPPER_COMBO)
{
extern BMAPPINGLocal bmap[];
for (int i = 0; bmap[i].init; ++i)
{
if (!stricmp(buf, bmap[i].name))
{
SendDlgItemMessage(hwnd, id, CB_SETCURSEL, i, 0);
*value = SendDlgItemMessage(hwnd, id, CB_GETITEMDATA, i, 0);
return true;
}
}
}
else
{
for (int i = 0; dropDownIdList[i]; ++i)
{
if (dropDownIdList[i] == id)
{
char** checkList = dropDownList[i];
for (int j = 0; checkList[j]; ++j)
{
if (!stricmp(buf, checkList[j]))
{
SendDlgItemMessage(hwnd, id, CB_SETCURSEL, j, 0);
*value = SendDlgItemMessage(hwnd, id, CB_GETITEMDATA, j, 0);
return true;
}
}
break;
}
}
}
}
return false;
}