Merge branch 'master' of ssh://amethyst/home/orion/projects2/bitcoin/bitcoin-qt

This commit is contained in:
Wladimir J. van der Laan 2011-06-05 16:11:44 +02:00
commit 9e0576587b
13 changed files with 260 additions and 48 deletions

View File

@ -14,20 +14,18 @@ This has been implemented:
- GUI only functionality (copy to clipboard, select address, address/transaction filter proxys) - GUI only functionality (copy to clipboard, select address, address/transaction filter proxys)
- Bitcoin core is made compatible with Qt4, and linked against - Bitcoin core is made compatible with Qt4
- Send coins dialog: address and input validation - Send coins dialog: address and input validation
- Address book and transactions views and models - Address book and transactions views and models
- Options dialog
- Sending coins - Sending coins
This has to be done: This has to be done:
- Settings are not remembered between invocations yet
- Minimize to tray / Minimize on close
- Start at system start - Start at system start
- Internationalization (convert WX language files) - Internationalization (convert WX language files)

View File

@ -62,7 +62,8 @@ HEADERS += gui/include/bitcoingui.h \
gui/include/transactionrecord.h \ gui/include/transactionrecord.h \
gui/include/guiconstants.h \ gui/include/guiconstants.h \
gui/include/optionsmodel.h \ gui/include/optionsmodel.h \
gui/include/monitoreddatamapper.h gui/include/monitoreddatamapper.h \
core/include/externui.h
SOURCES += gui/src/bitcoin.cpp gui/src/bitcoingui.cpp \ SOURCES += gui/src/bitcoin.cpp gui/src/bitcoingui.cpp \
gui/src/transactiontablemodel.cpp \ gui/src/transactiontablemodel.cpp \
gui/src/addresstablemodel.cpp \ gui/src/addresstablemodel.cpp \

45
core/include/externui.h Normal file
View File

@ -0,0 +1,45 @@
// Copyright (c) 2010 Satoshi Nakamoto
// Distributed under the MIT/X11 software license, see the accompanying
// file license.txt or http://www.opensource.org/licenses/mit-license.php.
#ifndef BITCOIN_EXTERNUI_H
#define BITCOIN_EXTERNUI_H
#include <string>
typedef void wxWindow;
#define wxYES 0x00000002
#define wxOK 0x00000004
#define wxNO 0x00000008
#define wxYES_NO (wxYES|wxNO)
#define wxCANCEL 0x00000010
#define wxAPPLY 0x00000020
#define wxCLOSE 0x00000040
#define wxOK_DEFAULT 0x00000000
#define wxYES_DEFAULT 0x00000000
#define wxNO_DEFAULT 0x00000080
#define wxCANCEL_DEFAULT 0x80000000
#define wxICON_EXCLAMATION 0x00000100
#define wxICON_HAND 0x00000200
#define wxICON_WARNING wxICON_EXCLAMATION
#define wxICON_ERROR wxICON_HAND
#define wxICON_QUESTION 0x00000400
#define wxICON_INFORMATION 0x00000800
#define wxICON_STOP wxICON_HAND
#define wxICON_ASTERISK wxICON_INFORMATION
#define wxICON_MASK (0x00000100|0x00000200|0x00000400|0x00000800)
#define wxFORWARD 0x00001000
#define wxBACKWARD 0x00002000
#define wxRESET 0x00004000
#define wxHELP 0x00008000
#define wxMORE 0x00010000
#define wxSETUP 0x00020000
extern int MyMessageBox(const std::string& message, const std::string& caption="Message", int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1);
#define wxMessageBox MyMessageBox
extern int ThreadSafeMessageBox(const std::string& message, const std::string& caption, int style=wxOK, wxWindow* parent=NULL, int x=-1, int y=-1);
extern bool ThreadSafeAskFee(int64 nFeeRequired, const std::string& strCaption, wxWindow* parent);
extern void CalledSetStatusBar(const std::string& strText, int nField);
extern void UIThreadCall(boost::function0<void> fn);
extern void MainFrameRepaint();
#endif

View File

@ -127,7 +127,7 @@
#include "uibase.h" #include "uibase.h"
#include "ui.h" #include "ui.h"
#else #else
#include "noui.h" #include "externui.h"
#endif #endif
#include "init.h" #include "init.h"

View File

@ -11,6 +11,8 @@ class ClientModel;
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QLabel; class QLabel;
class QLineEdit; class QLineEdit;
class QTableView;
class QAbstractItemModel;
QT_END_NAMESPACE QT_END_NAMESPACE
class BitcoinGUI : public QMainWindow class BitcoinGUI : public QMainWindow
@ -27,8 +29,12 @@ public:
Sent = 2, Sent = 2,
Received = 3 Received = 3
} TabIndex; } TabIndex;
protected:
void changeEvent(QEvent *e);
void closeEvent(QCloseEvent *event);
private: private:
TransactionTableModel *transaction_model;
ClientModel *model; ClientModel *model;
QLineEdit *address; QLineEdit *address;
@ -41,15 +47,17 @@ private:
QAction *sendcoins; QAction *sendcoins;
QAction *addressbook; QAction *addressbook;
QAction *about; QAction *about;
QAction *receiving_addresses; QAction *receivingAddresses;
QAction *options; QAction *options;
QAction *openBitCoin; QAction *openBitcoin;
QSystemTrayIcon *trayIcon; QSystemTrayIcon *trayIcon;
QList<QTableView *> transactionViews;
void createActions(); void createActions();
QWidget *createTabs(); QWidget *createTabs();
void createTrayIcon(); void createTrayIcon();
void setTabsModel(QAbstractItemModel *transaction_model);
public slots: public slots:
void setBalance(qint64 balance); void setBalance(qint64 balance);
@ -57,6 +65,7 @@ public slots:
void setNumConnections(int count); void setNumConnections(int count);
void setNumBlocks(int count); void setNumBlocks(int count);
void setNumTransactions(int count); void setNumTransactions(int count);
void error(const QString &title, const QString &message);
private slots: private slots:
void sendcoinsClicked(); void sendcoinsClicked();
@ -64,10 +73,9 @@ private slots:
void optionsClicked(); void optionsClicked();
void receivingAddressesClicked(); void receivingAddressesClicked();
void aboutClicked(); void aboutClicked();
void newAddressClicked(); void newAddressClicked();
void copyClipboardClicked(); void copyClipboardClicked();
void error(const QString &title, const QString &message); void trayIconActivated(QSystemTrayIcon::ActivationReason reason);
}; };
#endif #endif

View File

@ -5,6 +5,7 @@
class OptionsModel; class OptionsModel;
class AddressTableModel; class AddressTableModel;
class TransactionTableModel;
class ClientModel : public QObject class ClientModel : public QObject
{ {
@ -25,6 +26,7 @@ public:
OptionsModel *getOptionsModel(); OptionsModel *getOptionsModel();
AddressTableModel *getAddressTableModel(); AddressTableModel *getAddressTableModel();
TransactionTableModel *getTransactionTableModel();
qint64 getBalance(); qint64 getBalance();
QString getAddress(); QString getAddress();
@ -39,6 +41,7 @@ public:
private: private:
OptionsModel *optionsModel; OptionsModel *optionsModel;
AddressTableModel *addressTableModel; AddressTableModel *addressTableModel;
TransactionTableModel *transactionTableModel;
signals: signals:
void balanceChanged(qint64 balance); void balanceChanged(qint64 balance);

View File

@ -3,7 +3,12 @@
#include <QAbstractListModel> #include <QAbstractListModel>
/* Interface from QT to configuration data structure for bitcoin client */ /* Interface from QT to configuration data structure for bitcoin client.
To QT, the options are presented as a list with the different options
laid out vertically.
This can be changed to a tree once the settings become sufficiently
complex.
*/
class OptionsModel : public QAbstractListModel class OptionsModel : public QAbstractListModel
{ {
Q_OBJECT Q_OBJECT

View File

@ -1,11 +1,14 @@
#include "aboutdialog.h" #include "aboutdialog.h"
#include "ui_aboutdialog.h" #include "ui_aboutdialog.h"
#include "util.h"
AboutDialog::AboutDialog(QWidget *parent) : AboutDialog::AboutDialog(QWidget *parent) :
QDialog(parent), QDialog(parent),
ui(new Ui::AboutDialog) ui(new Ui::AboutDialog)
{ {
ui->setupUi(this); ui->setupUi(this);
ui->versionLabel->setText(QString::fromStdString(FormatFullVersion()));
} }
AboutDialog::~AboutDialog() AboutDialog::~AboutDialog()

View File

@ -5,21 +5,85 @@
#include "clientmodel.h" #include "clientmodel.h"
#include "util.h" #include "util.h"
#include "init.h" #include "init.h"
#include "externui.h"
#include <QApplication> #include <QApplication>
#include <QMessageBox>
// Need a global reference to process net thread
BitcoinGUI *guiref;
int MyMessageBox(const std::string& message, const std::string& caption, int style, wxWindow* parent, int x, int y)
{
// Message from main thread
printf("MyMessageBox\n");
if(guiref)
{
guiref->error(QString::fromStdString(caption),
QString::fromStdString(message));
}
else
{
QMessageBox::critical(0, QString::fromStdString(caption),
QString::fromStdString(message),
QMessageBox::Ok, QMessageBox::Ok);
}
return 4;
}
int ThreadSafeMessageBox(const std::string& message, const std::string& caption, int style, wxWindow* parent, int x, int y)
{
// Message from network thread
if(guiref)
{
QMetaObject::invokeMethod(guiref, "error", Qt::QueuedConnection,
Q_ARG(QString, QString::fromStdString(caption)),
Q_ARG(QString, QString::fromStdString(message)));
}
else
{
printf("%s: %s\n", caption.c_str(), message.c_str());
fprintf(stderr, "%s: %s\n", caption.c_str(), message.c_str());
}
return 4;
}
bool ThreadSafeAskFee(int64 nFeeRequired, const std::string& strCaption, wxWindow* parent)
{
// Query from network thread
// TODO
return true;
}
void CalledSetStatusBar(const std::string& strText, int nField)
{
// Only used for built-in mining, which is disabled, simple ignore
}
void UIThreadCall(boost::function0<void> fn)
{
// Only used for built-in mining, which is disabled, simple ignore
}
void MainFrameRepaint()
{
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
QApplication app(argc, argv); QApplication app(argc, argv);
app.setQuitOnLastWindowClosed(false);
BitcoinGUI window;
guiref = &window;
try { try {
if(AppInit2(argc, argv)) if(AppInit2(argc, argv))
{ {
ClientModel model; ClientModel model;
BitcoinGUI window;
window.setModel(&model); window.setModel(&model);
window.show(); window.show();
guiref = 0;
/* Depending on settings: QApplication::setQuitOnLastWindowClosed(false); */ /* Depending on settings: QApplication::setQuitOnLastWindowClosed(false); */
int retval = app.exec(); int retval = app.exec();

View File

@ -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 */
@ -93,8 +94,6 @@ BitcoinGUI::BitcoinGUI(QWidget *parent):
QVBoxLayout *vbox = new QVBoxLayout(); QVBoxLayout *vbox = new QVBoxLayout();
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());
@ -102,7 +101,7 @@ BitcoinGUI::BitcoinGUI(QWidget *parent):
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);
} }

View File

@ -3,11 +3,13 @@
#include "guiconstants.h" #include "guiconstants.h"
#include "optionsmodel.h" #include "optionsmodel.h"
#include "addresstablemodel.h" #include "addresstablemodel.h"
#include "transactiontablemodel.h"
#include <QTimer> #include <QTimer>
ClientModel::ClientModel(QObject *parent) : ClientModel::ClientModel(QObject *parent) :
QObject(parent), optionsModel(0), addressTableModel(0) QObject(parent), optionsModel(0), addressTableModel(0),
transactionTableModel(0)
{ {
/* Until signal notifications is built into the bitcoin core, /* Until signal notifications is built into the bitcoin core,
simply update everything after polling using a timer. simply update everything after polling using a timer.
@ -18,6 +20,7 @@ ClientModel::ClientModel(QObject *parent) :
optionsModel = new OptionsModel(this); optionsModel = new OptionsModel(this);
addressTableModel = new AddressTableModel(this); addressTableModel = new AddressTableModel(this);
transactionTableModel = new TransactionTableModel(this);
} }
qint64 ClientModel::getBalance() qint64 ClientModel::getBalance()
@ -140,3 +143,8 @@ AddressTableModel *ClientModel::getAddressTableModel()
{ {
return addressTableModel; return addressTableModel;
} }
TransactionTableModel *ClientModel::getTransactionTableModel()
{
return transactionTableModel;
}

View File

@ -207,6 +207,10 @@ MainOptionsPage::MainOptionsPage(QWidget *parent):
connect(connect_socks4, SIGNAL(toggled(bool)), proxy_ip, SLOT(setEnabled(bool))); connect(connect_socks4, SIGNAL(toggled(bool)), proxy_ip, SLOT(setEnabled(bool)));
connect(connect_socks4, SIGNAL(toggled(bool)), proxy_port, SLOT(setEnabled(bool))); connect(connect_socks4, SIGNAL(toggled(bool)), proxy_port, SLOT(setEnabled(bool)));
#ifndef USE_UPNP
map_port_upnp->setDisabled(true);
#endif
} }
void MainOptionsPage::setMapper(MonitoredDataMapper *mapper) void MainOptionsPage::setMapper(MonitoredDataMapper *mapper)

View File

@ -1,5 +1,6 @@
#include "optionsmodel.h" #include "optionsmodel.h"
#include "main.h" #include "main.h"
#include "net.h"
#include <QDebug> #include <QDebug>
@ -47,6 +48,7 @@ bool OptionsModel::setData(const QModelIndex & index, const QVariant & value, in
bool successful = true; /* set to false on parse error */ bool successful = true; /* set to false on parse error */
if(role == Qt::EditRole) if(role == Qt::EditRole)
{ {
CWalletDB walletdb;
switch(index.row()) switch(index.row())
{ {
case StartAtStartup: case StartAtStartup:
@ -54,15 +56,22 @@ bool OptionsModel::setData(const QModelIndex & index, const QVariant & value, in
break; break;
case MinimizeToTray: case MinimizeToTray:
fMinimizeToTray = value.toBool(); fMinimizeToTray = value.toBool();
walletdb.WriteSetting("fMinimizeToTray", fMinimizeToTray);
break; break;
case MapPortUPnP: case MapPortUPnP:
fUseUPnP = value.toBool(); fUseUPnP = value.toBool();
walletdb.WriteSetting("fUseUPnP", fUseUPnP);
#ifdef USE_UPNP
MapPort(fUseUPnP);
#endif
break; break;
case MinimizeOnClose: case MinimizeOnClose:
fMinimizeOnClose = value.toBool(); fMinimizeOnClose = value.toBool();
walletdb.WriteSetting("fMinimizeOnClose", fMinimizeOnClose);
break; break;
case ConnectSOCKS4: case ConnectSOCKS4:
fUseProxy = value.toBool(); fUseProxy = value.toBool();
walletdb.WriteSetting("fUseProxy", fUseProxy);
break; break;
case ProxyIP: case ProxyIP:
{ {
@ -71,6 +80,7 @@ bool OptionsModel::setData(const QModelIndex & index, const QVariant & value, in
if (addr.ip != INADDR_NONE) if (addr.ip != INADDR_NONE)
{ {
addrProxy.ip = addr.ip; addrProxy.ip = addr.ip;
walletdb.WriteSetting("addrProxy", addrProxy);
} else { } else {
successful = false; successful = false;
} }
@ -82,6 +92,7 @@ bool OptionsModel::setData(const QModelIndex & index, const QVariant & value, in
if (nPort > 0 && nPort < USHRT_MAX) if (nPort > 0 && nPort < USHRT_MAX)
{ {
addrProxy.port = htons(nPort); addrProxy.port = htons(nPort);
walletdb.WriteSetting("addrProxy", addrProxy);
} else { } else {
successful = false; successful = false;
} }
@ -92,6 +103,7 @@ bool OptionsModel::setData(const QModelIndex & index, const QVariant & value, in
if(ParseMoney(value.toString().toStdString(), retval)) if(ParseMoney(value.toString().toStdString(), retval))
{ {
nTransactionFee = retval; nTransactionFee = retval;
walletdb.WriteSetting("nTransactionFee", nTransactionFee);
} else { } else {
successful = false; /* parse error */ successful = false; /* parse error */
} }
@ -111,12 +123,12 @@ qint64 OptionsModel::getTransactionFee()
return nTransactionFee; return nTransactionFee;
} }
bool getMinimizeToTray() bool OptionsModel::getMinimizeToTray()
{ {
return fMinimizeToTray; return fMinimizeToTray;
} }
bool getMinimizeOnClose() bool OptionsModel::getMinimizeOnClose()
{ {
return fMinimizeOnClose; return fMinimizeOnClose;
} }