mirror of
https://github.com/fadden/6502bench.git
synced 2024-11-19 21:31:30 +00:00
f33cd7d8a6
The previous code output a character in single-quotes if it was standard ASCII, double-quotes if high ASCII, or hex if it was neither of those. If a flag was set, high ASCII would also be output as hex. The new system takes the character value and an encoding identifier. The identifier selects the character converter and delimiter pattern, and puts the two together to generate the operand. While doing this I realized that I could trivially support high ASCII character arguments in all assemblers by setting the delimiter pattern to "'#' | $80". In FormatDescriptor, I had previously renamed the "Ascii" sub-type "LowAscii" so it wouldn't be confused, but I dislike filling the project file with "LowAscii" when "Ascii" is more accurate and less confusing. So I switched it back, and we now check the project file version number when deciding what to do with an ASCII item. The CharEncoding tests/converters were also renamed. Moved the default delimiter patterns to the string table. Widened the delimiter pattern input fields slightly. Added a read- only TextBox with assorted non-typewriter quotes and things so people have something to copy text from.
986 lines
40 KiB
C#
986 lines
40 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(strReverseTextBox, "StrReverse"),
|
|
new TextBoxPropertyMap(strLen8TextBox, "StrLen8"),
|
|
new TextBoxPropertyMap(strLen16TextBox, "StrLen16"),
|
|
new TextBoxPropertyMap(strNullTermTextBox, "StrNullTerm"),
|
|
new TextBoxPropertyMap(strDciTextBox, "StrDci"),
|
|
};
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
public string AsciiDelimPat {
|
|
get {
|
|
return mSettings.GetString(AppSettings.CHR_ASCII_DELIM_PAT,
|
|
Res.Strings.DEFAULT_ASCII_DELIM_PAT);
|
|
}
|
|
set {
|
|
mSettings.SetString(AppSettings.CHR_ASCII_DELIM_PAT, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public string HighAsciiDelimPat {
|
|
get {
|
|
return mSettings.GetString(AppSettings.CHR_HIGH_ASCII_DELIM_PAT,
|
|
Res.Strings.DEFAULT_HIGH_ASCII_DELIM_PAT);
|
|
}
|
|
set {
|
|
mSettings.SetString(AppSettings.CHR_HIGH_ASCII_DELIM_PAT, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public string PetsciiDelimPat {
|
|
get {
|
|
return mSettings.GetString(AppSettings.CHR_C64_PETSCII_DELIM_PAT,
|
|
Res.Strings.DEFAULT_C64_PETSCII_DELIM_PAT);
|
|
}
|
|
set {
|
|
mSettings.SetString(AppSettings.CHR_C64_PETSCII_DELIM_PAT, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
public string ScreenCodeDelimPat {
|
|
get {
|
|
return mSettings.GetString(AppSettings.CHR_C64_SCREEN_CODE_DELIM_PAT,
|
|
Res.Strings.DEFAULT_C64_SCREEN_CODE_DELIM_PAT);
|
|
}
|
|
set {
|
|
mSettings.SetString(AppSettings.CHR_C64_SCREEN_CODE_DELIM_PAT, value);
|
|
OnPropertyChanged();
|
|
IsDirty = true;
|
|
}
|
|
}
|
|
|
|
private void DefaultTextDelimitersButton_Click(object sender, RoutedEventArgs e) {
|
|
AsciiDelimPat = Res.Strings.DEFAULT_ASCII_DELIM_PAT;
|
|
HighAsciiDelimPat = Res.Strings.DEFAULT_HIGH_ASCII_DELIM_PAT;
|
|
PetsciiDelimPat = Res.Strings.DEFAULT_C64_PETSCII_DELIM_PAT;
|
|
ScreenCodeDelimPat = Res.Strings.DEFAULT_C64_SCREEN_CODE_DELIM_PAT;
|
|
}
|
|
|
|
#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;
|
|
}
|
|
|
|
}
|
|
|
|
private void AsmExePathTextBox_TextChanged(object sender, TextChangedEventArgs e) {
|
|
if (IsLoaded) {
|
|
// We don't really need to be updating AssemblerConfig every time they type
|
|
// a character, but it's fine.
|
|
AssemblerInfo asmInfo = (AssemblerInfo)asmConfigComboBox.SelectedItem;
|
|
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;
|
|
|
|
// Create text field listeners.
|
|
foreach (TextBoxPropertyMap pmap in mPseudoNameMap) {
|
|
pmap.TextBox.TextChanged += PseudoOpTextChanged;
|
|
}
|
|
}
|
|
|
|
/// <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
|
|
}
|
|
|
|
#region Validation rules
|
|
|
|
/// <summary>
|
|
/// Text entry validation rule for assembler column widths.
|
|
/// </summary>
|
|
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 + "'");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Text entry validation rule for text string delimiter patterns.
|
|
/// </summary>
|
|
public class StringDelimiterRule : ValidationRule {
|
|
public override ValidationResult Validate(object value, CultureInfo cultureInfo) {
|
|
string strValue = Convert.ToString(value);
|
|
int firstHash = strValue.IndexOf('#');
|
|
if (firstHash < 0) {
|
|
return new ValidationResult(false, "Must include exactly one '#'");
|
|
}
|
|
if (strValue.LastIndexOf('#') != firstHash) {
|
|
return new ValidationResult(false, "Found more than one '#'");
|
|
}
|
|
return ValidationResult.ValidResult;
|
|
}
|
|
}
|
|
|
|
#endregion Validation rules
|
|
}
|