mirror of
https://github.com/libretro/Play-.git
synced 2025-01-07 09:00:25 +00:00
445 lines
11 KiB
C++
445 lines
11 KiB
C++
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "win32/ClientDeviceContext.h"
|
|
#include "win32/DefaultWndClass.h"
|
|
#include "win32/Font.h"
|
|
#include "win32/DpiUtils.h"
|
|
#include "string_cast.h"
|
|
#include "lexical_cast_ex.h"
|
|
#include "MemoryView.h"
|
|
|
|
#define ADDRESSCHARS 8
|
|
#define PAGESIZE 10
|
|
|
|
CMemoryView::CMemoryView(HWND parentWnd, const RECT& rect)
|
|
: m_font(reinterpret_cast<HFONT>(GetStockObject(ANSI_FIXED_FONT)))
|
|
{
|
|
{
|
|
m_renderMetrics.xmargin = Framework::Win32::PointsToPixels(5);
|
|
m_renderMetrics.ymargin = Framework::Win32::PointsToPixels(5);
|
|
m_renderMetrics.yspace = Framework::Win32::PointsToPixels(0);
|
|
m_renderMetrics.byteSpacing = Framework::Win32::PointsToPixels(3);
|
|
m_renderMetrics.lineSectionSpacing = Framework::Win32::PointsToPixels(10);
|
|
}
|
|
|
|
Create(WS_EX_CLIENTEDGE, Framework::Win32::CDefaultWndClass::GetName(), _T(""), WS_VISIBLE | WS_VSCROLL | WS_CHILD, rect, parentWnd, NULL);
|
|
SetClassPtr();
|
|
|
|
UpdateScrollRange();
|
|
}
|
|
|
|
CMemoryView::~CMemoryView()
|
|
{
|
|
|
|
}
|
|
|
|
uint32 CMemoryView::GetSelection()
|
|
{
|
|
return m_selectionStart;
|
|
}
|
|
|
|
void CMemoryView::SetSelectionStart(unsigned int newAddress)
|
|
{
|
|
newAddress = std::max<int>(newAddress, 0);
|
|
newAddress = std::min<int>(newAddress, m_size - 1);
|
|
|
|
m_selectionStart = newAddress;
|
|
EnsureSelectionVisible();
|
|
UpdateCaretPosition();
|
|
OnSelectionChange(m_selectionStart);
|
|
}
|
|
|
|
void CMemoryView::UpdateScrollRange()
|
|
{
|
|
auto renderParams = GetRenderParams();
|
|
|
|
unsigned int totalLines = 0;
|
|
if(renderParams.bytesPerLine != 0)
|
|
{
|
|
totalLines = (m_size / renderParams.bytesPerLine);
|
|
if((m_size % renderParams.bytesPerLine) != 0) totalLines++;
|
|
}
|
|
else
|
|
{
|
|
totalLines = 0;
|
|
}
|
|
|
|
SCROLLINFO si;
|
|
memset(&si, 0, sizeof(SCROLLINFO));
|
|
si.cbSize = sizeof(SCROLLINFO);
|
|
si.nMin = 0;
|
|
if(totalLines <= renderParams.totallyVisibleLines)
|
|
{
|
|
si.nMax = 0;
|
|
}
|
|
else
|
|
{
|
|
si.nMax = totalLines - renderParams.totallyVisibleLines;
|
|
}
|
|
si.fMask = SIF_RANGE | SIF_DISABLENOSCROLL;
|
|
SetScrollInfo(m_hWnd, SB_VERT, &si, TRUE);
|
|
}
|
|
|
|
unsigned int CMemoryView::GetScrollOffset()
|
|
{
|
|
SCROLLINFO si;
|
|
memset(&si, 0, sizeof(SCROLLINFO));
|
|
si.cbSize = sizeof(SCROLLINFO);
|
|
si.fMask = SIF_POS;
|
|
GetScrollInfo(m_hWnd, SB_VERT, &si);
|
|
return si.nPos;
|
|
}
|
|
|
|
unsigned int CMemoryView::GetScrollThumbPosition()
|
|
{
|
|
SCROLLINFO si;
|
|
memset(&si, 0, sizeof(SCROLLINFO));
|
|
si.cbSize = sizeof(SCROLLINFO);
|
|
si.fMask = SIF_TRACKPOS;
|
|
GetScrollInfo(m_hWnd, SB_VERT, &si);
|
|
return si.nTrackPos;
|
|
}
|
|
|
|
HMENU CMemoryView::CreateContextualMenu()
|
|
{
|
|
HMENU bytesPerLineMenu = CreatePopupMenu();
|
|
AppendMenu(bytesPerLineMenu, MF_STRING | ((m_bytesPerLine == 0) ? MF_CHECKED : MF_UNCHECKED), ID_MEMORYVIEW_COLUMNS_AUTO, _T("Auto"));
|
|
AppendMenu(bytesPerLineMenu, MF_STRING | ((m_bytesPerLine == 16) ? MF_CHECKED : MF_UNCHECKED), ID_MEMORYVIEW_COLUMNS_16BYTES, _T("16 bytes"));
|
|
AppendMenu(bytesPerLineMenu, MF_STRING | ((m_bytesPerLine == 32) ? MF_CHECKED : MF_UNCHECKED), ID_MEMORYVIEW_COLUMNS_32BYTES, _T("32 bytes"));
|
|
|
|
HMENU menu = CreatePopupMenu();
|
|
AppendMenu(menu, MF_STRING | MF_POPUP, reinterpret_cast<UINT_PTR>(bytesPerLineMenu), _T("Bytes Per Line"));
|
|
|
|
return menu;
|
|
}
|
|
|
|
void CMemoryView::Paint(HDC hDC)
|
|
{
|
|
Framework::Win32::CDeviceContext deviceContext(hDC);
|
|
|
|
auto rwin = GetClientRect();
|
|
BitBlt(deviceContext, 0, 0, rwin.Right(), rwin.Bottom(), NULL, 0, 0, WHITENESS);
|
|
|
|
deviceContext.SelectObject(m_font);
|
|
|
|
auto fontSize = Framework::Win32::GetFixedFontSize(m_font);
|
|
|
|
auto renderParams = GetRenderParams();
|
|
unsigned int y = m_renderMetrics.ymargin;
|
|
uint32 address = renderParams.address;
|
|
|
|
for(unsigned int i = 0; i < renderParams.lines; i++)
|
|
{
|
|
if(address >= m_size)
|
|
{
|
|
break;
|
|
}
|
|
|
|
unsigned int bytesForCurrentLine = renderParams.bytesPerLine;
|
|
if((address + bytesForCurrentLine) >= m_size)
|
|
{
|
|
bytesForCurrentLine = (m_size - address);
|
|
}
|
|
|
|
unsigned int x = m_renderMetrics.xmargin;
|
|
deviceContext.TextOut(x, y, lexical_cast_hex<std::tstring>(address, ADDRESSCHARS).c_str());
|
|
x += (ADDRESSCHARS * fontSize.cx) + m_renderMetrics.lineSectionSpacing;
|
|
|
|
for(unsigned int j = 0; j < bytesForCurrentLine; j++)
|
|
{
|
|
deviceContext.TextOut(x, y, lexical_cast_hex<std::tstring>(GetByte(address + j), 2).c_str());
|
|
x += (2 * fontSize.cx) + m_renderMetrics.byteSpacing;
|
|
}
|
|
//Compensate for incomplete lines (when bytesForCurrentLine < bytesPerLine)
|
|
x += (renderParams.bytesPerLine - bytesForCurrentLine) * (2 * fontSize.cx + m_renderMetrics.byteSpacing);
|
|
|
|
x += m_renderMetrics.lineSectionSpacing;
|
|
|
|
for(unsigned int j = 0; j < bytesForCurrentLine; j++)
|
|
{
|
|
uint8 value = GetByte(address + j);
|
|
if(value < 0x20 || value > 0x7F)
|
|
{
|
|
value = '.';
|
|
}
|
|
|
|
char valueString[2];
|
|
valueString[0] = value;
|
|
valueString[1] = 0x00;
|
|
|
|
deviceContext.TextOut(x, y, string_cast<std::tstring>(valueString).c_str());
|
|
x += fontSize.cx;
|
|
}
|
|
|
|
y += (fontSize.cy + m_renderMetrics.yspace);
|
|
address += bytesForCurrentLine;
|
|
}
|
|
}
|
|
|
|
void CMemoryView::SetMemorySize(uint32 size)
|
|
{
|
|
m_size = size;
|
|
UpdateScrollRange();
|
|
}
|
|
|
|
uint32 CMemoryView::GetBytesPerLine() const
|
|
{
|
|
return m_bytesPerLine;
|
|
}
|
|
|
|
void CMemoryView::SetBytesPerLine(uint32 bytesPerLine)
|
|
{
|
|
m_bytesPerLine = bytesPerLine;
|
|
UpdateScrollRange();
|
|
EnsureSelectionVisible();
|
|
UpdateCaretPosition();
|
|
Redraw();
|
|
}
|
|
|
|
long CMemoryView::OnVScroll(unsigned int scrollType, unsigned int)
|
|
{
|
|
SCROLLINFO si;
|
|
int offset = (int)GetScrollOffset();
|
|
switch(scrollType)
|
|
{
|
|
case SB_LINEDOWN:
|
|
offset++;
|
|
break;
|
|
case SB_LINEUP:
|
|
offset--;
|
|
break;
|
|
case SB_PAGEDOWN:
|
|
offset += PAGESIZE;
|
|
break;
|
|
case SB_PAGEUP:
|
|
offset -= PAGESIZE;
|
|
break;
|
|
case SB_THUMBPOSITION:
|
|
case SB_THUMBTRACK:
|
|
offset = GetScrollThumbPosition();
|
|
break;
|
|
default:
|
|
return FALSE;
|
|
break;
|
|
}
|
|
|
|
memset(&si, 0, sizeof(SCROLLINFO));
|
|
si.cbSize = sizeof(SCROLLINFO);
|
|
si.nPos = offset;
|
|
si.fMask = SIF_POS;
|
|
SetScrollInfo(m_hWnd, SB_VERT, &si, TRUE);
|
|
|
|
UpdateCaretPosition();
|
|
|
|
Redraw();
|
|
return TRUE;
|
|
}
|
|
|
|
long CMemoryView::OnCommand(unsigned short id, unsigned short, HWND)
|
|
{
|
|
switch(id)
|
|
{
|
|
case ID_MEMORYVIEW_COLUMNS_AUTO:
|
|
SetBytesPerLine(0);
|
|
break;
|
|
case ID_MEMORYVIEW_COLUMNS_16BYTES:
|
|
SetBytesPerLine(0x10);
|
|
break;
|
|
case ID_MEMORYVIEW_COLUMNS_32BYTES:
|
|
SetBytesPerLine(0x20);
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
long CMemoryView::OnSize(unsigned int type, unsigned int x, unsigned int y)
|
|
{
|
|
UpdateScrollRange();
|
|
UpdateCaretPosition();
|
|
CCustomDrawn::OnSize(type, x, y);
|
|
return TRUE;
|
|
}
|
|
|
|
long CMemoryView::OnSetFocus()
|
|
{
|
|
auto fontSize = Framework::Win32::GetFixedFontSize(m_font);
|
|
|
|
CreateCaret(m_hWnd, NULL, 2, fontSize.cy);
|
|
ShowCaret(m_hWnd);
|
|
|
|
UpdateCaretPosition();
|
|
|
|
Redraw();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnKillFocus()
|
|
{
|
|
HideCaret(m_hWnd);
|
|
DestroyCaret();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnMouseWheel(int x, int y, short z)
|
|
{
|
|
if(z < 0)
|
|
{
|
|
OnVScroll(SB_LINEDOWN, 0);
|
|
}
|
|
else
|
|
{
|
|
OnVScroll(SB_LINEUP, 0);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnLeftButtonDown(int x, int y)
|
|
{
|
|
SetFocus();
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnLeftButtonUp(int x, int y)
|
|
{
|
|
auto fontSize = Framework::Win32::GetFixedFontSize(m_font);
|
|
|
|
y -= m_renderMetrics.ymargin;
|
|
x -= m_renderMetrics.xmargin + (ADDRESSCHARS * fontSize.cx) + m_renderMetrics.lineSectionSpacing;
|
|
|
|
if(y < 0) return FALSE;
|
|
if(x < 0) return FALSE;
|
|
|
|
unsigned int selectedLine = y / (fontSize.cy + m_renderMetrics.yspace);
|
|
unsigned int selectedByte = x / ((2 * fontSize.cx) + m_renderMetrics.byteSpacing);
|
|
|
|
auto renderParams = GetRenderParams();
|
|
|
|
if(selectedByte >= renderParams.bytesPerLine) return FALSE;
|
|
|
|
SetSelectionStart(renderParams.address + selectedByte + (selectedLine * renderParams.bytesPerLine));
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnRightButtonUp(int x, int y)
|
|
{
|
|
HMENU contextualMenu = CreateContextualMenu();
|
|
|
|
POINT pt;
|
|
pt.x = x;
|
|
pt.y = y;
|
|
ClientToScreen(m_hWnd, &pt);
|
|
|
|
TrackPopupMenu(contextualMenu, 0, pt.x, pt.y, 0, m_hWnd, NULL);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
long CMemoryView::OnKeyDown(WPARAM key, LPARAM)
|
|
{
|
|
switch(key)
|
|
{
|
|
case VK_RIGHT:
|
|
SetSelectionStart(m_selectionStart + 1);
|
|
break;
|
|
case VK_LEFT:
|
|
SetSelectionStart(m_selectionStart - 1);
|
|
break;
|
|
case VK_UP:
|
|
case VK_DOWN:
|
|
{
|
|
auto renderParams = GetRenderParams();
|
|
SetSelectionStart(m_selectionStart + ((key == VK_UP) ? (-static_cast<int>(renderParams.bytesPerLine)) : (renderParams.bytesPerLine)));
|
|
}
|
|
break;
|
|
case VK_PRIOR:
|
|
case VK_NEXT:
|
|
{
|
|
auto renderParams = GetRenderParams();
|
|
SetSelectionStart(m_selectionStart + ((key == VK_PRIOR) ? (-static_cast<int>(renderParams.bytesPerLine * PAGESIZE)) : (renderParams.bytesPerLine * PAGESIZE)));
|
|
}
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void CMemoryView::UpdateCaretPosition()
|
|
{
|
|
auto fontSize = Framework::Win32::GetFixedFontSize(m_font);
|
|
auto renderParams = GetRenderParams();
|
|
if(
|
|
(renderParams.bytesPerLine != 0) &&
|
|
(m_selectionStart >= renderParams.address) &&
|
|
(m_selectionStart <= (renderParams.address + (renderParams.lines * renderParams.bytesPerLine)))
|
|
)
|
|
{
|
|
unsigned int selectionStart = m_selectionStart - renderParams.address;
|
|
int x = m_renderMetrics.xmargin + (ADDRESSCHARS * fontSize.cx) + m_renderMetrics.lineSectionSpacing + (selectionStart % renderParams.bytesPerLine) * ((2 * fontSize.cx) + m_renderMetrics.byteSpacing);
|
|
int y = m_renderMetrics.ymargin + (fontSize.cy + m_renderMetrics.yspace) * (selectionStart / renderParams.bytesPerLine);
|
|
SetCaretPos(x, y);
|
|
}
|
|
else
|
|
{
|
|
SetCaretPos(-20, -20);
|
|
}
|
|
}
|
|
|
|
void CMemoryView::EnsureSelectionVisible()
|
|
{
|
|
auto renderParams = GetRenderParams();
|
|
uint32 address = m_selectionStart;
|
|
|
|
if(renderParams.bytesPerLine == 0) return;
|
|
if(address >= m_size) return;
|
|
|
|
uint32 visibleMin = renderParams.address;
|
|
uint32 visibleMax = renderParams.address + (renderParams.bytesPerLine * renderParams.totallyVisibleLines);
|
|
|
|
//Already visible
|
|
if(address >= visibleMin && address < visibleMax)
|
|
{
|
|
return;
|
|
}
|
|
|
|
unsigned int line = (address / renderParams.bytesPerLine);
|
|
|
|
SCROLLINFO si;
|
|
memset(&si, 0, sizeof(SCROLLINFO));
|
|
si.cbSize = sizeof(SCROLLINFO);
|
|
si.nPos = line;
|
|
si.fMask = SIF_POS;
|
|
SetScrollInfo(m_hWnd, SB_VERT, &si, TRUE);
|
|
|
|
Redraw();
|
|
}
|
|
|
|
CMemoryView::RENDERPARAMS CMemoryView::GetRenderParams()
|
|
{
|
|
auto fontSize = Framework::Win32::GetFixedFontSize(m_font);
|
|
|
|
RENDERPARAMS renderParams;
|
|
|
|
RECT clientRect(GetClientRect());
|
|
|
|
renderParams.totallyVisibleLines = (clientRect.bottom - (m_renderMetrics.ymargin * 2)) / (fontSize.cy + m_renderMetrics.yspace);
|
|
renderParams.lines = renderParams.totallyVisibleLines + 1;
|
|
|
|
if(m_bytesPerLine == 0)
|
|
{
|
|
//lineSize = (2 * m_renderMetrics.xmargin) + (2 * m_renderMetrics.lineSectionSpacing) + (ADDRESSCHARS * cx) + bytesPerLine * (2 * cx + m_renderMetrics.byteSpacing) + bytesPerLine * cx
|
|
renderParams.bytesPerLine = clientRect.right - (2 * m_renderMetrics.xmargin) - (2 * m_renderMetrics.lineSectionSpacing) - (ADDRESSCHARS * fontSize.cx);
|
|
renderParams.bytesPerLine /= ((2 * fontSize.cx + m_renderMetrics.byteSpacing) + (fontSize.cx));
|
|
}
|
|
else
|
|
{
|
|
renderParams.bytesPerLine = m_bytesPerLine;
|
|
}
|
|
|
|
renderParams.address = GetScrollOffset() * renderParams.bytesPerLine;
|
|
|
|
return renderParams;
|
|
}
|