mirror of
https://github.com/fadden/6502bench.git
synced 2024-12-01 22:50:35 +00:00
901 lines
37 KiB
C#
901 lines
37 KiB
C#
/*
|
|
* Copyright 2019 faddenSoft
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Diagnostics;
|
|
using System.Globalization;
|
|
using System.Reflection;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using Microsoft.Win32;
|
|
|
|
using CommonUtil;
|
|
|
|
using AssemblerInfo = SourceGen.AsmGen.AssemblerInfo;
|
|
using AssemblerConfig = SourceGen.AsmGen.AssemblerConfig;
|
|
using ExpressionMode = Asm65.Formatter.FormatConfig.ExpressionMode;
|
|
|
|
namespace SourceGen.WpfGui {
|
|
/// <summary>
|
|
/// Application settings dialog.
|
|
/// </summary>
|
|
public partial class EditAppSettings : Window, INotifyPropertyChanged {
|
|
/// <summary>
|
|
/// Reference to main window. Needed for examination of the code list font and
|
|
/// column widths.
|
|
/// </summary>
|
|
private MainWindow mMainWin;
|
|
|
|
/// <summary>
|
|
/// Reference to main controller. Needed to push settings out when Apply/OK is clicked.
|
|
/// </summary>
|
|
private MainController mMainCtrl;
|
|
|
|
/// <summary>
|
|
/// Copy of settings that we make changes to. On "Apply" or "OK", this is pushed
|
|
/// into the global settings object, and applied to the ProjectView.
|
|
/// </summary>
|
|
private AppSettings mSettings;
|
|
|
|
/// <summary>
|
|
/// Dirty flag, set when anything in mSettings changes. Determines whether or not
|
|
/// the Apply button is enabled.
|
|
/// </summary>
|
|
public bool IsDirty {
|
|
get { return mIsDirty; }
|
|
set {
|
|
mIsDirty = value;
|
|
OnPropertyChanged();
|
|
}
|
|
}
|
|
private bool mIsDirty;
|
|
|
|
/// <summary>
|
|
/// Tab page enumeration.
|
|
/// </summary>
|
|
public enum Tab {
|
|
Unknown = -1,
|
|
CodeView = 0,
|
|
AsmConfig = 1,
|
|
DisplayFormat = 2,
|
|
PseudoOp = 3
|
|
}
|
|
|
|
/// <summary>
|
|
/// Tab to show when dialog is first opened.
|
|
/// </summary>
|
|
private Tab mInitialTab;
|
|
|
|
/// <summary>
|
|
/// Assembler to initially select in combo boxes.
|
|
/// </summary>
|
|
private AssemblerInfo.Id mInitialAsmId;
|
|
|
|
public List<AssemblerInfo> AssemblerList { get; private set; }
|
|
|
|
|
|
// INotifyPropertyChanged implementation
|
|
public event PropertyChangedEventHandler PropertyChanged;
|
|
private void OnPropertyChanged([CallerMemberName] string propertyName = "") {
|
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
|
}
|
|
|
|
|
|
public EditAppSettings(Window owner, MainWindow mainWin, MainController mainCtrl,
|
|
Tab initialTab, AssemblerInfo.Id initialAsmId) {
|
|
InitializeComponent();
|
|
Owner = owner;
|
|
DataContext = this;
|
|
|
|
mMainWin = mainWin;
|
|
mMainCtrl = mainCtrl;
|
|
mInitialTab = initialTab;
|
|
mInitialAsmId = initialAsmId;
|
|
|
|
// Make a work copy, so we can discard changes if the user cancels out of the dialog.
|
|
// Column width changes don't actively update settings, so grab them.
|
|
mMainWin.CaptureColumnWidths();
|
|
mSettings = AppSettings.Global.GetCopy();
|
|
|
|
// Put column-width buttons in an array.
|
|
mColButtons = new Button[] {
|
|
showCol0, showCol1, showCol2, showCol3, showCol4,
|
|
showCol5, showCol6, showCol7, showCol8 };
|
|
Debug.Assert(NUM_COLUMNS == 9);
|
|
|
|
// Extract format strings from column-width button labels.
|
|
for (int i = 0; i < NUM_COLUMNS; i++) {
|
|
//mColButtons[i].Click += ColumnVisibilityButtonClick;
|
|
mColumnFormats[i] = (string) mColButtons[i].Content;
|
|
}
|
|
|
|
// Create an assembler list for the assembler-config combo box and the two
|
|
// "quick set" combo boxes.
|
|
AssemblerList = new List<AssemblerInfo>();
|
|
IEnumerator<AssemblerInfo> iter = AssemblerInfo.GetInfoEnumerator();
|
|
while (iter.MoveNext()) {
|
|
AssemblerList.Add(iter.Current);
|
|
}
|
|
// Can't set the selected item yet.
|
|
|
|
// Map text boxes to PseudoOpName fields.
|
|
mPseudoNameMap = new TextBoxPropertyMap[] {
|
|
new TextBoxPropertyMap(equDirectiveTextBox, "EquDirective"),
|
|
new TextBoxPropertyMap(orgDirectiveTextBox, "OrgDirective"),
|
|
new TextBoxPropertyMap(regWidthDirectiveTextBox, "RegWidthDirective"),
|
|
new TextBoxPropertyMap(defineData1TextBox, "DefineData1"),
|
|
new TextBoxPropertyMap(defineData2TextBox, "DefineData2"),
|
|
new TextBoxPropertyMap(defineData3TextBox, "DefineData3"),
|
|
new TextBoxPropertyMap(defineData4TextBox, "DefineData4"),
|
|
new TextBoxPropertyMap(defineBigData2TextBox, "DefineBigData2"),
|
|
new TextBoxPropertyMap(fillTextBox, "Fill"),
|
|
new TextBoxPropertyMap(denseTextBox, "Dense"),
|
|
new TextBoxPropertyMap(strGenericTextBox, "StrGeneric"),
|
|
new TextBoxPropertyMap(strGenericHiTextBox, "StrGenericHi"),
|
|
new TextBoxPropertyMap(strReverseTextBox, "StrReverse"),
|
|
new TextBoxPropertyMap(strReverseHiTextBox, "StrReverseHi"),
|
|
new TextBoxPropertyMap(strLen8TextBox, "StrLen8"),
|
|
new TextBoxPropertyMap(strLen8HiTextBox, "StrLen8Hi"),
|
|
new TextBoxPropertyMap(strLen16TextBox, "StrLen16"),
|
|
new TextBoxPropertyMap(strLen16HiTextBox, "StrLen16Hi"),
|
|
new TextBoxPropertyMap(strNullTermTextBox, "StrNullTerm"),
|
|
new TextBoxPropertyMap(strNullTermHiTextBox, "StrNullTermHi"),
|
|
new TextBoxPropertyMap(strDciTextBox, "StrDci"),
|
|
new TextBoxPropertyMap(strDciHiTextBox, "StrDciHi"),
|
|
};
|
|
}
|
|
|
|
private void Window_Loaded(object sender, RoutedEventArgs e) {
|
|
Loaded_CodeView();
|
|
Loaded_AsmConfig();
|
|
Loaded_DisplayFormat();
|
|
Loaded_PseudoOp();
|
|
|
|
switch (mInitialTab) {
|
|
case Tab.CodeView:
|
|
tabControl.SelectedItem = codeViewTab;
|
|
break;
|
|
case Tab.AsmConfig:
|
|
tabControl.SelectedItem = asmConfigTab;
|
|
break;
|
|
case Tab.DisplayFormat:
|
|
tabControl.SelectedItem = displayFormatTab;
|
|
break;
|
|
case Tab.PseudoOp:
|
|
tabControl.SelectedItem = pseudoOpTab;
|
|
break;
|
|
case Tab.Unknown:
|
|
break;
|
|
default:
|
|
Debug.Assert(false);
|
|
break;
|
|
|
|
}
|
|
|
|
// The various control initializers probably triggered events. Reset the dirty flag.
|
|
IsDirty = false;
|
|
}
|
|
|
|
private void OkButton_Click(object sender, RoutedEventArgs e) {
|
|
ApplySettings();
|
|
DialogResult = true;
|
|
}
|
|
|
|
private void ApplyButton_Click(object sender, RoutedEventArgs e) {
|
|
ApplySettings();
|
|
}
|
|
|
|
private void ApplySettings() {
|
|
PseudoOp.PseudoOpNames opNames = ExportPseudoOpNames();
|
|
string pseudoCereal = opNames.Serialize();
|
|
mSettings.SetString(AppSettings.FMT_PSEUDO_OP_NAMES, pseudoCereal);
|
|
|
|
mMainCtrl.SetAppSettings(mSettings);
|
|
AsmGen.AssemblerVersionCache.QueryVersions();
|
|
IsDirty = false;
|
|
}
|
|
|
|
|
|
#region Code View
|
|
|
|
/// <summary>
|
|
/// Entries for the clipboard format item combo box.
|
|
/// </summary>
|
|
public class ClipboardFormatItem {
|
|
public string Name { get; private set; }
|
|
public MainController.ClipLineFormat Value { get; private set; }
|
|
|
|
public ClipboardFormatItem(string name, MainController.ClipLineFormat value) {
|
|
Name = name;
|
|
Value = value;
|
|
}
|
|
}
|
|
// NOTE: in the current implementation, the array index must match the enum value
|
|
private static ClipboardFormatItem[] sClipboardFormatItems = {
|
|
new ClipboardFormatItem(Res.Strings.CLIPFORMAT_ASSEMBLER_SOURCE,
|
|
MainController.ClipLineFormat.AssemblerSource),
|
|
new ClipboardFormatItem(Res.Strings.CLIPFORMAT_DISASSEMBLY,
|
|
MainController.ClipLineFormat.Disassembly)
|
|
};
|
|
// ItemsSource for combo box
|
|
public ClipboardFormatItem[] ClipboardFormatItems {
|
|
get { return sClipboardFormatItems; }
|
|
}
|
|
|
|
private void Loaded_CodeView() {
|
|
// Column widths. We called CaptureColumnWidths() during init, so this
|
|
// should always be a valid serialized string.
|
|
string widthStr = mSettings.GetString(AppSettings.CDLV_COL_WIDTHS, null);
|
|
Debug.Assert(!string.IsNullOrEmpty(widthStr));
|
|
int[] widths = TextUtil.DeserializeIntArray(widthStr);
|
|
Debug.Assert(widths != null);
|
|
for (int i = 0; i < NUM_COLUMNS; i++) {
|
|
SetShowHideButton(i, widths[i]);
|
|
}
|
|
|
|
// Set the string. Note this shows what is currently set in the pending settings
|
|
// object, *not* what's currently being rendered (until you hit Apply).
|
|
string fontFamilyName = mSettings.GetString(AppSettings.CDLV_FONT_FAMILY, "BROKEN");
|
|
int fontSize = mSettings.GetInt(AppSettings.CDLV_FONT_SIZE, 2);
|
|
codeListFontDesc.Text = string.Format(Res.Strings.FONT_DESCRIPTOR_FMT,
|
|
fontSize, fontFamilyName);
|
|
|
|
// Upper-case formatting.
|
|
UpperHexValues = mSettings.GetBool(AppSettings.FMT_UPPER_HEX_DIGITS, false);
|
|
UpperOpcodes = mSettings.GetBool(AppSettings.FMT_UPPER_OP_MNEMONIC, false);
|
|
UpperPseudoOps = mSettings.GetBool(AppSettings.FMT_UPPER_PSEUDO_OP_MNEMONIC, false);
|
|
UpperOperandA = mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_A, false);
|
|
UpperOperandS = mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_S, false);
|
|
UpperOperandXY = mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_XY, false);
|
|
|
|
Debug.Assert(clipboardFormatComboBox.Items.Count == sClipboardFormatItems.Length);
|
|
int clipIndex = mSettings.GetEnum(AppSettings.CLIP_LINE_FORMAT,
|
|
typeof(MainController.ClipLineFormat), 0);
|
|
if (clipIndex >= 0 && clipIndex < sClipboardFormatItems.Length) {
|
|
// require Value == clipIndex because we're lazy and don't want to search
|
|
Debug.Assert((int)sClipboardFormatItems[clipIndex].Value == clipIndex);
|
|
clipboardFormatComboBox.SelectedIndex = clipIndex;
|
|
}
|
|
|
|
SpacesBetweenBytes = mSettings.GetBool(AppSettings.FMT_SPACES_BETWEEN_BYTES, false);
|
|
EnableDebugMenu = mSettings.GetBool(AppSettings.DEBUG_MENU_ENABLED, false);
|
|
}
|
|
|
|
// Map buttons to column show/hide buttons.
|
|
private const int NUM_COLUMNS = (int)MainController.CodeListColumn.COUNT;
|
|
private string[] mColumnFormats = new string[NUM_COLUMNS];
|
|
private Button[] mColButtons;
|
|
|
|
/// <summary>
|
|
/// Updates the text on a show/hide column button.
|
|
/// </summary>
|
|
/// <param name="index">Column index.</param>
|
|
/// <param name="width">New width.</param>
|
|
private void SetShowHideButton(int index, int width) {
|
|
Button button = mColButtons[index];
|
|
string fmt = mColumnFormats[index];
|
|
string show = Res.Strings.SHOW_COL;
|
|
string hide = Res.Strings.HIDE_COL;
|
|
button.Content = string.Format(fmt, (width == 0) ? show : hide);
|
|
}
|
|
|
|
private void ColumnVisibilityButton_Click(object sender, RoutedEventArgs e) {
|
|
int index = -1;
|
|
for (int i = 0; i < mColButtons.Length; i++) {
|
|
if (sender == mColButtons[i]) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
Debug.Assert(index != -1);
|
|
|
|
string widthStr = mSettings.GetString(AppSettings.CDLV_COL_WIDTHS, null);
|
|
Debug.Assert(!string.IsNullOrEmpty(widthStr));
|
|
int[] widths = TextUtil.DeserializeIntArray(widthStr);
|
|
if (widths[index] == 0) {
|
|
// Expand to default width. The default width changes when the font
|
|
// changes, so it's best to just reacquire the default width set as needed.
|
|
int[] defaultWidths = mMainWin.GetDefaultCodeListColumnWidths();
|
|
widths[index] = defaultWidths[index];
|
|
} else {
|
|
widths[index] = 0;
|
|
}
|
|
widthStr = TextUtil.SerializeIntArray(widths);
|
|
mSettings.SetString(AppSettings.CDLV_COL_WIDTHS, widthStr);
|
|
SetShowHideButton(index, widths[index]);
|
|
|
|
IsDirty = true;
|
|
}
|
|
|
|
private void SelectFontButton_Click(object sender, RoutedEventArgs e) {
|
|
FontPicker dlg = new FontPicker(this,
|
|
mSettings.GetString(AppSettings.CDLV_FONT_FAMILY, string.Empty),
|
|
mSettings.GetInt(AppSettings.CDLV_FONT_SIZE, 12));
|
|
if (dlg.ShowDialog() == true) {
|
|
string familyName = dlg.SelectedFamily.ToString();
|
|
mSettings.SetString(AppSettings.CDLV_FONT_FAMILY, familyName);
|
|
mSettings.SetInt(AppSettings.CDLV_FONT_SIZE, dlg.SelectedSize);
|
|
|
|
codeListFontDesc.Text = string.Format(Res.Strings.FONT_DESCRIPTOR_FMT,
|
|
dlg.SelectedSize, familyName);
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
public bool UpperHexValues {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_HEX_DIGITS, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_HEX_DIGITS, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool UpperOpcodes {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_OP_MNEMONIC, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_OP_MNEMONIC, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool UpperPseudoOps {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_PSEUDO_OP_MNEMONIC, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_PSEUDO_OP_MNEMONIC, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool UpperOperandA {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_A, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_OPERAND_A, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool UpperOperandS {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_S, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_OPERAND_S, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool UpperOperandXY {
|
|
get { return mSettings.GetBool(AppSettings.FMT_UPPER_OPERAND_XY, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_UPPER_OPERAND_XY, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
private void AllLower_Click(object sender, RoutedEventArgs e) {
|
|
UpperHexValues = UpperOpcodes = UpperPseudoOps =
|
|
UpperOperandA = UpperOperandS = UpperOperandXY = false;
|
|
}
|
|
|
|
private void AllUpper_Click(object sender, RoutedEventArgs e) {
|
|
UpperHexValues = UpperOpcodes = UpperPseudoOps =
|
|
UpperOperandA = UpperOperandS = UpperOperandXY = true;
|
|
}
|
|
|
|
private void ClipboardFormatComboBox_SelectionChanged(object sender,
|
|
SelectionChangedEventArgs e) {
|
|
ClipboardFormatItem item = (ClipboardFormatItem)clipboardFormatComboBox.SelectedItem;
|
|
mSettings.SetEnum(AppSettings.CLIP_LINE_FORMAT, typeof(MainController.ClipLineFormat),
|
|
(int)item.Value);
|
|
IsDirty = true;
|
|
}
|
|
|
|
public bool SpacesBetweenBytes {
|
|
get { return mSettings.GetBool(AppSettings.FMT_SPACES_BETWEEN_BYTES, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.FMT_SPACES_BETWEEN_BYTES, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
public bool EnableDebugMenu {
|
|
get { return mSettings.GetBool(AppSettings.DEBUG_MENU_ENABLED, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.DEBUG_MENU_ENABLED, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
#endregion Code View
|
|
|
|
#region Asm Config
|
|
|
|
public const int ASM_COL_MIN_WIDTH = 1;
|
|
public const int ASM_COL_MAX_WIDTH = 200;
|
|
|
|
//
|
|
// Numeric input fields, bound directly to TextBox.Text. This performs the basic
|
|
// validation, but we also do our own so we can cap min/max. The custom validation
|
|
// rule seems to fire ahead of the field assignment, so if our rule fails we won't
|
|
// try to assign here at all.
|
|
//
|
|
// The validation approach seems to make the most sense here because we don't dim the
|
|
// Apply/OK buttons when invalid input is present. Instead we just set the width to
|
|
// the minimum value when validation fails.
|
|
//
|
|
// See also https://stackoverflow.com/a/44586784/294248
|
|
//
|
|
private int mAsmLabelColWidth;
|
|
public int AsmLabelColWidth {
|
|
get { return mAsmLabelColWidth; }
|
|
set {
|
|
if (mAsmLabelColWidth != value) {
|
|
mAsmLabelColWidth = value;
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
}
|
|
private int mAsmOpcodeColWidth;
|
|
public int AsmOpcodeColWidth {
|
|
get { return mAsmOpcodeColWidth; }
|
|
set {
|
|
if (mAsmOpcodeColWidth != value) {
|
|
mAsmOpcodeColWidth = value;
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
}
|
|
private int mAsmOperandColWidth;
|
|
public int AsmOperandColWidth {
|
|
get { return mAsmOperandColWidth; }
|
|
set {
|
|
if (mAsmOperandColWidth != value) {
|
|
mAsmOperandColWidth = value;
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
}
|
|
private int mAsmCommentColWidth;
|
|
public int AsmCommentColWidth {
|
|
get { return mAsmCommentColWidth; }
|
|
set {
|
|
if (mAsmCommentColWidth != value) {
|
|
mAsmCommentColWidth = value;
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// checkboxes
|
|
public bool ShowCycleCounts {
|
|
get { return mSettings.GetBool(AppSettings.SRCGEN_SHOW_CYCLE_COUNTS, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.SRCGEN_SHOW_CYCLE_COUNTS, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool LongLabelNewLine {
|
|
get { return mSettings.GetBool(AppSettings.SRCGEN_LONG_LABEL_NEW_LINE, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.SRCGEN_LONG_LABEL_NEW_LINE, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool AddIdentComment {
|
|
get { return mSettings.GetBool(AppSettings.SRCGEN_ADD_IDENT_COMMENT, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.SRCGEN_ADD_IDENT_COMMENT, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public bool DisableLabelLocalization {
|
|
get { return mSettings.GetBool(AppSettings.SRCGEN_DISABLE_LABEL_LOCALIZATION, false); }
|
|
set {
|
|
mSettings.SetBool(AppSettings.SRCGEN_DISABLE_LABEL_LOCALIZATION, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
private void Loaded_AsmConfig() {
|
|
asmConfigComboBox.SelectedItem = AssemblerInfo.GetAssemblerInfo(mInitialAsmId);
|
|
if (asmConfigComboBox.SelectedIndex < 0) {
|
|
Debug.Assert(mInitialAsmId == AssemblerInfo.Id.Unknown);
|
|
asmConfigComboBox.SelectedIndex = 0;
|
|
}
|
|
|
|
ShowCycleCounts =
|
|
mSettings.GetBool(AppSettings.SRCGEN_SHOW_CYCLE_COUNTS, false);
|
|
LongLabelNewLine =
|
|
mSettings.GetBool(AppSettings.SRCGEN_LONG_LABEL_NEW_LINE, false);
|
|
AddIdentComment =
|
|
mSettings.GetBool(AppSettings.SRCGEN_ADD_IDENT_COMMENT, false);
|
|
DisableLabelLocalization =
|
|
mSettings.GetBool(AppSettings.SRCGEN_DISABLE_LABEL_LOCALIZATION, false);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Populates the UI elements from the asm config item in the settings. If that doesn't
|
|
/// exist, use the default config.
|
|
/// </summary>
|
|
private void PopulateAsmConfigItems() {
|
|
AssemblerInfo info = (AssemblerInfo)asmConfigComboBox.SelectedItem;
|
|
|
|
AssemblerConfig config = AssemblerConfig.GetConfig(mSettings, info.AssemblerId);
|
|
if (config == null) {
|
|
AsmGen.IAssembler asm = AssemblerInfo.GetAssembler(info.AssemblerId);
|
|
config = asm.GetDefaultConfig();
|
|
}
|
|
|
|
asmExePathTextBox.Text = config.ExecutablePath;
|
|
asmLabelColWidthTextBox.Text = config.ColumnWidths[0].ToString();
|
|
asmOpcodeColWidthTextBox.Text = config.ColumnWidths[1].ToString();
|
|
asmOperandColWidthTextBox.Text = config.ColumnWidths[2].ToString();
|
|
asmCommentColWidthTextBox.Text = config.ColumnWidths[3].ToString();
|
|
}
|
|
|
|
private void AsmConfigComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e) {
|
|
// They're switching to a different asm config. Changing the boxes will cause
|
|
// the dirty flag to be raised, which isn't right, so we save/restore it.
|
|
bool oldDirty = IsDirty;
|
|
PopulateAsmConfigItems();
|
|
IsDirty = oldDirty;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Updates the assembler config settings whenever one of the width text fields is edited.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This fires 4x every time the combo box selection changes, as the new fields are
|
|
/// populated. That means we'll have incorrect intermediate states, but should
|
|
/// finish up correctly.
|
|
/// </remarks>
|
|
private void AsmColWidthTextBox_TextChanged(object sender, TextChangedEventArgs e) {
|
|
AssemblerInfo asm = (AssemblerInfo)asmConfigComboBox.SelectedItem;
|
|
if (asm == null) {
|
|
// fires during dialog initialization, before anything is selected
|
|
return;
|
|
}
|
|
AssemblerConfig.SetConfig(mSettings, asm.AssemblerId, GetAsmConfigFromUi());
|
|
IsDirty = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Extracts the asm configuration items (exe path, column widths) from the UI.
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
private AssemblerConfig GetAsmConfigFromUi() {
|
|
int[] widths = new int[4];
|
|
|
|
if (!Validation.GetHasError(asmLabelColWidthTextBox)) {
|
|
widths[0] = AsmLabelColWidth;
|
|
} else {
|
|
widths[0] = ASM_COL_MIN_WIDTH;
|
|
}
|
|
if (!Validation.GetHasError(asmOpcodeColWidthTextBox)) {
|
|
widths[1] = AsmOpcodeColWidth;
|
|
} else {
|
|
widths[1] = ASM_COL_MIN_WIDTH;
|
|
}
|
|
if (!Validation.GetHasError(asmOperandColWidthTextBox)) {
|
|
widths[2] = AsmOperandColWidth;
|
|
} else {
|
|
widths[2] = ASM_COL_MIN_WIDTH;
|
|
}
|
|
if (!Validation.GetHasError(asmCommentColWidthTextBox)) {
|
|
widths[3] = AsmCommentColWidth;
|
|
} else {
|
|
widths[3] = ASM_COL_MIN_WIDTH;
|
|
}
|
|
|
|
return new AssemblerConfig(asmExePathTextBox.Text, widths);
|
|
}
|
|
|
|
private void AsmExeBrowseButton_Click(object sender, RoutedEventArgs e) {
|
|
AssemblerInfo asmInfo = (AssemblerInfo)asmConfigComboBox.SelectedItem;
|
|
|
|
// Figure out what we're looking for. For example, cc65 needs "cl65".
|
|
AsmGen.IAssembler asm = AssemblerInfo.GetAssembler(asmInfo.AssemblerId);
|
|
asm.GetExeIdentifiers(out string humanName, out string exeName);
|
|
|
|
// Ask the user to find it.
|
|
string pathName = BrowseForExecutable(humanName, exeName);
|
|
if (pathName != null) {
|
|
asmExePathTextBox.Text = pathName;
|
|
AssemblerConfig.SetConfig(mSettings, asmInfo.AssemblerId, GetAsmConfigFromUi());
|
|
IsDirty = true;
|
|
}
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a file dialog to search for a specific executable.
|
|
/// </summary>
|
|
/// <param name="prefix">Human-readable filter string for UI.</param>
|
|
/// <param name="name">Filename of executable.</param>
|
|
/// <returns>Path of executable, or null if dialog was canceled.</returns>
|
|
private string BrowseForExecutable(string prefix, string name) {
|
|
string pathName = null;
|
|
|
|
if (Environment.OSVersion.Platform == PlatformID.Win32NT) {
|
|
name += ".exe";
|
|
}
|
|
|
|
OpenFileDialog dlg = new OpenFileDialog() {
|
|
FileName = name,
|
|
Filter = prefix + "|" + name,
|
|
RestoreDirectory = true
|
|
};
|
|
if (dlg.ShowDialog() == true) {
|
|
pathName = dlg.FileName;
|
|
}
|
|
|
|
return pathName;
|
|
}
|
|
|
|
#endregion AsmConfig
|
|
|
|
#region Display Format
|
|
|
|
/// <summary>
|
|
/// Holds an item for the expression style selection combo box.
|
|
/// </summary>
|
|
public class ExpressionStyleItem {
|
|
// Enumerated mode.
|
|
public ExpressionMode ExpMode { get; private set; }
|
|
|
|
// Human-readable name for display.
|
|
public string Name { get; private set; }
|
|
|
|
public ExpressionStyleItem(ExpressionMode expMode, string name) {
|
|
ExpMode = expMode;
|
|
Name = name;
|
|
}
|
|
}
|
|
private static ExpressionStyleItem[] sExpStyleItems = new ExpressionStyleItem[] {
|
|
new ExpressionStyleItem(ExpressionMode.Common, "Common"),
|
|
new ExpressionStyleItem(ExpressionMode.Cc65, "cc65"),
|
|
new ExpressionStyleItem(ExpressionMode.Merlin, "Merlin"),
|
|
};
|
|
public ExpressionStyleItem[] ExpressionStyleItems {
|
|
get { return sExpStyleItems; }
|
|
}
|
|
|
|
private void Loaded_DisplayFormat() {
|
|
PopulateWidthDisamSettings();
|
|
|
|
string exprMode = mSettings.GetString(AppSettings.FMT_EXPRESSION_MODE, string.Empty);
|
|
ExpressionMode mode;
|
|
if (!Enum.TryParse<ExpressionMode>(exprMode, out mode)) {
|
|
mode = ExpressionMode.Common;
|
|
}
|
|
SelectExpressionStyle(mode);
|
|
|
|
// No need to set this to anything specific.
|
|
displayFmtQuickComboBox.SelectedIndex = 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Populates the width disambiguation text boxes.
|
|
/// </summary>
|
|
private void PopulateWidthDisamSettings() {
|
|
// Operand width disambiguation. This is a little tricky -- we have to query all
|
|
// settings then set all controls, or the field-updated callback may interfere
|
|
// with us by changing AppSettings.
|
|
string opcSuffixAbs = mSettings.GetString(AppSettings.FMT_OPCODE_SUFFIX_ABS,
|
|
string.Empty);
|
|
string opcSuffixLong = mSettings.GetString(AppSettings.FMT_OPCODE_SUFFIX_LONG,
|
|
string.Empty);
|
|
string opPrefixAbs = mSettings.GetString(AppSettings.FMT_OPERAND_PREFIX_ABS,
|
|
string.Empty);
|
|
string opPrefixLong = mSettings.GetString(AppSettings.FMT_OPERAND_PREFIX_LONG,
|
|
string.Empty);
|
|
|
|
disambSuffix16TextBox.Text = opcSuffixAbs;
|
|
disambSuffix24TextBox.Text = opcSuffixLong;
|
|
disambPrefix16TextBox.Text = opPrefixAbs;
|
|
disambPrefix24TextBox.Text = opPrefixLong;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets all of the width disambiguation settings. Used for the quick-set feature.
|
|
/// </summary>
|
|
private void SetWidthDisamSettings(string opcodeSuffixAbs, string opcodeSuffixLong,
|
|
string operandPrefixAbs, string operandPrefixLong) {
|
|
mSettings.SetString(AppSettings.FMT_OPCODE_SUFFIX_ABS, opcodeSuffixAbs);
|
|
mSettings.SetString(AppSettings.FMT_OPCODE_SUFFIX_LONG, opcodeSuffixLong);
|
|
mSettings.SetString(AppSettings.FMT_OPERAND_PREFIX_ABS, operandPrefixAbs);
|
|
mSettings.SetString(AppSettings.FMT_OPERAND_PREFIX_LONG, operandPrefixLong);
|
|
PopulateWidthDisamSettings();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Exports the current state of the width controls to the settings object whenever
|
|
/// text is typed.
|
|
/// </summary>
|
|
private void WidthDisamControlChanged(object sender, TextChangedEventArgs e) {
|
|
if (mSettings == null) {
|
|
// initialization
|
|
return;
|
|
}
|
|
mSettings.SetString(AppSettings.FMT_OPCODE_SUFFIX_ABS, disambSuffix16TextBox.Text);
|
|
mSettings.SetString(AppSettings.FMT_OPCODE_SUFFIX_LONG, disambSuffix24TextBox.Text);
|
|
mSettings.SetString(AppSettings.FMT_OPERAND_PREFIX_ABS, disambPrefix16TextBox.Text);
|
|
mSettings.SetString(AppSettings.FMT_OPERAND_PREFIX_LONG, disambPrefix24TextBox.Text);
|
|
IsDirty = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Changes the combo box selection to the desired mode.
|
|
/// </summary>
|
|
private void SelectExpressionStyle(ExpressionMode mode) {
|
|
foreach (ExpressionStyleItem esi in expressionStyleComboBox.Items) {
|
|
if (esi.ExpMode == mode) {
|
|
expressionStyleComboBox.SelectedItem = esi;
|
|
return;
|
|
}
|
|
}
|
|
Debug.Assert(false, "Expression mode " + mode + " not found");
|
|
expressionStyleComboBox.SelectedIndex = 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Handles a change to the expression style.
|
|
/// </summary>
|
|
private void ExpressionStyleComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e) {
|
|
ExpressionStyleItem esi = (ExpressionStyleItem)expressionStyleComboBox.SelectedItem;
|
|
mSettings.SetString(AppSettings.FMT_EXPRESSION_MODE, esi.ExpMode.ToString());
|
|
IsDirty = true;
|
|
}
|
|
|
|
private void DisplayFmtSetButton_Click(object sender, RoutedEventArgs e) {
|
|
AssemblerInfo asmInfo = (AssemblerInfo)displayFmtQuickComboBox.SelectedItem;
|
|
AsmGen.IGenerator gen = AssemblerInfo.GetGenerator(asmInfo.AssemblerId);
|
|
|
|
PseudoOp.PseudoOpNames opNames;
|
|
Asm65.Formatter.FormatConfig formatConfig;
|
|
gen.GetDefaultDisplayFormat(out opNames, out formatConfig);
|
|
|
|
SetWidthDisamSettings(formatConfig.mForceAbsOpcodeSuffix,
|
|
formatConfig.mForceLongOpcodeSuffix,
|
|
formatConfig.mForceAbsOperandPrefix,
|
|
formatConfig.mForceLongOperandPrefix);
|
|
SelectExpressionStyle(formatConfig.mExpressionMode);
|
|
// dirty flag set by change watchers if one or more fields have changed
|
|
}
|
|
|
|
private void QuickFmtDefaultButton_Click(object sender, RoutedEventArgs e) {
|
|
SetWidthDisamSettings(null, "l", "a:", "f:");
|
|
SelectExpressionStyle(ExpressionMode.Common);
|
|
// dirty flag set by change watchers if one or more fields have changed
|
|
}
|
|
|
|
#endregion Display Format
|
|
|
|
#region PseudoOp
|
|
|
|
/// <summary>
|
|
/// Map pseudo-op text entry fields to PseudoOpName properties.
|
|
/// </summary>
|
|
private class TextBoxPropertyMap {
|
|
public TextBox TextBox { get; private set; }
|
|
public PropertyInfo PropInfo { get; private set; }
|
|
|
|
public TextBoxPropertyMap(TextBox textBox, string propName) {
|
|
TextBox = textBox;
|
|
PropInfo = typeof(PseudoOp.PseudoOpNames).GetProperty(propName);
|
|
}
|
|
}
|
|
private TextBoxPropertyMap[] mPseudoNameMap;
|
|
|
|
private void Loaded_PseudoOp() {
|
|
string opStrCereal = mSettings.GetString(AppSettings.FMT_PSEUDO_OP_NAMES, null);
|
|
if (!string.IsNullOrEmpty(opStrCereal)) {
|
|
PseudoOp.PseudoOpNames opNames = PseudoOp.PseudoOpNames.Deserialize(opStrCereal);
|
|
ImportPseudoOpNames(opNames);
|
|
} else {
|
|
// no data available, populate with blanks
|
|
//PseudoOp.PseudoOpNames opNames = PseudoOp.sDefaultPseudoOpNames;
|
|
ImportPseudoOpNames(new PseudoOp.PseudoOpNames());
|
|
}
|
|
|
|
// No need to set this to anything specific.
|
|
pseudoOpQuickComboBox.SelectedIndex = 0;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Imports values from PseudoOpNames struct into text fields.
|
|
/// </summary>
|
|
private void ImportPseudoOpNames(PseudoOp.PseudoOpNames opNames) {
|
|
for (int i = 0; i < mPseudoNameMap.Length; i++) {
|
|
string str = (string)mPseudoNameMap[i].PropInfo.GetValue(opNames);
|
|
mPseudoNameMap[i].TextBox.Text = (str == null) ? string.Empty : str;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Exports values from text fields to a PseudoOpNames object.
|
|
/// </summary>
|
|
private PseudoOp.PseudoOpNames ExportPseudoOpNames() {
|
|
PseudoOp.PseudoOpNames opNames = new PseudoOp.PseudoOpNames();
|
|
for (int i = 0; i < mPseudoNameMap.Length; i++) {
|
|
// NOTE: PseudoOpNames must be a class (not a struct) or this will fail.
|
|
// SetValue() would be invoked on a boxed copy that is discarded afterward.
|
|
mPseudoNameMap[i].PropInfo.SetValue(opNames, mPseudoNameMap[i].TextBox.Text);
|
|
}
|
|
return opNames;
|
|
}
|
|
|
|
// Invoked when text is changed in any pseudo-op text box.
|
|
private void PseudoOpTextChanged(object sender, EventArgs e) {
|
|
// Just set the dirty flag. The (somewhat expensive) export will happen
|
|
// on Apply/OK.
|
|
IsDirty = true;
|
|
}
|
|
|
|
private void PseudoOpDefaultButton_Click(object sender, RoutedEventArgs e) {
|
|
ImportPseudoOpNames(new PseudoOp.PseudoOpNames());
|
|
}
|
|
|
|
private void PseudoOpSetButton_Click(object sender, RoutedEventArgs e) {
|
|
AssemblerInfo asmInfo = (AssemblerInfo)pseudoOpQuickComboBox.SelectedItem;
|
|
AsmGen.IGenerator gen = AssemblerInfo.GetGenerator(asmInfo.AssemblerId);
|
|
|
|
PseudoOp.PseudoOpNames opNames;
|
|
Asm65.Formatter.FormatConfig formatConfig;
|
|
gen.GetDefaultDisplayFormat(out opNames, out formatConfig);
|
|
ImportPseudoOpNames(opNames);
|
|
|
|
// dirty flag set by change watchers if one or more fields have changed
|
|
}
|
|
|
|
#endregion PseudoOp
|
|
}
|
|
|
|
public class AsmColWidthRule : ValidationRule {
|
|
public override ValidationResult Validate(object value, CultureInfo cultureInfo) {
|
|
// Validating TextBox input, so value should always be a string. Check anyway.
|
|
string strValue = Convert.ToString(value);
|
|
if (string.IsNullOrEmpty(strValue)) {
|
|
//Debug.WriteLine("VVV not string");
|
|
return new ValidationResult(false, "Could not convert to string");
|
|
}
|
|
|
|
if (int.TryParse(strValue, out int result)) {
|
|
if (result >= EditAppSettings.ASM_COL_MIN_WIDTH &&
|
|
result <= EditAppSettings.ASM_COL_MAX_WIDTH) {
|
|
return ValidationResult.ValidResult;
|
|
}
|
|
//Debug.WriteLine("VVV out of range: '" + strValue + "' (" + result + ")");
|
|
return new ValidationResult(false, "Column width out of range");
|
|
}
|
|
|
|
//Debug.WriteLine("VVV not valid integer: '" + strValue + "'");
|
|
return new ValidationResult(false, "Invalid integer value: '" + strValue + "'");
|
|
}
|
|
}
|
|
}
|