2014-02-08 22:50:24 +01:00
// Copyright (c) 2011-2014 The Bitcoin developers
2015-01-15 11:31:35 +01:00
// Copyright (c) 2014-2015 The Darkcoin developers
2013-11-04 16:20:43 +01:00
// Distributed under the MIT/X11 software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2011-06-30 18:05:29 +02:00
# include "walletmodel.h"
2013-04-13 07:13:08 +02:00
2011-06-30 18:05:29 +02:00
# include "addresstablemodel.h"
2013-04-13 07:13:08 +02:00
# include "guiconstants.h"
2013-11-05 18:03:05 +01:00
# include "recentrequeststablemodel.h"
2013-12-16 22:54:02 +01:00
# include "transactiontablemodel.h"
2011-06-30 18:05:29 +02:00
2013-04-13 07:13:08 +02:00
# include "base58.h"
# include "db.h"
# include "keystore.h"
# include "main.h"
# include "sync.h"
2012-04-16 14:56:45 +02:00
# include "ui_interface.h"
2013-04-13 07:13:08 +02:00
# include "wallet.h"
2012-04-15 23:39:49 +02:00
# include "walletdb.h" // for BackupWallet
2015-02-11 15:47:21 +01:00
# include "spork.h"
2011-06-30 18:05:29 +02:00
2013-04-13 07:13:08 +02:00
# include <stdint.h>
# include <QDebug>
2011-07-16 19:01:05 +02:00
# include <QSet>
2012-07-05 17:43:28 +02:00
# include <QTimer>
2011-06-30 18:05:29 +02:00
2011-07-29 14:36:35 +02:00
WalletModel : : WalletModel ( CWallet * wallet , OptionsModel * optionsModel , QObject * parent ) :
QObject ( parent ) , wallet ( wallet ) , optionsModel ( optionsModel ) , addressTableModel ( 0 ) ,
2011-07-17 14:06:43 +02:00
transactionTableModel ( 0 ) ,
2013-11-05 18:03:05 +01:00
recentRequestsTableModel ( 0 ) ,
2012-02-14 12:08:00 +01:00
cachedBalance ( 0 ) , cachedUnconfirmedBalance ( 0 ) , cachedImmatureBalance ( 0 ) ,
cachedNumTransactions ( 0 ) ,
2012-07-05 17:43:28 +02:00
cachedEncryptionStatus ( Unencrypted ) ,
cachedNumBlocks ( 0 )
2011-06-30 18:05:29 +02:00
{
2015-02-10 14:24:05 +01:00
fForceCheckBalanceChanged = false ;
2011-06-30 18:05:29 +02:00
addressTableModel = new AddressTableModel ( wallet , this ) ;
transactionTableModel = new TransactionTableModel ( wallet , this ) ;
2013-11-05 18:03:05 +01:00
recentRequestsTableModel = new RecentRequestsTableModel ( wallet , this ) ;
2012-05-06 19:40:58 +02:00
2012-07-10 15:19:57 +02:00
// This timer will be fired repeatedly to update the balance
2012-07-05 17:43:28 +02:00
pollTimer = new QTimer ( this ) ;
connect ( pollTimer , SIGNAL ( timeout ( ) ) , this , SLOT ( pollBalanceChanged ( ) ) ) ;
2012-07-10 15:19:57 +02:00
pollTimer - > start ( MODEL_UPDATE_DELAY ) ;
2012-07-05 17:43:28 +02:00
2012-05-06 19:40:58 +02:00
subscribeToCoreSignals ( ) ;
}
WalletModel : : ~ WalletModel ( )
{
unsubscribeFromCoreSignals ( ) ;
2011-06-30 18:05:29 +02:00
}
2013-08-12 17:03:03 +02:00
qint64 WalletModel : : getBalance ( const CCoinControl * coinControl ) const
2011-06-30 18:05:29 +02:00
{
2013-08-12 17:03:03 +02:00
if ( coinControl )
{
qint64 nBalance = 0 ;
std : : vector < COutput > vCoins ;
wallet - > AvailableCoins ( vCoins , true , coinControl ) ;
BOOST_FOREACH ( const COutput & out , vCoins )
nBalance + = out . tx - > vout [ out . i ] . nValue ;
return nBalance ;
}
2011-06-30 18:05:29 +02:00
return wallet - > GetBalance ( ) ;
}
2014-12-09 02:17:57 +01:00
qint64 WalletModel : : getAnonymizedBalance ( ) const
{
2015-02-11 00:50:35 +01:00
return wallet - > GetAnonymizedBalance ( ) ;
2014-12-09 02:17:57 +01:00
}
2011-07-11 20:42:10 +02:00
qint64 WalletModel : : getUnconfirmedBalance ( ) const
{
return wallet - > GetUnconfirmedBalance ( ) ;
}
2012-02-14 12:08:00 +01:00
qint64 WalletModel : : getImmatureBalance ( ) const
{
return wallet - > GetImmatureBalance ( ) ;
}
2011-06-30 18:05:29 +02:00
int WalletModel : : getNumTransactions ( ) const
{
int numTransactions = 0 ;
{
2012-04-06 18:39:12 +02:00
LOCK ( wallet - > cs_wallet ) ;
2013-04-01 14:43:50 +02:00
// the size of mapWallet contains the number of unique transaction IDs
// (e.g. payments to yourself generate 2 transactions, but both share the same transaction ID)
2011-06-30 18:05:29 +02:00
numTransactions = wallet - > mapWallet . size ( ) ;
}
return numTransactions ;
}
2012-05-05 16:07:14 +02:00
void WalletModel : : updateStatus ( )
2011-06-30 18:05:29 +02:00
{
2012-05-05 16:07:14 +02:00
EncryptionStatus newEncryptionStatus = getEncryptionStatus ( ) ;
if ( cachedEncryptionStatus ! = newEncryptionStatus )
emit encryptionStatusChanged ( newEncryptionStatus ) ;
}
2012-07-05 17:43:28 +02:00
void WalletModel : : pollBalanceChanged ( )
2012-05-05 16:07:14 +02:00
{
2014-04-23 08:40:48 +02:00
// Get required locks upfront. This avoids the GUI from getting stuck on
// periodical polls if the core is holding the locks for a longer time -
// for example, during a wallet rescan.
TRY_LOCK ( cs_main , lockMain ) ;
if ( ! lockMain )
return ;
TRY_LOCK ( wallet - > cs_wallet , lockWallet ) ;
if ( ! lockWallet )
return ;
2015-02-01 21:04:20 +01:00
if ( chainActive . Height ( ) ! = cachedNumBlocks | | nDarksendRounds ! = cachedDarksendRounds | | cachedTxLocks ! = nCompleteTXLocks )
2012-07-10 15:19:57 +02:00
{
2014-04-23 08:40:48 +02:00
// Balance and number of transactions might have changed
cachedNumBlocks = chainActive . Height ( ) ;
2014-12-09 02:17:57 +01:00
cachedDarksendRounds = nDarksendRounds ;
2014-04-23 08:40:48 +02:00
2012-07-05 17:43:28 +02:00
checkBalanceChanged ( ) ;
2015-02-02 18:33:52 +01:00
if ( transactionTableModel ) {
2014-04-15 17:38:25 +02:00
transactionTableModel - > updateConfirmations ( ) ;
2015-02-02 18:33:52 +01:00
}
2012-07-05 17:43:28 +02:00
}
}
void WalletModel : : checkBalanceChanged ( )
{
2011-07-17 14:06:43 +02:00
qint64 newBalance = getBalance ( ) ;
qint64 newUnconfirmedBalance = getUnconfirmedBalance ( ) ;
2012-02-14 12:08:00 +01:00
qint64 newImmatureBalance = getImmatureBalance ( ) ;
2014-12-09 02:17:57 +01:00
qint64 newAnonymizedBalance = getAnonymizedBalance ( ) ;
2011-07-17 14:06:43 +02:00
2015-02-02 16:04:09 +01:00
if ( cachedBalance ! = newBalance | | cachedUnconfirmedBalance ! = newUnconfirmedBalance | | cachedImmatureBalance ! = newImmatureBalance | | cachedAnonymizedBalance ! = newAnonymizedBalance | | cachedTxLocks ! = nCompleteTXLocks )
2012-07-10 15:19:57 +02:00
{
2012-07-05 17:43:28 +02:00
cachedBalance = newBalance ;
cachedUnconfirmedBalance = newUnconfirmedBalance ;
cachedImmatureBalance = newImmatureBalance ;
2014-12-09 02:17:57 +01:00
cachedAnonymizedBalance = newAnonymizedBalance ;
2015-02-02 16:04:09 +01:00
cachedTxLocks = nCompleteTXLocks ;
2014-12-09 02:17:57 +01:00
emit balanceChanged ( newBalance , newUnconfirmedBalance , newImmatureBalance , newAnonymizedBalance ) ;
2012-07-05 17:43:28 +02:00
}
}
2012-02-14 12:08:00 +01:00
2015-02-10 14:24:05 +01:00
void WalletModel : : updateTransaction ( )
2012-07-05 17:43:28 +02:00
{
// Balance and number of transactions might have changed
checkBalanceChanged ( ) ;
int newNumTransactions = getNumTransactions ( ) ;
2012-07-10 15:19:57 +02:00
if ( cachedNumTransactions ! = newNumTransactions )
{
2012-07-05 17:43:28 +02:00
cachedNumTransactions = newNumTransactions ;
2012-07-10 15:19:57 +02:00
emit numTransactionsChanged ( newNumTransactions ) ;
2012-07-05 17:43:28 +02:00
}
2011-06-30 18:05:29 +02:00
}
2013-09-04 11:52:45 +02:00
void WalletModel : : updateAddressBook ( const QString & address , const QString & label ,
2013-08-29 16:19:43 +02:00
bool isMine , const QString & purpose , int status )
2012-03-24 18:48:18 +01:00
{
2012-05-05 16:07:14 +02:00
if ( addressTableModel )
2013-08-29 16:19:43 +02:00
addressTableModel - > updateEntry ( address , label , isMine , purpose , status ) ;
2012-03-24 18:48:18 +01:00
}
2011-07-16 19:01:05 +02:00
bool WalletModel : : validateAddress ( const QString & address )
2011-06-30 18:05:29 +02:00
{
2011-07-26 15:38:31 +02:00
CBitcoinAddress addressParsed ( address . toStdString ( ) ) ;
return addressParsed . IsValid ( ) ;
2011-07-16 19:01:05 +02:00
}
2013-08-12 17:03:03 +02:00
WalletModel : : SendCoinsReturn WalletModel : : prepareTransaction ( WalletModelTransaction & transaction , const CCoinControl * coinControl )
2011-07-16 19:01:05 +02:00
{
qint64 total = 0 ;
2013-08-30 20:04:48 +02:00
QList < SendCoinsRecipient > recipients = transaction . getRecipients ( ) ;
2013-04-13 07:13:08 +02:00
std : : vector < std : : pair < CScript , int64_t > > vecSend ;
2011-07-16 19:01:05 +02:00
if ( recipients . empty ( ) )
2011-06-30 18:05:29 +02:00
{
2011-07-16 19:01:05 +02:00
return OK ;
2011-06-30 18:05:29 +02:00
}
2014-12-09 02:17:57 +01:00
if ( isAnonymizeOnlyUnlocked ( ) )
{
return AnonymizeOnlyUnlocked ;
}
2013-07-22 08:50:39 +02:00
QSet < QString > setAddress ; // Used to detect duplicates
int nAddresses = 0 ;
2011-07-16 19:01:05 +02:00
// Pre-check input data for validity
foreach ( const SendCoinsRecipient & rcp , recipients )
2011-06-30 18:05:29 +02:00
{
2013-07-22 08:50:39 +02:00
if ( rcp . paymentRequest . IsInitialized ( ) )
2013-09-28 19:29:44 +02:00
{ // PaymentRequest...
2013-04-13 07:13:08 +02:00
int64_t subtotal = 0 ;
2013-07-22 08:50:39 +02:00
const payments : : PaymentDetails & details = rcp . paymentRequest . getDetails ( ) ;
for ( int i = 0 ; i < details . outputs_size ( ) ; i + + )
{
const payments : : Output & out = details . outputs ( i ) ;
if ( out . amount ( ) < = 0 ) continue ;
subtotal + = out . amount ( ) ;
const unsigned char * scriptStr = ( const unsigned char * ) out . script ( ) . data ( ) ;
CScript scriptPubKey ( scriptStr , scriptStr + out . script ( ) . size ( ) ) ;
2013-04-13 07:13:08 +02:00
vecSend . push_back ( std : : pair < CScript , int64_t > ( scriptPubKey , out . amount ( ) ) ) ;
2013-07-22 08:50:39 +02:00
}
if ( subtotal < = 0 )
{
return InvalidAmount ;
}
total + = subtotal ;
2011-07-16 19:01:05 +02:00
}
2013-07-22 08:50:39 +02:00
else
2014-12-12 12:44:25 +01:00
{ // User-entered darkcoin address / amount:
2013-07-22 08:50:39 +02:00
if ( ! validateAddress ( rcp . address ) )
{
return InvalidAddress ;
}
if ( rcp . amount < = 0 )
{
return InvalidAmount ;
}
setAddress . insert ( rcp . address ) ;
+ + nAddresses ;
2011-07-16 19:01:05 +02:00
2013-07-22 08:50:39 +02:00
CScript scriptPubKey ;
scriptPubKey . SetDestination ( CBitcoinAddress ( rcp . address . toStdString ( ) ) . Get ( ) ) ;
2013-04-13 07:13:08 +02:00
vecSend . push_back ( std : : pair < CScript , int64_t > ( scriptPubKey , rcp . amount ) ) ;
2013-07-22 08:50:39 +02:00
total + = rcp . amount ;
2011-07-16 19:01:05 +02:00
}
2011-06-30 18:05:29 +02:00
}
2013-07-22 08:50:39 +02:00
if ( setAddress . size ( ) ! = nAddresses )
2011-07-16 19:01:05 +02:00
{
return DuplicateAddress ;
}
2013-08-12 17:03:03 +02:00
qint64 nBalance = getBalance ( coinControl ) ;
if ( total > nBalance )
2011-06-30 18:05:29 +02:00
{
return AmountExceedsBalance ;
}
2013-08-12 17:03:03 +02:00
if ( ( total + nTransactionFee ) > nBalance )
2011-06-30 18:05:29 +02:00
{
2013-08-30 20:04:48 +02:00
transaction . setTransactionFee ( nTransactionFee ) ;
return SendCoinsReturn ( AmountWithFeeExceedsBalance ) ;
2011-06-30 18:05:29 +02:00
}
{
2012-04-06 18:39:12 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-30 20:04:48 +02:00
transaction . newPossibleKeyChange ( wallet ) ;
2013-04-13 07:13:08 +02:00
int64_t nFeeRequired = 0 ;
2013-04-25 23:31:22 +02:00
std : : string strFailReason ;
2013-08-30 20:04:48 +02:00
CWalletTx * newTx = transaction . getTransaction ( ) ;
CReserveKey * keyChange = transaction . getPossibleKeyChange ( ) ;
2014-12-09 02:17:57 +01:00
2015-02-11 15:47:21 +01:00
if ( recipients [ 0 ] . useInstantX & & total > GetSporkValue ( SPORK_5_MAX_VALUE ) * COIN ) {
emit message ( tr ( " Send Coins " ) , tr ( " InstantX doesn't support sending values that high yet. Transactions are currently limited to %n DRK. " , " " , GetSporkValue ( SPORK_5_MAX_VALUE ) ) ,
CClientUIInterface : : MSG_ERROR ) ;
return TransactionCreationFailed ;
}
2015-02-08 15:00:23 +01:00
bool fCreated = wallet - > CreateTransaction ( vecSend , * newTx , * keyChange , nFeeRequired , strFailReason , coinControl , recipients [ 0 ] . inputType , recipients [ 0 ] . useInstantX ) ;
2013-08-30 20:04:48 +02:00
transaction . setTransactionFee ( nFeeRequired ) ;
2011-06-30 18:05:29 +02:00
2011-07-16 19:01:05 +02:00
if ( ! fCreated )
2011-06-30 18:05:29 +02:00
{
2013-08-12 17:03:03 +02:00
if ( ( total + nFeeRequired ) > nBalance )
2011-07-16 19:01:05 +02:00
{
2013-08-30 20:04:48 +02:00
return SendCoinsReturn ( AmountWithFeeExceedsBalance ) ;
2011-07-16 19:01:05 +02:00
}
2013-04-25 23:31:22 +02:00
emit message ( tr ( " Send Coins " ) , QString : : fromStdString ( strFailReason ) ,
CClientUIInterface : : MSG_ERROR ) ;
2011-07-16 19:01:05 +02:00
return TransactionCreationFailed ;
2011-06-30 18:05:29 +02:00
}
2013-08-30 20:04:48 +02:00
}
return SendCoinsReturn ( OK ) ;
}
WalletModel : : SendCoinsReturn WalletModel : : sendCoins ( WalletModelTransaction & transaction )
{
QByteArray transaction_array ; /* store serialized transaction */
2014-12-09 02:17:57 +01:00
if ( isAnonymizeOnlyUnlocked ( ) )
{
return AnonymizeOnlyUnlocked ;
}
2013-08-30 20:04:48 +02:00
{
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
CWalletTx * newTx = transaction . getTransaction ( ) ;
2015-02-16 00:42:08 +01:00
QList < SendCoinsRecipient > recipients = transaction . getRecipients ( ) ;
2013-08-30 20:04:48 +02:00
2013-07-22 08:50:39 +02:00
// Store PaymentRequests in wtx.vOrderForm in wallet.
2015-02-16 00:42:08 +01:00
foreach ( const SendCoinsRecipient & rcp , recipients )
2013-07-22 08:50:39 +02:00
{
if ( rcp . paymentRequest . IsInitialized ( ) )
{
std : : string key ( " PaymentRequest " ) ;
std : : string value ;
rcp . paymentRequest . SerializeToString ( & value ) ;
2013-08-30 20:04:48 +02:00
newTx - > vOrderForm . push_back ( make_pair ( key , value ) ) ;
2013-07-22 08:50:39 +02:00
}
2014-12-12 12:44:25 +01:00
else if ( ! rcp . message . isEmpty ( ) ) // Message from normal darkcoin:URI (darkcoin:XyZ...?message=example)
2014-12-09 02:17:57 +01:00
{
2014-01-21 23:39:29 +01:00
newTx - > vOrderForm . push_back ( make_pair ( " Message " , rcp . message . toStdString ( ) ) ) ;
2014-12-09 02:17:57 +01:00
}
2011-06-30 18:05:29 +02:00
}
2013-08-30 20:04:48 +02:00
CReserveKey * keyChange = transaction . getPossibleKeyChange ( ) ;
2014-12-09 02:17:57 +01:00
transaction . getRecipients ( ) ;
2015-02-16 00:42:08 +01:00
if ( ! wallet - > CommitTransaction ( * newTx , * keyChange , ( recipients [ 0 ] . useInstantX ) ? " txlreq " : " tx " ) )
2011-07-16 19:01:05 +02:00
return TransactionCommitFailed ;
2013-07-22 08:50:39 +02:00
2013-08-30 20:04:48 +02:00
CTransaction * t = ( CTransaction * ) newTx ;
2013-07-22 08:50:39 +02:00
CDataStream ssTx ( SER_NETWORK , PROTOCOL_VERSION ) ;
ssTx < < * t ;
2013-08-30 20:04:48 +02:00
transaction_array . append ( & ( ssTx [ 0 ] ) , ssTx . size ( ) ) ;
2011-06-30 18:05:29 +02:00
}
2013-07-22 08:50:39 +02:00
// Add addresses / update labels that we've sent to to the address book,
2013-08-30 20:04:48 +02:00
// and emit coinsSent signal for each recipient
foreach ( const SendCoinsRecipient & rcp , transaction . getRecipients ( ) )
2011-06-30 18:05:29 +02:00
{
2013-10-24 16:02:39 +02:00
// Don't touch the address book when we have a payment request
if ( ! rcp . paymentRequest . IsInitialized ( ) )
2011-07-16 19:01:05 +02:00
{
2013-10-08 14:23:57 +02:00
std : : string strAddress = rcp . address . toStdString ( ) ;
CTxDestination dest = CBitcoinAddress ( strAddress ) . Get ( ) ;
std : : string strLabel = rcp . label . toStdString ( ) ;
2012-05-03 14:52:15 +02:00
{
2013-10-08 14:23:57 +02:00
LOCK ( wallet - > cs_wallet ) ;
std : : map < CTxDestination , CAddressBookData > : : iterator mi = wallet - > mapAddressBook . find ( dest ) ;
// Check if we have a new address or an updated label
if ( mi = = wallet - > mapAddressBook . end ( ) )
{
wallet - > SetAddressBook ( dest , strLabel , " send " ) ;
}
else if ( mi - > second . name ! = strLabel )
{
wallet - > SetAddressBook ( dest , strLabel , " " ) ; // "" means don't change purpose
}
2012-05-03 14:52:15 +02:00
}
2011-07-16 19:01:05 +02:00
}
2013-08-30 20:04:48 +02:00
emit coinsSent ( wallet , rcp , transaction_array ) ;
2011-06-30 18:05:29 +02:00
}
2011-07-30 19:28:41 +02:00
2013-08-30 20:04:48 +02:00
return SendCoinsReturn ( OK ) ;
2011-06-30 18:05:29 +02:00
}
OptionsModel * WalletModel : : getOptionsModel ( )
{
return optionsModel ;
}
AddressTableModel * WalletModel : : getAddressTableModel ( )
{
return addressTableModel ;
}
TransactionTableModel * WalletModel : : getTransactionTableModel ( )
{
return transactionTableModel ;
}
2011-07-02 13:45:59 +02:00
2013-11-05 18:03:05 +01:00
RecentRequestsTableModel * WalletModel : : getRecentRequestsTableModel ( )
{
return recentRequestsTableModel ;
}
2011-08-23 20:08:42 +02:00
WalletModel : : EncryptionStatus WalletModel : : getEncryptionStatus ( ) const
{
if ( ! wallet - > IsCrypted ( ) )
{
return Unencrypted ;
}
else if ( wallet - > IsLocked ( ) )
{
return Locked ;
}
2014-12-09 02:17:57 +01:00
else if ( wallet - > fWalletUnlockAnonymizeOnly )
{
return UnlockedForAnonymizationOnly ;
}
2011-08-23 20:08:42 +02:00
else
{
return Unlocked ;
}
}
2011-08-24 22:07:26 +02:00
2011-11-26 07:02:04 +01:00
bool WalletModel : : setWalletEncrypted ( bool encrypted , const SecureString & passphrase )
2011-08-24 22:07:26 +02:00
{
if ( encrypted )
{
// Encrypt
return wallet - > EncryptWallet ( passphrase ) ;
}
else
{
// Decrypt -- TODO; not supported yet
return false ;
}
}
2014-12-09 02:17:57 +01:00
bool WalletModel : : setWalletLocked ( bool locked , const SecureString & passPhrase , bool anonymizeOnly )
2011-08-24 22:07:26 +02:00
{
if ( locked )
{
// Lock
return wallet - > Lock ( ) ;
}
else
{
// Unlock
2014-12-09 02:17:57 +01:00
return wallet - > Unlock ( passPhrase , anonymizeOnly ) ;
2011-08-24 22:07:26 +02:00
}
}
2014-12-09 02:17:57 +01:00
bool WalletModel : : isAnonymizeOnlyUnlocked ( )
{
return wallet - > fWalletUnlockAnonymizeOnly ;
}
2011-11-26 07:02:04 +01:00
bool WalletModel : : changePassphrase ( const SecureString & oldPass , const SecureString & newPass )
2011-08-24 22:07:26 +02:00
{
bool retval ;
{
2012-04-06 18:39:12 +02:00
LOCK ( wallet - > cs_wallet ) ;
2011-08-24 22:07:26 +02:00
wallet - > Lock ( ) ; // Make sure wallet is locked before attempting pass change
retval = wallet - > ChangeWalletPassphrase ( oldPass , newPass ) ;
}
return retval ;
}
2012-02-14 13:14:43 +01:00
bool WalletModel : : backupWallet ( const QString & filename )
{
return BackupWallet ( * wallet , filename . toLocal8Bit ( ) . data ( ) ) ;
}
2012-05-06 19:40:58 +02:00
// Handlers for core signals
static void NotifyKeyStoreStatusChanged ( WalletModel * walletmodel , CCryptoKeyStore * wallet )
{
2013-09-04 11:52:45 +02:00
qDebug ( ) < < " NotifyKeyStoreStatusChanged " ;
2012-05-06 19:40:58 +02:00
QMetaObject : : invokeMethod ( walletmodel , " updateStatus " , Qt : : QueuedConnection ) ;
}
2013-08-29 16:19:43 +02:00
static void NotifyAddressBookChanged ( WalletModel * walletmodel , CWallet * wallet ,
const CTxDestination & address , const std : : string & label , bool isMine ,
const std : : string & purpose , ChangeType status )
2012-05-06 19:40:58 +02:00
{
2013-09-04 11:52:45 +02:00
QString strAddress = QString : : fromStdString ( CBitcoinAddress ( address ) . ToString ( ) ) ;
QString strLabel = QString : : fromStdString ( label ) ;
QString strPurpose = QString : : fromStdString ( purpose ) ;
qDebug ( ) < < " NotifyAddressBookChanged : " + strAddress + " " + strLabel + " isMine= " + QString : : number ( isMine ) + " purpose= " + strPurpose + " status= " + QString : : number ( status ) ;
2012-05-06 19:40:58 +02:00
QMetaObject : : invokeMethod ( walletmodel , " updateAddressBook " , Qt : : QueuedConnection ,
2013-09-04 11:52:45 +02:00
Q_ARG ( QString , strAddress ) ,
Q_ARG ( QString , strLabel ) ,
2012-05-06 22:41:35 +02:00
Q_ARG ( bool , isMine ) ,
2013-09-04 11:52:45 +02:00
Q_ARG ( QString , strPurpose ) ,
2012-05-06 19:40:58 +02:00
Q_ARG ( int , status ) ) ;
}
static void NotifyTransactionChanged ( WalletModel * walletmodel , CWallet * wallet , const uint256 & hash , ChangeType status )
{
2015-02-10 14:24:05 +01:00
Q_UNUSED ( wallet ) ;
Q_UNUSED ( hash ) ;
Q_UNUSED ( status ) ;
QMetaObject : : invokeMethod ( walletmodel , " updateTransaction " , Qt : : QueuedConnection ) ;
2012-05-06 19:40:58 +02:00
}
2014-03-19 00:26:14 +01:00
static void ShowProgress ( WalletModel * walletmodel , const std : : string & title , int nProgress )
{
// emits signal "showProgress"
QMetaObject : : invokeMethod ( walletmodel , " showProgress " , Qt : : QueuedConnection ,
Q_ARG ( QString , QString : : fromStdString ( title ) ) ,
Q_ARG ( int , nProgress ) ) ;
}
2012-05-06 19:40:58 +02:00
void WalletModel : : subscribeToCoreSignals ( )
{
// Connect signals to wallet
wallet - > NotifyStatusChanged . connect ( boost : : bind ( & NotifyKeyStoreStatusChanged , this , _1 ) ) ;
2013-08-29 16:19:43 +02:00
wallet - > NotifyAddressBookChanged . connect ( boost : : bind ( NotifyAddressBookChanged , this , _1 , _2 , _3 , _4 , _5 , _6 ) ) ;
2012-05-06 19:40:58 +02:00
wallet - > NotifyTransactionChanged . connect ( boost : : bind ( NotifyTransactionChanged , this , _1 , _2 , _3 ) ) ;
2014-03-19 00:26:14 +01:00
wallet - > ShowProgress . connect ( boost : : bind ( ShowProgress , this , _1 , _2 ) ) ;
2012-05-06 19:40:58 +02:00
}
void WalletModel : : unsubscribeFromCoreSignals ( )
{
// Disconnect signals from wallet
wallet - > NotifyStatusChanged . disconnect ( boost : : bind ( & NotifyKeyStoreStatusChanged , this , _1 ) ) ;
2013-08-29 16:19:43 +02:00
wallet - > NotifyAddressBookChanged . disconnect ( boost : : bind ( NotifyAddressBookChanged , this , _1 , _2 , _3 , _4 , _5 , _6 ) ) ;
2012-05-06 19:40:58 +02:00
wallet - > NotifyTransactionChanged . disconnect ( boost : : bind ( NotifyTransactionChanged , this , _1 , _2 , _3 ) ) ;
2014-03-19 00:26:14 +01:00
wallet - > ShowProgress . disconnect ( boost : : bind ( ShowProgress , this , _1 , _2 ) ) ;
2012-05-06 19:40:58 +02:00
}
2011-08-24 22:07:26 +02:00
// WalletModel::UnlockContext implementation
2014-12-09 02:17:57 +01:00
WalletModel : : UnlockContext WalletModel : : requestUnlock ( bool relock )
2011-08-24 22:07:26 +02:00
{
bool was_locked = getEncryptionStatus ( ) = = Locked ;
2014-12-09 02:17:57 +01:00
if ( ! was_locked & & isAnonymizeOnlyUnlocked ( ) )
{
setWalletLocked ( true ) ;
was_locked = getEncryptionStatus ( ) = = Locked ;
}
2011-08-24 22:07:26 +02:00
if ( was_locked )
{
// Request UI to unlock wallet
emit requireUnlock ( ) ;
}
// If wallet is still locked, unlock was failed or cancelled, mark context as invalid
bool valid = getEncryptionStatus ( ) ! = Locked ;
2014-12-09 02:17:57 +01:00
return UnlockContext ( this , valid , relock ) ;
// return UnlockContext(this, valid, was_locked && !isAnonymizeOnlyUnlocked());
2011-08-24 22:07:26 +02:00
}
WalletModel : : UnlockContext : : UnlockContext ( WalletModel * wallet , bool valid , bool relock ) :
wallet ( wallet ) ,
valid ( valid ) ,
relock ( relock )
{
}
WalletModel : : UnlockContext : : ~ UnlockContext ( )
{
if ( valid & & relock )
{
wallet - > setWalletLocked ( true ) ;
}
}
void WalletModel : : UnlockContext : : CopyFrom ( const UnlockContext & rhs )
{
// Transfer context; old object no longer relocks wallet
* this = rhs ;
rhs . relock = false ;
}
2013-08-12 17:03:03 +02:00
bool WalletModel : : getPubKey ( const CKeyID & address , CPubKey & vchPubKeyOut ) const
{
return wallet - > GetPubKey ( address , vchPubKeyOut ) ;
}
// returns a list of COutputs from COutPoints
void WalletModel : : getOutputs ( const std : : vector < COutPoint > & vOutpoints , std : : vector < COutput > & vOutputs )
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-12 17:03:03 +02:00
BOOST_FOREACH ( const COutPoint & outpoint , vOutpoints )
{
if ( ! wallet - > mapWallet . count ( outpoint . hash ) ) continue ;
2014-02-12 19:43:07 +01:00
int nDepth = wallet - > mapWallet [ outpoint . hash ] . GetDepthInMainChain ( ) ;
if ( nDepth < 0 ) continue ;
COutput out ( & wallet - > mapWallet [ outpoint . hash ] , outpoint . n , nDepth ) ;
2013-08-12 17:03:03 +02:00
vOutputs . push_back ( out ) ;
}
}
2014-02-15 22:38:28 +01:00
bool WalletModel : : isSpent ( const COutPoint & outpoint ) const
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2014-02-15 22:38:28 +01:00
return wallet - > IsSpent ( outpoint . hash , outpoint . n ) ;
}
2013-08-12 17:03:03 +02:00
// AvailableCoins + LockedCoins grouped by wallet address (put change in one group with wallet address)
void WalletModel : : listCoins ( std : : map < QString , std : : vector < COutput > > & mapCoins ) const
{
std : : vector < COutput > vCoins ;
wallet - > AvailableCoins ( vCoins ) ;
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ; // ListLockedCoins, mapWallet
2013-08-12 17:03:03 +02:00
std : : vector < COutPoint > vLockedCoins ;
wallet - > ListLockedCoins ( vLockedCoins ) ;
// add locked coins
BOOST_FOREACH ( const COutPoint & outpoint , vLockedCoins )
{
if ( ! wallet - > mapWallet . count ( outpoint . hash ) ) continue ;
2014-02-12 19:43:07 +01:00
int nDepth = wallet - > mapWallet [ outpoint . hash ] . GetDepthInMainChain ( ) ;
if ( nDepth < 0 ) continue ;
COutput out ( & wallet - > mapWallet [ outpoint . hash ] , outpoint . n , nDepth ) ;
2013-08-12 17:03:03 +02:00
vCoins . push_back ( out ) ;
}
BOOST_FOREACH ( const COutput & out , vCoins )
{
COutput cout = out ;
while ( wallet - > IsChange ( cout . tx - > vout [ cout . i ] ) & & cout . tx - > vin . size ( ) > 0 & & wallet - > IsMine ( cout . tx - > vin [ 0 ] ) )
{
if ( ! wallet - > mapWallet . count ( cout . tx - > vin [ 0 ] . prevout . hash ) ) break ;
cout = COutput ( & wallet - > mapWallet [ cout . tx - > vin [ 0 ] . prevout . hash ] , cout . tx - > vin [ 0 ] . prevout . n , 0 ) ;
}
CTxDestination address ;
if ( ! ExtractDestination ( cout . tx - > vout [ cout . i ] . scriptPubKey , address ) ) continue ;
mapCoins [ CBitcoinAddress ( address ) . ToString ( ) . c_str ( ) ] . push_back ( out ) ;
}
}
bool WalletModel : : isLockedCoin ( uint256 hash , unsigned int n ) const
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-12 17:03:03 +02:00
return wallet - > IsLockedCoin ( hash , n ) ;
}
void WalletModel : : lockCoin ( COutPoint & output )
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-12 17:03:03 +02:00
wallet - > LockCoin ( output ) ;
}
void WalletModel : : unlockCoin ( COutPoint & output )
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-12 17:03:03 +02:00
wallet - > UnlockCoin ( output ) ;
}
void WalletModel : : listLockedCoins ( std : : vector < COutPoint > & vOutpts )
{
2014-04-15 17:38:25 +02:00
LOCK2 ( cs_main , wallet - > cs_wallet ) ;
2013-08-12 17:03:03 +02:00
wallet - > ListLockedCoins ( vOutpts ) ;
}
2014-01-14 05:05:43 +01:00
void WalletModel : : loadReceiveRequests ( std : : vector < std : : string > & vReceiveRequests )
{
LOCK ( wallet - > cs_wallet ) ;
BOOST_FOREACH ( const PAIRTYPE ( CTxDestination , CAddressBookData ) & item , wallet - > mapAddressBook )
BOOST_FOREACH ( const PAIRTYPE ( std : : string , std : : string ) & item2 , item . second . destdata )
if ( item2 . first . size ( ) > 2 & & item2 . first . substr ( 0 , 2 ) = = " rr " ) // receive request
vReceiveRequests . push_back ( item2 . second ) ;
}
bool WalletModel : : saveReceiveRequest ( const std : : string & sAddress , const int64_t nId , const std : : string & sRequest )
{
CTxDestination dest = CBitcoinAddress ( sAddress ) . Get ( ) ;
std : : stringstream ss ;
ss < < nId ;
std : : string key = " rr " + ss . str ( ) ; // "rr" prefix = "receive request" in destdata
LOCK ( wallet - > cs_wallet ) ;
if ( sRequest . empty ( ) )
return wallet - > EraseDestData ( dest , key ) ;
else
return wallet - > AddDestData ( dest , key , sRequest ) ;
}