Mesen-X/GUI.NET/Debugger/frmPpuViewer.cs

454 lines
15 KiB
C#

using System;
using System.Drawing;
using System.Threading.Tasks;
using System.Windows.Forms;
using Mesen.GUI.Config;
using Mesen.GUI.Forms;
namespace Mesen.GUI.Debugger
{
public partial class frmPpuViewer : BaseForm
{
private DateTime _lastUpdate = DateTime.MinValue;
private InteropEmu.NotificationListener _notifListener;
private TabPage _selectedTab;
private bool _refreshing = false;
private Size _originalSize;
private bool _isCompact = false;
private bool _isZoomed = false;
private PpuViewerMode _mode;
private static int _nextPpuViewerId = 0;
private int _ppuViewerId = 0;
public frmPpuViewer(PpuViewerMode mode = PpuViewerMode.Combined)
{
InitializeComponent();
_ppuViewerId = GetNextPpuViewerId();
_mode = mode;
if(Program.IsMono) {
btnToggleView.Top -= 1;
chkToggleZoom.Top -= 1;
}
this._selectedTab = this.tpgNametableViewer;
this.mnuAutoRefresh.Checked = ConfigManager.Config.DebugInfo.PpuAutoRefresh;
this.mnuRefreshOnBreak.Checked = ConfigManager.Config.DebugInfo.PpuRefreshOnBreak;
this.mnuShowInformationOverlay.Checked = ConfigManager.Config.DebugInfo.PpuShowInformationOverlay;
this.ctrlNametableViewer.Connect(this.ctrlChrViewer);
Point? startupLocation;
switch(_mode) {
case PpuViewerMode.NametableViewer: startupLocation = ConfigManager.Config.DebugInfo.PpuNametableViewerLocation; break;
case PpuViewerMode.ChrViewer: startupLocation = ConfigManager.Config.DebugInfo.PpuChrViewerLocation; break;
case PpuViewerMode.SpriteViewer: startupLocation = ConfigManager.Config.DebugInfo.PpuSpriteViewerLocation; break;
case PpuViewerMode.PaletteViewer: startupLocation = ConfigManager.Config.DebugInfo.PpuPaletteViewerLocation; break;
default: startupLocation = ConfigManager.Config.DebugInfo.PpuWindowLocation; break;
}
RestoreLocation(startupLocation);
}
public static int GetNextPpuViewerId()
{
return _nextPpuViewerId++;
}
private void InitShortcuts()
{
mnuRefresh.InitShortcut(this, nameof(DebuggerShortcutsConfig.Refresh));
}
protected override void OnLoad(EventArgs e)
{
base.OnLoad(e);
if(!this.DesignMode) {
this._notifListener = new InteropEmu.NotificationListener(ConfigManager.Config.DebugInfo.DebugConsoleId);
this._notifListener.OnNotification += this._notifListener_OnNotification;
this.ctrlScanlineCycle.Initialize(_ppuViewerId, ConfigManager.Config.DebugInfo.PpuDisplayScanline, ConfigManager.Config.DebugInfo.PpuDisplayCycle);
GetData();
this.ctrlNametableViewer.RefreshViewer();
this.ctrlChrViewer.RefreshViewer();
this.ctrlSpriteViewer.RefreshViewer();
this.ctrlPaletteViewer.RefreshViewer();
this.InitShortcuts();
this.UpdateRefreshSpeedMenu();
string toggleViewTooltip = "Toggle Compact/Normal View";
if(ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleView != Keys.None) {
toggleViewTooltip += " (" + DebuggerShortcutsConfig.GetShortcutDisplay(ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleView) + ")";
}
this.toolTip.SetToolTip(this.btnToggleView, toggleViewTooltip);
string toggleZoomTooltip = "Toggle 2x Zoom";
if(ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleZoom != Keys.None) {
toggleZoomTooltip += " (" + DebuggerShortcutsConfig.GetShortcutDisplay(ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleZoom) + ")";
}
this.toolTip.SetToolTip(this.chkToggleZoom, toggleZoomTooltip);
_selectedTab = tabMain.SelectedTab;
if(_mode != PpuViewerMode.Combined) {
TabPage page = _selectedTab;
switch(_mode) {
case PpuViewerMode.NametableViewer: page = tpgNametableViewer; break;
case PpuViewerMode.ChrViewer: page = tpgChrViewer; break;
case PpuViewerMode.SpriteViewer: page = tpgSpriteViewer; break;
case PpuViewerMode.PaletteViewer: page = tpgPaletteViewer; break;
}
_selectedTab = page;
tabMain.SelectTab(page);
ToggleView();
}
}
}
protected override void OnFormClosing(FormClosingEventArgs e)
{
base.OnFormClosing(e);
this._notifListener.OnNotification -= this._notifListener_OnNotification;
Point location = this.WindowState != FormWindowState.Normal ? this.RestoreBounds.Location : this.Location;
switch(_mode) {
case PpuViewerMode.NametableViewer: ConfigManager.Config.DebugInfo.PpuNametableViewerLocation = location; break;
case PpuViewerMode.ChrViewer: ConfigManager.Config.DebugInfo.PpuChrViewerLocation = location; break;
case PpuViewerMode.SpriteViewer: ConfigManager.Config.DebugInfo.PpuSpriteViewerLocation = location; break;
case PpuViewerMode.PaletteViewer: ConfigManager.Config.DebugInfo.PpuPaletteViewerLocation = location; break;
default: ConfigManager.Config.DebugInfo.PpuWindowLocation = location; break;
}
ConfigManager.Config.DebugInfo.PpuDisplayScanline = ctrlScanlineCycle.Scanline;
ConfigManager.Config.DebugInfo.PpuDisplayCycle = ctrlScanlineCycle.Cycle;
ConfigManager.ApplyChanges();
InteropEmu.DebugClearPpuViewerSettings(_ppuViewerId);
}
private void _notifListener_OnNotification(InteropEmu.NotificationEventArgs e)
{
switch(e.NotificationType) {
case InteropEmu.ConsoleNotificationType.CodeBreak:
case InteropEmu.ConsoleNotificationType.GamePaused:
if(ConfigManager.Config.DebugInfo.PpuRefreshOnBreak) {
this.GetData();
this.BeginInvoke((MethodInvoker)(() => this.RefreshViewers()));
}
break;
case InteropEmu.ConsoleNotificationType.PpuViewerDisplayFrame:
if(e.Parameter.ToInt32() == _ppuViewerId) {
int refreshDelay = 60;
switch(ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed) {
case RefreshSpeed.Low: refreshDelay= 60; break;
case RefreshSpeed.Normal: refreshDelay = 30; break;
case RefreshSpeed.High: refreshDelay = 12; break;
}
if(ConfigManager.Config.DebugInfo.PpuAutoRefresh && !_refreshing && (DateTime.Now - _lastUpdate).Milliseconds > refreshDelay) {
_lastUpdate = DateTime.Now;
this.GetData();
this.BeginInvoke((MethodInvoker)(() => this.RefreshViewers()));
}
}
break;
case InteropEmu.ConsoleNotificationType.GameLoaded:
//Configuration is lost when debugger is restarted (when switching game or power cycling)
ctrlScanlineCycle.RefreshSettings();
break;
}
}
private void GetData()
{
if(_isCompact) {
//In compact mode, only load data for the current tab
if(_selectedTab == this.tpgNametableViewer) {
this.ctrlNametableViewer.GetData();
} else if(_selectedTab == this.tpgChrViewer) {
this.ctrlChrViewer.GetData();
} else if(_selectedTab == this.tpgSpriteViewer) {
this.ctrlSpriteViewer.GetData();
} else if(_selectedTab == this.tpgPaletteViewer) {
this.ctrlPaletteViewer.GetData();
}
} else {
this.ctrlNametableViewer.GetData();
this.ctrlChrViewer.GetData();
this.ctrlSpriteViewer.GetData();
this.ctrlPaletteViewer.GetData();
}
}
private void RefreshViewers()
{
_refreshing = true;
if(_selectedTab == this.tpgNametableViewer) {
this.ctrlNametableViewer.RefreshViewer();
} else if(_selectedTab == this.tpgChrViewer) {
this.ctrlChrViewer.RefreshViewer();
} else if(_selectedTab == this.tpgSpriteViewer) {
this.ctrlSpriteViewer.RefreshViewer();
} else if(_selectedTab == this.tpgPaletteViewer) {
this.ctrlPaletteViewer.RefreshViewer();
}
_refreshing = false;
}
private void mnuRefresh_Click(object sender, EventArgs e)
{
this.GetData();
this.RefreshViewers();
}
private void mnuClose_Click(object sender, EventArgs e)
{
this.Close();
}
private void mnuAutoRefresh_Click(object sender, EventArgs e)
{
ConfigManager.Config.DebugInfo.PpuAutoRefresh = this.mnuAutoRefresh.Checked;
ConfigManager.ApplyChanges();
}
private void mnuRefreshOnBreak_Click(object sender, EventArgs e)
{
ConfigManager.Config.DebugInfo.PpuRefreshOnBreak = this.mnuRefreshOnBreak.Checked;
ConfigManager.ApplyChanges();
}
private void mnuShowInformationOverlay_Click(object sender, EventArgs e)
{
ConfigManager.Config.DebugInfo.PpuShowInformationOverlay = this.mnuShowInformationOverlay.Checked;
ConfigManager.ApplyChanges();
}
private void tabMain_SelectedIndexChanged(object sender, EventArgs e)
{
bool wasZoomedIn = _isZoomed;
if(wasZoomedIn) {
this.ToggleZoom();
}
this._selectedTab = this.tabMain.SelectedTab;
if(wasZoomedIn) {
this.ToggleZoom();
}
if(InteropEmu.DebugIsExecutionStopped()) {
//Refresh data when changing tabs when not running
this.RefreshViewers();
}
}
private void ToggleZoom()
{
ICompactControl ctrl = null;
if(_selectedTab == tpgChrViewer) {
ctrl = ctrlChrViewer;
} else if(_selectedTab == tpgPaletteViewer) {
ctrl = ctrlPaletteViewer;
} else if(_selectedTab == tpgSpriteViewer) {
ctrl = ctrlSpriteViewer;
} else if(_selectedTab == tpgNametableViewer) {
ctrl = ctrlNametableViewer;
}
if(!_isZoomed) {
Size pictureSize = ctrl.GetCompactSize(false);
this.Size += pictureSize;
_originalSize += pictureSize;
ctrl.ScaleImage(2);
_isZoomed = true;
} else {
Size pictureSize = ctrl.GetCompactSize(false);
Size halfSize = new Size(pictureSize.Width / 2, pictureSize.Height / 2);
this.Size -= halfSize;
_originalSize -= halfSize;
ctrl.ScaleImage(0.5);
_isZoomed = false;
}
chkToggleZoom.Checked = _isZoomed;
}
protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
{
if(keyData == ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleZoom) {
ToggleZoom();
return true;
} else if(keyData == ConfigManager.Config.DebugInfo.Shortcuts.PpuViewer_ToggleView) {
ToggleView();
return true;
}
if(!this.ctrlScanlineCycle.ContainsFocus) {
if(this.tabMain.SelectedTab == tpgChrViewer) {
bool shift = keyData.HasFlag(Keys.Shift);
keyData &= ~Keys.Shift;
if(keyData >= Keys.D1 && keyData <= Keys.D9) {
if(shift) {
this.ctrlChrViewer.SelectPalette(keyData - Keys.D1);
} else {
this.ctrlChrViewer.SelectColor((keyData - Keys.D1) % 4);
}
return true;
}
if(keyData >= Keys.NumPad1 && keyData <= Keys.NumPad9) {
if(shift) {
this.ctrlChrViewer.SelectPalette(keyData - Keys.NumPad1);
} else {
this.ctrlChrViewer.SelectColor((keyData - Keys.NumPad1) % 4);
}
return true;
}
}
}
return base.ProcessCmdKey(ref msg, keyData);
}
public void SelectChrTile(int tileIndex, int paletteIndex, bool allowOpenWindow)
{
if(_isCompact && allowOpenWindow) {
//If in compact mode, don't move to the CHR tab, open or use another window instead
frmPpuViewer otherPpuViewer = null;
foreach(BaseForm frm in DebugWindowManager.GetWindows()) {
if(frm != this && frm is frmPpuViewer && (!((frmPpuViewer)frm)._isCompact || ((frmPpuViewer)frm)._selectedTab == ((frmPpuViewer)frm).tpgChrViewer)) {
//If a window exists and is either not in compact mode, or in compact mode and showing the CHR viewer, use it
otherPpuViewer = frm as frmPpuViewer;
break;
}
}
if(otherPpuViewer == null) {
//Open up a new viewer, in compact mode
otherPpuViewer = DebugWindowManager.OpenPpuViewer(PpuViewerMode.ChrViewer);
otherPpuViewer.SelectChrTile(tileIndex, paletteIndex, false);
} else {
//Reuse an existing viewer that's not in compact mode
otherPpuViewer.SelectChrTile(tileIndex, paletteIndex, false);
otherPpuViewer.BringToFront();
}
} else {
if(!InteropEmu.DebugIsExecutionStopped() || ConfigManager.Config.DebugInfo.PpuRefreshOnBreak) {
//Only change the palette if execution is not stopped (or if we're configured to refresh the viewer on break/pause)
//Otherwise, the CHR viewer will refresh its data (and it might not match the data we loaded at the specified scanline/cycle anymore)
ctrlChrViewer.SelectedPaletteIndex = paletteIndex;
}
ctrlChrViewer.SelectedTileIndex = tileIndex;
tabMain.SelectTab(tpgChrViewer);
_selectedTab = tpgChrViewer;
}
}
private void ctrlNametableViewer_OnSelectChrTile(int tileIndex, int paletteIndex)
{
SelectChrTile(tileIndex, paletteIndex, true);
}
private void ctrlSpriteViewer_OnSelectTilePalette(int tileIndex, int paletteIndex)
{
SelectChrTile(tileIndex, paletteIndex, true);
}
private void UpdateRefreshSpeedMenu()
{
mnuAutoRefreshLow.Checked = ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed == RefreshSpeed.Low;
mnuAutoRefreshNormal.Checked = ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed == RefreshSpeed.Normal;
mnuAutoRefreshHigh.Checked = ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed == RefreshSpeed.High;
}
private void mnuAutoRefreshSpeed_Click(object sender, EventArgs e)
{
if(sender == mnuAutoRefreshLow) {
ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed = RefreshSpeed.Low;
} else if(sender == mnuAutoRefreshNormal) {
ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed = RefreshSpeed.Normal;
} else if(sender == mnuAutoRefreshHigh) {
ConfigManager.Config.DebugInfo.PpuAutoRefreshSpeed = RefreshSpeed.High;
}
ConfigManager.ApplyChanges();
UpdateRefreshSpeedMenu();
}
private void ToggleCompactMode(PpuViewerMode mode, ICompactControl control, TabPage tab, string title)
{
if(!_isCompact) {
Point tabTopLeft = tabMain.PointToScreen(Point.Empty);
Point tabContentTopLeft = tab.PointToScreen(Point.Empty);
int heightGap = tabContentTopLeft.Y - tabTopLeft.Y + ctrlScanlineCycle.Height;
_isCompact = true;
_originalSize = this.Size;
Size size = control.GetCompactSize(true);
int widthDiff = ((Control)control).Width - size.Width;
int heightDiff = ((Control)control).Height - size.Height;
this.Controls.Add((Control)control);
((Control)control).BringToFront();
tabMain.Visible = false;
ctrlScanlineCycle.Visible = false;
this.Text = title;
this.Size = new Size(this.Width - widthDiff, this.Height - heightDiff - heightGap + 3);
} else {
_mode = PpuViewerMode.Combined;
_isCompact = false;
this.Size = _originalSize;
tabMain.Visible = true;
tab.Controls.Add((Control)control);
ctrlScanlineCycle.Visible = true;
this.Text = "PPU Viewer";
}
btnToggleView.Image = _isCompact ? Properties.Resources.Expand : Properties.Resources.Collapse;
}
private void ToggleView()
{
if(_selectedTab == tpgChrViewer) {
ToggleCompactMode(PpuViewerMode.ChrViewer, ctrlChrViewer, tpgChrViewer, "CHR Viewer");
} else if(_selectedTab == tpgPaletteViewer) {
ToggleCompactMode(PpuViewerMode.PaletteViewer, ctrlPaletteViewer, tpgPaletteViewer, "Palette Viewer");
} else if(_selectedTab == tpgSpriteViewer) {
ToggleCompactMode(PpuViewerMode.SpriteViewer, ctrlSpriteViewer, tpgSpriteViewer, "Sprite Viewer");
} else if(_selectedTab == tpgNametableViewer) {
ToggleCompactMode(PpuViewerMode.NametableViewer, ctrlNametableViewer, tpgNametableViewer, "Nametable Viewer");
}
}
private void btnToggleView_Click(object sender, EventArgs e)
{
ToggleView();
}
private void chkToggleZoom_Click(object sender, EventArgs e)
{
ToggleZoom();
}
}
public enum PpuViewerMode
{
Combined = 0,
NametableViewer = 1,
ChrViewer = 2,
SpriteViewer = 3,
PaletteViewer = 4,
}
public interface ICompactControl
{
Size GetCompactSize(bool includeMargins);
void ScaleImage(double scale);
}
}