Merge pull request #1269 from laanwj/2012_05_consoleimprovements
UI console improvements
This commit is contained in:
commit
85a2229264
3 changed files with 80 additions and 100 deletions
|
@ -7,7 +7,7 @@
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>706</width>
|
<width>706</width>
|
||||||
<height>382</height>
|
<height>446</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
|
@ -327,30 +327,22 @@
|
||||||
<number>3</number>
|
<number>3</number>
|
||||||
</property>
|
</property>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QTableWidget" name="messagesWidget">
|
<widget class="QTextEdit" name="messagesWidget">
|
||||||
<property name="minimumSize">
|
<property name="minimumSize">
|
||||||
<size>
|
<size>
|
||||||
<width>0</width>
|
<width>0</width>
|
||||||
<height>100</height>
|
<height>100</height>
|
||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<property name="tabKeyNavigation">
|
<property name="readOnly">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
<property name="tabKeyNavigation" stdset="0">
|
||||||
<bool>false</bool>
|
<bool>false</bool>
|
||||||
</property>
|
</property>
|
||||||
<property name="selectionBehavior">
|
<property name="columnCount" stdset="0">
|
||||||
<enum>QAbstractItemView::SelectRows</enum>
|
|
||||||
</property>
|
|
||||||
<property name="columnCount">
|
|
||||||
<number>2</number>
|
<number>2</number>
|
||||||
</property>
|
</property>
|
||||||
<attribute name="horizontalHeaderVisible">
|
|
||||||
<bool>false</bool>
|
|
||||||
</attribute>
|
|
||||||
<attribute name="verticalHeaderVisible">
|
|
||||||
<bool>false</bool>
|
|
||||||
</attribute>
|
|
||||||
<column/>
|
|
||||||
<column/>
|
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QTextEdit>
|
#include <QTextEdit>
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
#include <QUrl>
|
||||||
|
|
||||||
#include <boost/tokenizer.hpp>
|
#include <boost/tokenizer.hpp>
|
||||||
|
|
||||||
|
@ -19,6 +20,19 @@
|
||||||
const int CONSOLE_SCROLLBACK = 50;
|
const int CONSOLE_SCROLLBACK = 50;
|
||||||
const int CONSOLE_HISTORY = 50;
|
const int CONSOLE_HISTORY = 50;
|
||||||
|
|
||||||
|
const QSize ICON_SIZE(24, 24);
|
||||||
|
|
||||||
|
const struct {
|
||||||
|
const char *url;
|
||||||
|
const char *source;
|
||||||
|
} ICON_MAPPING[] = {
|
||||||
|
{"cmd-request", ":/icons/tx_input"},
|
||||||
|
{"cmd-reply", ":/icons/tx_output"},
|
||||||
|
{"cmd-error", ":/icons/tx_output"},
|
||||||
|
{"misc", ":/icons/tx_inout"},
|
||||||
|
{NULL, NULL}
|
||||||
|
};
|
||||||
|
|
||||||
/* Object for executing console RPC commands in a separate thread.
|
/* Object for executing console RPC commands in a separate thread.
|
||||||
*/
|
*/
|
||||||
class RPCExecutor: public QObject
|
class RPCExecutor: public QObject
|
||||||
|
@ -41,19 +55,26 @@ void RPCExecutor::start()
|
||||||
void RPCExecutor::request(const QString &command)
|
void RPCExecutor::request(const QString &command)
|
||||||
{
|
{
|
||||||
// Parse shell-like command line into separate arguments
|
// Parse shell-like command line into separate arguments
|
||||||
boost::escaped_list_separator<char> els('\\',' ','\"');
|
|
||||||
std::string strCommand = command.toStdString();
|
|
||||||
boost::tokenizer<boost::escaped_list_separator<char> > tok(strCommand, els);
|
|
||||||
|
|
||||||
std::string strMethod;
|
std::string strMethod;
|
||||||
std::vector<std::string> strParams;
|
std::vector<std::string> strParams;
|
||||||
int n = 0;
|
try {
|
||||||
for(boost::tokenizer<boost::escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end();++beg,++n)
|
boost::escaped_list_separator<char> els('\\',' ','\"');
|
||||||
|
std::string strCommand = command.toStdString();
|
||||||
|
boost::tokenizer<boost::escaped_list_separator<char> > tok(strCommand, els);
|
||||||
|
|
||||||
|
int n = 0;
|
||||||
|
for(boost::tokenizer<boost::escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end();++beg,++n)
|
||||||
|
{
|
||||||
|
if(n == 0) // First parameter is the command
|
||||||
|
strMethod = *beg;
|
||||||
|
else
|
||||||
|
strParams.push_back(*beg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch(boost::escaped_list_error &e)
|
||||||
{
|
{
|
||||||
if(n == 0) // First parameter is the command
|
emit reply(RPCConsole::CMD_ERROR, QString("Parse error"));
|
||||||
strMethod = *beg;
|
return;
|
||||||
else
|
|
||||||
strParams.push_back(*beg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
@ -83,12 +104,9 @@ void RPCExecutor::request(const QString &command)
|
||||||
RPCConsole::RPCConsole(QWidget *parent) :
|
RPCConsole::RPCConsole(QWidget *parent) :
|
||||||
QDialog(parent),
|
QDialog(parent),
|
||||||
ui(new Ui::RPCConsole),
|
ui(new Ui::RPCConsole),
|
||||||
firstLayout(true),
|
|
||||||
historyPtr(0)
|
historyPtr(0)
|
||||||
{
|
{
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
ui->messagesWidget->horizontalHeader()->setResizeMode(1, QHeaderView::Stretch);
|
|
||||||
ui->messagesWidget->setContextMenuPolicy(Qt::ActionsContextMenu);
|
|
||||||
|
|
||||||
#ifndef WIN32
|
#ifndef WIN32
|
||||||
// Show Debug logfile label and Open button only for Windows
|
// Show Debug logfile label and Open button only for Windows
|
||||||
|
@ -99,13 +117,6 @@ RPCConsole::RPCConsole(QWidget *parent) :
|
||||||
// Install event filter for up and down arrow
|
// Install event filter for up and down arrow
|
||||||
ui->lineEdit->installEventFilter(this);
|
ui->lineEdit->installEventFilter(this);
|
||||||
|
|
||||||
// Add "Copy message" to context menu explicitly
|
|
||||||
QAction *copyMessageAction = new QAction(tr("&Copy"), this);
|
|
||||||
copyMessageAction->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_C));
|
|
||||||
copyMessageAction->setShortcutContext(Qt::WidgetShortcut);
|
|
||||||
connect(copyMessageAction, SIGNAL(triggered()), this, SLOT(copyMessage()));
|
|
||||||
ui->messagesWidget->addAction(copyMessageAction);
|
|
||||||
|
|
||||||
connect(ui->clearButton, SIGNAL(clicked()), this, SLOT(clear()));
|
connect(ui->clearButton, SIGNAL(clicked()), this, SLOT(clear()));
|
||||||
connect(ui->openDebugLogfileButton, SIGNAL(clicked()), this, SLOT(on_openDebugLogfileButton_clicked()));
|
connect(ui->openDebugLogfileButton, SIGNAL(clicked()), this, SLOT(on_openDebugLogfileButton_clicked()));
|
||||||
|
|
||||||
|
@ -159,68 +170,62 @@ void RPCConsole::setClientModel(ClientModel *model)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static QColor categoryColor(int category)
|
static QString categoryClass(int category)
|
||||||
{
|
{
|
||||||
switch(category)
|
switch(category)
|
||||||
{
|
{
|
||||||
case RPCConsole::MC_ERROR: return QColor(255,0,0); break;
|
case RPCConsole::CMD_REQUEST: return "cmd-request"; break;
|
||||||
case RPCConsole::MC_DEBUG: return QColor(192,192,192); break;
|
case RPCConsole::CMD_REPLY: return "cmd-reply"; break;
|
||||||
case RPCConsole::CMD_REQUEST: return QColor(128,128,128); break;
|
case RPCConsole::CMD_ERROR: return "cmd-error"; break;
|
||||||
case RPCConsole::CMD_REPLY: return QColor(128,255,128); break;
|
default: return "misc";
|
||||||
case RPCConsole::CMD_ERROR: return QColor(255,128,128); break;
|
|
||||||
default: return QColor(0,0,0);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void RPCConsole::clear()
|
void RPCConsole::clear()
|
||||||
{
|
{
|
||||||
ui->messagesWidget->clear();
|
ui->messagesWidget->clear();
|
||||||
ui->messagesWidget->setRowCount(0);
|
|
||||||
ui->lineEdit->clear();
|
ui->lineEdit->clear();
|
||||||
ui->lineEdit->setFocus();
|
ui->lineEdit->setFocus();
|
||||||
|
|
||||||
message(CMD_REPLY, tr("Welcome to the bitcoin RPC console.")+"\n"+
|
// Add smoothly scaled icon images.
|
||||||
tr("Use up and down arrows to navigate history, and Ctrl-L to clear screen.")+"\n"+
|
// (when using width/height on an img, Qt uses nearest instead of linear interpolation)
|
||||||
tr("Type \"help\" for an overview of available commands."));
|
for(int i=0; ICON_MAPPING[i].url; ++i)
|
||||||
}
|
|
||||||
|
|
||||||
void RPCConsole::message(int category, const QString &message)
|
|
||||||
{
|
|
||||||
// Add row to messages widget
|
|
||||||
int row = ui->messagesWidget->rowCount();
|
|
||||||
ui->messagesWidget->setRowCount(row+1);
|
|
||||||
|
|
||||||
QTime time = QTime::currentTime();
|
|
||||||
QTableWidgetItem *newTime = new QTableWidgetItem(time.toString());
|
|
||||||
newTime->setData(Qt::DecorationRole, categoryColor(category));
|
|
||||||
newTime->setForeground(QColor(128,128,128));
|
|
||||||
newTime->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); // make non-editable
|
|
||||||
|
|
||||||
int numLines = message.count("\n") + 1;
|
|
||||||
// As Qt doesn't like very tall cells (they break scrolling) keep only short messages in
|
|
||||||
// the cell text, longer messages trigger a display widget with scroll bar
|
|
||||||
if(numLines < 5)
|
|
||||||
{
|
{
|
||||||
QTableWidgetItem *newItem = new QTableWidgetItem(message);
|
ui->messagesWidget->document()->addResource(
|
||||||
newItem->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); // make non-editable
|
QTextDocument::ImageResource,
|
||||||
if(category == CMD_ERROR) // Coloring error messages in red
|
QUrl(ICON_MAPPING[i].url),
|
||||||
newItem->setForeground(QColor(255,16,16));
|
QImage(ICON_MAPPING[i].source).scaled(ICON_SIZE, Qt::IgnoreAspectRatio, Qt::SmoothTransformation));
|
||||||
ui->messagesWidget->setItem(row, 1, newItem);
|
|
||||||
} else {
|
|
||||||
QTextEdit *newWidget = new QTextEdit;
|
|
||||||
newWidget->setText(message);
|
|
||||||
newWidget->setMaximumHeight(100);
|
|
||||||
newWidget->setReadOnly(true);
|
|
||||||
ui->messagesWidget->setCellWidget(row, 1, newWidget);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ui->messagesWidget->setItem(row, 0, newTime);
|
// Set default style sheet
|
||||||
ui->messagesWidget->resizeRowToContents(row);
|
ui->messagesWidget->document()->setDefaultStyleSheet(
|
||||||
// Preserve only limited scrollback buffer
|
"table { }"
|
||||||
while(ui->messagesWidget->rowCount() > CONSOLE_SCROLLBACK)
|
"td.time { color: #808080; padding-top: 3px; } "
|
||||||
ui->messagesWidget->removeRow(0);
|
"td.message { font-family: Monospace; font-size: 12px; } "
|
||||||
// Scroll to bottom after table is updated
|
"td.cmd-request { color: #006060; } "
|
||||||
QTimer::singleShot(0, ui->messagesWidget, SLOT(scrollToBottom()));
|
"td.cmd-error { color: red; } "
|
||||||
|
"b { color: #006060; } "
|
||||||
|
);
|
||||||
|
|
||||||
|
message(CMD_REPLY, tr("Welcome to the Bitcoin RPC console.<br>"
|
||||||
|
"Use up and down arrows to navigate history, and <b>Ctrl-L</b> to clear screen.<br>"
|
||||||
|
"Type <b>help</b> for an overview of available commands."), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RPCConsole::message(int category, const QString &message, bool html)
|
||||||
|
{
|
||||||
|
QTime time = QTime::currentTime();
|
||||||
|
QString timeString = time.toString();
|
||||||
|
QString out;
|
||||||
|
out += "<table><tr><td class=\"time\" width=\"65\">" + timeString + "</td>";
|
||||||
|
out += "<td class=\"icon\" width=\"32\"><img src=\"" + categoryClass(category) + "\"></td>";
|
||||||
|
out += "<td class=\"message " + categoryClass(category) + "\" valign=\"middle\">";
|
||||||
|
if(html)
|
||||||
|
out += message;
|
||||||
|
else
|
||||||
|
out += GUIUtil::HtmlEscape(message, true);
|
||||||
|
out += "</td></tr></table>";
|
||||||
|
ui->messagesWidget->append(out);
|
||||||
}
|
}
|
||||||
|
|
||||||
void RPCConsole::setNumConnections(int count)
|
void RPCConsole::setNumConnections(int count)
|
||||||
|
@ -298,24 +303,10 @@ void RPCConsole::startExecutor()
|
||||||
thread->start();
|
thread->start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void RPCConsole::copyMessage()
|
|
||||||
{
|
|
||||||
GUIUtil::copyEntryData(ui->messagesWidget, 1, Qt::EditRole);
|
|
||||||
}
|
|
||||||
|
|
||||||
void RPCConsole::on_tabWidget_currentChanged(int index)
|
void RPCConsole::on_tabWidget_currentChanged(int index)
|
||||||
{
|
{
|
||||||
if(ui->tabWidget->widget(index) == ui->tab_console)
|
if(ui->tabWidget->widget(index) == ui->tab_console)
|
||||||
{
|
{
|
||||||
if(firstLayout)
|
|
||||||
{
|
|
||||||
// Work around QTableWidget issue:
|
|
||||||
// Call resizeRowsToContents on first Layout request with widget visible,
|
|
||||||
// to make sure multiline messages that were added before the console was shown
|
|
||||||
// have the right height.
|
|
||||||
firstLayout = false;
|
|
||||||
ui->messagesWidget->resizeRowsToContents();
|
|
||||||
}
|
|
||||||
ui->lineEdit->setFocus();
|
ui->lineEdit->setFocus();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,15 +37,13 @@ private slots:
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void clear();
|
void clear();
|
||||||
void message(int category, const QString &message);
|
void message(int category, const QString &message, bool html = false);
|
||||||
/** Set number of connections shown in the UI */
|
/** Set number of connections shown in the UI */
|
||||||
void setNumConnections(int count);
|
void setNumConnections(int count);
|
||||||
/** Set number of blocks shown in the UI */
|
/** Set number of blocks shown in the UI */
|
||||||
void setNumBlocks(int count);
|
void setNumBlocks(int count);
|
||||||
/** Go forward or back in history */
|
/** Go forward or back in history */
|
||||||
void browseHistory(int offset);
|
void browseHistory(int offset);
|
||||||
/** Copy currently selected message to clipboard */
|
|
||||||
void copyMessage();
|
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
// For RPC command executor
|
// For RPC command executor
|
||||||
|
@ -55,7 +53,6 @@ signals:
|
||||||
private:
|
private:
|
||||||
Ui::RPCConsole *ui;
|
Ui::RPCConsole *ui;
|
||||||
ClientModel *clientModel;
|
ClientModel *clientModel;
|
||||||
bool firstLayout;
|
|
||||||
QStringList history;
|
QStringList history;
|
||||||
int historyPtr;
|
int historyPtr;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue