|
|
@ -12,6 +12,7 @@ |
|
|
|
#include "clientmodel.h" |
|
|
|
#include "clientmodel.h" |
|
|
|
#include "guiutil.h" |
|
|
|
#include "guiutil.h" |
|
|
|
#include "editaddressdialog.h" |
|
|
|
#include "editaddressdialog.h" |
|
|
|
|
|
|
|
#include "optionsmodel.h" |
|
|
|
|
|
|
|
|
|
|
|
#include "main.h" |
|
|
|
#include "main.h" |
|
|
|
|
|
|
|
|
|
|
@ -48,26 +49,26 @@ BitcoinGUI::BitcoinGUI(QWidget *parent): |
|
|
|
|
|
|
|
|
|
|
|
createActions(); |
|
|
|
createActions(); |
|
|
|
|
|
|
|
|
|
|
|
/* Menus */ |
|
|
|
// Menus
|
|
|
|
QMenu *file = menuBar()->addMenu("&File"); |
|
|
|
QMenu *file = menuBar()->addMenu("&File"); |
|
|
|
file->addAction(sendcoins); |
|
|
|
file->addAction(sendcoins); |
|
|
|
file->addSeparator(); |
|
|
|
file->addSeparator(); |
|
|
|
file->addAction(quit); |
|
|
|
file->addAction(quit); |
|
|
|
|
|
|
|
|
|
|
|
QMenu *settings = menuBar()->addMenu("&Settings"); |
|
|
|
QMenu *settings = menuBar()->addMenu("&Settings"); |
|
|
|
settings->addAction(receiving_addresses); |
|
|
|
settings->addAction(receivingAddresses); |
|
|
|
settings->addAction(options); |
|
|
|
settings->addAction(options); |
|
|
|
|
|
|
|
|
|
|
|
QMenu *help = menuBar()->addMenu("&Help"); |
|
|
|
QMenu *help = menuBar()->addMenu("&Help"); |
|
|
|
help->addAction(about); |
|
|
|
help->addAction(about); |
|
|
|
|
|
|
|
|
|
|
|
/* Toolbar */ |
|
|
|
// Toolbar
|
|
|
|
QToolBar *toolbar = addToolBar("Main toolbar"); |
|
|
|
QToolBar *toolbar = addToolBar("Main toolbar"); |
|
|
|
toolbar->setToolButtonStyle(Qt::ToolButtonTextBesideIcon); |
|
|
|
toolbar->setToolButtonStyle(Qt::ToolButtonTextBesideIcon); |
|
|
|
toolbar->addAction(sendcoins); |
|
|
|
toolbar->addAction(sendcoins); |
|
|
|
toolbar->addAction(addressbook); |
|
|
|
toolbar->addAction(addressbook); |
|
|
|
|
|
|
|
|
|
|
|
/* Address: <address>: New... : Paste to clipboard */ |
|
|
|
// Address: <address>: New... : Paste to clipboard
|
|
|
|
QHBoxLayout *hbox_address = new QHBoxLayout(); |
|
|
|
QHBoxLayout *hbox_address = new QHBoxLayout(); |
|
|
|
hbox_address->addWidget(new QLabel(tr("Your Bitcoin Address:"))); |
|
|
|
hbox_address->addWidget(new QLabel(tr("Your Bitcoin Address:"))); |
|
|
|
address = new QLineEdit(); |
|
|
|
address = new QLineEdit(); |
|
|
@ -80,7 +81,7 @@ BitcoinGUI::BitcoinGUI(QWidget *parent): |
|
|
|
hbox_address->addWidget(button_new); |
|
|
|
hbox_address->addWidget(button_new); |
|
|
|
hbox_address->addWidget(button_clipboard); |
|
|
|
hbox_address->addWidget(button_clipboard); |
|
|
|
|
|
|
|
|
|
|
|
/* Balance: <balance> */ |
|
|
|
// Balance: <balance>
|
|
|
|
QHBoxLayout *hbox_balance = new QHBoxLayout(); |
|
|
|
QHBoxLayout *hbox_balance = new QHBoxLayout(); |
|
|
|
hbox_balance->addWidget(new QLabel(tr("Balance:"))); |
|
|
|
hbox_balance->addWidget(new QLabel(tr("Balance:"))); |
|
|
|
hbox_balance->addSpacing(5);/* Add some spacing between the label and the text */ |
|
|
|
hbox_balance->addSpacing(5);/* Add some spacing between the label and the text */ |
|
|
@ -94,15 +95,13 @@ BitcoinGUI::BitcoinGUI(QWidget *parent): |
|
|
|
vbox->addLayout(hbox_address); |
|
|
|
vbox->addLayout(hbox_address); |
|
|
|
vbox->addLayout(hbox_balance); |
|
|
|
vbox->addLayout(hbox_balance); |
|
|
|
|
|
|
|
|
|
|
|
transaction_model = new TransactionTableModel(this); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
vbox->addWidget(createTabs()); |
|
|
|
vbox->addWidget(createTabs()); |
|
|
|
|
|
|
|
|
|
|
|
QWidget *centralwidget = new QWidget(this); |
|
|
|
QWidget *centralwidget = new QWidget(this); |
|
|
|
centralwidget->setLayout(vbox); |
|
|
|
centralwidget->setLayout(vbox); |
|
|
|
setCentralWidget(centralwidget); |
|
|
|
setCentralWidget(centralwidget); |
|
|
|
|
|
|
|
|
|
|
|
/* Create status bar */ |
|
|
|
// Create status bar
|
|
|
|
statusBar(); |
|
|
|
statusBar(); |
|
|
|
|
|
|
|
|
|
|
|
labelConnections = new QLabel(); |
|
|
|
labelConnections = new QLabel(); |
|
|
@ -121,7 +120,7 @@ BitcoinGUI::BitcoinGUI(QWidget *parent): |
|
|
|
statusBar()->addPermanentWidget(labelBlocks); |
|
|
|
statusBar()->addPermanentWidget(labelBlocks); |
|
|
|
statusBar()->addPermanentWidget(labelTransactions); |
|
|
|
statusBar()->addPermanentWidget(labelTransactions); |
|
|
|
|
|
|
|
|
|
|
|
/* Action bindings */ |
|
|
|
// Action bindings
|
|
|
|
connect(button_new, SIGNAL(clicked()), this, SLOT(newAddressClicked())); |
|
|
|
connect(button_new, SIGNAL(clicked()), this, SLOT(newAddressClicked())); |
|
|
|
connect(button_clipboard, SIGNAL(clicked()), this, SLOT(copyClipboardClicked())); |
|
|
|
connect(button_clipboard, SIGNAL(clicked()), this, SLOT(copyClipboardClicked())); |
|
|
|
|
|
|
|
|
|
|
@ -134,22 +133,24 @@ void BitcoinGUI::createActions() |
|
|
|
sendcoins = new QAction(QIcon(":/icons/send"), tr("&Send coins"), this); |
|
|
|
sendcoins = new QAction(QIcon(":/icons/send"), tr("&Send coins"), this); |
|
|
|
addressbook = new QAction(QIcon(":/icons/address-book"), tr("&Address Book"), this); |
|
|
|
addressbook = new QAction(QIcon(":/icons/address-book"), tr("&Address Book"), this); |
|
|
|
about = new QAction(QIcon(":/icons/bitcoin"), tr("&About"), this); |
|
|
|
about = new QAction(QIcon(":/icons/bitcoin"), tr("&About"), this); |
|
|
|
receiving_addresses = new QAction(QIcon(":/icons/receiving-addresses"), tr("Your &Receiving Addresses..."), this); |
|
|
|
receivingAddresses = new QAction(QIcon(":/icons/receiving-addresses"), tr("Your &Receiving Addresses..."), this); |
|
|
|
options = new QAction(QIcon(":/icons/options"), tr("&Options..."), this); |
|
|
|
options = new QAction(QIcon(":/icons/options"), tr("&Options..."), this); |
|
|
|
openBitCoin = new QAction(QIcon(":/icons/bitcoin"), "Open Bitcoin", this); |
|
|
|
openBitcoin = new QAction(QIcon(":/icons/bitcoin"), "Open &Bitcoin", this); |
|
|
|
|
|
|
|
|
|
|
|
connect(quit, SIGNAL(triggered()), qApp, SLOT(quit())); |
|
|
|
connect(quit, SIGNAL(triggered()), qApp, SLOT(quit())); |
|
|
|
connect(sendcoins, SIGNAL(triggered()), this, SLOT(sendcoinsClicked())); |
|
|
|
connect(sendcoins, SIGNAL(triggered()), this, SLOT(sendcoinsClicked())); |
|
|
|
connect(addressbook, SIGNAL(triggered()), this, SLOT(addressbookClicked())); |
|
|
|
connect(addressbook, SIGNAL(triggered()), this, SLOT(addressbookClicked())); |
|
|
|
connect(receiving_addresses, SIGNAL(triggered()), this, SLOT(receivingAddressesClicked())); |
|
|
|
connect(receivingAddresses, SIGNAL(triggered()), this, SLOT(receivingAddressesClicked())); |
|
|
|
connect(options, SIGNAL(triggered()), this, SLOT(optionsClicked())); |
|
|
|
connect(options, SIGNAL(triggered()), this, SLOT(optionsClicked())); |
|
|
|
connect(about, SIGNAL(triggered()), this, SLOT(aboutClicked())); |
|
|
|
connect(about, SIGNAL(triggered()), this, SLOT(aboutClicked())); |
|
|
|
|
|
|
|
connect(openBitcoin, SIGNAL(triggered()), this, SLOT(show())); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::setModel(ClientModel *model) |
|
|
|
void BitcoinGUI::setModel(ClientModel *model) |
|
|
|
{ |
|
|
|
{ |
|
|
|
this->model = model; |
|
|
|
this->model = model; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keep up to date with client
|
|
|
|
setBalance(model->getBalance()); |
|
|
|
setBalance(model->getBalance()); |
|
|
|
connect(model, SIGNAL(balanceChanged(qint64)), this, SLOT(setBalance(qint64))); |
|
|
|
connect(model, SIGNAL(balanceChanged(qint64)), this, SLOT(setBalance(qint64))); |
|
|
|
|
|
|
|
|
|
|
@ -165,14 +166,17 @@ void BitcoinGUI::setModel(ClientModel *model) |
|
|
|
setAddress(model->getAddress()); |
|
|
|
setAddress(model->getAddress()); |
|
|
|
connect(model, SIGNAL(addressChanged(QString)), this, SLOT(setAddress(QString))); |
|
|
|
connect(model, SIGNAL(addressChanged(QString)), this, SLOT(setAddress(QString))); |
|
|
|
|
|
|
|
|
|
|
|
/* Report errors from network/worker thread */ |
|
|
|
// Report errors from network/worker thread
|
|
|
|
connect(model, SIGNAL(error(QString,QString)), this, SLOT(error(QString,QString))); |
|
|
|
connect(model, SIGNAL(error(QString,QString)), this, SLOT(error(QString,QString))); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Put transaction list in tabs
|
|
|
|
|
|
|
|
setTabsModel(model->getTransactionTableModel()); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::createTrayIcon() |
|
|
|
void BitcoinGUI::createTrayIcon() |
|
|
|
{ |
|
|
|
{ |
|
|
|
QMenu *trayIconMenu = new QMenu(this); |
|
|
|
QMenu *trayIconMenu = new QMenu(this); |
|
|
|
trayIconMenu->addAction(openBitCoin); |
|
|
|
trayIconMenu->addAction(openBitcoin); |
|
|
|
trayIconMenu->addAction(sendcoins); |
|
|
|
trayIconMenu->addAction(sendcoins); |
|
|
|
trayIconMenu->addAction(options); |
|
|
|
trayIconMenu->addAction(options); |
|
|
|
trayIconMenu->addSeparator(); |
|
|
|
trayIconMenu->addSeparator(); |
|
|
@ -181,23 +185,48 @@ void BitcoinGUI::createTrayIcon() |
|
|
|
trayIcon = new QSystemTrayIcon(this); |
|
|
|
trayIcon = new QSystemTrayIcon(this); |
|
|
|
trayIcon->setContextMenu(trayIconMenu); |
|
|
|
trayIcon->setContextMenu(trayIconMenu); |
|
|
|
trayIcon->setIcon(QIcon(":/icons/toolbar")); |
|
|
|
trayIcon->setIcon(QIcon(":/icons/toolbar")); |
|
|
|
|
|
|
|
connect(trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)), |
|
|
|
|
|
|
|
this, SLOT(trayIconActivated(QSystemTrayIcon::ActivationReason))); |
|
|
|
trayIcon->show(); |
|
|
|
trayIcon->show(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::trayIconActivated(QSystemTrayIcon::ActivationReason reason) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if(reason == QSystemTrayIcon::DoubleClick) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
// Doubleclick on system tray icon triggers "open bitcoin"
|
|
|
|
|
|
|
|
openBitcoin->trigger(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
QWidget *BitcoinGUI::createTabs() |
|
|
|
QWidget *BitcoinGUI::createTabs() |
|
|
|
{ |
|
|
|
{ |
|
|
|
QStringList tab_filters, tab_labels; |
|
|
|
QStringList tab_labels; |
|
|
|
tab_filters << "^." |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Sent+TransactionTableModel::Received+"]" |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Sent+"]" |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Received+"]"; |
|
|
|
|
|
|
|
tab_labels << tr("All transactions") |
|
|
|
tab_labels << tr("All transactions") |
|
|
|
<< tr("Sent/Received") |
|
|
|
<< tr("Sent/Received") |
|
|
|
<< tr("Sent") |
|
|
|
<< tr("Sent") |
|
|
|
<< tr("Received"); |
|
|
|
<< tr("Received"); |
|
|
|
QTabWidget *tabs = new QTabWidget(this); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
QTabWidget *tabs = new QTabWidget(this); |
|
|
|
for(int i = 0; i < tab_labels.size(); ++i) |
|
|
|
for(int i = 0; i < tab_labels.size(); ++i) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
QTableView *view = new QTableView(this); |
|
|
|
|
|
|
|
tabs->addTab(view, tab_labels.at(i)); |
|
|
|
|
|
|
|
transactionViews.append(view); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return tabs; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::setTabsModel(QAbstractItemModel *transaction_model) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
QStringList tab_filters; |
|
|
|
|
|
|
|
tab_filters << "^." |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Sent+TransactionTableModel::Received+"]" |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Sent+"]" |
|
|
|
|
|
|
|
<< "^["+TransactionTableModel::Received+"]"; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for(int i = 0; i < transactionViews.size(); ++i) |
|
|
|
{ |
|
|
|
{ |
|
|
|
QSortFilterProxyModel *proxy_model = new QSortFilterProxyModel(this); |
|
|
|
QSortFilterProxyModel *proxy_model = new QSortFilterProxyModel(this); |
|
|
|
proxy_model->setSourceModel(transaction_model); |
|
|
|
proxy_model->setSourceModel(transaction_model); |
|
|
@ -206,7 +235,7 @@ QWidget *BitcoinGUI::createTabs() |
|
|
|
proxy_model->setFilterRegExp(QRegExp(tab_filters.at(i))); |
|
|
|
proxy_model->setFilterRegExp(QRegExp(tab_filters.at(i))); |
|
|
|
proxy_model->setSortRole(Qt::EditRole); |
|
|
|
proxy_model->setSortRole(Qt::EditRole); |
|
|
|
|
|
|
|
|
|
|
|
QTableView *transaction_table = new QTableView(this); |
|
|
|
QTableView *transaction_table = transactionViews.at(i); |
|
|
|
transaction_table->setModel(proxy_model); |
|
|
|
transaction_table->setModel(proxy_model); |
|
|
|
transaction_table->setSelectionBehavior(QAbstractItemView::SelectRows); |
|
|
|
transaction_table->setSelectionBehavior(QAbstractItemView::SelectRows); |
|
|
|
transaction_table->setSelectionMode(QAbstractItemView::ExtendedSelection); |
|
|
|
transaction_table->setSelectionMode(QAbstractItemView::ExtendedSelection); |
|
|
@ -224,10 +253,7 @@ QWidget *BitcoinGUI::createTabs() |
|
|
|
TransactionTableModel::Debit, 79); |
|
|
|
TransactionTableModel::Debit, 79); |
|
|
|
transaction_table->horizontalHeader()->resizeSection( |
|
|
|
transaction_table->horizontalHeader()->resizeSection( |
|
|
|
TransactionTableModel::Credit, 79); |
|
|
|
TransactionTableModel::Credit, 79); |
|
|
|
|
|
|
|
|
|
|
|
tabs->addTab(transaction_table, tab_labels.at(i)); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
return tabs; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::sendcoinsClicked() |
|
|
|
void BitcoinGUI::sendcoinsClicked() |
|
|
@ -273,7 +299,7 @@ void BitcoinGUI::newAddressClicked() |
|
|
|
if(dlg.exec()) |
|
|
|
if(dlg.exec()) |
|
|
|
{ |
|
|
|
{ |
|
|
|
QString newAddress = dlg.saveCurrentRow(); |
|
|
|
QString newAddress = dlg.saveCurrentRow(); |
|
|
|
/* Set returned address as new default address */ |
|
|
|
// Set returned address as new default addres
|
|
|
|
if(!newAddress.isEmpty()) |
|
|
|
if(!newAddress.isEmpty()) |
|
|
|
{ |
|
|
|
{ |
|
|
|
model->setAddress(newAddress); |
|
|
|
model->setAddress(newAddress); |
|
|
@ -283,7 +309,7 @@ void BitcoinGUI::newAddressClicked() |
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::copyClipboardClicked() |
|
|
|
void BitcoinGUI::copyClipboardClicked() |
|
|
|
{ |
|
|
|
{ |
|
|
|
/* Copy text in address to clipboard */ |
|
|
|
// Copy text in address to clipboard
|
|
|
|
QApplication::clipboard()->setText(address->text()); |
|
|
|
QApplication::clipboard()->setText(address->text()); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -314,8 +340,43 @@ void BitcoinGUI::setNumTransactions(int count) |
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::error(const QString &title, const QString &message) |
|
|
|
void BitcoinGUI::error(const QString &title, const QString &message) |
|
|
|
{ |
|
|
|
{ |
|
|
|
/* Report errors from network/worker thread */ |
|
|
|
// Report errors from network/worker thread
|
|
|
|
QMessageBox::critical(this, title, |
|
|
|
if(trayIcon->supportsMessages()) |
|
|
|
message, |
|
|
|
{ |
|
|
|
QMessageBox::Ok, QMessageBox::Ok); |
|
|
|
// Show as "balloon" message if possible
|
|
|
|
|
|
|
|
trayIcon->showMessage(title, message, QSystemTrayIcon::Critical); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
// Fall back to old fashioned popup dialog if not
|
|
|
|
|
|
|
|
QMessageBox::critical(this, title, |
|
|
|
|
|
|
|
message, |
|
|
|
|
|
|
|
QMessageBox::Ok, QMessageBox::Ok); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::changeEvent(QEvent *e) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (e->type() == QEvent::WindowStateChange) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if(model->getOptionsModel()->getMinimizeToTray()) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if (isMinimized()) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
hide(); |
|
|
|
|
|
|
|
e->ignore(); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
e->accept(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
QMainWindow::changeEvent(e); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BitcoinGUI::closeEvent(QCloseEvent *event) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
if(!model->getOptionsModel()->getMinimizeToTray() && |
|
|
|
|
|
|
|
!model->getOptionsModel()->getMinimizeOnClose()) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
qApp->quit(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
QMainWindow::closeEvent(event); |
|
|
|
} |
|
|
|
} |
|
|
|