2019-01-18 21:20:55 +01:00
|
|
|
// Copyright (c) 2019 The Bitcoin Core developers
|
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2021-11-06 20:14:18 +01:00
|
|
|
#include <qt/guiutil.h>
|
2019-01-18 21:20:55 +01:00
|
|
|
#include <qt/walletcontroller.h>
|
|
|
|
|
|
|
|
#include <interfaces/handler.h>
|
|
|
|
#include <interfaces/node.h>
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
2019-03-22 11:07:35 +01:00
|
|
|
#include <QApplication>
|
2019-02-12 19:19:05 +01:00
|
|
|
#include <QMessageBox>
|
2019-01-18 21:20:55 +01:00
|
|
|
#include <QMutexLocker>
|
|
|
|
#include <QThread>
|
2019-03-22 11:07:35 +01:00
|
|
|
#include <QWindow>
|
2019-01-18 21:20:55 +01:00
|
|
|
|
|
|
|
WalletController::WalletController(interfaces::Node& node, OptionsModel* options_model, QObject* parent)
|
|
|
|
: QObject(parent)
|
|
|
|
, m_node(node)
|
|
|
|
, m_options_model(options_model)
|
|
|
|
{
|
|
|
|
m_handler_load_wallet = m_node.handleLoadWallet([this](std::unique_ptr<interfaces::Wallet> wallet) {
|
|
|
|
getOrCreateWallet(std::move(wallet));
|
|
|
|
});
|
|
|
|
|
|
|
|
for (std::unique_ptr<interfaces::Wallet>& wallet : m_node.getWallets()) {
|
|
|
|
getOrCreateWallet(std::move(wallet));
|
|
|
|
}
|
2019-02-12 19:19:05 +01:00
|
|
|
|
|
|
|
m_activity_thread.start();
|
2019-01-18 21:20:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Not using the default destructor because not all member types definitions are
|
|
|
|
// available in the header, just forward declared.
|
2019-02-12 19:19:05 +01:00
|
|
|
WalletController::~WalletController()
|
|
|
|
{
|
|
|
|
m_activity_thread.quit();
|
|
|
|
m_activity_thread.wait();
|
|
|
|
}
|
2019-01-18 21:20:55 +01:00
|
|
|
|
2019-07-08 16:27:18 +02:00
|
|
|
std::vector<WalletModel*> WalletController::getOpenWallets() const
|
2019-01-18 21:20:55 +01:00
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
return m_wallets;
|
|
|
|
}
|
|
|
|
|
2019-07-08 16:27:18 +02:00
|
|
|
std::map<std::string, bool> WalletController::listWalletDir() const
|
2019-02-12 19:19:05 +01:00
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
2019-07-08 16:27:18 +02:00
|
|
|
std::map<std::string, bool> wallets;
|
|
|
|
for (const std::string& name : m_node.listWalletDir()) {
|
|
|
|
wallets[name] = false;
|
|
|
|
}
|
2019-02-12 19:19:05 +01:00
|
|
|
for (WalletModel* wallet_model : m_wallets) {
|
2019-07-08 16:27:18 +02:00
|
|
|
auto it = wallets.find(wallet_model->wallet().getWalletName());
|
|
|
|
if (it != wallets.end()) it->second = true;
|
2019-02-12 19:19:05 +01:00
|
|
|
}
|
|
|
|
return wallets;
|
|
|
|
}
|
|
|
|
|
|
|
|
OpenWalletActivity* WalletController::openWallet(const std::string& name, QWidget* parent)
|
|
|
|
{
|
|
|
|
OpenWalletActivity* activity = new OpenWalletActivity(this, name);
|
|
|
|
activity->moveToThread(&m_activity_thread);
|
|
|
|
return activity;
|
|
|
|
}
|
|
|
|
|
2019-02-14 21:46:22 +01:00
|
|
|
void WalletController::closeWallet(WalletModel* wallet_model, QWidget* parent)
|
|
|
|
{
|
|
|
|
QMessageBox box(parent);
|
|
|
|
box.setWindowTitle(tr("Close wallet"));
|
2021-11-06 20:14:18 +01:00
|
|
|
box.setText(tr("Are you sure you wish to close wallet <i>%1</i>?").arg(GUIUtil::HtmlEscape(wallet_model->getDisplayName())));
|
2019-02-14 21:46:22 +01:00
|
|
|
box.setInformativeText(tr("Closing the wallet for too long can result in having to resync the entire chain if pruning is enabled."));
|
|
|
|
box.setStandardButtons(QMessageBox::Yes|QMessageBox::Cancel);
|
|
|
|
box.setDefaultButton(QMessageBox::Yes);
|
|
|
|
if (box.exec() != QMessageBox::Yes) return;
|
|
|
|
|
|
|
|
// First remove wallet from node.
|
|
|
|
wallet_model->wallet().remove();
|
|
|
|
// Now release the model.
|
|
|
|
removeAndDeleteWallet(wallet_model);
|
|
|
|
}
|
|
|
|
|
2019-01-18 21:20:55 +01:00
|
|
|
WalletModel* WalletController::getOrCreateWallet(std::unique_ptr<interfaces::Wallet> wallet)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
|
|
|
|
// Return model instance if exists.
|
|
|
|
if (!m_wallets.empty()) {
|
|
|
|
std::string name = wallet->getWalletName();
|
|
|
|
for (WalletModel* wallet_model : m_wallets) {
|
|
|
|
if (wallet_model->wallet().getWalletName() == name) {
|
|
|
|
return wallet_model;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instantiate model and register it.
|
|
|
|
WalletModel* wallet_model = new WalletModel(std::move(wallet), m_node, m_options_model, nullptr);
|
2019-08-12 14:15:48 +02:00
|
|
|
// Handler callback runs in a different thread so fix wallet model thread affinity.
|
|
|
|
wallet_model->moveToThread(thread());
|
|
|
|
wallet_model->setParent(this);
|
2019-01-18 21:20:55 +01:00
|
|
|
m_wallets.push_back(wallet_model);
|
|
|
|
|
2019-10-26 13:00:01 +02:00
|
|
|
// WalletModel::startPollBalance needs to be called in a thread managed by
|
|
|
|
// Qt because of startTimer. Considering the current thread can be a RPC
|
|
|
|
// thread, better delegate the calling to Qt with Qt::AutoConnection.
|
|
|
|
const bool called = QMetaObject::invokeMethod(wallet_model, "startPollBalance");
|
|
|
|
assert(called);
|
|
|
|
|
2019-01-18 21:20:55 +01:00
|
|
|
connect(wallet_model, &WalletModel::unload, [this, wallet_model] {
|
2019-03-22 11:07:35 +01:00
|
|
|
// Defer removeAndDeleteWallet when no modal widget is active.
|
|
|
|
// TODO: remove this workaround by removing usage of QDiallog::exec.
|
|
|
|
if (QApplication::activeModalWidget()) {
|
|
|
|
connect(qApp, &QApplication::focusWindowChanged, wallet_model, [this, wallet_model]() {
|
|
|
|
if (!QApplication::activeModalWidget()) {
|
|
|
|
removeAndDeleteWallet(wallet_model);
|
|
|
|
}
|
|
|
|
}, Qt::QueuedConnection);
|
|
|
|
} else {
|
|
|
|
removeAndDeleteWallet(wallet_model);
|
|
|
|
}
|
2019-01-18 21:20:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
// Re-emit coinsSent signal from wallet model.
|
|
|
|
connect(wallet_model, &WalletModel::coinsSent, this, &WalletController::coinsSent);
|
|
|
|
|
|
|
|
// Notify walletAdded signal on the GUI thread.
|
2019-08-12 14:15:48 +02:00
|
|
|
Q_EMIT walletAdded(wallet_model);
|
2019-01-18 21:20:55 +01:00
|
|
|
|
|
|
|
return wallet_model;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WalletController::removeAndDeleteWallet(WalletModel* wallet_model)
|
|
|
|
{
|
|
|
|
// Unregister wallet model.
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
m_wallets.erase(std::remove(m_wallets.begin(), m_wallets.end(), wallet_model));
|
|
|
|
}
|
|
|
|
Q_EMIT walletRemoved(wallet_model);
|
|
|
|
// Currently this can trigger the unload since the model can hold the last
|
|
|
|
// CWallet shared pointer.
|
|
|
|
delete wallet_model;
|
|
|
|
}
|
2019-02-12 19:19:05 +01:00
|
|
|
|
|
|
|
|
|
|
|
OpenWalletActivity::OpenWalletActivity(WalletController* wallet_controller, const std::string& name)
|
|
|
|
: m_wallet_controller(wallet_controller)
|
|
|
|
, m_name(name)
|
|
|
|
{}
|
|
|
|
|
|
|
|
void OpenWalletActivity::open()
|
|
|
|
{
|
|
|
|
std::string error, warning;
|
|
|
|
std::unique_ptr<interfaces::Wallet> wallet = m_wallet_controller->m_node.loadWallet(m_name, error, warning);
|
|
|
|
if (!warning.empty()) {
|
|
|
|
Q_EMIT message(QMessageBox::Warning, QString::fromStdString(warning));
|
|
|
|
}
|
|
|
|
if (wallet) {
|
|
|
|
Q_EMIT opened(m_wallet_controller->getOrCreateWallet(std::move(wallet)));
|
|
|
|
} else {
|
|
|
|
Q_EMIT message(QMessageBox::Critical, QString::fromStdString(error));
|
|
|
|
}
|
|
|
|
Q_EMIT finished();
|
|
|
|
}
|