// 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. #include #include #include #include #include #include #include WalletController::WalletController(interfaces::Node& node, const PlatformStyle* platform_style, OptionsModel* options_model, QObject* parent) : QObject(parent) , m_node(node) , m_platform_style(platform_style) , m_options_model(options_model) { m_handler_load_wallet = m_node.handleLoadWallet([this](std::unique_ptr wallet) { getOrCreateWallet(std::move(wallet)); }); for (std::unique_ptr& wallet : m_node.getWallets()) { getOrCreateWallet(std::move(wallet)); } m_activity_thread.start(); } // Not using the default destructor because not all member types definitions are // available in the header, just forward declared. WalletController::~WalletController() { m_activity_thread.quit(); m_activity_thread.wait(); } std::vector WalletController::getWallets() const { QMutexLocker locker(&m_mutex); return m_wallets; } std::vector WalletController::getWalletsAvailableToOpen() const { QMutexLocker locker(&m_mutex); std::vector wallets = m_node.listWalletDir(); for (WalletModel* wallet_model : m_wallets) { auto it = std::remove(wallets.begin(), wallets.end(), wallet_model->wallet().getWalletName()); if (it != wallets.end()) wallets.erase(it); } return wallets; } WalletModel* WalletController::openWallet(const std::string& name, QWidget* parent) { std::string error, warning; WalletModel* wallet_model = getOrCreateWallet(m_node.loadWallet(name, error, warning)); if (!wallet_model) { QMessageBox::warning(parent, tr("Open Wallet"), QString::fromStdString(error)); } if (!warning.empty()) { QMessageBox::information(parent, tr("Open Wallet"), QString::fromStdString(warning)); } return wallet_model; } WalletModel* WalletController::getOrCreateWallet(std::unique_ptr 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_platform_style, m_options_model, nullptr); m_wallets.push_back(wallet_model); connect(wallet_model, &WalletModel::unload, [this, wallet_model] { removeAndDeleteWallet(wallet_model); }); // Re-emit coinsSent signal from wallet model. connect(wallet_model, &WalletModel::coinsSent, this, &WalletController::coinsSent); // Notify walletAdded signal on the GUI thread. if (QThread::currentThread() == thread()) { addWallet(wallet_model); } else { // Handler callback runs in a different thread so fix wallet model thread affinity. wallet_model->moveToThread(thread()); QMetaObject::invokeMethod(this, "addWallet", Qt::QueuedConnection, Q_ARG(WalletModel*, wallet_model)); } return wallet_model; } void WalletController::addWallet(WalletModel* wallet_model) { // Take ownership of the wallet model and register it. wallet_model->setParent(this); Q_EMIT walletAdded(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; }