1678 lines
56 KiB
C++
1678 lines
56 KiB
C++
/////////////////////////////////////////////////////////////////
|
|
// $Id$
|
|
/////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2002-2023 The Bochs Project
|
|
//
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
// License as published by the Free Software Foundation; either
|
|
// version 2 of the License, or (at your option) any later version.
|
|
//
|
|
// This library 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
|
|
// Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this library; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
//
|
|
/////////////////////////////////////////////////////////////////
|
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
#define BX_PLUGGABLE
|
|
|
|
#include "config.h" // definitions based on configure script
|
|
#include "param_names.h"
|
|
|
|
#if BX_WITH_WX
|
|
|
|
// For compilers that support precompilation, includes <wx/wx.h>.
|
|
#include <wx/wxprec.h>
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
#ifndef WX_PRECOMP
|
|
#include <wx/wx.h>
|
|
#endif
|
|
#include <wx/spinctrl.h>
|
|
#include <wx/config.h>
|
|
#include <wx/confbase.h>
|
|
#include <wx/notebook.h>
|
|
|
|
#include "osdep.h" // workarounds for missing stuff
|
|
#include "gui/paramtree.h" // config parameter tree
|
|
#include "gui/siminterface.h" // interface to the simulator
|
|
#include "logio.h" // log level definitions
|
|
#include "wxdialog.h" // custom dialog boxes
|
|
#include "wxmain.h" // wxwidgets shared stuff
|
|
|
|
#if !defined(wxADJUST_MINSIZE)
|
|
#define wxADJUST_MINSIZE 0
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// constants, prototypes
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
enum {
|
|
ID_ShowDialog_1 = 1,
|
|
ID_ShowDialog_2,
|
|
ID_ShowDialog_3,
|
|
ID_Button1,
|
|
ID_Button2,
|
|
ID_MY_LAST_ID
|
|
};
|
|
|
|
wxSize longTextSize (300, -1); // width=300, height=default
|
|
wxSize normalTextSize (180, -1); // width=200, height=default
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// LogMsgAskDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Structure:
|
|
// vertSizer:
|
|
// context text field,
|
|
// message text field
|
|
// don't-ask checkbox
|
|
// buttonSizer:
|
|
// continue button
|
|
// die button
|
|
// dumpcore button
|
|
// debugger button
|
|
// help button
|
|
//
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(LogMsgAskDialog, wxDialog)
|
|
EVT_BUTTON(ID_Continue, LogMsgAskDialog::OnEvent)
|
|
EVT_BUTTON(ID_Die, LogMsgAskDialog::OnEvent)
|
|
EVT_BUTTON(ID_DumpCore, LogMsgAskDialog::OnEvent)
|
|
EVT_BUTTON(ID_Debugger, LogMsgAskDialog::OnEvent)
|
|
EVT_BUTTON(wxID_HELP, LogMsgAskDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
|
|
LogMsgAskDialog::LogMsgAskDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id,
|
|
const wxString& title)
|
|
: wxDialog (parent, id, title, wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
|
|
{
|
|
for (int i=0; i<N_BUTTONS; i++) enabled[i] = TRUE;
|
|
vertSizer = new wxBoxSizer(wxVERTICAL);
|
|
context = new wxStaticText(this, -1, wxT(""));
|
|
wxFont font = context->GetFont();
|
|
font.SetWeight(wxFONTWEIGHT_BOLD);
|
|
font.SetPointSize(2 + font.GetPointSize());
|
|
context->SetFont(font);
|
|
message = new wxStaticText(this, -1, wxT(""));
|
|
message->SetFont(font);
|
|
dontAsk = new wxCheckBox(this, -1, LOG_MSG_DONT_ASK_STRING);
|
|
btnSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
// fill vertical sizer
|
|
vertSizer->Add(context, 0, wxGROW|wxALIGN_LEFT|wxLEFT|wxTOP, 30);
|
|
vertSizer->Add(message, 0, wxGROW|wxALIGN_LEFT|wxLEFT, 30);
|
|
vertSizer->Add(dontAsk, 0, wxALIGN_CENTER|wxTOP, 30);
|
|
vertSizer->Add(btnSizer, 0, wxALIGN_CENTER|wxTOP, 30);
|
|
// Some object creation and layout is postponed until Init()
|
|
// so that caller has time to configure the dialog.
|
|
}
|
|
|
|
void LogMsgAskDialog::SetContext(wxString s)
|
|
{
|
|
ChangeStaticText(vertSizer, context, wxString(LOG_MSG_CONTEXT) + s);
|
|
}
|
|
|
|
void LogMsgAskDialog::SetMessage(wxString s)
|
|
{
|
|
ChangeStaticText(vertSizer, message, wxString(LOG_MSG_MSG) + s);
|
|
}
|
|
|
|
void LogMsgAskDialog::Init()
|
|
{
|
|
static const int ids[N_BUTTONS] = LOG_MSG_ASK_IDS;
|
|
static const wxString names[N_BUTTONS] = LOG_MSG_ASK_NAMES;
|
|
for (int i=0; i<N_BUTTONS; i++) {
|
|
if (!enabled[i]) continue;
|
|
wxButton *btn = new wxButton(this, ids[i], names[i]);
|
|
btnSizer->Add(btn, 1, wxALL, 5);
|
|
}
|
|
SetAutoLayout(TRUE);
|
|
SetSizer(vertSizer);
|
|
vertSizer->Fit(this);
|
|
wxSize size = vertSizer->GetMinSize();
|
|
int margin = 10;
|
|
SetSizeHints (size.GetWidth () + margin, size.GetHeight () + margin);
|
|
Center ();
|
|
}
|
|
|
|
// Event handler for dialog buttons. Just translate the wx ids into
|
|
// enum values and return them with EndModal() to make the dialog
|
|
// go away.
|
|
void LogMsgAskDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
int ret = -1;
|
|
switch (id) {
|
|
case ID_Continue: ret = BX_LOG_ASK_CHOICE_CONTINUE; break;
|
|
case ID_Die: ret = BX_LOG_ASK_CHOICE_DIE; break;
|
|
case ID_DumpCore: ret = BX_LOG_ASK_CHOICE_DUMP_CORE; break;
|
|
case ID_Debugger: ret = BX_LOG_ASK_CHOICE_ENTER_DEBUG; break;
|
|
case wxID_HELP: ShowHelp(); return;
|
|
default:
|
|
return; // without EndModal
|
|
}
|
|
EndModal(ret);
|
|
}
|
|
|
|
void LogMsgAskDialog::ShowHelp()
|
|
{
|
|
wxMessageBox(MSG_NO_HELP, MSG_NO_HELP_CAPTION, wxOK | wxICON_ERROR, this);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// AdvancedLogOptionsDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Structure:
|
|
// vertSizer:
|
|
// logfileSizer
|
|
// prompt
|
|
// logfile
|
|
// browse button
|
|
// prompt (multiline)
|
|
// applyDefault button
|
|
// scrollWin
|
|
// scrollpanel
|
|
// gridSizer 5 columns
|
|
// device
|
|
// debug
|
|
// info
|
|
// error
|
|
// panic
|
|
// etc.
|
|
// buttonSizer:
|
|
// help
|
|
// cancel
|
|
// ok
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(AdvancedLogOptionsDialog, wxDialog)
|
|
EVT_BUTTON(-1, AdvancedLogOptionsDialog::OnEvent)
|
|
EVT_CHECKBOX(-1, AdvancedLogOptionsDialog::OnEvent)
|
|
EVT_TEXT(-1, AdvancedLogOptionsDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
AdvancedLogOptionsDialog::AdvancedLogOptionsDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
|
|
{
|
|
static wxString names[] = ADVLOG_OPTS_TYPE_NAMES;
|
|
SetTitle(ADVLOG_OPTS_TITLE);
|
|
vertSizer = new wxBoxSizer(wxVERTICAL);
|
|
// top level objects
|
|
logfileSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
vertSizer->Add(logfileSizer, 0, wxTOP|wxLEFT, 20);
|
|
wxStaticText *text = new wxStaticText(this, -1, ADVLOG_OPTS_PROMPT);
|
|
vertSizer->Add(text, 0, wxALL, 10);
|
|
applyDefault = new wxButton(this, ID_ApplyDefault, ADVLOG_DEFAULTS);
|
|
vertSizer->Add(applyDefault, 0, wxALL|wxALIGN_RIGHT, 10);
|
|
headerSizer = new wxGridSizer(ADVLOG_OPTS_N_TYPES + 1);
|
|
vertSizer->Add(headerSizer, 0, wxALL|wxGROW, 10);
|
|
scrollWin = new wxScrolledWindow(this, -1);
|
|
vertSizer->Add(scrollWin, 1, wxALL|wxGROW, 10);
|
|
buttonSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
vertSizer->Add(buttonSizer, 0, wxALIGN_RIGHT);
|
|
|
|
// logfileSizer contents
|
|
text = new wxStaticText(this, -1, ADVLOG_OPTS_LOGFILE);
|
|
logfileSizer->Add(text, 0, wxALL | wxALIGN_CENTER_VERTICAL, 5);
|
|
logfile = new wxTextCtrl(this, -1, wxT(""), wxDefaultPosition, longTextSize);
|
|
logfileSizer->Add(logfile, 0, wxALL | wxALIGN_CENTER_VERTICAL);
|
|
wxButton *btn = new wxButton(this, ID_Browse, BTNLABEL_BROWSE);
|
|
logfileSizer->Add(btn, 0, wxALL, 5);
|
|
if (runtime) {
|
|
text->Enable(false);
|
|
logfile->Enable(false);
|
|
btn->Enable(false);
|
|
}
|
|
|
|
// to get the scrollWin geometry right, first build everything on a wxPanel,
|
|
// with gridSizer as the main sizer.
|
|
scrollPanel = new wxPanel(scrollWin, -1);
|
|
gridSizer = new wxGridSizer(ADVLOG_OPTS_N_TYPES + 1);
|
|
// add title row
|
|
int typemax = ADVLOG_OPTS_N_TYPES;
|
|
text = new wxStaticText(this, -1, wxT("Device"));
|
|
headerSizer->Add(text, 0, wxALIGN_LEFT);
|
|
int type;
|
|
for (type=0; type < typemax; type++) {
|
|
text = new wxStaticText(this, -1, names[type]);
|
|
headerSizer->Add(text, 0, wxALIGN_LEFT);
|
|
}
|
|
// add rows of choice boxes, one for each device
|
|
int devmax = SIM->get_n_log_modules();
|
|
action = new wxChoice** [devmax]; // array of pointers
|
|
for (int dev=0; dev<devmax; dev++) {
|
|
if (strcmp(SIM->get_logfn_name(dev), "?")) {
|
|
action[dev] = new wxChoice* [ADVLOG_OPTS_N_TYPES];
|
|
// name of device in first column
|
|
gridSizer->Add(new wxStaticText(scrollPanel, -1, wxString(SIM->get_logfn_name(dev), wxConvUTF8)),
|
|
0, wxALL | wxALIGN_CENTER_VERTICAL);
|
|
// wxChoice in every other column
|
|
for (type=0; type < typemax; type++) {
|
|
action[dev][type] = makeLogOptionChoiceBox(scrollPanel, -1, type);
|
|
gridSizer->Add(action[dev][type], 1, wxALL|wxGROW|wxADJUST_MINSIZE, 2);
|
|
}
|
|
} else {
|
|
action[dev] = NULL;
|
|
}
|
|
}
|
|
headerSizer->Fit(this);
|
|
headerSizer->SetSizeHints(this);
|
|
scrollPanel->SetAutoLayout(TRUE);
|
|
scrollPanel->SetSizer(gridSizer);
|
|
gridSizer->Fit(scrollPanel);
|
|
gridSizer->SetSizeHints(scrollPanel);
|
|
wxSize size = scrollPanel->GetBestSize();
|
|
// now we know how big the panel wants to be, and we can set the scrollbar
|
|
// and scrollWin size accordingly
|
|
|
|
// finally set up the scroll window outside
|
|
scrollWin->SetScrollbars(1, 1, size.GetWidth(), size.GetHeight());
|
|
|
|
// now that we know the desired width of the panel, use it to set the
|
|
// width of the scrollWin. I tried several things before arriving at
|
|
// a solution, and I'll list them for educational purposes.
|
|
//
|
|
// failure #1: this had no effect at all. sizer took precedence.
|
|
// scrollWin->SetSize(500, 500);
|
|
// failure #2: this changed scrollWin size but sizer was not notified so
|
|
// the overall window didn't expand to make space for it.
|
|
// scrollWin->SetSizeHints(500, 500);
|
|
// success: tell the sizer to change the scrollWin's size, and it works
|
|
vertSizer->SetItemMinSize(scrollWin, size.GetWidth()+30, 400);
|
|
|
|
// buttonSizer contents
|
|
btn = new wxButton(this, wxID_HELP, BTNLABEL_HELP);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
// use wxID_CANCEL because pressing ESC produces this same code
|
|
btn = new wxButton(this, wxID_CANCEL, BTNLABEL_CANCEL);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
btn = new wxButton(this, wxID_OK, BTNLABEL_OK);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
|
|
runtime = 0;
|
|
}
|
|
|
|
AdvancedLogOptionsDialog::~AdvancedLogOptionsDialog()
|
|
{
|
|
int dev, ndev = SIM->get_n_log_modules();
|
|
for (dev=0; dev<ndev; dev++) {
|
|
delete [] action[dev];
|
|
}
|
|
delete [] action;
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::Init()
|
|
{
|
|
CopyParamToGui();
|
|
// lay it out!
|
|
SetAutoLayout(TRUE);
|
|
SetSizer(vertSizer);
|
|
vertSizer->Fit(this);
|
|
wxSize size = vertSizer->GetMinSize();
|
|
int margin = 5;
|
|
SetSizeHints(size.GetWidth() + margin, size.GetHeight() + margin);
|
|
Center();
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::CopyParamToGui()
|
|
{
|
|
bx_param_string_c *logfile = SIM->get_param_string(BXPN_LOG_FILENAME);
|
|
SetLogfile(wxString(logfile->getptr(), wxConvUTF8));
|
|
// copy log action settings from siminterface to gui
|
|
int dev, ndev = SIM->get_n_log_modules();
|
|
int type, ntype = SIM->get_max_log_level();
|
|
for (dev=0; dev<ndev; dev++) {
|
|
for (type=0; type<ntype; type++) {
|
|
SetAction(dev, type, SIM->get_log_action(dev, type));
|
|
}
|
|
}
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::CopyGuiToParam()
|
|
{
|
|
char buf[1024];
|
|
safeWxStrcpy(buf, GetLogfile(), sizeof(buf));
|
|
bx_param_string_c *logfile = SIM->get_param_string(BXPN_LOG_FILENAME);
|
|
logfile->set(buf);
|
|
// copy log action settings from gui to siminterface
|
|
int dev, ndev = SIM->get_n_log_modules();
|
|
int type, ntype = SIM->get_max_log_level();
|
|
for (dev=0; dev<ndev; dev++) {
|
|
for (type=0; type<ntype; type++) {
|
|
SIM->set_log_action(dev, type, GetAction(dev, type));
|
|
}
|
|
}
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::SetAction(int dev, int evtype, int act)
|
|
{
|
|
if (action[dev] == NULL) return;
|
|
// find the choice whose client data matches "act".
|
|
int *ptr;
|
|
// wxLogDebug(wxT("SetAction dev=%d type=%d act=%d"), dev, evtype, act);
|
|
wxChoice *control = action[dev][evtype];
|
|
for (int i=0; i < (int)control->GetCount(); i++) {
|
|
// wxLogDebug(wxT("reading action[%d][%d]->GetClientData(%d)"), dev, evtype, i);
|
|
ptr = (int*) control->GetClientData(i);
|
|
if (ptr == NULL) continue;
|
|
if (act == *ptr) { // found it!
|
|
control->SetSelection(i);
|
|
return;
|
|
}
|
|
}
|
|
// this can happen if one of the choices that is excluded by
|
|
// BX_LOG_OPTS_EXCLUDE() is used, for example.
|
|
wxLogDebug(wxT("warning: SetAction type=%d act=%d not found"), evtype, act);
|
|
}
|
|
|
|
int AdvancedLogOptionsDialog::GetAction(int dev, int evtype)
|
|
{
|
|
if (action[dev] == NULL) return LOG_OPTS_NO_CHANGE;
|
|
int sel = action[dev][evtype]->GetSelection();
|
|
int *ptrToChoice = (int*)action[dev][evtype]->GetClientData(sel);
|
|
wxASSERT(ptrToChoice != NULL);
|
|
return *ptrToChoice;
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
switch (id) {
|
|
case ID_Browse:
|
|
BrowseTextCtrl(logfile);
|
|
break;
|
|
case ID_ApplyDefault: {
|
|
int lev, nlev = SIM->get_max_log_level();
|
|
// copy default settings to every device
|
|
for (lev=0; lev<nlev; lev++) {
|
|
int action = SIM->get_default_log_action(lev);
|
|
int dev, ndev = SIM->get_n_log_modules();
|
|
for (dev=0; dev<ndev; dev++)
|
|
SetAction(dev, lev, action);
|
|
}
|
|
break;
|
|
}
|
|
case wxID_OK:
|
|
CopyGuiToParam();
|
|
EndModal(wxID_OK);
|
|
break;
|
|
case wxID_CANCEL:
|
|
EndModal(wxID_CANCEL);
|
|
break;
|
|
case wxID_HELP:
|
|
ShowHelp();
|
|
break;
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
void AdvancedLogOptionsDialog::ShowHelp()
|
|
{
|
|
wxMessageBox(MSG_NO_HELP, MSG_NO_HELP_CAPTION, wxOK | wxICON_ERROR, this);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// PluginControlDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(PluginControlDialog, wxDialog)
|
|
EVT_BUTTON(-1, PluginControlDialog::OnEvent)
|
|
EVT_CHECKBOX(-1, PluginControlDialog::OnEvent)
|
|
EVT_TEXT(-1, PluginControlDialog::OnEvent)
|
|
EVT_LISTBOX(-1, PluginControlDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
PluginControlDialog::PluginControlDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE)
|
|
{
|
|
SetTitle(wxT("Optional Plugin Control"));
|
|
vertSizer = new wxBoxSizer(wxVERTICAL);
|
|
horzSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
leftSizer = new wxBoxSizer(wxVERTICAL);
|
|
centerSizer = new wxBoxSizer(wxVERTICAL);
|
|
rightSizer = new wxBoxSizer(wxVERTICAL);
|
|
buttonSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
horzSizer->Add(leftSizer, 0, wxALIGN_LEFT);
|
|
horzSizer->Add(centerSizer, 0, wxALIGN_CENTER);
|
|
horzSizer->Add(rightSizer, 0, wxALIGN_RIGHT);
|
|
vertSizer->Add(horzSizer, 0, wxALIGN_CENTER);
|
|
vertSizer->Add(buttonSizer, 0, wxALIGN_CENTER);
|
|
// leftSizer contents
|
|
plugtxt1 = new wxStaticText(this, -1, wxT("Available"));
|
|
pluglist1 = new wxListBox(this, ID_PluginList1, wxDefaultPosition, wxSize(120, 200));
|
|
leftSizer->Add(plugtxt1, 0, wxALL | wxALIGN_CENTER, 10);
|
|
leftSizer->Add(pluglist1, 0, wxALL, 10);
|
|
// rightSizer contents
|
|
plugtxt2 = new wxStaticText(this, -1, wxT("Loaded"));
|
|
pluglist2 = new wxListBox(this, ID_PluginList2, wxDefaultPosition, wxSize(120, 200));
|
|
rightSizer->Add(plugtxt2, 0, wxALL | wxALIGN_CENTER, 10);
|
|
rightSizer->Add(pluglist2, 0, wxALL, 10);
|
|
// centerSizer contents
|
|
btn_load = new wxButton(this, ID_Load, wxT(">> Load >>"));
|
|
centerSizer->Add(btn_load, 0, wxALL | wxALIGN_RIGHT, 5);
|
|
btn_unload = new wxButton(this, ID_Unload, wxT("<< Unload <<"));
|
|
centerSizer->Add(btn_unload, 0, wxALL | wxALIGN_RIGHT, 5);
|
|
// buttonSizer contents
|
|
wxButton *btn = new wxButton(this, wxID_HELP, BTNLABEL_HELP);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
btn = new wxButton(this, wxID_OK, BTNLABEL_OK);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
// add plugin names to the listboxes
|
|
bx_list_c *plugin_ctrl = (bx_list_c*) SIM->get_param(BXPN_PLUGIN_CTRL);
|
|
int a = 0, b = 0;
|
|
for (int i = 0; i < plugin_ctrl->get_size(); i++) {
|
|
bx_param_bool_c *plugin = (bx_param_bool_c*)plugin_ctrl->get(i);
|
|
if (plugin->get()) {
|
|
pluglist2->Insert(wxString(plugin->get_name(), wxConvUTF8), a++);
|
|
} else {
|
|
pluglist1->Insert(wxString(plugin->get_name(), wxConvUTF8), b++);
|
|
}
|
|
}
|
|
btn_load->Enable(0);
|
|
btn_unload->Enable(0);
|
|
}
|
|
|
|
void PluginControlDialog::Init()
|
|
{
|
|
SetSizer(vertSizer);
|
|
vertSizer->Fit(this);
|
|
wxSize size = vertSizer->GetMinSize();
|
|
int margin = 5;
|
|
SetSizeHints(size.GetWidth() + margin, size.GetHeight() + margin);
|
|
Center();
|
|
}
|
|
|
|
void PluginControlDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
char buf[1024];
|
|
int i;
|
|
|
|
int id = event.GetId();
|
|
switch (id) {
|
|
case ID_PluginList1:
|
|
if (event.GetEventType() == wxEVT_COMMAND_LISTBOX_SELECTED) {
|
|
pluglist2->SetSelection(-1);
|
|
btn_load->Enable(1);
|
|
btn_unload->Enable(0);
|
|
}
|
|
break;
|
|
case ID_PluginList2:
|
|
if (event.GetEventType() == wxEVT_COMMAND_LISTBOX_SELECTED) {
|
|
pluglist1->SetSelection(-1);
|
|
btn_load->Enable(0);
|
|
btn_unload->Enable(1);
|
|
}
|
|
break;
|
|
case ID_Load:
|
|
{
|
|
i = pluglist1->GetSelection();
|
|
wxString tmpname = pluglist1->GetString(i);
|
|
strncpy(buf, tmpname.mb_str(wxConvUTF8), sizeof(buf) - 1);
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
if (SIM->opt_plugin_ctrl(buf, 1)) {
|
|
tmpname.Printf(wxT("Plugin '%s' loaded"), buf);
|
|
wxMessageBox(tmpname, wxT("Plugin Control"), wxOK | wxICON_INFORMATION, this);
|
|
pluglist1->Delete(i);
|
|
pluglist2->Insert(wxString(buf, wxConvUTF8), pluglist2->GetCount());
|
|
pluglist1->SetSelection(-1);
|
|
btn_load->Enable(0);
|
|
}
|
|
}
|
|
break;
|
|
case ID_Unload:
|
|
{
|
|
i = pluglist2->GetSelection();
|
|
wxString tmpname = pluglist2->GetString(i);
|
|
strncpy(buf, tmpname.mb_str(wxConvUTF8), sizeof(buf) - 1);
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
if (SIM->opt_plugin_ctrl(buf, 0)) {
|
|
tmpname.Printf(wxT("Plugin '%s' unloaded"), buf);
|
|
wxMessageBox(tmpname, wxT("Plugin Control"), wxOK | wxICON_INFORMATION, this);
|
|
pluglist1->Insert(wxString(buf, wxConvUTF8), pluglist1->GetCount());
|
|
pluglist2->Delete(i);
|
|
pluglist2->SetSelection(-1);
|
|
btn_unload->Enable(0);
|
|
}
|
|
}
|
|
break;
|
|
case wxID_OK:
|
|
EndModal(wxID_OK);
|
|
break;
|
|
case wxID_HELP:
|
|
ShowHelp();
|
|
break;
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
void PluginControlDialog::ShowHelp()
|
|
{
|
|
wxMessageBox(MSG_NO_HELP, MSG_NO_HELP_CAPTION, wxOK | wxICON_ERROR, this);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// LogViewDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(LogViewDialog, wxDialog)
|
|
EVT_BUTTON(-1, LogViewDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
LogViewDialog::LogViewDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE)
|
|
{
|
|
lengthMax = LOG_VIEW_DEFAULT_LENGTH_MAX;
|
|
lengthTolerance = LOG_VIEW_DEFAULT_TOLERANCE;
|
|
SetTitle(wxT("Bochs Log Viewer"));
|
|
mainSizer = new wxBoxSizer(wxVERTICAL);
|
|
logSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
buttonSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
mainSizer->Add(logSizer, 0, wxALIGN_CENTER);
|
|
mainSizer->Add(buttonSizer, 0, wxALIGN_CENTER);
|
|
log = new wxTextCtrl(this, -1, wxT(""),
|
|
wxDefaultPosition, wxSize(575, 300),
|
|
wxTE_MULTILINE | wxTE_RICH | wxTE_READONLY);
|
|
wxFont font(8, wxFONTFAMILY_TELETYPE, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL);
|
|
wxTextAttr attr;
|
|
attr.SetFont(font);
|
|
log->SetDefaultStyle(attr);
|
|
logSizer->Add(log, 1, wxALL|wxGROW, 10);
|
|
// buttonSizer contents
|
|
wxButton *btn = new wxButton(this, wxID_OK, BTNLABEL_CLOSE);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
}
|
|
|
|
void LogViewDialog::Init()
|
|
{
|
|
SetSizer(mainSizer);
|
|
mainSizer->Fit(this);
|
|
wxSize size = mainSizer->GetMinSize();
|
|
int margin = 5;
|
|
SetSizeHints(size.GetWidth() + margin, size.GetHeight() + margin);
|
|
Center();
|
|
}
|
|
|
|
bool LogViewDialog::Show(bool val)
|
|
{
|
|
SIM->set_log_viewer(val);
|
|
if (val) wxDialog::Raise();
|
|
return wxDialog::Show(val);
|
|
}
|
|
|
|
void LogViewDialog::CheckLogLength()
|
|
{
|
|
// truncate the text control periodically to avoid a
|
|
// serious memory leak.
|
|
wxString str = log->GetValue();
|
|
Bit32u len = str.Length();
|
|
if (len > lengthMax + lengthTolerance) {
|
|
// Truncate the string. Start from length - lengthMax, search
|
|
// forward until we find the first \n.
|
|
for (Bit32u i = len - lengthMax; i<len-1; i++) {
|
|
if (str.GetChar(i) == '\n') {
|
|
// remove the \n and everything before it.
|
|
log->Remove(0, i+1);
|
|
return;
|
|
}
|
|
}
|
|
// no newline found?!
|
|
log->Remove(0, len - lengthMax);
|
|
}
|
|
}
|
|
|
|
void LogViewDialog::AppendText(int level, wxString msg)
|
|
{
|
|
if ((level == LOGLEV_ERROR) || (level == LOGLEV_PANIC)) {
|
|
log->SetDefaultStyle(wxTextAttr(*wxRED));
|
|
} else {
|
|
log->SetDefaultStyle(wxTextAttr(*wxBLACK));
|
|
}
|
|
log->AppendText(msg);
|
|
int n = log->GetLastPosition();
|
|
if (n>0) n--;
|
|
log->ShowPosition(n);
|
|
CheckLogLength();
|
|
}
|
|
|
|
void LogViewDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
switch (id) {
|
|
case wxID_OK:
|
|
Show(false);
|
|
break;
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////
|
|
// ParamDialog
|
|
/////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(ParamDialog, wxDialog)
|
|
EVT_BUTTON(-1, ParamDialog::OnEvent)
|
|
EVT_CHECKBOX(-1, ParamDialog::OnEvent)
|
|
EVT_CHOICE(-1, ParamDialog::OnEvent)
|
|
EVT_TEXT(-1, ParamDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
ParamDialog::ParamDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
|
|
{
|
|
idHash = new wxHashTable(wxKEY_INTEGER);
|
|
paramHash = new wxHashTable(wxKEY_INTEGER);
|
|
nbuttons = 0;
|
|
runtime = 0;
|
|
|
|
// top level objects
|
|
mainSizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
// info sizer only used in floppy and log options dialog
|
|
infoSizer = NULL;
|
|
|
|
// create buttonSizer, which will hold all the buttons.
|
|
buttonSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
}
|
|
|
|
ParamDialog::~ParamDialog()
|
|
{
|
|
paramHash->BeginFind();
|
|
wxHashTable::Node *node;
|
|
while ((node = paramHash->Next()) != NULL) {
|
|
// assume that no ParamStruct appears in the hash table under multiple
|
|
// keys. If so, we will delete it twice and corrupt memory.
|
|
ParamStruct *pstr = (ParamStruct*) node->GetData();
|
|
// wxLogDebug(wxT("deleting ParamStruct id=%d for param %s"), pstr->id, pstr->param->get_name());
|
|
delete pstr;
|
|
}
|
|
delete idHash;
|
|
delete paramHash;
|
|
}
|
|
|
|
wxButton* ParamDialog::AddButton(int id, wxString label)
|
|
{
|
|
wxButton *btn = new wxButton(this, id, label);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
nbuttons++;
|
|
return btn;
|
|
}
|
|
|
|
// add the standard HELP, CANCEL, OK buttons.
|
|
void ParamDialog::AddDefaultButtons()
|
|
{
|
|
AddButton(wxID_HELP, BTNLABEL_HELP);
|
|
AddButton(wxID_CANCEL, BTNLABEL_CANCEL);
|
|
AddButton(wxID_OK, BTNLABEL_OK);
|
|
}
|
|
|
|
void ParamDialog::Init()
|
|
{
|
|
// add info sizer if present
|
|
if (infoSizer != NULL) {
|
|
mainSizer->Add(infoSizer, 0, wxALIGN_CENTER);
|
|
}
|
|
// if nobody has made any buttons, then create some now
|
|
if (nbuttons==0) AddDefaultButtons();
|
|
mainSizer->Add(buttonSizer, 0, wxALIGN_RIGHT);
|
|
EnableChanged();
|
|
// lay it out!
|
|
SetAutoLayout(TRUE);
|
|
SetSizer(mainSizer);
|
|
mainSizer->Fit(this);
|
|
wxSize size = mainSizer->GetMinSize();
|
|
int margin = 5;
|
|
SetSizeHints(size.GetWidth() + margin, size.GetHeight() + margin);
|
|
Center();
|
|
}
|
|
|
|
static int _next_id = ID_LAST_USER_DEFINED;
|
|
|
|
int ParamDialog::genId()
|
|
{
|
|
return ++_next_id;
|
|
}
|
|
|
|
bool ParamDialog::isGeneratedId(int id)
|
|
{
|
|
return (id >= ID_LAST_USER_DEFINED && id <= _next_id);
|
|
}
|
|
|
|
void ParamDialog::AddParamList(const char *nameList[], bx_param_c *base, wxFlexGridSizer *sizer, bool plain)
|
|
{
|
|
int i = 0;
|
|
while (nameList[i] != NULL) {
|
|
bx_param_c *param = SIM->get_param(nameList[i], base);
|
|
if (param != NULL) {
|
|
AddParam(param, sizer, plain);
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// support "legacy" addparam functions. Easier than changing them.
|
|
void ParamDialog::AddParam(bx_param_c *param, wxFlexGridSizer *sizer, bool plain)
|
|
{
|
|
AddParamContext context;
|
|
context.depth = 0;
|
|
context.parent = this;
|
|
context.vertSizer = mainSizer;
|
|
context.gridSizer = sizer;
|
|
AddParam(param, plain, &context);
|
|
}
|
|
|
|
void ParamDialog::AddParam(
|
|
bx_param_c *param_generic,
|
|
bool plain,
|
|
AddParamContext *context)
|
|
{
|
|
AddParamContext defaultContext;
|
|
if (context == NULL) {
|
|
context = &defaultContext;
|
|
context->depth = 0;
|
|
context->parent = this;
|
|
context->vertSizer = mainSizer;
|
|
context->gridSizer = NULL;
|
|
}
|
|
wxASSERT(context->parent != NULL);
|
|
wxASSERT(context->vertSizer != NULL);
|
|
if (param_generic == NULL)
|
|
return; // param not registered, probably this option was not compiled in
|
|
wxLogDebug(wxT("AddParam for param '%s'"), param_generic->get_name());
|
|
if (context->gridSizer == NULL) {
|
|
// create a gridSizer if none exists yet. add it to default vertSizer.
|
|
context->gridSizer = new wxFlexGridSizer(3);
|
|
context->vertSizer->Add(context->gridSizer);
|
|
}
|
|
wxFlexGridSizer *sizer = context->gridSizer;
|
|
|
|
ParamStruct *pstr = new ParamStruct();
|
|
pstr->param = param_generic;
|
|
pstr->id = genId();
|
|
pstr->label = NULL;
|
|
pstr->u.window = NULL;
|
|
pstr->browseButton = NULL;
|
|
int type = param_generic->get_type();
|
|
const char *prompt;
|
|
if (type == BXT_LIST) {
|
|
bx_list_c *list = (bx_list_c*)pstr->param;
|
|
prompt = list->get_title();
|
|
} else {
|
|
prompt = pstr->param->get_label();
|
|
}
|
|
if (!prompt) prompt = pstr->param->get_name();
|
|
const char *description = pstr->param->get_description();
|
|
wxASSERT(prompt != NULL);
|
|
#define ADD_LABEL(x) sizer->Add(pstr->label = new wxStaticText(context->parent, -1, wxString(x, wxConvUTF8)), 0, wxALIGN_RIGHT|wxALIGN_CENTER_VERTICAL|wxALL, 3)
|
|
switch (type) {
|
|
case BXT_PARAM_BOOL:
|
|
{
|
|
bx_param_bool_c *param = (bx_param_bool_c*) param_generic;
|
|
if (!plain) ADD_LABEL(prompt);
|
|
wxCheckBox *ckbx = new wxCheckBox(context->parent, pstr->id, wxT(""));
|
|
ckbx->SetValue(param->get());
|
|
if (description) ckbx->SetToolTip(wxString(description, wxConvUTF8));
|
|
sizer->Add(ckbx, 0, wxALL, 2);
|
|
if (!plain) sizer->Add(1, 1); // spacer
|
|
pstr->u.checkbox = ckbx;
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
break;
|
|
}
|
|
case BXT_PARAM_NUM:
|
|
{
|
|
bx_param_num_c *param = (bx_param_num_c*) param_generic;
|
|
if (!plain) ADD_LABEL(prompt);
|
|
if (param->get_options() & param->USE_SPIN_CONTROL) {
|
|
wxSpinCtrl *spinctrl = new wxSpinCtrl(context->parent, pstr->id);
|
|
spinctrl->SetValue(param->get());
|
|
int max = (param->get_max() < (1<<24))?param->get_max():(1<<24)-1;
|
|
spinctrl->SetRange(param->get_min(), SPINCTRL_FIX_MAX(max));
|
|
if (description) spinctrl->SetToolTip(wxString(description, wxConvUTF8));
|
|
sizer->Add(spinctrl, 0, wxALL, 2);
|
|
if (!plain) sizer->Add(1, 1); // spacer
|
|
pstr->u.spin = spinctrl;
|
|
} else {
|
|
wxTextCtrl *textctrl = new wxTextCtrl(context->parent, pstr->id, wxT(""), wxDefaultPosition, normalTextSize);
|
|
const char *format = param->get_format();
|
|
if (!format)
|
|
format = strdup(param->get_base() == 16 ? "0x%X" : "%d");
|
|
SetTextCtrl(textctrl, format, param->get());
|
|
if (description) textctrl->SetToolTip(wxString(description, wxConvUTF8));
|
|
sizer->Add(textctrl, 0, wxALL, 2);
|
|
if (!plain) sizer->Add(1, 1); // spacer
|
|
pstr->u.text = textctrl;
|
|
}
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
break;
|
|
}
|
|
case BXT_PARAM_ENUM:
|
|
{
|
|
bx_param_enum_c *param = (bx_param_enum_c*) param_generic;
|
|
if (!plain) ADD_LABEL(prompt);
|
|
wxChoice *choice = new wxChoice(context->parent, pstr->id);
|
|
if (description) choice->SetToolTip(wxString(description, wxConvUTF8));
|
|
sizer->Add(choice, 0, wxADJUST_MINSIZE, 2);
|
|
if (!plain) sizer->Add(1, 1); // spacer
|
|
// fill in the choices
|
|
int i=0;
|
|
const char *ptr;
|
|
while (NULL != (ptr = param->get_choice(i++)))
|
|
choice->Append(wxString(ptr, wxConvUTF8));
|
|
choice->SetSelection(param->get() - param->get_min());
|
|
pstr->u.choice = choice;
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
break;
|
|
}
|
|
case BXT_PARAM_STRING:
|
|
case BXT_PARAM_BYTESTRING:
|
|
{
|
|
bx_param_string_c *param = (bx_param_string_c*) param_generic;
|
|
char value[1024];
|
|
if (!plain) ADD_LABEL(prompt);
|
|
bool isFilename = param->get_options() & param->IS_FILENAME;
|
|
wxTextCtrl *txtctrl = new wxTextCtrl(context->parent, pstr->id, wxT(""), wxDefaultPosition, isFilename? longTextSize : normalTextSize);
|
|
if (description) txtctrl->SetToolTip(wxString(description, wxConvUTF8));
|
|
param->dump_param(value, 1024);
|
|
txtctrl->SetValue(wxString(value, wxConvUTF8));
|
|
if (type != BXT_PARAM_BYTESTRING) {
|
|
txtctrl->SetMaxLength(param->get_maxsize());
|
|
}
|
|
sizer->Add(txtctrl, 0, wxALL, 2);
|
|
if (!plain) {
|
|
if (isFilename) {
|
|
// create Browse button
|
|
pstr->browseButtonId = genId();
|
|
pstr->browseButton = new wxButton(context->parent,
|
|
pstr->browseButtonId, BTNLABEL_BROWSE);
|
|
sizer->Add(pstr->browseButton, 0, wxALL, 2);
|
|
idHash->Put(pstr->browseButtonId, pstr); // register under button id
|
|
} else {
|
|
sizer->Add(1, 1); // spacer
|
|
}
|
|
}
|
|
pstr->u.text = txtctrl;
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
break;
|
|
}
|
|
case BXT_LIST:
|
|
{
|
|
bx_list_c *list = (bx_list_c*) param_generic;
|
|
if (list->get_options() & bx_list_c::USE_TAB_WINDOW) {
|
|
// put each item in a separate tab of a tabbed window
|
|
wxNotebook *notebook = new wxNotebook(context->parent, -1);
|
|
#if wxMAJOR_VERSION == 2 && wxMINOR_VERSION < 6
|
|
wxNotebookSizer *nbsizer = new wxNotebookSizer(notebook);
|
|
#endif
|
|
// put all items in the list into a separate page of the notebook.
|
|
for (int i=0; i<list->get_size(); i++) {
|
|
bx_list_c *child = (bx_list_c*)list->get(i);
|
|
wxASSERT(child->get_type() == BXT_LIST);
|
|
// the child must be a list! I could support other things but
|
|
// I don't see any reason to. It wouldn't make sense to devote
|
|
// a whole tab to a single parameter.
|
|
wxPanel *panel = new wxPanel(notebook);
|
|
wxBoxSizer *boxsz = new wxBoxSizer(wxVERTICAL);
|
|
AddParamContext newcontext;
|
|
newcontext.depth = 1 + context->depth;
|
|
newcontext.parent = panel;
|
|
newcontext.vertSizer = boxsz;
|
|
newcontext.gridSizer = NULL; // will be created if needed
|
|
// the child itself is a list. Add the child's children in
|
|
// this new context.
|
|
bx_list_c *childl = (bx_list_c *)child;
|
|
for (int j=0; j<childl->get_size(); j++)
|
|
AddParam(childl->get(j), plain, &newcontext);
|
|
const char *pagename = child->get_title();
|
|
if (!pagename) pagename = child->get_name();
|
|
panel->SetAutoLayout(TRUE);
|
|
panel->SetSizer(boxsz);
|
|
notebook->AddPage(panel, wxString(pagename, wxConvUTF8));
|
|
}
|
|
#if wxMAJOR_VERSION == 2 && wxMINOR_VERSION < 6
|
|
context->vertSizer->Add(nbsizer, 0, wxALL|wxGROW, 10);
|
|
#else
|
|
context->vertSizer->Add(notebook, 0, wxALL|wxGROW, 10);
|
|
#endif
|
|
// clear gridSizer variable so that any future parameters force
|
|
// creation of a new one.
|
|
context->gridSizer = NULL;
|
|
// add to hashes
|
|
pstr->u.notebook = notebook;
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
} else {
|
|
wxString boxTitle;
|
|
wxScrolledWindow *scrollWin = NULL;
|
|
wxPanel *scrollPanel = NULL;
|
|
wxBoxSizer *scrollsz = NULL;
|
|
wxStaticBox *box = NULL;
|
|
wxStaticBoxSizer *boxsz = NULL;
|
|
if (list->get_options() & bx_list_c::USE_BOX_TITLE) {
|
|
boxTitle = wxString(prompt, wxConvUTF8);
|
|
} else {
|
|
boxTitle = wxT("");
|
|
}
|
|
AddParamContext newcontext;
|
|
newcontext.depth = 1 + context->depth;
|
|
newcontext.gridSizer = NULL; // it will be created if necessary
|
|
if (list->get_options() & bx_list_c::USE_SCROLL_WINDOW) {
|
|
scrollWin = new wxScrolledWindow(context->parent, -1);
|
|
scrollPanel = new wxPanel(scrollWin, -1);
|
|
scrollsz = new wxBoxSizer(wxVERTICAL);
|
|
newcontext.parent = scrollPanel;
|
|
newcontext.vertSizer = scrollsz;
|
|
} else {
|
|
box = new wxStaticBox(context->parent, -1, boxTitle);
|
|
boxsz = new wxStaticBoxSizer(box, wxVERTICAL);
|
|
newcontext.parent = context->parent;
|
|
newcontext.vertSizer = boxsz;
|
|
}
|
|
// put all items in the list inside the boxsz sizer.
|
|
for (int i=0; i<list->get_size(); i++) {
|
|
bx_param_c *child = list->get(i);
|
|
AddParam(child, plain, &newcontext);
|
|
}
|
|
if (list->get_options() & bx_list_c::USE_SCROLL_WINDOW) {
|
|
scrollPanel->SetAutoLayout(TRUE);
|
|
scrollPanel->SetSizer(scrollsz);
|
|
scrollsz->Fit(scrollPanel);
|
|
scrollsz->SetSizeHints(scrollPanel);
|
|
wxSize size = scrollPanel->GetBestSize();
|
|
scrollWin->SetScrollbars(1, 1, size.GetWidth(), size.GetHeight());
|
|
context->vertSizer->Add(scrollWin, 0, wxALL|wxGROW, 10);
|
|
context->vertSizer->SetItemMinSize(scrollWin, size.GetWidth() + 30, 400);
|
|
} else {
|
|
// add the boxsz to vertSizer
|
|
context->vertSizer->Add(boxsz, 0, wxALL|wxGROW, 10);
|
|
}
|
|
// clear gridSizer variable so that any future parameters force
|
|
// creation of a new one.
|
|
context->gridSizer = NULL;
|
|
// add to hashes
|
|
pstr->u.staticbox = box;
|
|
idHash->Put(pstr->id, pstr);
|
|
paramHash->Put(pstr->param->get_id(), pstr);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
wxLogError(wxT("ParamDialog::AddParam called with unsupported param type id=%d"), (int)type);
|
|
}
|
|
if (pstr->label) pstr->label->Enable(pstr->param->get_enabled());
|
|
if (pstr->u.window) pstr->u.window->Enable(pstr->param->get_enabled());
|
|
if (pstr->browseButton) pstr->browseButton->Enable(pstr->param->get_enabled());
|
|
}
|
|
|
|
bool ParamDialog::CopyGuiToParam()
|
|
{
|
|
bool sim_running = 0;
|
|
|
|
if (runtime) {
|
|
sim_running = theFrame->SimThreadControl(0);
|
|
}
|
|
// loop through all the parameters
|
|
idHash->BeginFind();
|
|
wxHashTable::Node *node;
|
|
while ((node = idHash->Next()) != NULL) {
|
|
ParamStruct *pstr = (ParamStruct*) node->GetData();
|
|
wxLogDebug(wxT("commit changes for param %s"), pstr->param->get_name());
|
|
CopyGuiToParam(pstr->param);
|
|
if (pstr->param->get_type() == BXT_LIST) break;
|
|
}
|
|
if (runtime && sim_running) {
|
|
SIM->update_runtime_options();
|
|
theFrame->SimThreadControl(1);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool ParamDialog::CopyGuiToParam(bx_param_c *param)
|
|
{
|
|
int i;
|
|
|
|
if (param == NULL) return false;
|
|
ParamStruct *pstr = (ParamStruct*) paramHash->Get(param->get_id());
|
|
wxLogDebug(wxT("commit changes for param %s"), param->get_name());
|
|
int type = param->get_type();
|
|
switch (type) {
|
|
case BXT_PARAM_BOOL: {
|
|
bx_param_bool_c *boolp = (bx_param_bool_c*) pstr->param;
|
|
bool val = pstr->u.checkbox->GetValue();
|
|
if (val != boolp->get()) boolp->set(val);
|
|
break;
|
|
}
|
|
case BXT_PARAM_NUM: {
|
|
bx_param_num_c *nump = (bx_param_num_c*) pstr->param;
|
|
bool valid;
|
|
int n;
|
|
wxString complaint(wxT("Invalid integer for '"));
|
|
complaint += wxString(pstr->param->get_name(), wxConvUTF8);
|
|
complaint += wxT("'.");
|
|
if (nump->get_options() & nump->USE_SPIN_CONTROL) {
|
|
n = pstr->u.spin->GetValue();
|
|
} else {
|
|
n = GetTextCtrlInt(pstr->u.text, &valid, true, complaint);
|
|
}
|
|
if ((n < nump->get_min()) || (n > nump->get_max())) {
|
|
wxMessageBox(wxT("Numerical parameter out of range"), wxT("Error"), wxOK | wxICON_ERROR, this);
|
|
return false;
|
|
}
|
|
if (n != nump->get()) nump->set(n);
|
|
break;
|
|
}
|
|
case BXT_PARAM_ENUM: {
|
|
bx_param_enum_c *enump = (bx_param_enum_c*) pstr->param;
|
|
int value = pstr->u.choice->GetSelection() + enump->get_min();
|
|
if (value != enump->get()) enump->set(value);
|
|
break;
|
|
}
|
|
case BXT_PARAM_STRING: {
|
|
bx_param_string_c *stringp = (bx_param_string_c*) pstr->param;
|
|
char buf[1024];
|
|
wxString tmp(pstr->u.text->GetValue());
|
|
strncpy(buf, tmp.mb_str(wxConvUTF8), sizeof(buf) - 1);
|
|
buf[sizeof(buf)-1] = 0;
|
|
if (!stringp->equals(buf)) stringp->set(buf);
|
|
break;
|
|
}
|
|
case BXT_PARAM_BYTESTRING: {
|
|
bx_param_bytestring_c *stringp = (bx_param_bytestring_c*) pstr->param;
|
|
wxString tmp(pstr->u.text->GetValue());
|
|
char src[1024];
|
|
strcpy(src, tmp.mb_str(wxConvUTF8));
|
|
if (!stringp->parse_param(src)) {
|
|
wxMessageBox(wxT("Illegal raw byte format"), wxT("Error"), wxOK | wxICON_ERROR, this);
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
case BXT_LIST: {
|
|
bx_list_c *list = (bx_list_c*) param;
|
|
for (i = 0; i < list->get_size(); i++) {
|
|
bx_param_c *item = list->get(i);
|
|
if (!CopyGuiToParam(item)) break;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
wxLogError(wxT("ParamDialog::CopyGuiToParam: unsupported param type id=%d"), (int)type);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void ParamDialog::EnableChanged()
|
|
{
|
|
idHash->BeginFind();
|
|
wxHashTable::Node *node;
|
|
while ((node = idHash->Next()) != NULL) {
|
|
ParamStruct *pstr = (ParamStruct*) node->GetData();
|
|
if (runtime) {
|
|
if ((pstr->param->get_type() != BXT_LIST) && !pstr->param->get_runtime_param())
|
|
EnableParam(pstr->param->get_id(), false);
|
|
}
|
|
// special cases that can't be handled in the usual way
|
|
}
|
|
}
|
|
|
|
void ParamDialog::EnableChanged(ParamStruct *pstr)
|
|
{
|
|
wxLogDebug(wxT("EnableChanged on param %s"), pstr->param->get_name());
|
|
ProcessDependentList(pstr, true);
|
|
}
|
|
|
|
void ParamDialog::EnableParam(int param_id, bool enabled)
|
|
{
|
|
ParamStruct *pstr = (ParamStruct*) paramHash->Get(param_id);
|
|
if (!pstr) return;
|
|
if (pstr->label) pstr->label->Enable(enabled);
|
|
if (pstr->browseButton) pstr->browseButton->Enable(enabled);
|
|
if (pstr->u.window) pstr->u.window->Enable(enabled);
|
|
}
|
|
|
|
void ParamDialog::ProcessDependentList(ParamStruct *pstrChanged, bool enabled)
|
|
{
|
|
bx_param_c *dparam;
|
|
bx_param_string_c *sparam;
|
|
param_enable_handler enable_handler;
|
|
ParamStruct *pstr;
|
|
Bit64s value;
|
|
bool en;
|
|
int i;
|
|
|
|
bx_list_c *list = pstrChanged->param->get_dependent_list();
|
|
if (list) {
|
|
if (pstrChanged->param->get_type() == BXT_PARAM_ENUM) {
|
|
bx_param_enum_c *enump = (bx_param_enum_c*)pstrChanged->param;
|
|
value = pstrChanged->u.choice->GetSelection() + enump->get_min();
|
|
Bit64u enable_bitmap = enump->get_dependent_bitmap(value);
|
|
Bit64u mask = 0x1;
|
|
for (i = 0; i < list->get_size(); i++) {
|
|
dparam = list->get(i);
|
|
if (dparam != enump) {
|
|
en = (enable_bitmap & mask) && enabled;
|
|
if (dparam->get_type() == BXT_PARAM_STRING) {
|
|
sparam = (bx_param_string_c*)dparam;
|
|
enable_handler = sparam->get_enable_handler();
|
|
if (enable_handler) {
|
|
en = enable_handler(sparam, en);
|
|
}
|
|
}
|
|
pstr = (ParamStruct*) paramHash->Get(dparam->get_id());
|
|
if (pstr) {
|
|
if (en != pstr->u.window->IsEnabled()) {
|
|
EnableParam(dparam->get_id(), en);
|
|
ProcessDependentList(pstr, en);
|
|
}
|
|
}
|
|
}
|
|
mask <<= 1;
|
|
}
|
|
} else if ((pstrChanged->param->get_type() == BXT_PARAM_BOOL) ||
|
|
(pstrChanged->param->get_type() == BXT_PARAM_NUM) ||
|
|
(pstrChanged->param->get_type() == BXT_PARAM_STRING) ||
|
|
(pstrChanged->param->get_type() == BXT_PARAM_BYTESTRING)) {
|
|
bx_param_c *param = pstrChanged->param;
|
|
if (param->get_type() == BXT_PARAM_BOOL) {
|
|
value = pstrChanged->u.checkbox->GetValue();
|
|
} else if (param->get_type() == BXT_PARAM_NUM) {
|
|
bx_param_num_c *nump = (bx_param_num_c*)param;
|
|
if (nump->get_options() & nump->USE_SPIN_CONTROL) {
|
|
value = (pstrChanged->u.spin->GetValue() > 0);
|
|
} else {
|
|
bool valid;
|
|
value = (GetTextCtrlInt(pstrChanged->u.text, &valid, true, wxT("")) > 0);
|
|
}
|
|
} else {
|
|
wxString tmp(pstrChanged->u.text->GetValue());
|
|
value = !tmp.IsEmpty() && tmp.compare(wxT("none"));
|
|
}
|
|
for (i = 0; i < list->get_size(); i++) {
|
|
dparam = list->get(i);
|
|
if (dparam != param) {
|
|
en = (value && enabled);
|
|
pstr = (ParamStruct*) paramHash->Get(dparam->get_id());
|
|
if (pstr) {
|
|
if (en != pstr->u.window->IsEnabled()) {
|
|
EnableParam(dparam->get_id(), en);
|
|
ProcessDependentList(pstr, en);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// if any parameters changed, update the associated control
|
|
void ParamDialog::CopyParamToGui()
|
|
{
|
|
// loop through all the parameters
|
|
idHash->BeginFind();
|
|
wxHashTable::Node *node;
|
|
while ((node = idHash->Next()) != NULL) {
|
|
ParamStruct *pstr = (ParamStruct*) node->GetData();
|
|
IFDBG_DLG(wxLogDebug(wxT("refresh param %s"), pstr->param->get_name()));
|
|
int type = pstr->param->get_type();
|
|
switch (type) {
|
|
case BXT_PARAM_BOOL: {
|
|
bx_param_bool_c *boolp = (bx_param_bool_c*) pstr->param;
|
|
pstr->u.checkbox->SetValue(boolp->get());
|
|
break;
|
|
}
|
|
case BXT_PARAM_NUM: {
|
|
bx_param_num_c *nump = (bx_param_num_c*) pstr->param;
|
|
const char *format = nump->get_format();
|
|
if (!format)
|
|
format = strdup(nump->get_base() == 16 ? "0x%X" : "%d");
|
|
SetTextCtrl(pstr->u.text, format, nump->get());
|
|
break;
|
|
}
|
|
case BXT_PARAM_ENUM: {
|
|
bx_param_enum_c *enump = (bx_param_enum_c*) pstr->param;
|
|
pstr->u.choice->SetSelection(enump->get() - enump->get_min());
|
|
break;
|
|
}
|
|
case BXT_PARAM_STRING:
|
|
case BXT_PARAM_BYTESTRING: {
|
|
bx_param_string_c *stringp = (bx_param_string_c*) pstr->param;
|
|
pstr->u.text->SetValue(wxString(stringp->getptr(), wxConvUTF8));
|
|
break;
|
|
}
|
|
case BXT_LIST:
|
|
break;
|
|
default:
|
|
wxLogError(wxT("ParamDialog::CopyParamToGui(): unsupported param type id=%d"), (int)type);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ParamDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
if (isGeneratedId(id)) {
|
|
ParamStruct *pstr = (ParamStruct*) idHash->Get(id);
|
|
if (pstr == NULL) {
|
|
wxLogDebug(wxT("ParamStruct not found for id=%d"), id);
|
|
return;
|
|
}
|
|
if (id == pstr->id) {
|
|
IFDBG_DLG(wxLogDebug(wxT("event came from window %p (id=%d) controlled by parameter '%s'"), pstr->u.window, id, pstr->param->get_name()));
|
|
switch (pstr->param->get_type()) {
|
|
case BXT_PARAM_BOOL:
|
|
case BXT_PARAM_NUM:
|
|
case BXT_PARAM_ENUM:
|
|
case BXT_PARAM_STRING:
|
|
case BXT_PARAM_BYTESTRING:
|
|
EnableChanged(pstr);
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
if (id == pstr->browseButtonId) {
|
|
wxLogDebug(wxT("browse button id=%d attached to wxTextCtrl %p"), id, pstr->u.text);
|
|
BrowseTextCtrl(pstr->u.text);
|
|
return;
|
|
}
|
|
wxLogDebug(wxT("id was key to ParamStruct but doesn't match either id inside"));
|
|
}
|
|
switch (id) {
|
|
case wxID_OK:
|
|
if (IsModal()) {
|
|
if (CopyGuiToParam())
|
|
EndModal(wxID_OK);
|
|
} else {
|
|
CopyParamToGui();
|
|
}
|
|
break;
|
|
case wxID_CANCEL:
|
|
if (IsModal())
|
|
EndModal(wxID_CANCEL);
|
|
else
|
|
Show(FALSE);
|
|
break;
|
|
case wxID_HELP:
|
|
ShowHelp();
|
|
break;
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
void ParamDialog::ShowHelp()
|
|
{
|
|
wxMessageBox(MSG_NO_HELP, MSG_NO_HELP_CAPTION, wxOK | wxICON_ERROR, this);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// FloppyConfigDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(FloppyConfigDialog, wxDialog)
|
|
EVT_BUTTON(-1, FloppyConfigDialog::OnEvent)
|
|
EVT_CHECKBOX(-1, FloppyConfigDialog::OnEvent)
|
|
EVT_CHOICE(-1, FloppyConfigDialog::OnEvent)
|
|
EVT_TEXT(-1, FloppyConfigDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
FloppyConfigDialog::FloppyConfigDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: ParamDialog(parent, id)
|
|
{
|
|
infoSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
infoSizer->Add(new wxStaticText(this, -1, wxString("Clicking OK signals a media change for this drive.", wxConvUTF8)), 0, wxALIGN_CENTER|wxALL, 3);
|
|
createButton = AddButton(ID_Create, BTNLABEL_CREATE_IMG);
|
|
AddDefaultButtons();
|
|
}
|
|
|
|
void FloppyConfigDialog::Setup(bx_list_c *list)
|
|
{
|
|
int devtype_id, path_id, media_id, status_id, readonly_id;
|
|
|
|
devtype_id = list->get_by_name("devtype")->get_id();
|
|
path_id = list->get_by_name("path")->get_id();
|
|
media_id = list->get_by_name("type")->get_id();
|
|
status_id = list->get_by_name("status")->get_id();
|
|
readonly_id = list->get_by_name("readonly")->get_id();
|
|
AddParam(list);
|
|
pstrDevice = (ParamStruct*) paramHash->Get(devtype_id);
|
|
pstrPath = (ParamStruct*) paramHash->Get(path_id);
|
|
pstrMedia = (ParamStruct*) paramHash->Get(media_id);
|
|
pstrStatus = (ParamStruct*) paramHash->Get(status_id);
|
|
pstrReadonly = (ParamStruct*) paramHash->Get(readonly_id);
|
|
}
|
|
|
|
void FloppyConfigDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
if (isGeneratedId(id)) {
|
|
ParamStruct *pstr = (ParamStruct*) idHash->Get(id);
|
|
if (pstr == NULL) {
|
|
wxLogDebug(wxT("ParamStruct not found for id=%d"), id);
|
|
return;
|
|
}
|
|
if (id == pstr->id) {
|
|
if ((pstr == pstrDevice) || (pstr == pstrMedia)) {
|
|
int val1 = pstrDevice->u.choice->GetSelection() + ((bx_param_num_c*)pstrDevice->param)->get_min();
|
|
int val2 = pstrMedia->u.choice->GetSelection() + ((bx_param_num_c*)pstrMedia->param)->get_min();
|
|
createButton->Enable((val1 != BX_FDD_NONE) && (val2 != BX_FLOPPY_NONE));
|
|
} else if ((pstr == pstrPath) && (!pstrPath->u.text->IsModified())) {
|
|
pstrMedia->u.choice->SetSelection(pstrMedia->u.choice->FindString(wxT("auto")));
|
|
pstrStatus->u.choice->SetSelection(BX_INSERTED);
|
|
}
|
|
}
|
|
ParamDialog::OnEvent(event);
|
|
} else {
|
|
switch (id) {
|
|
case ID_Create:
|
|
{
|
|
int cap = pstrMedia->u.choice->GetSelection();
|
|
char name[1024];
|
|
strncpy(name, pstrPath->u.text->GetValue().mb_str(wxConvUTF8), sizeof(name) - 1);
|
|
name[sizeof(name) - 1] = '\0';
|
|
if ((floppy_type_n_sectors[cap] > 0) && (strlen(name) > 0) && (strcmp(name, "none"))) {
|
|
if (CreateImage(0, floppy_type_n_sectors[cap], name)) {
|
|
wxString msg(wxT("Created a "));
|
|
msg += pstrMedia->u.choice->GetString(cap);
|
|
msg += wxT(" disk image called '");
|
|
msg += pstrPath->u.text->GetValue();
|
|
msg += wxT("'.");
|
|
wxMessageBox(msg, wxT("Image Created"), wxOK | wxICON_INFORMATION, this);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case wxID_OK:
|
|
// force a media change
|
|
((bx_param_enum_c*)pstrStatus->param)->set(BX_EJECTED);
|
|
default:
|
|
ParamDialog::OnEvent(event);
|
|
}
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// LogOptionsDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(LogOptionsDialog, wxDialog)
|
|
EVT_BUTTON(-1, LogOptionsDialog::OnEvent)
|
|
EVT_CHECKBOX(-1, LogOptionsDialog::OnEvent)
|
|
EVT_TEXT(-1, LogOptionsDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
LogOptionsDialog::LogOptionsDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id)
|
|
: ParamDialog(parent, id)
|
|
{
|
|
static wxString names[] = LOG_OPTS_TYPE_NAMES;
|
|
SetTitle(LOG_OPTS_TITLE);
|
|
AddParam(SIM->get_param("log"));
|
|
wxStaticText *text = new wxStaticText(this, -1, LOG_OPTS_PROMPT);
|
|
mainSizer->Add(text, 0, wxALL, 10);
|
|
gridSizer = new wxFlexGridSizer(2);
|
|
mainSizer->Add(gridSizer, 1, wxLEFT, 40);
|
|
infoSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
text = new wxStaticText(this, -1, LOG_OPTS_ADV);
|
|
infoSizer->Add(text, 0, wxALIGN_CENTER|wxALL, 3);
|
|
|
|
// gridSizer contents
|
|
gridSizer->AddGrowableCol(1);
|
|
for (int evtype=0; evtype<LOG_OPTS_N_TYPES; evtype++) {
|
|
gridSizer->Add(new wxStaticText(this, -1, names[evtype]), 0, wxALL|wxALIGN_CENTER_VERTICAL, 5);
|
|
action[evtype] = makeLogOptionChoiceBox(this, -1, evtype, true);
|
|
gridSizer->Add(action[evtype], 1, wxALL|wxGROW|wxADJUST_MINSIZE, 5);
|
|
}
|
|
}
|
|
|
|
void LogOptionsDialog::SetAction(int evtype, int a)
|
|
{
|
|
// find the choice whose client data matches "a".
|
|
int *ptr;
|
|
//wxLogDebug ("SetAction type=%d a=%d", evtype, a);
|
|
for (int i=0; i < (int)action[evtype]->GetCount(); i++) {
|
|
//wxLogDebug ("reading action[%d]->GetClientData(%d)", evtype, i);
|
|
ptr = (int*) action[evtype]->GetClientData(i);
|
|
if (ptr == NULL) continue;
|
|
if (a == *ptr) { // found it!
|
|
action[evtype]->SetSelection(i);
|
|
return;
|
|
}
|
|
}
|
|
// this can happen if one of the choices that is excluded by
|
|
// BX_LOG_OPTS_EXCLUDE() is used, for example.
|
|
wxLogDebug(wxT("SetAction type=%d a=%d not found"), evtype, a);
|
|
}
|
|
|
|
int LogOptionsDialog::GetAction(int evtype)
|
|
{
|
|
int sel = action[evtype]->GetSelection();
|
|
int *ptrToChoice = (int*)action[evtype]->GetClientData(sel);
|
|
wxASSERT(ptrToChoice != NULL);
|
|
return *ptrToChoice;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// ModelessDialog implementation
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// all events go to OnEvent method
|
|
BEGIN_EVENT_TABLE(ModelessDialog, wxDialog)
|
|
EVT_BUTTON(-1, ModelessDialog::OnEvent)
|
|
END_EVENT_TABLE()
|
|
|
|
ModelessDialog::ModelessDialog(
|
|
wxWindow* parent,
|
|
wxWindowID id,
|
|
const wxString& title,
|
|
const wxString& msg)
|
|
: wxDialog(parent, id, title, wxDefaultPosition, wxDefaultSize,
|
|
wxDEFAULT_DIALOG_STYLE)
|
|
{
|
|
mainSizer = new wxBoxSizer(wxVERTICAL);
|
|
msgSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
buttonSizer = new wxBoxSizer(wxHORIZONTAL);
|
|
mainSizer->Add(msgSizer, 0, wxALIGN_CENTER);
|
|
mainSizer->Add(buttonSizer, 0, wxALIGN_CENTER);
|
|
// msgSizer contents
|
|
message = new wxStaticText(this, -1, msg);
|
|
wxFont font = message->GetFont();
|
|
font.SetWeight(wxFONTWEIGHT_BOLD);
|
|
font.SetPointSize(2 + font.GetPointSize());
|
|
message->SetFont(font);
|
|
msgSizer->Add(message, 0, wxGROW|wxALIGN_LEFT|wxLEFT, 30);
|
|
// buttonSizer contents
|
|
wxButton *btn = new wxButton(this, wxID_OK, BTNLABEL_CLOSE);
|
|
buttonSizer->Add(btn, 0, wxALL, 5);
|
|
}
|
|
|
|
void ModelessDialog::Init()
|
|
{
|
|
SetSizer(mainSizer);
|
|
mainSizer->Fit(this);
|
|
wxSize size = mainSizer->GetMinSize();
|
|
int margin = 5;
|
|
SetSizeHints(size.GetWidth() + margin, size.GetHeight() + margin);
|
|
Center();
|
|
}
|
|
|
|
bool ModelessDialog::Show(bool val)
|
|
{
|
|
if (val) {
|
|
Init();
|
|
wxDialog::Raise();
|
|
}
|
|
return wxDialog::Show(val);
|
|
}
|
|
|
|
void ModelessDialog::OnEvent(wxCommandEvent& event)
|
|
{
|
|
int id = event.GetId();
|
|
switch (id) {
|
|
case wxID_OK:
|
|
Show(false);
|
|
break;
|
|
default:
|
|
event.Skip();
|
|
}
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////
|
|
// utility
|
|
/////////////////////////////////////////////////////////////////
|
|
|
|
// Unfortunately this step is necessary if you change the text of
|
|
// a wxStaticText. Otherwise the sizer that contains the text never realizes
|
|
// that the size has changed, and the layout is never updated. The
|
|
// SetItemMinSize trick was reported on comp.soft-sys.wxwindows by
|
|
// Dirk Birnhardt.
|
|
void ChangeStaticText(wxSizer *sizer, wxStaticText *win, wxString newtext)
|
|
{
|
|
win->SetLabel(newtext);
|
|
wxSize sz = win->GetSize();
|
|
sizer->SetItemMinSize(win, sz.GetWidth(), sz.GetHeight());
|
|
}
|
|
|
|
// CreateImage produces a disk image. It's in the utility function
|
|
// area because it's used by both floppy and hard disk image creation.
|
|
bool CreateImage(int harddisk, int sectors, const char *filename)
|
|
{
|
|
if (sectors<1) {
|
|
wxMessageBox(wxT("The disk size is invalid."), wxT("Invalid Size"), wxOK | wxICON_ERROR);
|
|
return false;
|
|
}
|
|
wxLogDebug(wxT("filename = '%s'\n"), filename);
|
|
if (strlen(filename) < 1) {
|
|
wxMessageBox(wxT("You must type a file name for the new disk image."), wxT("Bad Filename"), wxOK | wxICON_ERROR);
|
|
return false;
|
|
}
|
|
// create disk image with name and capacity determined by the filename
|
|
// and sector args. Call SIM->create_image(filename, sectors, overwrite=0)
|
|
// first which will create the file if it doesn't already exist. If it
|
|
// exists, it will instead return -1, and we can ask the user "are you sure
|
|
// you want to overwrite?". If yes, call again with overwrite=1.
|
|
int ret = SIM->create_disk_image(filename, sectors, 0);
|
|
if (ret == -1) { // already exists
|
|
int answer = wxMessageBox(wxT("File exists. Do you want to overwrite it?"),
|
|
wxT("File exists"), wxYES_NO | wxCENTER);
|
|
if (answer == wxYES)
|
|
ret = SIM->create_disk_image(filename, sectors, 1);
|
|
else
|
|
return false; // wxNO
|
|
}
|
|
if (ret == -2) {
|
|
wxMessageBox(wxT("I could not create the disk image. Check for permission problems or available disk space."), wxT("Failed"), wxOK | wxICON_ERROR);
|
|
return false;
|
|
}
|
|
wxASSERT(ret==0);
|
|
return true;
|
|
}
|
|
|
|
void SetTextCtrl(wxTextCtrl *ctrl, const char *format, int val)
|
|
{
|
|
wxString tmp;
|
|
tmp.Printf(wxString(format, wxConvUTF8), val);
|
|
ctrl->SetValue(tmp);
|
|
}
|
|
|
|
int GetTextCtrlInt(wxTextCtrl *ctrl,
|
|
bool *valid,
|
|
bool complain,
|
|
wxString complaint)
|
|
{
|
|
wxString tmp(ctrl->GetValue());
|
|
char buf[1024];
|
|
strncpy(buf, tmp.mb_str(wxConvUTF8), sizeof(buf) - 1);
|
|
buf[sizeof(buf)-1] = '\0';
|
|
int n = strtol(buf, NULL, 0);
|
|
if (n != LONG_MIN && n != LONG_MAX) {
|
|
if (valid) *valid = true;
|
|
return n;
|
|
}
|
|
if (valid) *valid = false;
|
|
if (complain) {
|
|
wxMessageBox(complaint, wxT("Invalid"), wxOK | wxICON_ERROR);
|
|
ctrl->SetFocus();
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
bool BrowseTextCtrl(wxTextCtrl *text, wxString prompt, long style)
|
|
{
|
|
// try to configure the dialog to show hidden files
|
|
wxConfigBase::Get() ->Write(wxT("/wxWidgets/wxFileDialog/ShowHidden"), true);
|
|
wxFileDialog *fdialog = new wxFileDialog(text->GetParent(), prompt, wxT(""), text->GetValue(), wxT("*.*"), style);
|
|
int result = fdialog->ShowModal();
|
|
if (result == wxID_OK)
|
|
text->SetValue(fdialog->GetPath());
|
|
delete fdialog;
|
|
return (result == wxID_OK);
|
|
}
|
|
|
|
wxChoice *makeLogOptionChoiceBox(wxWindow *parent,
|
|
wxWindowID id,
|
|
int evtype,
|
|
bool includeNoChange)
|
|
{
|
|
static wxString choices[] = LOG_OPTS_CHOICES;
|
|
static int integers[LOG_OPTS_N_CHOICES] = {0, 1, 2, 3, 4, 5};
|
|
wxChoice *control = new wxChoice(parent, id, wxDefaultPosition, wxDefaultSize);
|
|
int lastChoice = 0; // remember index of last choice
|
|
int nchoice = includeNoChange? LOG_OPTS_N_CHOICES : LOG_OPTS_N_CHOICES_NORMAL;
|
|
for (int choice=0; choice<nchoice; choice++) {
|
|
// the exclude expression allows some choices not being available if they
|
|
// don't make any sense. For example, it would be stupid to ignore a panic.
|
|
if (!BX_LOG_OPTS_EXCLUDE(evtype, choice)) {
|
|
control->Append(choices[choice], &integers[choice]);
|
|
// the client data is an int* that points to the choice number.
|
|
// This is what will be returned by GetAction().
|
|
lastChoice++;
|
|
}
|
|
}
|
|
control->SetSelection(lastChoice-1);
|
|
return control;
|
|
}
|
|
|
|
#endif /* if BX_WITH_WX */
|