mirror of
https://github.com/AppleWin/AppleWin.git
synced 2025-01-07 07:29:20 +00:00
edf65762cd
. replaced most GetObj() with GetRef() . so dynamic_cast'ing to a reference now (instead of a pointer)
628 lines
19 KiB
C++
628 lines
19 KiB
C++
/*
|
|
AppleWin : An Apple //e emulator for Windows
|
|
|
|
Copyright (C) 1994-1996, Michael O'Brien
|
|
Copyright (C) 1999-2001, Oliver Schmidt
|
|
Copyright (C) 2002-2005, Tom Charlesworth
|
|
Copyright (C) 2006-2012, Tom Charlesworth, Michael Pohoreski
|
|
|
|
AppleWin is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
AppleWin is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with AppleWin; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "StdAfx.h"
|
|
|
|
#include "../Applewin.h" // g_nAppMode, g_uScrollLockToggle, sg_PropertySheet
|
|
#include "../CardManager.h"
|
|
#include "../Disk.h"
|
|
#include "../Frame.h"
|
|
#include "../Log.h"
|
|
#include "../Registry.h"
|
|
#include "../SaveState.h"
|
|
#include "IPropertySheet.h"
|
|
#include "PropertySheetHelper.h"
|
|
|
|
/*
|
|
Config causing AfterClose msgs:
|
|
===============================
|
|
|
|
Page: Action: Comment:
|
|
------------------------------------------------------------------------
|
|
Config
|
|
. Computer WM_USER_RESTART
|
|
. Benchmark PSBTN_OK -> WM_USER_BENCHMARK Forces PSBTN_OK
|
|
Input
|
|
. Mouse WM_USER_RESTART
|
|
. CP/M WM_USER_RESTART
|
|
Sound
|
|
. MB/Phasor/SAM/None WM_USER_RESTART
|
|
Disk
|
|
. HDD enable WM_USER_RESTART
|
|
Advanced
|
|
. Save State WM_USER_SAVESTATE
|
|
. Load State WM_USER_LOADSTATE
|
|
. Clone WM_USER_RESTART
|
|
. MrFreeze Rom WM_USER_RESTART
|
|
|
|
Requirements:
|
|
-------------
|
|
. Want to change multiple HW at once.
|
|
. Allow change Computer & Load State (ie. multiple AfterClose msgs)
|
|
|
|
Design:
|
|
-------
|
|
. At PropSheet init, copy original config state (for above items)
|
|
. On last Page close, compare new config state, if changed:
|
|
- Show 1 restart + confirm msg (if necessary)
|
|
- Cancel will rollback to original config (for above items), but other items will be applied
|
|
. Load State button
|
|
- Don't action it immediately.
|
|
- .aws should contain config (but doesn't), so should override any config changes.
|
|
- so this can just discard any config changes
|
|
- if any config change, then show msg box to say they won't be applied
|
|
. Save State button
|
|
- Don't action it immediately.
|
|
- save state applies to current config (prior to restart).
|
|
- so this can just discard any config changes
|
|
- if any config change, then show msg box to say they won't be applied
|
|
. Benchmark button
|
|
- Action it immediately.
|
|
- If config has changed:
|
|
- Prompt to either do benchmark (and lose new config) or cancel benchmark (and drop back to Configuration dialog).
|
|
*/
|
|
|
|
void CPropertySheetHelper::FillComboBox(HWND window, int controlid, LPCTSTR choices, int currentchoice)
|
|
{
|
|
_ASSERT(choices);
|
|
HWND combowindow = GetDlgItem(window, controlid);
|
|
SendMessage(combowindow, CB_RESETCONTENT, 0, 0);
|
|
while (choices && *choices)
|
|
{
|
|
SendMessage(combowindow, CB_ADDSTRING, 0, (LPARAM)(LPCTSTR)choices);
|
|
choices += _tcslen(choices)+1;
|
|
}
|
|
|
|
if (SendMessage(combowindow, CB_SETCURSEL, currentchoice, 0) == CB_ERR && currentchoice != -1)
|
|
{
|
|
_ASSERT(0);
|
|
SendMessage(combowindow, CB_SETCURSEL, 0, 0); // GH#434: Failed to set currentchoice, so select item-0
|
|
}
|
|
}
|
|
|
|
void CPropertySheetHelper::SaveComputerType(eApple2Type NewApple2Type)
|
|
{
|
|
if (NewApple2Type == A2TYPE_CLONE) // Clone picked from Config tab, but no specific one picked from Advanced tab
|
|
NewApple2Type = A2TYPE_PRAVETS82;
|
|
|
|
ConfigSaveApple2Type(NewApple2Type);
|
|
}
|
|
|
|
void CPropertySheetHelper::ConfigSaveApple2Type(eApple2Type apple2Type)
|
|
{
|
|
REGSAVE(TEXT(REGVALUE_APPLE2_TYPE), apple2Type);
|
|
LogFileOutput("Config: Apple2 Type changed to %d\n", apple2Type);
|
|
}
|
|
|
|
void CPropertySheetHelper::SaveCpuType(eCpuType NewCpuType)
|
|
{
|
|
REGSAVE(TEXT(REGVALUE_CPU_TYPE), NewCpuType);
|
|
}
|
|
|
|
void CPropertySheetHelper::SetSlot(UINT slot, SS_CARDTYPE newCardType)
|
|
{
|
|
_ASSERT(slot < NUM_SLOTS);
|
|
if (slot >= NUM_SLOTS)
|
|
return;
|
|
|
|
// Two paths:
|
|
// 1) Via Config dialog: card not inserted yet
|
|
// 2) Snapshot_LoadState_v2(): card already inserted
|
|
if (g_CardMgr.QuerySlot(slot) != newCardType)
|
|
g_CardMgr.Insert(slot, newCardType);
|
|
|
|
std::string slotText;
|
|
switch (slot)
|
|
{
|
|
case 0: slotText = REGVALUE_SLOT0; break;
|
|
case 1: slotText = REGVALUE_SLOT1; break;
|
|
case 2: slotText = REGVALUE_SLOT2; break;
|
|
case 3: slotText = REGVALUE_SLOT3; break;
|
|
case 4: slotText = REGVALUE_SLOT4; break;
|
|
case 5: slotText = REGVALUE_SLOT5; break;
|
|
case 6: slotText = REGVALUE_SLOT6; break;
|
|
case 7: slotText = REGVALUE_SLOT7; break;
|
|
}
|
|
|
|
REGSAVE(slotText.c_str(), (DWORD)newCardType);
|
|
}
|
|
|
|
// Looks like a (bad) C&P from SaveStateSelectImage()
|
|
// - eg. see "RAPCS" tags below...
|
|
// Used by:
|
|
// . CPageDisk: IDC_CIDERPRESS_BROWSE
|
|
// . CPageAdvanced: IDC_PRINTER_DUMP_FILENAME_BROWSE
|
|
std::string CPropertySheetHelper::BrowseToFile(HWND hWindow, TCHAR* pszTitle, TCHAR* REGVALUE, TCHAR* FILEMASKS)
|
|
{
|
|
static std::string PathToFile; //This is a really awkward way to prevent mixing CiderPress and SaveStated values (RAPCS), but it seem the quickest. Here is its Line 1.
|
|
PathToFile = Snapshot_GetFilename(); //RAPCS, line 2.
|
|
TCHAR szDirectory[MAX_PATH] = TEXT("");
|
|
TCHAR szFilename[MAX_PATH];
|
|
RegLoadString(TEXT("Configuration"), REGVALUE, 1, szFilename, MAX_PATH, TEXT(""));
|
|
std::string PathName = szFilename;
|
|
|
|
OPENFILENAME ofn;
|
|
ZeroMemory(&ofn,sizeof(OPENFILENAME));
|
|
|
|
ofn.lStructSize = sizeof(OPENFILENAME);
|
|
ofn.hwndOwner = hWindow;
|
|
ofn.hInstance = g_hInstance;
|
|
ofn.lpstrFilter = FILEMASKS;
|
|
/*ofn.lpstrFilter = TEXT("Applications (*.exe)\0*.exe\0")
|
|
TEXT("Text files (*.txt)\0*.txt\0")
|
|
TEXT("All Files\0*.*\0");*/
|
|
ofn.lpstrFile = szFilename;
|
|
ofn.nMaxFile = MAX_PATH;
|
|
ofn.lpstrInitialDir = szDirectory;
|
|
ofn.Flags = OFN_PATHMUSTEXIST | OFN_HIDEREADONLY;
|
|
ofn.lpstrTitle = pszTitle;
|
|
|
|
int nRes = GetOpenFileName(&ofn);
|
|
if(nRes) // Okay is pressed
|
|
{
|
|
m_szNewFilename = &szFilename[ofn.nFileOffset]; // TODO:TC: m_szNewFilename not used! (Was g_szNewFilename)
|
|
|
|
szFilename[ofn.nFileOffset] = 0;
|
|
if (_tcsicmp(szDirectory, szFilename))
|
|
m_szSSNewDirectory = szFilename; // TODO:TC: m_szSSNewDirectory looks dodgy! (Was g_szSSNewDirectory)
|
|
|
|
PathName = szFilename;
|
|
PathName.append (m_szNewFilename);
|
|
}
|
|
else // Cancel is pressed
|
|
{
|
|
RegLoadString(TEXT("Configuration"), REGVALUE, 1, szFilename, MAX_PATH, TEXT(""));
|
|
PathName = szFilename;
|
|
}
|
|
|
|
m_szNewFilename = PathToFile; //RAPCS, line 3 (last).
|
|
return PathName;
|
|
}
|
|
|
|
void CPropertySheetHelper::SaveStateUpdate()
|
|
{
|
|
if (m_bSSNewFilename)
|
|
{
|
|
Snapshot_SetFilename(m_szSSNewPathname);
|
|
|
|
RegSaveString(TEXT(REG_CONFIG), REGVALUE_SAVESTATE_FILENAME, 1, m_szSSNewPathname);
|
|
|
|
if(!m_szSSNewDirectory.empty())
|
|
RegSaveString(TEXT(REG_PREFS), REGVALUE_PREF_START_DIR, 1, m_szSSNewDirectory);
|
|
}
|
|
}
|
|
|
|
void CPropertySheetHelper::GetDiskBaseNameWithAWS(std::string & pszFilename)
|
|
{
|
|
if (g_CardMgr.QuerySlot(SLOT6) != CT_Disk2)
|
|
return;
|
|
|
|
const std::string& diskName = dynamic_cast<Disk2InterfaceCard&>(g_CardMgr.GetRef(SLOT6)).GetBaseName(DRIVE_1);
|
|
if (!diskName.empty())
|
|
{
|
|
pszFilename = diskName + ".aws.yaml";
|
|
}
|
|
}
|
|
|
|
// NB. OK'ing this property sheet will call Snapshot_SetFilename() with this new filename
|
|
int CPropertySheetHelper::SaveStateSelectImage(HWND hWindow, TCHAR* pszTitle, bool bSave)
|
|
{
|
|
std::string szDirectory;
|
|
std::string tempFilename;
|
|
|
|
if (bSave)
|
|
{
|
|
// Attempt to use drive1's image name as the name for the .aws file
|
|
// Else Attempt to use the Prop Sheet's filename
|
|
GetDiskBaseNameWithAWS(tempFilename);
|
|
if (tempFilename.empty())
|
|
{
|
|
tempFilename = Snapshot_GetFilename();
|
|
}
|
|
}
|
|
else // Load (or Browse)
|
|
{
|
|
// Attempt to use the Prop Sheet's filename first
|
|
// Else attempt to use drive1's image name as the name for the .aws file
|
|
tempFilename = Snapshot_GetFilename();
|
|
if (tempFilename.empty())
|
|
{
|
|
GetDiskBaseNameWithAWS(tempFilename);
|
|
}
|
|
|
|
szDirectory = Snapshot_GetPath();
|
|
}
|
|
|
|
if (szDirectory.empty())
|
|
szDirectory = g_sCurrentDir;
|
|
|
|
// convert tempFilename to char * for the rest of the function
|
|
TCHAR szFilename[MAX_PATH] = {0};
|
|
strcpy(szFilename, tempFilename.c_str());
|
|
tempFilename.clear(); // do NOT use this any longer
|
|
|
|
//
|
|
|
|
OPENFILENAME ofn;
|
|
ZeroMemory(&ofn,sizeof(OPENFILENAME));
|
|
|
|
ofn.lStructSize = sizeof(OPENFILENAME);
|
|
ofn.hwndOwner = hWindow;
|
|
ofn.hInstance = g_hInstance;
|
|
if (bSave)
|
|
{
|
|
ofn.lpstrFilter = TEXT("Save State files (*.aws.yaml)\0*.aws.yaml\0");
|
|
TEXT("All Files\0*.*\0");
|
|
}
|
|
else
|
|
{
|
|
ofn.lpstrFilter = TEXT("Save State files (*.aws,*.aws.yaml)\0*.aws;*.aws.yaml\0");
|
|
TEXT("All Files\0*.*\0");
|
|
}
|
|
ofn.lpstrFile = szFilename; // Dialog strips the last .EXT from this string (eg. file.aws.yaml is displayed as: file.aws
|
|
ofn.nMaxFile = MAX_PATH;
|
|
ofn.lpstrInitialDir = szDirectory.c_str();
|
|
ofn.Flags = OFN_PATHMUSTEXIST | OFN_HIDEREADONLY;
|
|
ofn.lpstrTitle = pszTitle;
|
|
|
|
int nRes = bSave ? GetSaveFileName(&ofn) : GetOpenFileName(&ofn);
|
|
|
|
if(nRes)
|
|
{
|
|
if (bSave) // Only for saving (allow loading of any file for backwards compatibility)
|
|
{
|
|
// Append .aws.yaml if it's not there
|
|
const char szAWS_EXT1[] = ".aws";
|
|
const char szAWS_EXT2[] = ".yaml";
|
|
const char szAWS_EXT3[] = ".aws.yaml";
|
|
const UINT uStrLenFile = strlen(&szFilename[ofn.nFileOffset]);
|
|
const UINT uStrLenExt1 = strlen(szAWS_EXT1);
|
|
const UINT uStrLenExt2 = strlen(szAWS_EXT2);
|
|
const UINT uStrLenExt3 = strlen(szAWS_EXT3);
|
|
if (uStrLenFile <= uStrLenExt1)
|
|
{
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile], szAWS_EXT3); // "file" += ".aws.yaml"
|
|
}
|
|
else if (uStrLenFile <= uStrLenExt2)
|
|
{
|
|
if (strcmp(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt1], szAWS_EXT1) == 0)
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt1], szAWS_EXT3); // "file.aws" -> "file" + ".aws.yaml"
|
|
else
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile], szAWS_EXT3); // "file" += ".aws.yaml"
|
|
}
|
|
else if ((uStrLenFile <= uStrLenExt3) || (strcmp(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt3], szAWS_EXT3) != 0))
|
|
{
|
|
if (strcmp(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt1], szAWS_EXT1) == 0)
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt1], szAWS_EXT3); // "file.aws" -> "file" + ".aws.yaml"
|
|
else if (strcmp(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt2], szAWS_EXT2) == 0)
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile-uStrLenExt2], szAWS_EXT3); // "file.yaml" -> "file" + ".aws.yaml"
|
|
else
|
|
strcpy(&szFilename[ofn.nFileOffset+uStrLenFile], szAWS_EXT3); // "file" += ".aws.yaml"
|
|
}
|
|
}
|
|
|
|
m_szSSNewFilename = &szFilename[ofn.nFileOffset];
|
|
m_szSSNewPathname = szFilename;
|
|
|
|
szFilename[ofn.nFileOffset] = 0;
|
|
if (_tcsicmp(szDirectory.c_str(), szFilename))
|
|
m_szSSNewDirectory = szFilename;
|
|
}
|
|
|
|
m_bSSNewFilename = nRes ? true : false;
|
|
return nRes;
|
|
}
|
|
|
|
// On OK: Optionally post a single "uAfterClose" msg after last page closes
|
|
void CPropertySheetHelper::PostMsgAfterClose(HWND hWnd, PAGETYPE page)
|
|
{
|
|
m_bmPages &= ~(1<<(UINT32)page);
|
|
if (m_bmPages)
|
|
return; // Still pages to close
|
|
|
|
//
|
|
|
|
if (m_ConfigNew.m_uSaveLoadStateMsg && IsOkToSaveLoadState(hWnd, IsConfigChanged()))
|
|
{
|
|
// Drop any config change, and do load/save state
|
|
PostMessage(g_hFrameWindow, m_ConfigNew.m_uSaveLoadStateMsg, 0, 0);
|
|
return;
|
|
}
|
|
|
|
if (m_bDoBenchmark)
|
|
{
|
|
// Drop any config change, and do benchmark
|
|
PostMessage(g_hFrameWindow, WM_USER_BENCHMARK, 0, 0); // NB. doesn't do WM_USER_RESTART
|
|
return;
|
|
}
|
|
|
|
UINT uAfterClose = 0;
|
|
|
|
if (m_ConfigNew.m_Apple2Type == A2TYPE_CLONE)
|
|
{
|
|
MessageBox(hWnd, "Error - Unable to change configuration\n\nReason: A specific clone wasn't selected from the Advanced tab", g_pAppTitle.c_str(), MB_ICONSTOP | MB_SETFOREGROUND);
|
|
return;
|
|
}
|
|
|
|
_ASSERT(m_ConfigNew.m_CpuType != CPU_UNKNOWN); // NB. Could only ever be CPU_UNKNOWN for a clone (and only if a mistake was made when adding a new clone)
|
|
if (m_ConfigNew.m_CpuType == CPU_UNKNOWN)
|
|
{
|
|
m_ConfigNew.m_CpuType = ProbeMainCpuDefault(m_ConfigNew.m_Apple2Type);
|
|
}
|
|
|
|
if (IsConfigChanged())
|
|
{
|
|
if (!CheckChangesForRestart(hWnd))
|
|
{
|
|
// Cancelled
|
|
RestoreCurrentConfig();
|
|
return;
|
|
}
|
|
|
|
ApplyNewConfig();
|
|
|
|
uAfterClose = WM_USER_RESTART;
|
|
}
|
|
|
|
if (uAfterClose)
|
|
PostMessage(g_hFrameWindow, uAfterClose, 0, 0);
|
|
}
|
|
|
|
bool CPropertySheetHelper::CheckChangesForRestart(HWND hWnd)
|
|
{
|
|
if (!HardwareConfigChanged(hWnd))
|
|
return false; // Cancelled
|
|
|
|
if (!IsOkToRestart(hWnd))
|
|
return false; // Cancelled
|
|
|
|
return true; // OK
|
|
}
|
|
|
|
#define CONFIG_CHANGED_LOCAL(var) \
|
|
(ConfigOld.var != ConfigNew.var)
|
|
|
|
// Apply changes to Registry
|
|
void CPropertySheetHelper::ApplyNewConfig(const CConfigNeedingRestart& ConfigNew, const CConfigNeedingRestart& ConfigOld)
|
|
{
|
|
if (CONFIG_CHANGED_LOCAL(m_Apple2Type))
|
|
{
|
|
SaveComputerType(ConfigNew.m_Apple2Type);
|
|
}
|
|
|
|
if (CONFIG_CHANGED_LOCAL(m_CpuType))
|
|
{
|
|
SaveCpuType(ConfigNew.m_CpuType);
|
|
}
|
|
|
|
UINT slot = 4;
|
|
if (CONFIG_CHANGED_LOCAL(m_Slot[slot]))
|
|
SetSlot(slot, ConfigNew.m_Slot[slot]);
|
|
|
|
slot = 5;
|
|
if (CONFIG_CHANGED_LOCAL(m_Slot[slot]))
|
|
SetSlot(slot, ConfigNew.m_Slot[slot]);
|
|
|
|
// slot = 7;
|
|
// if (CONFIG_CHANGED_LOCAL(m_Slot[slot]))
|
|
// SetSlot(slot, ConfigNew.m_Slot[slot]);
|
|
|
|
if (CONFIG_CHANGED_LOCAL(m_bEnableHDD))
|
|
{
|
|
REGSAVE(TEXT(REGVALUE_HDD_ENABLED), ConfigNew.m_bEnableHDD ? 1 : 0); // TODO: Change to REGVALUE_SLOT7
|
|
}
|
|
|
|
if (CONFIG_CHANGED_LOCAL(m_bEnableTheFreezesF8Rom))
|
|
{
|
|
REGSAVE(TEXT(REGVALUE_THE_FREEZES_F8_ROM), ConfigNew.m_bEnableTheFreezesF8Rom);
|
|
}
|
|
|
|
if (CONFIG_CHANGED_LOCAL(m_videoRefreshRate))
|
|
{
|
|
REGSAVE(TEXT(REGVALUE_VIDEO_REFRESH_RATE), ConfigNew.m_videoRefreshRate);
|
|
}
|
|
}
|
|
|
|
void CPropertySheetHelper::ApplyNewConfig(void)
|
|
{
|
|
ApplyNewConfig(m_ConfigNew, m_ConfigOld);
|
|
}
|
|
|
|
void CPropertySheetHelper::SaveCurrentConfig(void)
|
|
{
|
|
// NB. clone-type is encoded in g_Apple2Type
|
|
m_ConfigOld.m_Apple2Type = GetApple2Type();
|
|
m_ConfigOld.m_CpuType = GetMainCpu();
|
|
m_ConfigOld.m_Slot[SLOT4] = g_CardMgr.QuerySlot(SLOT4);
|
|
m_ConfigOld.m_Slot[SLOT5] = g_CardMgr.QuerySlot(SLOT5);
|
|
m_ConfigOld.m_bEnableHDD = HD_CardIsEnabled();
|
|
m_ConfigOld.m_bEnableTheFreezesF8Rom = sg_PropertySheet.GetTheFreezesF8Rom();
|
|
m_ConfigOld.m_videoRefreshRate = GetVideoRefreshRate();
|
|
|
|
// Reset flags each time:
|
|
m_ConfigOld.m_uSaveLoadStateMsg = 0;
|
|
m_bDoBenchmark = false;
|
|
|
|
// Setup ConfigNew
|
|
m_ConfigNew = m_ConfigOld;
|
|
}
|
|
|
|
void CPropertySheetHelper::RestoreCurrentConfig(void)
|
|
{
|
|
// NB. clone-type is encoded in g_Apple2Type
|
|
SetApple2Type(m_ConfigOld.m_Apple2Type);
|
|
SetMainCpu(m_ConfigOld.m_CpuType);
|
|
g_CardMgr.Insert(SLOT4, m_ConfigOld.m_Slot[SLOT4]);
|
|
g_CardMgr.Insert(SLOT5, m_ConfigOld.m_Slot[SLOT5]);
|
|
HD_SetEnabled(m_ConfigOld.m_bEnableHDD);
|
|
sg_PropertySheet.SetTheFreezesF8Rom(m_ConfigOld.m_bEnableTheFreezesF8Rom);
|
|
SetVideoRefreshRate(m_ConfigOld.m_videoRefreshRate);
|
|
}
|
|
|
|
bool CPropertySheetHelper::IsOkToSaveLoadState(HWND hWnd, const bool bConfigChanged)
|
|
{
|
|
if (bConfigChanged)
|
|
{
|
|
if (MessageBox(hWnd,
|
|
TEXT("The hardware configuration has changed. Save/Load state will lose these changes.\n\n")
|
|
TEXT("Are you sure you want to do this?"),
|
|
TEXT(REG_CONFIG),
|
|
MB_ICONQUESTION | MB_OKCANCEL | MB_SETFOREGROUND) == IDCANCEL)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPropertySheetHelper::IsOkToRestart(HWND hWnd)
|
|
{
|
|
if (g_nAppMode == MODE_LOGO)
|
|
return true;
|
|
|
|
if (MessageBox(hWnd,
|
|
TEXT("Restarting the emulator will reset the state ")
|
|
TEXT("of the emulated machine, causing you to lose any ")
|
|
TEXT("unsaved work.\n\n")
|
|
TEXT("Are you sure you want to do this?"),
|
|
TEXT(REG_CONFIG),
|
|
MB_ICONQUESTION | MB_OKCANCEL | MB_SETFOREGROUND) == IDCANCEL)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
#define CONFIG_CHANGED(var) \
|
|
(m_ConfigOld.var != m_ConfigNew.var)
|
|
|
|
bool CPropertySheetHelper::HardwareConfigChanged(HWND hWnd)
|
|
{
|
|
std::string strMsg("The emulator needs to restart as the hardware configuration has changed:\n");
|
|
strMsg += "\n";
|
|
|
|
std::string strMsgMain;
|
|
{
|
|
if (CONFIG_CHANGED(m_Apple2Type))
|
|
strMsgMain += ". Emulated computer has changed\n";
|
|
|
|
if (CONFIG_CHANGED(m_CpuType))
|
|
strMsgMain += ". Emulated main CPU has changed\n";
|
|
|
|
if (CONFIG_CHANGED(m_videoRefreshRate))
|
|
strMsgMain += ". Video refresh rate has changed\n";
|
|
|
|
if (CONFIG_CHANGED(m_Slot[4]))
|
|
strMsgMain += GetSlot(4);
|
|
|
|
if (CONFIG_CHANGED(m_Slot[5]))
|
|
strMsgMain += GetSlot(5);
|
|
|
|
if (CONFIG_CHANGED(m_bEnableHDD))
|
|
strMsgMain += ". Harddisk(s) have been plugged/unplugged\n";
|
|
|
|
if (CONFIG_CHANGED(m_bEnableTheFreezesF8Rom))
|
|
strMsgMain += ". F8 ROM changed (The Freeze's F8 Rom)\n";
|
|
}
|
|
|
|
std::string strMsgPost("\n");
|
|
strMsgPost += "This change will not take effect until the next time you restart the emulator.\n\n";
|
|
strMsgPost += "Would you like to restart the emulator now?";
|
|
|
|
strMsg += strMsgMain;
|
|
strMsg += strMsgPost;
|
|
|
|
if (MessageBox(hWnd,
|
|
strMsg.c_str(),
|
|
TEXT(REG_CONFIG),
|
|
MB_ICONQUESTION | MB_OKCANCEL | MB_SETFOREGROUND) == IDCANCEL)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
std::string CPropertySheetHelper::GetSlot(const UINT uSlot)
|
|
{
|
|
// strMsg = ". Slot n: ";
|
|
std::string strMsg(". Slot ");
|
|
strMsg += '0' + uSlot;
|
|
strMsg += ": ";
|
|
|
|
const SS_CARDTYPE OldCardType = m_ConfigOld.m_Slot[uSlot];
|
|
const SS_CARDTYPE NewCardType = m_ConfigNew.m_Slot[uSlot];
|
|
|
|
if ((OldCardType == CT_Empty) || (NewCardType == CT_Empty))
|
|
{
|
|
if (NewCardType == CT_Empty)
|
|
{
|
|
strMsg += GetCardName(OldCardType);
|
|
strMsg += " card removed\n";
|
|
}
|
|
else
|
|
{
|
|
strMsg += GetCardName(NewCardType);
|
|
strMsg += " card added\n";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
strMsg += GetCardName(OldCardType);
|
|
strMsg += " card removed & ";
|
|
strMsg += GetCardName(NewCardType);
|
|
strMsg += " card added\n";
|
|
}
|
|
|
|
return strMsg;
|
|
}
|
|
|
|
std::string CPropertySheetHelper::GetCardName(const SS_CARDTYPE CardType)
|
|
{
|
|
switch (CardType)
|
|
{
|
|
case CT_Empty:
|
|
return "Empty";
|
|
case CT_Disk2: // Apple Disk][
|
|
return "Disk][";
|
|
case CT_SSC: // Apple Super Serial Card
|
|
return "Super Serial";
|
|
case CT_MockingboardC: // Soundcard
|
|
return "Mockingboard";
|
|
case CT_GenericPrinter:
|
|
return "Printer";
|
|
case CT_GenericHDD: // Hard disk
|
|
return "Hard Disk";
|
|
case CT_GenericClock:
|
|
return "Clock";
|
|
case CT_MouseInterface:
|
|
return "Mouse";
|
|
case CT_Z80:
|
|
return "CP/M";
|
|
case CT_Phasor: // Soundcard
|
|
return "Phasor";
|
|
case CT_Echo: // Soundcard
|
|
return "Echo";
|
|
case CT_SAM: // Soundcard: Software Automated Mouth
|
|
return "SAM";
|
|
default:
|
|
return "Unknown";
|
|
}
|
|
}
|