Play-/Source/ui_win32/MemoryView.cpp
2015-05-06 01:27:59 -04:00

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;
}