2018-01-02 14:12:05 -03:00
|
|
|
// Copyright (c) 2011-2017 The Bitcoin Core developers
|
2014-12-13 01:09:33 -03:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2013-11-04 12:20:43 -03:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2014-11-03 12:16:40 -03:00
|
|
|
#ifndef BITCOIN_QT_WALLETMODEL_H
|
|
|
|
#define BITCOIN_QT_WALLETMODEL_H
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2017-09-19 22:12:25 -03:00
|
|
|
#include <amount.h>
|
|
|
|
#include <key.h>
|
|
|
|
#include <serialize.h>
|
|
|
|
#include <script/standard.h>
|
|
|
|
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <qt/paymentrequestplus.h>
|
|
|
|
#include <qt/walletmodeltransaction.h>
|
2011-11-26 03:02:04 -03:00
|
|
|
|
2018-04-07 04:42:02 -03:00
|
|
|
#include <interfaces/wallet.h>
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <support/allocators/secure.h>
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2013-08-12 11:03:03 -04:00
|
|
|
#include <map>
|
2013-11-16 13:37:31 -03:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <QObject>
|
2013-04-13 02:13:08 -03:00
|
|
|
|
2018-02-10 23:06:35 -03:00
|
|
|
enum class OutputType;
|
2018-01-16 17:11:40 -03:00
|
|
|
|
2011-06-30 12:05:29 -04:00
|
|
|
class AddressTableModel;
|
2013-04-13 02:13:08 -03:00
|
|
|
class OptionsModel;
|
2015-07-28 10:20:14 -03:00
|
|
|
class PlatformStyle;
|
2013-11-05 14:03:05 -03:00
|
|
|
class RecentRequestsTableModel;
|
2013-12-16 18:54:02 -03:00
|
|
|
class TransactionTableModel;
|
2013-08-30 14:04:48 -04:00
|
|
|
class WalletModelTransaction;
|
2013-11-16 13:37:31 -03:00
|
|
|
|
|
|
|
class CCoinControl;
|
2013-08-12 11:03:03 -04:00
|
|
|
class CKeyID;
|
|
|
|
class COutPoint;
|
2013-11-16 13:37:31 -03:00
|
|
|
class COutput;
|
|
|
|
class CPubKey;
|
|
|
|
class uint256;
|
2013-04-13 02:13:08 -03:00
|
|
|
|
2018-04-07 04:42:02 -03:00
|
|
|
namespace interfaces {
|
2017-04-17 19:56:44 -03:00
|
|
|
class Node;
|
2018-04-07 04:42:02 -03:00
|
|
|
} // namespace interfaces
|
2017-04-17 19:56:44 -03:00
|
|
|
|
2012-07-05 11:43:28 -04:00
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
class QTimer;
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
2012-03-18 19:14:03 -03:00
|
|
|
class SendCoinsRecipient
|
2011-07-16 13:01:05 -04:00
|
|
|
{
|
2012-03-18 19:14:03 -03:00
|
|
|
public:
|
2014-07-23 08:34:36 -04:00
|
|
|
explicit SendCoinsRecipient() : amount(0), fSubtractFeeFromAmount(false), nVersion(SendCoinsRecipient::CURRENT_VERSION) { }
|
2016-09-09 08:43:29 -03:00
|
|
|
explicit SendCoinsRecipient(const QString &addr, const QString &_label, const CAmount& _amount, const QString &_message):
|
|
|
|
address(addr), label(_label), amount(_amount), message(_message), fSubtractFeeFromAmount(false), nVersion(SendCoinsRecipient::CURRENT_VERSION) {}
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2015-01-08 10:42:04 -03:00
|
|
|
// If from an unauthenticated payment request, this is used for storing
|
2013-10-24 11:02:39 -03:00
|
|
|
// the addresses, e.g. address-A<br />address-B<br />address-C.
|
|
|
|
// Info: As we don't need to process addresses in here when using
|
|
|
|
// payment requests, we can abuse it for displaying an address list.
|
|
|
|
// Todo: This is a hack, should be replaced with a cleaner solution!
|
2011-07-16 13:01:05 -04:00
|
|
|
QString address;
|
|
|
|
QString label;
|
2014-04-22 19:46:19 -03:00
|
|
|
CAmount amount;
|
2013-10-27 17:52:01 -03:00
|
|
|
// If from a payment request, this is used for storing the memo
|
2013-10-18 08:44:27 -03:00
|
|
|
QString message;
|
2013-07-22 02:50:39 -04:00
|
|
|
|
|
|
|
// If from a payment request, paymentRequest.IsInitialized() will be true
|
|
|
|
PaymentRequestPlus paymentRequest;
|
2013-10-24 11:02:39 -03:00
|
|
|
// Empty if no authentication or invalid signature/cert/etc.
|
|
|
|
QString authenticatedMerchant;
|
2014-01-14 01:05:43 -03:00
|
|
|
|
2014-07-23 08:34:36 -04:00
|
|
|
bool fSubtractFeeFromAmount; // memory only
|
|
|
|
|
2014-01-22 05:46:15 -03:00
|
|
|
static const int CURRENT_VERSION = 1;
|
2014-01-14 01:05:43 -03:00
|
|
|
int nVersion;
|
|
|
|
|
2014-09-02 03:58:09 -04:00
|
|
|
ADD_SERIALIZE_METHODS;
|
overhaul serialization code
The implementation of each class' serialization/deserialization is no longer
passed within a macro. The implementation now lies within a template of form:
template <typename T, typename Stream, typename Operation>
inline static size_t SerializationOp(T thisPtr, Stream& s, Operation ser_action, int nType, int nVersion) {
size_t nSerSize = 0;
/* CODE */
return nSerSize;
}
In cases when codepath should depend on whether or not we are just deserializing
(old fGetSize, fWrite, fRead flags) an additional clause can be used:
bool fRead = boost::is_same<Operation, CSerActionUnserialize>();
The IMPLEMENT_SERIALIZE macro will now be a freestanding clause added within
class' body (similiar to Qt's Q_OBJECT) to implement GetSerializeSize,
Serialize and Unserialize. These are now wrappers around
the "SerializationOp" template.
2014-08-20 02:42:31 -04:00
|
|
|
|
2014-08-20 16:44:38 -04:00
|
|
|
template <typename Stream, typename Operation>
|
2016-10-28 20:29:17 -03:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
2014-09-01 15:36:46 -04:00
|
|
|
std::string sAddress = address.toStdString();
|
|
|
|
std::string sLabel = label.toStdString();
|
|
|
|
std::string sMessage = message.toStdString();
|
2014-01-14 01:05:43 -03:00
|
|
|
std::string sPaymentRequest;
|
2014-09-01 15:36:46 -04:00
|
|
|
if (!ser_action.ForRead() && paymentRequest.IsInitialized())
|
|
|
|
paymentRequest.SerializeToString(&sPaymentRequest);
|
|
|
|
std::string sAuthenticatedMerchant = authenticatedMerchant.toStdString();
|
2014-01-14 01:05:43 -03:00
|
|
|
|
2014-09-01 15:36:46 -04:00
|
|
|
READWRITE(this->nVersion);
|
2014-01-14 01:05:43 -03:00
|
|
|
READWRITE(sAddress);
|
|
|
|
READWRITE(sLabel);
|
2014-08-20 16:44:38 -04:00
|
|
|
READWRITE(amount);
|
2014-01-14 01:05:43 -03:00
|
|
|
READWRITE(sMessage);
|
|
|
|
READWRITE(sPaymentRequest);
|
|
|
|
READWRITE(sAuthenticatedMerchant);
|
|
|
|
|
2014-09-01 15:36:46 -04:00
|
|
|
if (ser_action.ForRead())
|
2014-01-14 01:05:43 -03:00
|
|
|
{
|
2014-09-01 15:36:46 -04:00
|
|
|
address = QString::fromStdString(sAddress);
|
|
|
|
label = QString::fromStdString(sLabel);
|
|
|
|
message = QString::fromStdString(sMessage);
|
2014-01-14 01:05:43 -03:00
|
|
|
if (!sPaymentRequest.empty())
|
2014-09-01 15:36:46 -04:00
|
|
|
paymentRequest.parse(QByteArray::fromRawData(sPaymentRequest.data(), sPaymentRequest.size()));
|
|
|
|
authenticatedMerchant = QString::fromStdString(sAuthenticatedMerchant);
|
2014-01-14 01:05:43 -03:00
|
|
|
}
|
overhaul serialization code
The implementation of each class' serialization/deserialization is no longer
passed within a macro. The implementation now lies within a template of form:
template <typename T, typename Stream, typename Operation>
inline static size_t SerializationOp(T thisPtr, Stream& s, Operation ser_action, int nType, int nVersion) {
size_t nSerSize = 0;
/* CODE */
return nSerSize;
}
In cases when codepath should depend on whether or not we are just deserializing
(old fGetSize, fWrite, fRead flags) an additional clause can be used:
bool fRead = boost::is_same<Operation, CSerActionUnserialize>();
The IMPLEMENT_SERIALIZE macro will now be a freestanding clause added within
class' body (similiar to Qt's Q_OBJECT) to implement GetSerializeSize,
Serialize and Unserialize. These are now wrappers around
the "SerializationOp" template.
2014-08-20 02:42:31 -04:00
|
|
|
}
|
2011-07-16 13:01:05 -04:00
|
|
|
};
|
|
|
|
|
2011-11-13 09:19:52 -03:00
|
|
|
/** Interface to Bitcoin wallet from Qt view code. */
|
2011-06-30 12:05:29 -04:00
|
|
|
class WalletModel : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
2013-01-23 17:51:02 -03:00
|
|
|
|
2011-06-30 12:05:29 -04:00
|
|
|
public:
|
2018-04-07 04:42:02 -03:00
|
|
|
explicit WalletModel(std::unique_ptr<interfaces::Wallet> wallet, interfaces::Node& node, const PlatformStyle *platformStyle, OptionsModel *optionsModel, QObject *parent = 0);
|
2012-05-06 13:40:58 -04:00
|
|
|
~WalletModel();
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2011-08-31 11:08:31 -03:00
|
|
|
enum StatusCode // Returned by sendCoins
|
2011-06-30 12:05:29 -04:00
|
|
|
{
|
|
|
|
OK,
|
|
|
|
InvalidAmount,
|
|
|
|
InvalidAddress,
|
|
|
|
AmountExceedsBalance,
|
|
|
|
AmountWithFeeExceedsBalance,
|
2011-07-16 13:01:05 -04:00
|
|
|
DuplicateAddress,
|
2011-08-31 11:08:31 -03:00
|
|
|
TransactionCreationFailed, // Error returned when wallet is still locked
|
2014-11-01 20:14:47 -03:00
|
|
|
TransactionCommitFailed,
|
2015-01-30 23:54:55 -03:00
|
|
|
AbsurdFee,
|
2015-01-08 10:42:04 -03:00
|
|
|
PaymentRequestExpired
|
2011-06-30 12:05:29 -04:00
|
|
|
};
|
|
|
|
|
2011-08-23 15:08:42 -03:00
|
|
|
enum EncryptionStatus
|
|
|
|
{
|
|
|
|
Unencrypted, // !wallet->IsCrypted()
|
|
|
|
Locked, // wallet->IsCrypted() && wallet->IsLocked()
|
|
|
|
Unlocked // wallet->IsCrypted() && !wallet->IsLocked()
|
|
|
|
};
|
|
|
|
|
2011-06-30 12:05:29 -04:00
|
|
|
OptionsModel *getOptionsModel();
|
|
|
|
AddressTableModel *getAddressTableModel();
|
|
|
|
TransactionTableModel *getTransactionTableModel();
|
2013-11-05 14:03:05 -03:00
|
|
|
RecentRequestsTableModel *getRecentRequestsTableModel();
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2011-08-23 15:08:42 -03:00
|
|
|
EncryptionStatus getEncryptionStatus() const;
|
|
|
|
|
2011-07-16 13:01:05 -04:00
|
|
|
// Check address for validity
|
|
|
|
bool validateAddress(const QString &address);
|
|
|
|
|
2011-08-08 11:38:17 -04:00
|
|
|
// Return status record for SendCoins, contains error id + information
|
2011-07-16 13:01:05 -04:00
|
|
|
struct SendCoinsReturn
|
|
|
|
{
|
2016-10-30 14:22:22 -03:00
|
|
|
SendCoinsReturn(StatusCode _status = OK, QString _reasonCommitFailed = "")
|
|
|
|
: status(_status),
|
|
|
|
reasonCommitFailed(_reasonCommitFailed)
|
|
|
|
{
|
|
|
|
}
|
2011-07-16 13:01:05 -04:00
|
|
|
StatusCode status;
|
2016-10-30 14:22:22 -03:00
|
|
|
QString reasonCommitFailed;
|
2011-07-16 13:01:05 -04:00
|
|
|
};
|
|
|
|
|
2013-08-30 14:04:48 -04:00
|
|
|
// prepare transaction for getting txfee before sending coins
|
2017-06-28 16:41:55 -04:00
|
|
|
SendCoinsReturn prepareTransaction(WalletModelTransaction &transaction, const CCoinControl& coinControl);
|
2013-08-30 14:04:48 -04:00
|
|
|
|
2011-08-08 11:38:17 -04:00
|
|
|
// Send coins to a list of recipients
|
2013-08-30 14:04:48 -04:00
|
|
|
SendCoinsReturn sendCoins(WalletModelTransaction &transaction);
|
2011-08-24 17:07:26 -03:00
|
|
|
|
|
|
|
// Wallet encryption
|
2011-11-26 03:02:04 -03:00
|
|
|
bool setWalletEncrypted(bool encrypted, const SecureString &passphrase);
|
2011-08-24 17:07:26 -03:00
|
|
|
// Passphrase only needed when unlocking
|
2011-11-26 03:02:04 -03:00
|
|
|
bool setWalletLocked(bool locked, const SecureString &passPhrase=SecureString());
|
|
|
|
bool changePassphrase(const SecureString &oldPass, const SecureString &newPass);
|
2011-08-24 17:07:26 -03:00
|
|
|
|
|
|
|
// RAI object for unlocking wallet, returned by requestUnlock()
|
|
|
|
class UnlockContext
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
UnlockContext(WalletModel *wallet, bool valid, bool relock);
|
|
|
|
~UnlockContext();
|
|
|
|
|
|
|
|
bool isValid() const { return valid; }
|
|
|
|
|
2011-08-31 11:08:31 -03:00
|
|
|
// Copy operator and constructor transfer the context
|
|
|
|
UnlockContext(const UnlockContext& obj) { CopyFrom(obj); }
|
|
|
|
UnlockContext& operator=(const UnlockContext& rhs) { CopyFrom(rhs); return *this; }
|
2011-08-24 17:07:26 -03:00
|
|
|
private:
|
|
|
|
WalletModel *wallet;
|
|
|
|
bool valid;
|
|
|
|
mutable bool relock; // mutable, as it can be set to false by copying
|
|
|
|
|
|
|
|
void CopyFrom(const UnlockContext& rhs);
|
|
|
|
};
|
|
|
|
|
|
|
|
UnlockContext requestUnlock();
|
|
|
|
|
2014-01-14 01:05:43 -03:00
|
|
|
void loadReceiveRequests(std::vector<std::string>& vReceiveRequests);
|
|
|
|
bool saveReceiveRequest(const std::string &sAddress, const int64_t nId, const std::string &sRequest);
|
|
|
|
|
2017-02-03 18:04:39 -03:00
|
|
|
bool bumpFee(uint256 hash);
|
|
|
|
|
2016-09-21 07:37:00 -03:00
|
|
|
static bool isWalletEnabled();
|
|
|
|
|
2018-04-07 04:42:02 -03:00
|
|
|
interfaces::Node& node() const { return m_node; }
|
|
|
|
interfaces::Wallet& wallet() const { return *m_wallet; }
|
2016-10-21 05:24:03 -03:00
|
|
|
|
2018-03-06 01:26:40 -03:00
|
|
|
QString getWalletName() const;
|
|
|
|
|
2017-04-17 19:56:44 -03:00
|
|
|
bool isMultiwallet();
|
2011-06-30 12:05:29 -04:00
|
|
|
private:
|
2018-04-07 04:42:02 -03:00
|
|
|
std::unique_ptr<interfaces::Wallet> m_wallet;
|
|
|
|
std::unique_ptr<interfaces::Handler> m_handler_status_changed;
|
|
|
|
std::unique_ptr<interfaces::Handler> m_handler_address_book_changed;
|
|
|
|
std::unique_ptr<interfaces::Handler> m_handler_transaction_changed;
|
|
|
|
std::unique_ptr<interfaces::Handler> m_handler_show_progress;
|
|
|
|
std::unique_ptr<interfaces::Handler> m_handler_watch_only_changed;
|
|
|
|
interfaces::Node& m_node;
|
2017-04-17 19:56:44 -03:00
|
|
|
|
2014-07-26 15:05:11 -04:00
|
|
|
bool fHaveWatchOnly;
|
2014-08-16 20:34:42 -04:00
|
|
|
bool fForceCheckBalanceChanged;
|
2011-06-30 12:05:29 -04:00
|
|
|
|
|
|
|
// Wallet has an options model for wallet-specific options
|
|
|
|
// (transaction fee, for example)
|
|
|
|
OptionsModel *optionsModel;
|
|
|
|
|
|
|
|
AddressTableModel *addressTableModel;
|
|
|
|
TransactionTableModel *transactionTableModel;
|
2013-11-05 14:03:05 -03:00
|
|
|
RecentRequestsTableModel *recentRequestsTableModel;
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2011-08-31 11:08:31 -03:00
|
|
|
// Cache some values to be able to detect changes
|
2018-04-07 04:42:02 -03:00
|
|
|
interfaces::WalletBalances m_cached_balances;
|
2011-08-23 15:08:42 -03:00
|
|
|
EncryptionStatus cachedEncryptionStatus;
|
2012-07-05 11:43:28 -04:00
|
|
|
int cachedNumBlocks;
|
|
|
|
|
|
|
|
QTimer *pollTimer;
|
2011-07-17 08:06:43 -04:00
|
|
|
|
2012-05-06 13:40:58 -04:00
|
|
|
void subscribeToCoreSignals();
|
|
|
|
void unsubscribeFromCoreSignals();
|
2018-04-07 04:42:02 -03:00
|
|
|
void checkBalanceChanged(const interfaces::WalletBalances& new_balances);
|
2012-07-05 11:43:28 -04:00
|
|
|
|
2015-07-14 08:59:05 -03:00
|
|
|
Q_SIGNALS:
|
2011-08-31 11:08:31 -03:00
|
|
|
// Signal that balance in wallet changed
|
2018-04-07 04:42:02 -03:00
|
|
|
void balanceChanged(const interfaces::WalletBalances& balances);
|
2011-08-31 11:08:31 -03:00
|
|
|
|
|
|
|
// Encryption status of wallet changed
|
2016-10-24 04:21:51 -03:00
|
|
|
void encryptionStatusChanged();
|
2011-08-31 11:08:31 -03:00
|
|
|
|
|
|
|
// Signal emitted when wallet needs to be unlocked
|
|
|
|
// It is valid behaviour for listeners to keep the wallet locked after this signal;
|
|
|
|
// this means that the unlocking failed or was cancelled.
|
2011-08-24 17:07:26 -03:00
|
|
|
void requireUnlock();
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2013-10-24 11:02:39 -03:00
|
|
|
// Fired when a message should be reported to the user
|
2012-12-03 09:24:42 -03:00
|
|
|
void message(const QString &title, const QString &message, unsigned int style);
|
2011-06-30 12:05:29 -04:00
|
|
|
|
2013-07-22 02:50:39 -04:00
|
|
|
// Coins sent: from wallet, to recipient, in (serialized) transaction:
|
2018-03-23 18:14:39 -03:00
|
|
|
void coinsSent(WalletModel* wallet, SendCoinsRecipient recipient, QByteArray transaction);
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2014-03-18 20:26:14 -03:00
|
|
|
// Show progress dialog e.g. for rescan
|
|
|
|
void showProgress(const QString &title, int nProgress);
|
|
|
|
|
2014-07-26 15:05:11 -04:00
|
|
|
// Watch-only address added
|
|
|
|
void notifyWatchonlyChanged(bool fHaveWatchonly);
|
|
|
|
|
2015-07-14 08:59:05 -03:00
|
|
|
public Q_SLOTS:
|
2012-05-05 10:07:14 -04:00
|
|
|
/* Wallet status might have changed */
|
|
|
|
void updateStatus();
|
|
|
|
/* New transaction, or transaction changed status */
|
2014-10-28 15:52:21 -03:00
|
|
|
void updateTransaction();
|
2012-05-05 10:07:14 -04:00
|
|
|
/* New, updated or removed address book entry */
|
2013-08-29 10:19:43 -04:00
|
|
|
void updateAddressBook(const QString &address, const QString &label, bool isMine, const QString &purpose, int status);
|
2014-08-28 17:20:46 -04:00
|
|
|
/* Watch-only added */
|
2014-07-26 15:05:11 -04:00
|
|
|
void updateWatchOnlyFlag(bool fHaveWatchonly);
|
2012-07-05 11:43:28 -04:00
|
|
|
/* Current, immature or unconfirmed balance might have changed - emit 'balanceChanged' if so */
|
|
|
|
void pollBalanceChanged();
|
2011-06-30 12:05:29 -04:00
|
|
|
};
|
|
|
|
|
2014-11-03 12:16:40 -03:00
|
|
|
#endif // BITCOIN_QT_WALLETMODEL_H
|