fceux/src/drivers/Qt/RamSearch.cpp

810 lines
20 KiB
C++
Raw Normal View History

2020-10-11 15:25:35 +00:00
// RamSearch.cpp
//
#include <stdio.h>
#include <stdlib.h>
2020-10-11 15:25:35 +00:00
#include <stdint.h>
#include <string.h>
#include <ctype.h>
#include <string>
#include <SDL.h>
#include <QMenuBar>
#include <QAction>
#include <QHeaderView>
#include <QCloseEvent>
#include <QGroupBox>
#include <QLineEdit>
#include <QRadioButton>
#include <QFileDialog>
#include <QPainter>
#include "../../types.h"
#include "../../fceu.h"
#include "../../cheat.h"
#include "../../debug.h"
#include "Qt/main.h"
#include "Qt/dface.h"
#include "Qt/input.h"
#include "Qt/config.h"
#include "Qt/keyscan.h"
#include "Qt/fceuWrapper.h"
#include "Qt/RamWatch.h"
#include "Qt/RamSearch.h"
#include "Qt/CheatsConf.h"
#include "Qt/ConsoleUtilities.h"
2020-10-11 15:25:35 +00:00
static bool ShowROM = false;
static RamSearchDialog_t *ramSearchWin = NULL;
static uint64_t iterMask = 0x01;
struct memoryState_t
{
union {
int8_t i;
uint8_t u;
} v8;
union {
int16_t i;
uint16_t u;
} v16;
union {
int32_t i;
uint32_t u;
} v32;
2020-10-11 15:25:35 +00:00
};
struct memoryLocation_t
2020-10-11 15:25:35 +00:00
{
int addr;
2020-10-11 15:25:35 +00:00
memoryState_t val;
2020-10-11 15:25:35 +00:00
std::vector <memoryState_t> hist;
2020-10-11 15:25:35 +00:00
uint32_t chgCount;
uint64_t elimMask;
2020-10-11 15:25:35 +00:00
memoryLocation_t(void)
2020-10-11 15:25:35 +00:00
{
addr = 0; val.v32.u = 0;
chgCount = 0; elimMask = 0;
2020-10-11 15:25:35 +00:00
}
};
2020-10-12 04:09:00 +00:00
static struct memoryLocation_t memLoc[0x10000];
2020-10-11 15:25:35 +00:00
static std::list <struct memoryLocation_t*> actvSrchList;
2020-10-11 15:25:35 +00:00
static int dpySize = 'b';
static int dpyType = 's';
2020-10-12 04:09:00 +00:00
static bool chkMisAligned = false;
2020-10-11 15:25:35 +00:00
//----------------------------------------------------------------------------
void openRamSearchWindow( QWidget *parent )
{
if ( ramSearchWin != NULL )
{
return;
}
ramSearchWin = new RamSearchDialog_t(parent);
ramSearchWin->show();
}
//----------------------------------------------------------------------------
RamSearchDialog_t::RamSearchDialog_t(QWidget *parent)
: QDialog( parent )
{
QVBoxLayout *mainLayout;
QHBoxLayout *hbox, *hbox1, *hbox2, *hbox3;
QVBoxLayout *vbox, *vbox1, *vbox2;
QGridLayout *grid;
QGroupBox *frame;
setWindowTitle("RAM Search");
resize( 512, 512 );
mainLayout = new QVBoxLayout();
hbox1 = new QHBoxLayout();
mainLayout->addLayout( hbox1 );
grid = new QGridLayout();
ramView = new QRamSearchView(this);
vbar = new QScrollBar( Qt::Vertical, this );
hbar = new QScrollBar( Qt::Horizontal, this );
grid->addWidget( ramView, 0, 0 );
grid->addWidget( vbar , 0, 1 );
grid->addWidget( hbar , 1, 0 );
connect( hbar, SIGNAL(valueChanged(int)), this, SLOT(hbarChanged(int)) );
connect( vbar, SIGNAL(valueChanged(int)), this, SLOT(vbarChanged(int)) );
ramView->setScrollBars( hbar, vbar );
hbar->setMinimum(0);
hbar->setMaximum(100);
vbar->setMinimum(0);
2020-10-12 04:09:00 +00:00
vbar->setMaximum(ShowROM ? 0x10000 : 0x8000);
vbar->setValue(0);
vbox = new QVBoxLayout();
hbox1->addLayout( grid, 100);
hbox1->addLayout( vbox, 1 );
searchButton = new QPushButton( tr("Search") );
vbox->addWidget( searchButton );
connect( searchButton, SIGNAL(clicked(void)), this, SLOT(runSearch(void)));
resetButton = new QPushButton( tr("Reset") );
vbox->addWidget( resetButton );
connect( resetButton, SIGNAL(clicked(void)), this, SLOT(resetSearch(void)));
clearChangeButton = new QPushButton( tr("Clear Change") );
vbox->addWidget( clearChangeButton );
2020-10-12 04:09:00 +00:00
connect( clearChangeButton, SIGNAL(clicked(void)), this, SLOT(clearChangeCounts(void)));
2020-10-12 04:09:00 +00:00
undoButton = new QPushButton( tr("Undo") );
vbox->addWidget( undoButton );
//connect( undoButton, SIGNAL(clicked(void)), this, SLOT(removeWatchClicked(void)));
//undoButton->setEnabled(false);
searchROMCbox = new QCheckBox( tr("Search ROM") );
vbox->addWidget( searchROMCbox );
2020-10-12 04:09:00 +00:00
searchROMCbox->setChecked( ShowROM );
connect( searchROMCbox, SIGNAL(stateChanged(int)), this, SLOT(searchROMChanged(int)));
elimButton = new QPushButton( tr("Eliminate") );
vbox->addWidget( elimButton );
//connect( elimButton, SIGNAL(clicked(void)), this, SLOT(newWatchClicked(void)));
2020-10-12 04:09:00 +00:00
elimButton->setEnabled(false);
watchButton = new QPushButton( tr("Watch") );
vbox->addWidget( watchButton );
//connect( watchButton, SIGNAL(clicked(void)), this, SLOT(dupWatchClicked(void)));
watchButton->setEnabled(false);
addCheatButton = new QPushButton( tr("Add Cheat") );
vbox->addWidget( addCheatButton );
//connect( addCheatButton, SIGNAL(clicked(void)), this, SLOT(sepWatchClicked(void)));
addCheatButton->setEnabled(false);
hexEditButton = new QPushButton( tr("Hex Editor") );
vbox->addWidget( hexEditButton );
//connect( hexEditButton, SIGNAL(clicked(void)), this, SLOT(sepWatchClicked(void)));
hexEditButton->setEnabled(false);
hbox2 = new QHBoxLayout();
mainLayout->addLayout( hbox2 );
frame = new QGroupBox( tr("Comparison Operator") );
vbox = new QVBoxLayout();
hbox2->addWidget( frame );
frame->setLayout(vbox);
lt_btn = new QRadioButton( tr("Less Than") );
gt_btn = new QRadioButton( tr("Greater Than") );
le_btn = new QRadioButton( tr("Less Than or Equal To") );
ge_btn = new QRadioButton( tr("Greater Than or Equal To") );
eq_btn = new QRadioButton( tr("Equal To") );
ne_btn = new QRadioButton( tr("Not Equal To") );
df_btn = new QRadioButton( tr("Different By:") );
md_btn = new QRadioButton( tr("Modulo") );
eq_btn->setChecked(true);
diffByEdit = new QLineEdit();
moduloEdit = new QLineEdit();
vbox->addWidget( lt_btn );
vbox->addWidget( gt_btn );
vbox->addWidget( le_btn );
vbox->addWidget( ge_btn );
vbox->addWidget( eq_btn );
vbox->addWidget( ne_btn );
hbox = new QHBoxLayout();
vbox->addLayout( hbox );
hbox->addWidget( df_btn );
hbox->addWidget( diffByEdit );
hbox = new QHBoxLayout();
vbox->addLayout( hbox );
hbox->addWidget( md_btn );
hbox->addWidget( moduloEdit );
vbox1 = new QVBoxLayout();
grid = new QGridLayout();
hbox2->addLayout( vbox1 );
frame = new QGroupBox( tr("Compare To/By") );
frame->setLayout( grid );
vbox1->addWidget( frame );
pv_btn = new QRadioButton( tr("Previous Value") );
sv_btn = new QRadioButton( tr("Specific Value:") );
sa_btn = new QRadioButton( tr("Specific Address:") );
nc_btn = new QRadioButton( tr("Number of Changes:") );
pv_btn->setChecked(true);
specValEdit = new QLineEdit();
specAddrEdit = new QLineEdit();
numChangeEdit = new QLineEdit();
grid->addWidget( pv_btn , 0, 0, Qt::AlignLeft );
grid->addWidget( sv_btn , 1, 0, Qt::AlignLeft );
grid->addWidget( specValEdit , 1, 1, Qt::AlignLeft );
grid->addWidget( sa_btn , 2, 0, Qt::AlignLeft );
grid->addWidget( specAddrEdit , 2, 1, Qt::AlignLeft );
grid->addWidget( nc_btn , 3, 0, Qt::AlignLeft );
grid->addWidget( numChangeEdit, 3, 1, Qt::AlignLeft );
vbox = new QVBoxLayout();
hbox3 = new QHBoxLayout();
frame = new QGroupBox( tr("Data Size") );
frame->setLayout( vbox );
vbox1->addLayout( hbox3 );
hbox3->addWidget( frame );
ds1_btn = new QRadioButton( tr("1 Byte") );
ds2_btn = new QRadioButton( tr("2 Byte") );
ds4_btn = new QRadioButton( tr("4 Byte") );
misalignedCbox = new QCheckBox( tr("Check Misaligned") );
2020-10-12 04:09:00 +00:00
misalignedCbox->setEnabled(dpySize != 'b');
misalignedCbox->setChecked(chkMisAligned);
connect( misalignedCbox, SIGNAL(stateChanged(int)), this, SLOT(misalignedChanged(int)));
2020-10-12 04:09:00 +00:00
ds1_btn->setChecked( dpySize == 'b' );
ds2_btn->setChecked( dpySize == 'w' );
ds4_btn->setChecked( dpySize == 'd' );
connect( ds1_btn, SIGNAL(clicked(void)), this, SLOT(ds1Clicked(void)));
connect( ds2_btn, SIGNAL(clicked(void)), this, SLOT(ds2Clicked(void)));
connect( ds4_btn, SIGNAL(clicked(void)), this, SLOT(ds4Clicked(void)));
vbox->addWidget( ds1_btn );
vbox->addWidget( ds2_btn );
vbox->addWidget( ds4_btn );
vbox->addWidget( misalignedCbox );
vbox = new QVBoxLayout();
vbox2 = new QVBoxLayout();
frame = new QGroupBox( tr("Data Type / Display") );
frame->setLayout( vbox );
vbox2->addWidget( frame );
hbox3->addLayout( vbox2 );
signed_btn = new QRadioButton( tr("Signed") );
unsigned_btn = new QRadioButton( tr("Unsigned") );
hex_btn = new QRadioButton( tr("Hexadecimal") );
vbox->addWidget( signed_btn );
vbox->addWidget( unsigned_btn );
vbox->addWidget( hex_btn );
2020-10-12 04:09:00 +00:00
signed_btn->setChecked( dpyType == 's' );
unsigned_btn->setChecked( dpyType == 'u' );
hex_btn->setChecked( dpyType == 'h' );
connect( signed_btn , SIGNAL(clicked(void)), this, SLOT(signedTypeClicked(void)));
connect( unsigned_btn, SIGNAL(clicked(void)), this, SLOT(unsignedTypeClicked(void)));
connect( hex_btn , SIGNAL(clicked(void)), this, SLOT(hexTypeClicked(void)));
autoSearchCbox = new QCheckBox( tr("Auto-Search") );
autoSearchCbox->setEnabled(true);
vbox2->addWidget( autoSearchCbox );
setLayout( mainLayout );
resetSearch();
updateTimer = new QTimer( this );
connect( updateTimer, &QTimer::timeout, this, &RamSearchDialog_t::periodicUpdate );
updateTimer->start( 100 ); // 10hz
}
//----------------------------------------------------------------------------
RamSearchDialog_t::~RamSearchDialog_t(void)
{
updateTimer->stop();
printf("Destroy RAM Watch Config Window\n");
2020-10-11 15:25:35 +00:00
ramSearchWin = NULL;
actvSrchList.clear();
for (unsigned int addr=0; addr<0x08000; addr++)
{
memLoc[addr].hist.clear();
}
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::closeEvent(QCloseEvent *event)
{
printf("RAM Watch Close Window Event\n");
done(0);
deleteLater();
event->accept();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::closeWindow(void)
{
//printf("Close Window\n");
done(0);
deleteLater();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::periodicUpdate(void)
{
2020-10-12 04:09:00 +00:00
updateRamValues();
ramView->update();
}
//----------------------------------------------------
void RamSearchDialog_t::hbarChanged(int val)
{
ramView->update();
}
//----------------------------------------------------
void RamSearchDialog_t::vbarChanged(int val)
{
ramView->update();
}
2020-10-12 04:09:00 +00:00
//----------------------------------------------------
void RamSearchDialog_t::searchROMChanged(int state)
{
ShowROM = (state != Qt::Unchecked);
}
//----------------------------------------------------
void RamSearchDialog_t::misalignedChanged(int state)
{
chkMisAligned = (state != Qt::Unchecked);
calcRamList();
}
//----------------------------------------------------------------------------
static unsigned int ReadValueAtHardwareAddress(int address, unsigned int size)
{
unsigned int value = 0;
2020-10-12 04:09:00 +00:00
int maxAddr = ShowROM ? 0x10000 : 0x8000;
// read as little endian
for (unsigned int i = 0; i < size; i++)
{
2020-10-12 04:09:00 +00:00
if ( address < maxAddr )
{
value <<= 8;
value |= GetMem(address);
address++;
}
}
return value;
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::runSearch(void)
{
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::resetSearch(void)
{
actvSrchList.clear();
2020-10-12 04:09:00 +00:00
for (unsigned int addr=0; addr<0x10000; addr++)
{
memLoc[addr].hist.clear();
memLoc[addr].addr = addr;
memLoc[addr].val.v8.u = GetMem(addr);
memLoc[addr].val.v16.u = ReadValueAtHardwareAddress(addr, 2);
memLoc[addr].val.v32.u = ReadValueAtHardwareAddress(addr, 4);
memLoc[addr].elimMask = 0;
2020-10-12 04:09:00 +00:00
memLoc[addr].chgCount = 0;
memLoc[addr].hist.push_back( memLoc[addr].val );
}
iterMask = 0x01;
2020-10-12 04:09:00 +00:00
calcRamList();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::clearChangeCounts(void)
{
for (unsigned int addr=0; addr<0x10000; addr++)
{
memLoc[addr].chgCount = 0;
}
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::ds1Clicked(void)
{
dpySize = 'b';
misalignedCbox->setEnabled(false);
calcRamList();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::ds2Clicked(void)
{
dpySize = 'w';
misalignedCbox->setEnabled(true);
calcRamList();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::ds4Clicked(void)
{
dpySize = 'd';
misalignedCbox->setEnabled(true);
calcRamList();
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::signedTypeClicked(void)
{
dpyType = 's';
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::unsignedTypeClicked(void)
{
dpyType = 'u';
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::hexTypeClicked(void)
{
dpyType = 'h';
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::calcRamList(void)
{
int addr;
int dataSize = 1;
int endAddr = ShowROM ? 0x10000 : 0x8000;
if ( chkMisAligned )
{
dataSize = 1;
}
else if ( dpySize == 'd' )
{
dataSize = 4;
}
else if ( dpySize == 'w' )
{
dataSize = 2;
}
else
{
dataSize = 1;
}
actvSrchList.clear();
for (addr=0; addr<endAddr; addr += dataSize)
{
switch ( dpySize )
{
case 'd':
if ( (addr+3) < endAddr )
{
if ( (memLoc[addr].elimMask == 0) &&
(memLoc[addr+1].elimMask == 0) &&
(memLoc[addr+2].elimMask == 0) &&
(memLoc[addr+3].elimMask == 0) )
{
actvSrchList.push_back( &memLoc[addr] );
}
}
break;
case 'w':
if ( (addr+1) < endAddr )
{
if ( (memLoc[addr].elimMask == 0) &&
(memLoc[addr+1].elimMask == 0) )
{
actvSrchList.push_back( &memLoc[addr] );
}
}
break;
default:
case 'b':
if ( memLoc[addr].elimMask == 0 )
{
actvSrchList.push_back( &memLoc[addr] );
}
break;
}
}
vbar->setMaximum( actvSrchList.size() );
}
//----------------------------------------------------------------------------
void RamSearchDialog_t::updateRamValues(void)
{
std::list <struct memoryLocation_t*>::iterator it;
memoryLocation_t *loc = NULL;
memoryState_t val;
for (it = actvSrchList.begin(); it != actvSrchList.end(); it++)
{
loc = *it;
val.v8.u = GetMem(loc->addr);
val.v16.u = ReadValueAtHardwareAddress(loc->addr, 2);
val.v32.u = ReadValueAtHardwareAddress(loc->addr, 4);
if ( dpySize == 'd' )
{
if ( memLoc[loc->addr].val.v32.u != val.v32.u )
{
memLoc[loc->addr].val = val;
memLoc[loc->addr].chgCount++;
}
}
else if ( dpySize == 'w' )
{
if ( memLoc[loc->addr].val.v16.u != val.v16.u )
{
memLoc[loc->addr].val = val;
memLoc[loc->addr].chgCount++;
}
}
else
{
if ( memLoc[loc->addr].val.v8.u != val.v8.u )
{
memLoc[loc->addr].val = val;
memLoc[loc->addr].chgCount++;
}
}
}
}
//----------------------------------------------------------------------------
QRamSearchView::QRamSearchView(QWidget *parent)
: QWidget(parent)
{
QPalette pal;
QColor fg(0,0,0), bg(255,255,255);
pal = this->palette();
pal.setColor(QPalette::Base , bg );
pal.setColor(QPalette::Background, bg );
pal.setColor(QPalette::WindowText, fg );
this->setPalette(pal);
font.setFamily("Courier New");
font.setStyle( QFont::StyleNormal );
font.setStyleHint( QFont::Monospace );
calcFontData();
lineOffset = 0;
maxLineOffset = 0;
}
//----------------------------------------------------------------------------
QRamSearchView::~QRamSearchView(void)
{
}
//----------------------------------------------------------------------------
void QRamSearchView::calcFontData(void)
{
this->setFont(font);
QFontMetrics metrics(font);
#if QT_VERSION > QT_VERSION_CHECK(5, 11, 0)
pxCharWidth = metrics.horizontalAdvance(QLatin1Char('2'));
#else
pxCharWidth = metrics.width(QLatin1Char('2'));
#endif
pxCharHeight = metrics.height();
pxLineSpacing = metrics.lineSpacing() * 1.25;
pxLineLead = pxLineSpacing - pxCharHeight;
pxCursorHeight = pxCharHeight;
pxColWidth[0] = pxCharWidth * 10;
pxColWidth[1] = pxCharWidth * 15;
pxColWidth[2] = pxCharWidth * 15;
pxColWidth[3] = pxCharWidth * 15;
pxLineWidth = pxColWidth[0] + pxColWidth[1] + pxColWidth[2] + pxColWidth[3];
viewLines = (viewHeight / pxLineSpacing) + 1;
}
//----------------------------------------------------------------------------
void QRamSearchView::setScrollBars( QScrollBar *hbar, QScrollBar *vbar )
{
this->hbar = hbar; this->vbar = vbar;
}
//----------------------------------------------------
void QRamSearchView::resizeEvent(QResizeEvent *event)
{
viewWidth = event->size().width();
viewHeight = event->size().height();
//printf("QAsmView Resize: %ix%i\n", viewWidth, viewHeight );
viewLines = (viewHeight / pxLineSpacing) + 1;
//maxLineOffset = 0; // mb.numLines() - viewLines + 1;
if ( viewWidth >= pxLineWidth )
{
pxLineXScroll = 0;
}
else
{
pxLineXScroll = (int)(0.010f * (float)hbar->value() * (float)(pxLineWidth - viewWidth) );
}
}
//----------------------------------------------------------------------------
void QRamSearchView::paintEvent(QPaintEvent *event)
{
2020-10-12 04:09:00 +00:00
int i,x,y,row,nrow;
std::list <struct memoryLocation_t*>::iterator it;
char addrStr[32], valStr[32], prevStr[32], chgStr[32];
QPainter painter(this);
memoryLocation_t *loc = NULL;
int fieldWidth, fieldPad[4], fieldLen[4], fieldStart[4];
const char *fieldText[4];
painter.setFont(font);
viewWidth = event->rect().width();
viewHeight = event->rect().height();
fieldWidth = viewWidth / 4;
for (i=0; i<4; i++)
{
fieldStart[i] = fieldWidth * i;
}
nrow = (viewHeight / pxLineSpacing)-1;
if (nrow < 1 ) nrow = 1;
viewLines = nrow;
maxLineOffset = actvSrchList.size() - nrow;
if ( maxLineOffset < 1 ) maxLineOffset = 1;
lineOffset = vbar->value();
if ( lineOffset > maxLineOffset )
{
lineOffset = maxLineOffset;
vbar->setValue( lineOffset );
}
if ( lineOffset < 0 )
{
lineOffset = 0;
vbar->setValue( 0 );
}
i=0;
it = actvSrchList.begin();
while ( it != actvSrchList.end() )
{
if ( i == lineOffset )
{
break;
}
i++; it++;
}
painter.fillRect( 0, 0, viewWidth, viewHeight, this->palette().color(QPalette::Background) );
painter.setPen( this->palette().color(QPalette::WindowText));
pxLineXScroll = (int)(0.010f * (float)hbar->value() * (float)(pxLineWidth - viewWidth) );
x = -pxLineXScroll;
y = pxLineSpacing;
strcpy( addrStr, "Address");
strcpy( valStr , "Value");
strcpy( prevStr, "Previous");
strcpy( chgStr , "Changes");
fieldText[0] = addrStr;
fieldText[1] = valStr;
fieldText[2] = prevStr;
fieldText[3] = chgStr;
for (i=0; i<4; i++)
{
fieldLen[i] = strlen(fieldText[i]) * pxCharWidth;
fieldPad[i] = (fieldWidth - fieldLen[i]) / 2;
painter.drawText( x+fieldStart[i]+fieldPad[i], y, tr(fieldText[i]) );
}
painter.drawLine( 0, y+pxLineLead, viewWidth, y+pxLineLead );
painter.drawLine( x+fieldStart[1], 0, x+fieldStart[1], viewHeight );
painter.drawLine( x+fieldStart[2], 0, x+fieldStart[2], viewHeight );
painter.drawLine( x+fieldStart[3], 0, x+fieldStart[3], viewHeight );
y += pxLineSpacing;
for (row=0; row<nrow; row++)
{
if ( it != actvSrchList.end() )
{
loc = *it;
}
else
{
loc = NULL;
}
if ( loc == NULL )
{
continue;
}
it++;
sprintf (addrStr, "$%04X", loc->addr);
if ( dpySize == 'd' )
{
if ( dpyType == 'h' )
{
sprintf( valStr , "0x%08X", loc->val.v32.u );
sprintf( prevStr, "0x%08X", loc->hist.back().v32.u );
}
else if ( dpyType == 'u' )
{
sprintf( valStr , "%u", loc->val.v32.u );
sprintf( prevStr, "%u", loc->hist.back().v32.u );
}
else
{
sprintf( valStr , "%i", loc->val.v32.i );
sprintf( prevStr, "%i", loc->hist.back().v32.i );
}
}
else if ( dpySize == 'w' )
{
if ( dpyType == 'h' )
{
sprintf( valStr , "0x%04X", loc->val.v16.u );
sprintf( prevStr, "0x%04X", loc->hist.back().v16.u );
}
else if ( dpyType == 'u' )
{
sprintf( valStr , "%u", loc->val.v16.u );
sprintf( prevStr, "%u", loc->hist.back().v16.u );
}
else
{
sprintf( valStr , "%i", loc->val.v16.i );
sprintf( prevStr, "%i", loc->hist.back().v16.i );
}
}
else
{
if ( dpyType == 'h' )
{
sprintf( valStr , "0x%02X", loc->val.v8.u );
sprintf( prevStr, "0x%02X", loc->hist.back().v8.u );
}
else if ( dpyType == 'u' )
{
sprintf( valStr , "%u", loc->val.v8.u );
sprintf( prevStr, "%u", loc->hist.back().v8.u );
}
else
{
sprintf( valStr , "%i", loc->val.v8.i );
sprintf( prevStr, "%i", loc->hist.back().v8.i );
}
}
sprintf( chgStr, "%i", loc->chgCount );
for (i=0; i<4; i++)
{
fieldLen[i] = strlen(fieldText[i]) * pxCharWidth;
fieldPad[i] = (fieldWidth - fieldLen[i]) / 2;
painter.drawText( x+fieldStart[i]+fieldPad[i], y, tr(fieldText[i]) );
}
y += pxLineSpacing;
}
}
//----------------------------------------------------------------------------