2018-01-03 02:12:05 +09:00
// Copyright (c) 2011-2017 The Bitcoin Core developers
2014-12-13 12:09:33 +08:00
// Distributed under the MIT software license, see the accompanying
2013-11-04 16:20:43 +01:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-11-10 13:57:53 +13:00
# include <qt/sendcoinsdialog.h>
2017-08-15 17:31:26 +02:00
# include <qt/forms/ui_sendcoinsdialog.h>
2013-01-23 21:51:02 +01:00
2017-11-10 13:57:53 +13:00
# include <qt/addresstablemodel.h>
# include <qt/bitcoinunits.h>
# include <qt/clientmodel.h>
# include <qt/coincontroldialog.h>
# include <qt/guiutil.h>
# include <qt/optionsmodel.h>
# include <qt/platformstyle.h>
# include <qt/sendcoinsentry.h>
2013-04-13 00:13:08 -05:00
2017-11-10 13:57:53 +13:00
# include <chainparams.h>
2018-04-07 03:42:02 -04:00
# include <interfaces/node.h>
2017-09-19 18:12:25 -07:00
# include <key_io.h>
2017-11-10 13:57:53 +13:00
# include <wallet/coincontrol.h>
# include <ui_interface.h>
# include <txmempool.h>
# include <policy/fees.h>
# include <wallet/fees.h>
2011-05-12 17:55:24 +02:00
2017-02-17 08:41:57 +00:00
# include <QFontMetrics>
2011-12-07 00:00:04 -05:00
# include <QScrollBar>
2014-11-02 00:14:47 +01:00
# include <QSettings>
2013-04-13 00:13:08 -05:00
# include <QTextDocument>
2016-05-09 22:26:57 -04:00
2017-07-07 22:03:49 +02:00
static const std : : array < int , 9 > confTargets = { { 2 , 4 , 6 , 12 , 24 , 48 , 144 , 504 , 1008 } } ;
int getConfTargetForIndex ( int index ) {
if ( index + 1 > static_cast < int > ( confTargets . size ( ) ) ) {
return confTargets . back ( ) ;
}
if ( index < 0 ) {
return confTargets [ 0 ] ;
}
return confTargets [ index ] ;
}
int getIndexForConfTarget ( int target ) {
for ( unsigned int i = 0 ; i < confTargets . size ( ) ; i + + ) {
if ( confTargets [ i ] > = target ) {
return i ;
}
}
return confTargets . size ( ) - 1 ;
}
2016-09-09 13:43:29 +02:00
SendCoinsDialog : : SendCoinsDialog ( const PlatformStyle * _platformStyle , QWidget * parent ) :
2011-05-12 14:44:52 +02:00
QDialog ( parent ) ,
2011-05-30 20:20:12 +02:00
ui ( new Ui : : SendCoinsDialog ) ,
2014-11-20 13:06:10 +01:00
clientModel ( 0 ) ,
model ( 0 ) ,
fNewRecipientAllowed ( true ) ,
2015-07-28 15:20:14 +02:00
fFeeMinimized ( true ) ,
2016-09-09 13:43:29 +02:00
platformStyle ( _platformStyle )
2011-05-12 14:44:52 +02:00
{
ui - > setupUi ( this ) ;
2011-07-08 19:25:35 +02:00
2016-09-09 13:43:29 +02:00
if ( ! _platformStyle - > getImagesOnButtons ( ) ) {
2015-07-28 15:20:14 +02:00
ui - > addButton - > setIcon ( QIcon ( ) ) ;
ui - > clearButton - > setIcon ( QIcon ( ) ) ;
ui - > sendButton - > setIcon ( QIcon ( ) ) ;
} else {
2016-09-09 13:43:29 +02:00
ui - > addButton - > setIcon ( _platformStyle - > SingleColorIcon ( " :/icons/add " ) ) ;
ui - > clearButton - > setIcon ( _platformStyle - > SingleColorIcon ( " :/icons/remove " ) ) ;
ui - > sendButton - > setIcon ( _platformStyle - > SingleColorIcon ( " :/icons/send " ) ) ;
2015-07-28 15:20:14 +02:00
}
2013-11-20 15:56:51 +01:00
GUIUtil : : setupAddressWidget ( ui - > lineEditCoinControlChange , this ) ;
2011-10-07 13:21:45 +02:00
2011-07-16 19:01:05 +02:00
addEntry ( ) ;
2011-05-15 19:31:20 +02:00
2011-07-16 19:01:05 +02:00
connect ( ui - > addButton , SIGNAL ( clicked ( ) ) , this , SLOT ( addEntry ( ) ) ) ;
2011-09-27 17:46:19 +03:00
connect ( ui - > clearButton , SIGNAL ( clicked ( ) ) , this , SLOT ( clear ( ) ) ) ;
2011-12-23 20:27:12 -08:00
2013-08-12 17:03:03 +02:00
// Coin Control
connect ( ui - > pushButtonCoinControl , SIGNAL ( clicked ( ) ) , this , SLOT ( coinControlButtonClicked ( ) ) ) ;
connect ( ui - > checkBoxCoinControlChange , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( coinControlChangeChecked ( int ) ) ) ;
connect ( ui - > lineEditCoinControlChange , SIGNAL ( textEdited ( const QString & ) ) , this , SLOT ( coinControlChangeEdited ( const QString & ) ) ) ;
// Coin Control: clipboard actions
QAction * clipboardQuantityAction = new QAction ( tr ( " Copy quantity " ) , this ) ;
QAction * clipboardAmountAction = new QAction ( tr ( " Copy amount " ) , this ) ;
QAction * clipboardFeeAction = new QAction ( tr ( " Copy fee " ) , this ) ;
QAction * clipboardAfterFeeAction = new QAction ( tr ( " Copy after fee " ) , this ) ;
QAction * clipboardBytesAction = new QAction ( tr ( " Copy bytes " ) , this ) ;
2014-06-08 01:05:53 +02:00
QAction * clipboardLowOutputAction = new QAction ( tr ( " Copy dust " ) , this ) ;
2013-08-12 17:03:03 +02:00
QAction * clipboardChangeAction = new QAction ( tr ( " Copy change " ) , this ) ;
connect ( clipboardQuantityAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardQuantity ( ) ) ) ;
connect ( clipboardAmountAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardAmount ( ) ) ) ;
connect ( clipboardFeeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardFee ( ) ) ) ;
connect ( clipboardAfterFeeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardAfterFee ( ) ) ) ;
connect ( clipboardBytesAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardBytes ( ) ) ) ;
connect ( clipboardLowOutputAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardLowOutput ( ) ) ) ;
connect ( clipboardChangeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( coinControlClipboardChange ( ) ) ) ;
ui - > labelCoinControlQuantity - > addAction ( clipboardQuantityAction ) ;
ui - > labelCoinControlAmount - > addAction ( clipboardAmountAction ) ;
ui - > labelCoinControlFee - > addAction ( clipboardFeeAction ) ;
ui - > labelCoinControlAfterFee - > addAction ( clipboardAfterFeeAction ) ;
ui - > labelCoinControlBytes - > addAction ( clipboardBytesAction ) ;
ui - > labelCoinControlLowOutput - > addAction ( clipboardLowOutputAction ) ;
ui - > labelCoinControlChange - > addAction ( clipboardChangeAction ) ;
2014-11-02 00:14:47 +01:00
// init transaction fee section
QSettings settings ;
if ( ! settings . contains ( " fFeeSectionMinimized " ) )
settings . setValue ( " fFeeSectionMinimized " , true ) ;
if ( ! settings . contains ( " nFeeRadio " ) & & settings . contains ( " nTransactionFee " ) & & settings . value ( " nTransactionFee " ) . toLongLong ( ) > 0 ) // compatibility
settings . setValue ( " nFeeRadio " , 1 ) ; // custom
if ( ! settings . contains ( " nFeeRadio " ) )
settings . setValue ( " nFeeRadio " , 0 ) ; // recommended
if ( ! settings . contains ( " nSmartFeeSliderPosition " ) )
settings . setValue ( " nSmartFeeSliderPosition " , 0 ) ;
if ( ! settings . contains ( " nTransactionFee " ) )
2018-04-07 12:12:46 -04:00
settings . setValue ( " nTransactionFee " , ( qint64 ) DEFAULT_PAY_TX_FEE ) ;
2014-11-02 00:14:47 +01:00
if ( ! settings . contains ( " fPayOnlyMinFee " ) )
settings . setValue ( " fPayOnlyMinFee " , false ) ;
ui - > groupFee - > setId ( ui - > radioSmartFee , 0 ) ;
ui - > groupFee - > setId ( ui - > radioCustomFee , 1 ) ;
ui - > groupFee - > button ( ( int ) std : : max ( 0 , std : : min ( 1 , settings . value ( " nFeeRadio " ) . toInt ( ) ) ) ) - > setChecked ( true ) ;
ui - > customFee - > setValue ( settings . value ( " nTransactionFee " ) . toLongLong ( ) ) ;
ui - > checkBoxMinimumFee - > setChecked ( settings . value ( " fPayOnlyMinFee " ) . toBool ( ) ) ;
minimizeFeeSection ( settings . value ( " fFeeSectionMinimized " ) . toBool ( ) ) ;
2011-05-12 14:44:52 +02:00
}
2016-09-09 13:43:29 +02:00
void SendCoinsDialog : : setClientModel ( ClientModel * _clientModel )
2014-11-02 00:14:47 +01:00
{
2016-09-09 13:43:29 +02:00
this - > clientModel = _clientModel ;
2014-11-20 13:06:10 +01:00
2016-09-09 13:43:29 +02:00
if ( _clientModel ) {
connect ( _clientModel , SIGNAL ( numBlocksChanged ( int , QDateTime , double , bool ) ) , this , SLOT ( updateSmartFeeLabel ( ) ) ) ;
2014-11-20 13:06:10 +01:00
}
2014-11-02 00:14:47 +01:00
}
2016-09-09 13:43:29 +02:00
void SendCoinsDialog : : setModel ( WalletModel * _model )
2011-05-30 20:20:12 +02:00
{
2016-09-09 13:43:29 +02:00
this - > model = _model ;
2011-07-16 19:01:05 +02:00
2016-09-09 13:43:29 +02:00
if ( _model & & _model - > getOptionsModel ( ) )
2011-07-16 19:01:05 +02:00
{
2013-08-23 13:07:20 +02:00
for ( int i = 0 ; i < ui - > entries - > count ( ) ; + + i )
2011-07-16 19:01:05 +02:00
{
2013-08-23 13:07:20 +02:00
SendCoinsEntry * entry = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( i ) - > widget ( ) ) ;
if ( entry )
{
2016-09-09 13:43:29 +02:00
entry - > setModel ( _model ) ;
2013-08-23 13:07:20 +02:00
}
2011-07-16 19:01:05 +02:00
}
2013-08-23 13:07:20 +02:00
2018-04-07 03:42:02 -04:00
interfaces : : WalletBalances balances = _model - > wallet ( ) . getBalances ( ) ;
2018-03-31 06:41:33 -04:00
setBalance ( balances ) ;
2018-04-07 03:42:02 -04:00
connect ( _model , SIGNAL ( balanceChanged ( interfaces : : WalletBalances ) ) , this , SLOT ( setBalance ( interfaces : : WalletBalances ) ) ) ;
2016-09-09 13:43:29 +02:00
connect ( _model - > getOptionsModel ( ) , SIGNAL ( displayUnitChanged ( int ) ) , this , SLOT ( updateDisplayUnit ( ) ) ) ;
2014-11-02 00:14:47 +01:00
updateDisplayUnit ( ) ;
2013-08-12 17:03:03 +02:00
// Coin Control
2016-09-09 13:43:29 +02:00
connect ( _model - > getOptionsModel ( ) , SIGNAL ( displayUnitChanged ( int ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
connect ( _model - > getOptionsModel ( ) , SIGNAL ( coinControlFeaturesChanged ( bool ) ) , this , SLOT ( coinControlFeatureChanged ( bool ) ) ) ;
ui - > frameCoinControl - > setVisible ( _model - > getOptionsModel ( ) - > getCoinControlFeatures ( ) ) ;
2013-08-12 17:03:03 +02:00
coinControlUpdateLabels ( ) ;
2014-11-02 00:14:47 +01:00
// fee section
2017-07-16 02:07:52 +02:00
for ( const int n : confTargets ) {
2017-07-07 22:03:49 +02:00
ui - > confTargetSelector - > addItem ( tr ( " %1 (%2 blocks) " ) . arg ( GUIUtil : : formatNiceTimeOffset ( n * Params ( ) . GetConsensus ( ) . nPowTargetSpacing ) ) . arg ( n ) ) ;
}
connect ( ui - > confTargetSelector , SIGNAL ( currentIndexChanged ( int ) ) , this , SLOT ( updateSmartFeeLabel ( ) ) ) ;
connect ( ui - > confTargetSelector , SIGNAL ( currentIndexChanged ( int ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
2014-11-02 00:14:47 +01:00
connect ( ui - > groupFee , SIGNAL ( buttonClicked ( int ) ) , this , SLOT ( updateFeeSectionControls ( ) ) ) ;
connect ( ui - > groupFee , SIGNAL ( buttonClicked ( int ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
connect ( ui - > customFee , SIGNAL ( valueChanged ( ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
connect ( ui - > checkBoxMinimumFee , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( setMinimumFee ( ) ) ) ;
connect ( ui - > checkBoxMinimumFee , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( updateFeeSectionControls ( ) ) ) ;
connect ( ui - > checkBoxMinimumFee , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
2017-06-13 14:02:55 -04:00
connect ( ui - > optInRBF , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( updateSmartFeeLabel ( ) ) ) ;
connect ( ui - > optInRBF , SIGNAL ( stateChanged ( int ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
2018-04-07 12:12:46 -04:00
ui - > customFee - > setSingleStep ( model - > wallet ( ) . getRequiredFee ( 1000 ) ) ;
2014-11-02 00:14:47 +01:00
updateFeeSectionControls ( ) ;
updateMinFeeLabel ( ) ;
updateSmartFeeLabel ( ) ;
2016-10-21 10:53:37 +02:00
2017-04-17 19:03:45 -04:00
// set default rbf checkbox state
2017-12-22 09:18:05 +01:00
ui - > optInRBF - > setCheckState ( Qt : : Checked ) ;
2017-04-17 19:03:45 -04:00
2016-10-21 10:53:37 +02:00
// set the smartfee-sliders default value (wallets default conf.target or last stored value)
QSettings settings ;
2017-07-13 12:19:17 +02:00
if ( settings . value ( " nSmartFeeSliderPosition " ) . toInt ( ) ! = 0 ) {
// migrate nSmartFeeSliderPosition to nConfTarget
// nConfTarget is available since 0.15 (replaced nSmartFeeSliderPosition)
int nConfirmTarget = 25 - settings . value ( " nSmartFeeSliderPosition " ) . toInt ( ) ; // 25 == old slider range
settings . setValue ( " nConfTarget " , nConfirmTarget ) ;
settings . remove ( " nSmartFeeSliderPosition " ) ;
}
2017-07-07 22:03:49 +02:00
if ( settings . value ( " nConfTarget " ) . toInt ( ) = = 0 )
2018-04-07 12:12:46 -04:00
ui - > confTargetSelector - > setCurrentIndex ( getIndexForConfTarget ( model - > wallet ( ) . getConfirmTarget ( ) ) ) ;
2016-10-21 10:53:37 +02:00
else
2017-07-07 22:03:49 +02:00
ui - > confTargetSelector - > setCurrentIndex ( getIndexForConfTarget ( settings . value ( " nConfTarget " ) . toInt ( ) ) ) ;
2011-11-08 21:18:36 +01:00
}
2011-05-30 20:20:12 +02:00
}
2011-05-12 14:44:52 +02:00
SendCoinsDialog : : ~ SendCoinsDialog ( )
{
2014-11-02 00:14:47 +01:00
QSettings settings ;
settings . setValue ( " fFeeSectionMinimized " , fFeeMinimized ) ;
settings . setValue ( " nFeeRadio " , ui - > groupFee - > checkedId ( ) ) ;
2017-07-07 22:03:49 +02:00
settings . setValue ( " nConfTarget " , getConfTargetForIndex ( ui - > confTargetSelector - > currentIndex ( ) ) ) ;
2014-11-02 00:14:47 +01:00
settings . setValue ( " nTransactionFee " , ( qint64 ) ui - > customFee - > value ( ) ) ;
settings . setValue ( " fPayOnlyMinFee " , ui - > checkBoxMinimumFee - > isChecked ( ) ) ;
2011-05-12 14:44:52 +02:00
delete ui ;
}
2011-05-12 17:55:24 +02:00
void SendCoinsDialog : : on_sendButton_clicked ( )
{
2013-08-24 15:07:17 +02:00
if ( ! model | | ! model - > getOptionsModel ( ) )
return ;
2011-07-16 19:01:05 +02:00
QList < SendCoinsRecipient > recipients ;
bool valid = true ;
2011-11-08 21:18:36 +01:00
2011-07-16 19:01:05 +02:00
for ( int i = 0 ; i < ui - > entries - > count ( ) ; + + i )
{
SendCoinsEntry * entry = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( i ) - > widget ( ) ) ;
if ( entry )
{
2017-04-18 17:06:13 -04:00
if ( entry - > validate ( model - > node ( ) ) )
2011-07-16 19:01:05 +02:00
{
recipients . append ( entry - > getValue ( ) ) ;
}
else
{
valid = false ;
}
}
}
2011-05-30 20:20:12 +02:00
2011-07-16 19:01:05 +02:00
if ( ! valid | | recipients . isEmpty ( ) )
2011-05-14 17:25:05 +02:00
{
2011-05-15 19:31:20 +02:00
return ;
2011-05-14 17:25:05 +02:00
}
2011-05-27 21:43:05 +02:00
2014-06-10 03:14:13 +02:00
fNewRecipientAllowed = false ;
WalletModel : : UnlockContext ctx ( model - > requestUnlock ( ) ) ;
if ( ! ctx . isValid ( ) )
{
// Unlock wallet was cancelled
fNewRecipientAllowed = true ;
return ;
}
// prepare transaction for getting txFee earlier
WalletModelTransaction currentTransaction ( recipients ) ;
WalletModel : : SendCoinsReturn prepareStatus ;
2016-10-21 10:53:37 +02:00
// Always use a CCoinControl instance, use the CoinControlDialog instance if CoinControl has been enabled
CCoinControl ctrl ;
if ( model - > getOptionsModel ( ) - > getCoinControlFeatures ( ) )
2018-02-01 16:48:56 +00:00
ctrl = * CoinControlDialog : : coinControl ( ) ;
2017-06-28 19:24:28 -04:00
updateCoinControlState ( ctrl ) ;
2016-04-05 16:55:41 +02:00
2017-06-28 16:41:55 -04:00
prepareStatus = model - > prepareTransaction ( currentTransaction , ctrl ) ;
2014-06-10 03:14:13 +02:00
// process prepareStatus and on error generate message shown to user
processSendCoinsReturn ( prepareStatus ,
BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , currentTransaction . getTransactionFee ( ) ) ) ;
if ( prepareStatus . status ! = WalletModel : : OK ) {
fNewRecipientAllowed = true ;
return ;
}
CAmount txFee = currentTransaction . getTransactionFee ( ) ;
2011-07-16 19:01:05 +02:00
// Format confirmation message
QStringList formatted ;
2017-06-02 03:25:02 +02:00
for ( const SendCoinsRecipient & rcp : currentTransaction . getRecipients ( ) )
2011-07-16 19:01:05 +02:00
{
2018-03-06 12:56:21 +08:00
// generate bold amount string with wallet name in case of multiwallet
2014-05-09 23:50:09 +01:00
QString amount = " <b> " + BitcoinUnits : : formatHtmlWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , rcp . amount ) ;
2018-03-06 12:56:21 +08:00
if ( model - > isMultiwallet ( ) ) {
amount . append ( " <u> " + tr ( " from wallet %1 " ) . arg ( GUIUtil : : HtmlEscape ( model - > getWalletName ( ) ) ) + " </u> " ) ;
}
2013-09-13 16:49:35 +02:00
amount . append ( " </b> " ) ;
// generate monospace address string
QString address = " <span style='font-family: monospace;'> " + rcp . address ;
address . append ( " </span> " ) ;
QString recipientElement ;
2018-05-03 14:38:39 +01:00
recipientElement = " <br /> " ;
2013-09-13 16:49:35 +02:00
2013-10-24 16:02:39 +02:00
if ( ! rcp . paymentRequest . IsInitialized ( ) ) // normal payment
2013-07-22 16:50:39 +10:00
{
2013-09-13 16:49:35 +02:00
if ( rcp . label . length ( ) > 0 ) // label with address
2013-08-30 20:04:48 +02:00
{
2018-05-03 14:38:39 +01:00
recipientElement . append ( tr ( " %1 to %2 " ) . arg ( amount , GUIUtil : : HtmlEscape ( rcp . label ) ) ) ;
2013-09-13 16:49:35 +02:00
recipientElement . append ( QString ( " (%1) " ) . arg ( address ) ) ;
2013-08-30 20:04:48 +02:00
}
2013-09-13 16:49:35 +02:00
else // just address
2013-08-30 20:04:48 +02:00
{
2018-05-03 14:38:39 +01:00
recipientElement . append ( tr ( " %1 to %2 " ) . arg ( amount , address ) ) ;
2013-08-30 20:04:48 +02:00
}
2013-07-22 16:50:39 +10:00
}
2015-03-18 11:22:27 +01:00
else if ( ! rcp . authenticatedMerchant . isEmpty ( ) ) // authenticated payment request
2013-07-22 16:50:39 +10:00
{
2018-05-03 14:38:39 +01:00
recipientElement . append ( tr ( " %1 to %2 " ) . arg ( amount , GUIUtil : : HtmlEscape ( rcp . authenticatedMerchant ) ) ) ;
2013-07-22 16:50:39 +10:00
}
2015-03-18 11:22:27 +01:00
else // unauthenticated payment request
2013-10-24 16:02:39 +02:00
{
2018-05-03 14:38:39 +01:00
recipientElement . append ( tr ( " %1 to %2 " ) . arg ( amount , address ) ) ;
2013-10-24 16:02:39 +02:00
}
2013-09-13 16:49:35 +02:00
formatted . append ( recipientElement ) ;
2011-07-16 19:01:05 +02:00
}
2011-06-25 19:32:36 +02:00
2013-08-30 20:04:48 +02:00
QString questionString = tr ( " Are you sure you want to send? " ) ;
2018-05-03 14:38:39 +01:00
questionString . append ( " <br /><span style='font-size:10pt;'> " ) ;
questionString . append ( tr ( " Please, review your transaction. " ) ) ;
questionString . append ( " </span><br />%1 " ) ;
2013-08-30 20:04:48 +02:00
if ( txFee > 0 )
{
// append fee string if a fee is required
2018-05-03 14:38:39 +01:00
questionString . append ( " <hr /><b> " ) ;
questionString . append ( tr ( " Transaction fee " ) ) ;
questionString . append ( " </b> " ) ;
2014-11-02 00:14:47 +01:00
// append transaction size
2018-05-03 14:38:39 +01:00
questionString . append ( " ( " + QString : : number ( ( double ) currentTransaction . getTransactionSize ( ) / 1000 ) + " kB): " ) ;
// append transaction fee value
questionString . append ( " <span style='color:#aa0000; font-weight:bold;'> " ) ;
questionString . append ( BitcoinUnits : : formatHtmlWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , txFee ) ) ;
questionString . append ( " </span><br /> " ) ;
// append RBF message according to transaction's signalling
questionString . append ( " <span style='font-size:10pt; font-weight:normal;'> " ) ;
if ( ui - > optInRBF - > isChecked ( ) ) {
questionString . append ( tr ( " You can increase the fee later (signals Replace-By-Fee, BIP-125). " ) ) ;
} else {
questionString . append ( tr ( " Not signalling Replace-By-Fee, BIP-125. " ) ) ;
}
questionString . append ( " </span> " ) ;
2013-08-30 20:04:48 +02:00
}
2013-11-17 14:43:23 +01:00
// add total amount in all subdivision units
questionString . append ( " <hr /> " ) ;
2014-04-22 15:46:19 -07:00
CAmount totalAmount = currentTransaction . getTotalTransactionAmount ( ) + txFee ;
2013-11-17 14:43:23 +01:00
QStringList alternativeUnits ;
2017-06-02 03:25:02 +02:00
for ( BitcoinUnits : : Unit u : BitcoinUnits : : availableUnits ( ) )
2013-08-30 20:04:48 +02:00
{
2013-11-17 14:43:23 +01:00
if ( u ! = model - > getOptionsModel ( ) - > getDisplayUnit ( ) )
2014-05-09 23:50:09 +01:00
alternativeUnits . append ( BitcoinUnits : : formatHtmlWithUnit ( u , totalAmount ) ) ;
2013-08-30 20:04:48 +02:00
}
2018-05-03 14:38:39 +01:00
questionString . append ( QString ( " <b>%1</b>: <b>%2</b> " ) . arg ( tr ( " Total Amount " ) )
2015-11-06 13:22:00 +01:00
. arg ( BitcoinUnits : : formatHtmlWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , totalAmount ) ) ) ;
2018-05-03 14:38:39 +01:00
questionString . append ( QString ( " <br /><span style='font-size:10pt; font-weight:normal;'>(=%1)</span> " )
. arg ( alternativeUnits . join ( " " + tr ( " or " ) + " " ) ) ) ;
2016-04-05 16:55:41 +02:00
2016-05-09 22:26:57 -04:00
SendConfirmationDialog confirmationDialog ( tr ( " Confirm send coins " ) ,
questionString . arg ( formatted . join ( " <br /> " ) ) , SEND_CONFIRM_DELAY , this ) ;
confirmationDialog . exec ( ) ;
2017-06-01 16:13:35 +02:00
QMessageBox : : StandardButton retval = static_cast < QMessageBox : : StandardButton > ( confirmationDialog . result ( ) ) ;
2013-08-30 20:04:48 +02:00
if ( retval ! = QMessageBox : : Yes )
{
fNewRecipientAllowed = true ;
return ;
}
// now send the prepared transaction
2013-10-30 15:37:41 +01:00
WalletModel : : SendCoinsReturn sendStatus = model - > sendCoins ( currentTransaction ) ;
// process sendStatus and on error generate message shown to user
processSendCoinsReturn ( sendStatus ) ;
if ( sendStatus . status = = WalletModel : : OK )
2013-08-30 20:04:48 +02:00
{
2011-05-31 22:24:53 +02:00
accept ( ) ;
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > UnSelectAll ( ) ;
2013-08-12 17:03:03 +02:00
coinControlUpdateLabels ( ) ;
2017-04-17 18:56:44 -04:00
Q_EMIT coinsSent ( currentTransaction . getWtx ( ) - > get ( ) . GetHash ( ) ) ;
2011-05-14 17:25:05 +02:00
}
2011-12-23 20:27:12 -08:00
fNewRecipientAllowed = true ;
2011-05-12 17:55:24 +02:00
}
2011-07-16 19:01:05 +02:00
void SendCoinsDialog : : clear ( )
2011-05-12 17:55:24 +02:00
{
2018-02-14 14:23:43 +01:00
// Clear coin control settings
CoinControlDialog : : coinControl ( ) - > UnSelectAll ( ) ;
ui - > checkBoxCoinControlChange - > setChecked ( false ) ;
ui - > lineEditCoinControlChange - > clear ( ) ;
coinControlUpdateLabels ( ) ;
2011-07-16 19:01:05 +02:00
// Remove entries until only one left
2011-08-07 16:04:48 +02:00
while ( ui - > entries - > count ( ) )
2011-07-16 19:01:05 +02:00
{
2013-12-04 08:17:57 +01:00
ui - > entries - > takeAt ( 0 ) - > widget ( ) - > deleteLater ( ) ;
2011-07-16 19:01:05 +02:00
}
2011-08-07 16:04:48 +02:00
addEntry ( ) ;
2011-05-13 22:00:27 +02:00
2013-11-22 13:53:05 +01:00
updateTabsAndLabels ( ) ;
2011-07-07 17:33:15 +02:00
}
void SendCoinsDialog : : reject ( )
{
clear ( ) ;
}
void SendCoinsDialog : : accept ( )
{
clear ( ) ;
}
2011-07-16 19:01:05 +02:00
2011-08-07 16:04:48 +02:00
SendCoinsEntry * SendCoinsDialog : : addEntry ( )
2011-07-16 19:01:05 +02:00
{
2015-07-28 15:20:14 +02:00
SendCoinsEntry * entry = new SendCoinsEntry ( platformStyle , this ) ;
2011-07-16 19:01:05 +02:00
entry - > setModel ( model ) ;
ui - > entries - > addWidget ( entry ) ;
connect ( entry , SIGNAL ( removeEntry ( SendCoinsEntry * ) ) , this , SLOT ( removeEntry ( SendCoinsEntry * ) ) ) ;
2017-08-19 22:04:56 -07:00
connect ( entry , SIGNAL ( useAvailableBalance ( SendCoinsEntry * ) ) , this , SLOT ( useAvailableBalance ( SendCoinsEntry * ) ) ) ;
2013-08-12 17:03:03 +02:00
connect ( entry , SIGNAL ( payAmountChanged ( ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
2014-07-23 14:34:36 +02:00
connect ( entry , SIGNAL ( subtractFeeFromAmountChanged ( ) ) , this , SLOT ( coinControlUpdateLabels ( ) ) ) ;
2011-07-16 19:01:05 +02:00
// Focus the field, so that entry can start immediately
entry - > clear ( ) ;
2011-12-07 00:00:04 -05:00
entry - > setFocus ( ) ;
ui - > scrollAreaWidgetContents - > resize ( ui - > scrollAreaWidgetContents - > sizeHint ( ) ) ;
2013-04-02 17:30:14 +02:00
qApp - > processEvents ( ) ;
2011-12-07 00:00:04 -05:00
QScrollBar * bar = ui - > scrollArea - > verticalScrollBar ( ) ;
2012-06-09 15:41:21 +02:00
if ( bar )
2011-12-07 00:00:04 -05:00
bar - > setSliderPosition ( bar - > maximum ( ) ) ;
2016-01-04 09:44:36 +01:00
updateTabsAndLabels ( ) ;
2011-08-07 16:04:48 +02:00
return entry ;
2011-07-16 19:01:05 +02:00
}
2013-11-22 13:53:05 +01:00
void SendCoinsDialog : : updateTabsAndLabels ( )
2011-07-16 19:01:05 +02:00
{
setupTabChain ( 0 ) ;
2013-08-12 17:03:03 +02:00
coinControlUpdateLabels ( ) ;
2011-07-16 19:01:05 +02:00
}
void SendCoinsDialog : : removeEntry ( SendCoinsEntry * entry )
{
2013-12-10 12:01:54 +01:00
entry - > hide ( ) ;
2013-11-22 13:53:05 +01:00
2013-12-10 12:01:54 +01:00
// If the last entry is about to be removed add an empty one
if ( ui - > entries - > count ( ) = = 1 )
2013-11-22 13:53:05 +01:00
addEntry ( ) ;
2013-12-10 12:01:54 +01:00
entry - > deleteLater ( ) ;
2013-11-22 13:53:05 +01:00
updateTabsAndLabels ( ) ;
2011-07-16 19:01:05 +02:00
}
QWidget * SendCoinsDialog : : setupTabChain ( QWidget * prev )
{
for ( int i = 0 ; i < ui - > entries - > count ( ) ; + + i )
{
SendCoinsEntry * entry = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( i ) - > widget ( ) ) ;
if ( entry )
{
prev = entry - > setupTabChain ( prev ) ;
}
}
2014-01-29 14:41:41 +01:00
QWidget : : setTabOrder ( prev , ui - > sendButton ) ;
QWidget : : setTabOrder ( ui - > sendButton , ui - > clearButton ) ;
QWidget : : setTabOrder ( ui - > clearButton , ui - > addButton ) ;
return ui - > addButton ;
2011-07-16 19:01:05 +02:00
}
2011-08-07 16:04:48 +02:00
2013-01-25 18:46:53 +01:00
void SendCoinsDialog : : setAddress ( const QString & address )
{
SendCoinsEntry * entry = 0 ;
// Replace the first entry if it is still unused
if ( ui - > entries - > count ( ) = = 1 )
{
SendCoinsEntry * first = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( 0 ) - > widget ( ) ) ;
if ( first - > isClear ( ) )
{
entry = first ;
}
}
if ( ! entry )
{
entry = addEntry ( ) ;
}
entry - > setAddress ( address ) ;
}
2011-08-07 16:04:48 +02:00
void SendCoinsDialog : : pasteEntry ( const SendCoinsRecipient & rv )
{
2012-06-09 15:41:21 +02:00
if ( ! fNewRecipientAllowed )
2011-12-23 20:27:12 -08:00
return ;
2011-08-07 16:04:48 +02:00
SendCoinsEntry * entry = 0 ;
// Replace the first entry if it is still unused
if ( ui - > entries - > count ( ) = = 1 )
{
SendCoinsEntry * first = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( 0 ) - > widget ( ) ) ;
if ( first - > isClear ( ) )
{
entry = first ;
}
}
if ( ! entry )
{
entry = addEntry ( ) ;
}
entry - > setValue ( rv ) ;
2013-11-22 13:53:05 +01:00
updateTabsAndLabels ( ) ;
2011-08-07 16:04:48 +02:00
}
2013-07-22 16:50:39 +10:00
bool SendCoinsDialog : : handlePaymentRequest ( const SendCoinsRecipient & rv )
2011-08-07 16:04:48 +02:00
{
2013-11-16 01:54:29 +01:00
// Just paste the entry, all pre-checks
// are done in paymentserver.cpp.
2013-07-22 16:50:39 +10:00
pasteEntry ( rv ) ;
return true ;
2011-08-07 16:04:48 +02:00
}
2011-09-22 19:02:01 +02:00
2018-04-07 03:42:02 -04:00
void SendCoinsDialog : : setBalance ( const interfaces : : WalletBalances & balances )
2011-09-22 19:02:01 +02:00
{
2013-08-24 15:07:17 +02:00
if ( model & & model - > getOptionsModel ( ) )
{
2018-03-31 06:41:33 -04:00
ui - > labelBalance - > setText ( BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , balances . balance ) ) ;
2013-08-24 15:07:17 +02:00
}
2011-09-22 19:02:01 +02:00
}
2012-06-09 15:41:21 +02:00
void SendCoinsDialog : : updateDisplayUnit ( )
{
2018-03-31 06:41:33 -04:00
setBalance ( model - > wallet ( ) . getBalances ( ) ) ;
2014-11-02 00:14:47 +01:00
ui - > customFee - > setDisplayUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) ) ;
updateMinFeeLabel ( ) ;
updateSmartFeeLabel ( ) ;
2012-06-09 15:41:21 +02:00
}
2013-10-30 15:37:41 +01:00
void SendCoinsDialog : : processSendCoinsReturn ( const WalletModel : : SendCoinsReturn & sendCoinsReturn , const QString & msgArg )
{
QPair < QString , CClientUIInterface : : MessageBoxFlags > msgParams ;
// Default to a warning message, override if error message is needed
msgParams . second = CClientUIInterface : : MSG_WARNING ;
// This comment is specific to SendCoinsDialog usage of WalletModel::SendCoinsReturn.
// WalletModel::TransactionCommitFailed is used only in WalletModel::sendCoins()
// all others are used only in WalletModel::prepareTransaction()
switch ( sendCoinsReturn . status )
{
case WalletModel : : InvalidAddress :
2015-04-28 14:48:28 +00:00
msgParams . first = tr ( " The recipient address is not valid. Please recheck. " ) ;
2013-10-30 15:37:41 +01:00
break ;
case WalletModel : : InvalidAmount :
msgParams . first = tr ( " The amount to pay must be larger than 0. " ) ;
break ;
case WalletModel : : AmountExceedsBalance :
msgParams . first = tr ( " The amount exceeds your balance. " ) ;
break ;
case WalletModel : : AmountWithFeeExceedsBalance :
msgParams . first = tr ( " The total exceeds your balance when the %1 transaction fee is included. " ) . arg ( msgArg ) ;
break ;
case WalletModel : : DuplicateAddress :
2015-04-28 14:47:17 +00:00
msgParams . first = tr ( " Duplicate address found: addresses should only be used once each. " ) ;
2013-10-30 15:37:41 +01:00
break ;
case WalletModel : : TransactionCreationFailed :
msgParams . first = tr ( " Transaction creation failed! " ) ;
msgParams . second = CClientUIInterface : : MSG_ERROR ;
break ;
case WalletModel : : TransactionCommitFailed :
2016-10-30 18:22:22 +01:00
msgParams . first = tr ( " The transaction was rejected with the following reason: %1 " ) . arg ( sendCoinsReturn . reasonCommitFailed ) ;
2013-10-30 15:37:41 +01:00
msgParams . second = CClientUIInterface : : MSG_ERROR ;
break ;
2015-01-31 02:54:55 +00:00
case WalletModel : : AbsurdFee :
2017-04-20 12:28:58 -04:00
msgParams . first = tr ( " A fee higher than %1 is considered an absurdly high fee. " ) . arg ( BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , model - > node ( ) . getMaxTxFee ( ) ) ) ;
2014-11-02 00:14:47 +01:00
break ;
2015-01-08 14:42:04 +01:00
case WalletModel : : PaymentRequestExpired :
2015-02-03 22:44:33 +01:00
msgParams . first = tr ( " Payment request expired. " ) ;
2015-01-08 14:42:04 +01:00
msgParams . second = CClientUIInterface : : MSG_ERROR ;
break ;
2013-12-16 09:46:55 +01:00
// included to prevent a compiler warning.
2013-10-30 15:37:41 +01:00
case WalletModel : : OK :
default :
return ;
}
2015-07-14 13:59:05 +02:00
Q_EMIT message ( tr ( " Send Coins " ) , msgParams . first , msgParams . second ) ;
2013-10-30 15:37:41 +01:00
}
2013-08-12 17:03:03 +02:00
2014-11-02 00:14:47 +01:00
void SendCoinsDialog : : minimizeFeeSection ( bool fMinimize )
{
ui - > labelFeeMinimized - > setVisible ( fMinimize ) ;
ui - > buttonChooseFee - > setVisible ( fMinimize ) ;
ui - > buttonMinimizeFee - > setVisible ( ! fMinimize ) ;
ui - > frameFeeSelection - > setVisible ( ! fMinimize ) ;
ui - > horizontalLayoutSmartFee - > setContentsMargins ( 0 , ( fMinimize ? 0 : 6 ) , 0 , 0 ) ;
fFeeMinimized = fMinimize ;
}
void SendCoinsDialog : : on_buttonChooseFee_clicked ( )
{
minimizeFeeSection ( false ) ;
}
void SendCoinsDialog : : on_buttonMinimizeFee_clicked ( )
{
updateFeeMinimizedLabel ( ) ;
minimizeFeeSection ( true ) ;
}
2017-08-19 22:04:56 -07:00
void SendCoinsDialog : : useAvailableBalance ( SendCoinsEntry * entry )
{
// Get CCoinControl instance if CoinControl is enabled or create a new one.
CCoinControl coin_control ;
if ( model - > getOptionsModel ( ) - > getCoinControlFeatures ( ) ) {
2018-02-01 16:48:56 +00:00
coin_control = * CoinControlDialog : : coinControl ( ) ;
2017-08-19 22:04:56 -07:00
}
// Calculate available amount to send.
2017-04-17 18:56:44 -04:00
CAmount amount = model - > wallet ( ) . getAvailableBalance ( coin_control ) ;
2017-08-19 22:04:56 -07:00
for ( int i = 0 ; i < ui - > entries - > count ( ) ; + + i ) {
SendCoinsEntry * e = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( i ) - > widget ( ) ) ;
if ( e & & ! e - > isHidden ( ) & & e ! = entry ) {
amount - = e - > getValue ( ) . amount ;
}
}
if ( amount > 0 ) {
entry - > checkSubtractFeeFromAmount ( ) ;
entry - > setAmount ( amount ) ;
} else {
entry - > setAmount ( 0 ) ;
}
}
2014-11-02 00:14:47 +01:00
void SendCoinsDialog : : setMinimumFee ( )
{
2018-04-07 12:12:46 -04:00
ui - > customFee - > setValue ( model - > wallet ( ) . getRequiredFee ( 1000 ) ) ;
2014-11-02 00:14:47 +01:00
}
void SendCoinsDialog : : updateFeeSectionControls ( )
{
2017-07-07 22:03:49 +02:00
ui - > confTargetSelector - > setEnabled ( ui - > radioSmartFee - > isChecked ( ) ) ;
2014-11-02 00:14:47 +01:00
ui - > labelSmartFee - > setEnabled ( ui - > radioSmartFee - > isChecked ( ) ) ;
ui - > labelSmartFee2 - > setEnabled ( ui - > radioSmartFee - > isChecked ( ) ) ;
ui - > labelSmartFee3 - > setEnabled ( ui - > radioSmartFee - > isChecked ( ) ) ;
ui - > labelFeeEstimation - > setEnabled ( ui - > radioSmartFee - > isChecked ( ) ) ;
ui - > checkBoxMinimumFee - > setEnabled ( ui - > radioCustomFee - > isChecked ( ) ) ;
ui - > labelMinFeeWarning - > setEnabled ( ui - > radioCustomFee - > isChecked ( ) ) ;
2017-09-15 00:07:45 -04:00
ui - > labelCustomPerKilobyte - > setEnabled ( ui - > radioCustomFee - > isChecked ( ) & & ! ui - > checkBoxMinimumFee - > isChecked ( ) ) ;
2014-11-02 00:14:47 +01:00
ui - > customFee - > setEnabled ( ui - > radioCustomFee - > isChecked ( ) & & ! ui - > checkBoxMinimumFee - > isChecked ( ) ) ;
}
void SendCoinsDialog : : updateFeeMinimizedLabel ( )
{
if ( ! model | | ! model - > getOptionsModel ( ) )
return ;
if ( ui - > radioSmartFee - > isChecked ( ) )
ui - > labelFeeMinimized - > setText ( ui - > labelSmartFee - > text ( ) ) ;
else {
2017-09-15 00:07:45 -04:00
ui - > labelFeeMinimized - > setText ( BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , ui - > customFee - > value ( ) ) + " /kB " ) ;
2014-11-02 00:14:47 +01:00
}
}
void SendCoinsDialog : : updateMinFeeLabel ( )
{
if ( model & & model - > getOptionsModel ( ) )
2015-10-25 02:47:04 +02:00
ui - > checkBoxMinimumFee - > setText ( tr ( " Pay only the required fee of %1 " ) . arg (
2018-04-07 12:12:46 -04:00
BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , model - > wallet ( ) . getRequiredFee ( 1000 ) ) + " /kB " )
2014-11-02 00:14:47 +01:00
) ;
}
2017-06-28 19:24:28 -04:00
void SendCoinsDialog : : updateCoinControlState ( CCoinControl & ctrl )
{
if ( ui - > radioCustomFee - > isChecked ( ) ) {
ctrl . m_feerate = CFeeRate ( ui - > customFee - > value ( ) ) ;
} else {
2017-06-29 11:29:34 -04:00
ctrl . m_feerate . reset ( ) ;
2017-06-28 19:24:28 -04:00
}
// Avoid using global defaults when sending money from the GUI
// Either custom fee will be used or if not selected, the confirmation target from dropdown box
ctrl . m_confirm_target = getConfTargetForIndex ( ui - > confTargetSelector - > currentIndex ( ) ) ;
2018-04-07 12:12:46 -04:00
ctrl . m_signal_bip125_rbf = ui - > optInRBF - > isChecked ( ) ;
2017-06-28 19:24:28 -04:00
}
2014-11-02 00:14:47 +01:00
void SendCoinsDialog : : updateSmartFeeLabel ( )
{
if ( ! model | | ! model - > getOptionsModel ( ) )
return ;
2017-06-28 19:24:28 -04:00
CCoinControl coin_control ;
updateCoinControlState ( coin_control ) ;
2017-06-29 11:29:34 -04:00
coin_control . m_feerate . reset ( ) ; // Explicitly use only fee estimation rate for smart fee labels
2017-04-20 12:28:58 -04:00
int returned_target ;
FeeReason reason ;
2018-04-07 12:12:46 -04:00
CFeeRate feeRate = CFeeRate ( model - > wallet ( ) . getMinimumFee ( 1000 , coin_control , & returned_target , & reason ) ) ;
2017-06-28 19:24:28 -04:00
2017-06-29 11:29:34 -04:00
ui - > labelSmartFee - > setText ( BitcoinUnits : : formatWithUnit ( model - > getOptionsModel ( ) - > getDisplayUnit ( ) , feeRate . GetFeePerK ( ) ) + " /kB " ) ;
2017-04-20 12:28:58 -04:00
if ( reason = = FeeReason : : FALLBACK ) {
2014-11-02 00:14:47 +01:00
ui - > labelSmartFee2 - > show ( ) ; // (Smart fee not initialized yet. This usually takes a few blocks...)
ui - > labelFeeEstimation - > setText ( " " ) ;
2017-01-06 10:42:27 +01:00
ui - > fallbackFeeWarningLabel - > setVisible ( true ) ;
2017-02-17 08:18:32 +00:00
int lightness = ui - > fallbackFeeWarningLabel - > palette ( ) . color ( QPalette : : WindowText ) . lightness ( ) ;
QColor warning_colour ( 255 - ( lightness / 5 ) , 176 - ( lightness / 3 ) , 48 - ( lightness / 14 ) ) ;
ui - > fallbackFeeWarningLabel - > setStyleSheet ( " QLabel { color: " + warning_colour . name ( ) + " ; } " ) ;
2017-02-17 08:41:57 +00:00
ui - > fallbackFeeWarningLabel - > setIndent ( QFontMetrics ( ui - > fallbackFeeWarningLabel - > font ( ) ) . width ( " x " ) ) ;
2014-11-02 00:14:47 +01:00
}
else
{
ui - > labelSmartFee2 - > hide ( ) ;
2017-04-20 12:28:58 -04:00
ui - > labelFeeEstimation - > setText ( tr ( " Estimated to begin confirmation within %n block(s). " , " " , returned_target ) ) ;
2017-01-06 10:42:27 +01:00
ui - > fallbackFeeWarningLabel - > setVisible ( false ) ;
2014-11-02 00:14:47 +01:00
}
updateFeeMinimizedLabel ( ) ;
}
2013-08-12 17:03:03 +02:00
// Coin Control: copy label "Quantity" to clipboard
void SendCoinsDialog : : coinControlClipboardQuantity ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlQuantity - > text ( ) ) ;
}
// Coin Control: copy label "Amount" to clipboard
void SendCoinsDialog : : coinControlClipboardAmount ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlAmount - > text ( ) . left ( ui - > labelCoinControlAmount - > text ( ) . indexOf ( " " ) ) ) ;
}
// Coin Control: copy label "Fee" to clipboard
void SendCoinsDialog : : coinControlClipboardFee ( )
{
2015-01-12 22:26:29 +00:00
GUIUtil : : setClipboard ( ui - > labelCoinControlFee - > text ( ) . left ( ui - > labelCoinControlFee - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// Coin Control: copy label "After fee" to clipboard
void SendCoinsDialog : : coinControlClipboardAfterFee ( )
{
2015-01-12 22:26:29 +00:00
GUIUtil : : setClipboard ( ui - > labelCoinControlAfterFee - > text ( ) . left ( ui - > labelCoinControlAfterFee - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// Coin Control: copy label "Bytes" to clipboard
void SendCoinsDialog : : coinControlClipboardBytes ( )
{
2015-01-12 22:26:29 +00:00
GUIUtil : : setClipboard ( ui - > labelCoinControlBytes - > text ( ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
2014-06-08 01:05:53 +02:00
// Coin Control: copy label "Dust" to clipboard
2013-08-12 17:03:03 +02:00
void SendCoinsDialog : : coinControlClipboardLowOutput ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlLowOutput - > text ( ) ) ;
}
// Coin Control: copy label "Change" to clipboard
void SendCoinsDialog : : coinControlClipboardChange ( )
{
2015-01-12 22:26:29 +00:00
GUIUtil : : setClipboard ( ui - > labelCoinControlChange - > text ( ) . left ( ui - > labelCoinControlChange - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// Coin Control: settings menu - coin control enabled/disabled by user
void SendCoinsDialog : : coinControlFeatureChanged ( bool checked )
{
ui - > frameCoinControl - > setVisible ( checked ) ;
if ( ! checked & & model ) // coin control features disabled
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > SetNull ( ) ;
2014-01-27 19:15:56 +01:00
2015-11-25 14:05:30 +01:00
coinControlUpdateLabels ( ) ;
2013-08-12 17:03:03 +02:00
}
// Coin Control: button inputs -> show actual coin control dialog
void SendCoinsDialog : : coinControlButtonClicked ( )
{
2015-07-28 15:20:14 +02:00
CoinControlDialog dlg ( platformStyle ) ;
2013-08-12 17:03:03 +02:00
dlg . setModel ( model ) ;
dlg . exec ( ) ;
coinControlUpdateLabels ( ) ;
}
// Coin Control: checkbox custom change address
void SendCoinsDialog : : coinControlChangeChecked ( int state )
{
2013-11-20 15:49:34 +01:00
if ( state = = Qt : : Unchecked )
2013-08-12 17:03:03 +02:00
{
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > destChange = CNoDestination ( ) ;
2013-11-20 15:49:34 +01:00
ui - > labelCoinControlChangeLabel - > clear ( ) ;
2013-08-12 17:03:03 +02:00
}
2013-11-20 15:49:34 +01:00
else
// use this to re-validate an already entered address
coinControlChangeEdited ( ui - > lineEditCoinControlChange - > text ( ) ) ;
2013-08-12 17:03:03 +02:00
ui - > lineEditCoinControlChange - > setEnabled ( ( state = = Qt : : Checked ) ) ;
}
// Coin Control: custom change address changed
2013-12-10 12:01:54 +01:00
void SendCoinsDialog : : coinControlChangeEdited ( const QString & text )
2013-08-12 17:03:03 +02:00
{
2013-12-11 15:12:13 +01:00
if ( model & & model - > getAddressTableModel ( ) )
2013-08-12 17:03:03 +02:00
{
2013-12-11 15:12:13 +01:00
// Default to no change address until verified
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > destChange = CNoDestination ( ) ;
2013-12-11 15:12:13 +01:00
ui - > labelCoinControlChangeLabel - > setStyleSheet ( " QLabel{color:red;} " ) ;
2017-08-22 18:02:33 -07:00
const CTxDestination dest = DecodeDestination ( text . toStdString ( ) ) ;
2013-08-12 17:03:03 +02:00
2013-12-11 15:12:13 +01:00
if ( text . isEmpty ( ) ) // Nothing entered
{
2013-08-12 17:03:03 +02:00
ui - > labelCoinControlChangeLabel - > setText ( " " ) ;
2013-12-11 15:12:13 +01:00
}
2017-08-22 18:02:33 -07:00
else if ( ! IsValidDestination ( dest ) ) // Invalid address
2013-08-12 17:03:03 +02:00
{
ui - > labelCoinControlChangeLabel - > setText ( tr ( " Warning: Invalid Bitcoin address " ) ) ;
}
2013-12-11 15:12:13 +01:00
else // Valid address
2013-08-12 17:03:03 +02:00
{
2017-04-17 18:56:44 -04:00
if ( ! model - > wallet ( ) . isSpendable ( dest ) ) {
2013-12-11 15:12:13 +01:00
ui - > labelCoinControlChangeLabel - > setText ( tr ( " Warning: Unknown change address " ) ) ;
2016-12-23 10:24:29 +01:00
// confirmation dialog
QMessageBox : : StandardButton btnRetVal = QMessageBox : : question ( this , tr ( " Confirm custom change address " ) , tr ( " The address you selected for change is not part of this wallet. Any or all funds in your wallet may be sent to this address. Are you sure? " ) ,
QMessageBox : : Yes | QMessageBox : : Cancel , QMessageBox : : Cancel ) ;
if ( btnRetVal = = QMessageBox : : Yes )
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > destChange = dest ;
2016-12-23 10:24:29 +01:00
else
{
ui - > lineEditCoinControlChange - > setText ( " " ) ;
ui - > labelCoinControlChangeLabel - > setStyleSheet ( " QLabel{color:black;} " ) ;
ui - > labelCoinControlChangeLabel - > setText ( " " ) ;
}
2013-12-11 15:12:13 +01:00
}
else // Known change address
{
ui - > labelCoinControlChangeLabel - > setStyleSheet ( " QLabel{color:black;} " ) ;
// Query label
QString associatedLabel = model - > getAddressTableModel ( ) - > labelForAddress ( text ) ;
if ( ! associatedLabel . isEmpty ( ) )
ui - > labelCoinControlChangeLabel - > setText ( associatedLabel ) ;
2013-08-12 17:03:03 +02:00
else
2013-12-11 15:12:13 +01:00
ui - > labelCoinControlChangeLabel - > setText ( tr ( " (no label) " ) ) ;
2018-02-01 16:48:56 +00:00
CoinControlDialog : : coinControl ( ) - > destChange = dest ;
2013-08-12 17:03:03 +02:00
}
}
}
}
// Coin Control: update labels
void SendCoinsDialog : : coinControlUpdateLabels ( )
{
2015-11-25 14:05:30 +01:00
if ( ! model | | ! model - > getOptionsModel ( ) )
2013-08-12 17:03:03 +02:00
return ;
2018-02-01 16:48:56 +00:00
updateCoinControlState ( * CoinControlDialog : : coinControl ( ) ) ;
2017-06-28 19:24:28 -04:00
2013-08-12 17:03:03 +02:00
// set pay amounts
CoinControlDialog : : payAmounts . clear ( ) ;
2014-07-23 14:34:36 +02:00
CoinControlDialog : : fSubtractFeeFromAmount = false ;
2017-06-13 15:47:08 -04:00
2013-08-12 17:03:03 +02:00
for ( int i = 0 ; i < ui - > entries - > count ( ) ; + + i )
{
SendCoinsEntry * entry = qobject_cast < SendCoinsEntry * > ( ui - > entries - > itemAt ( i ) - > widget ( ) ) ;
2016-01-04 09:44:36 +01:00
if ( entry & & ! entry - > isHidden ( ) )
2014-07-23 14:34:36 +02:00
{
SendCoinsRecipient rcp = entry - > getValue ( ) ;
CoinControlDialog : : payAmounts . append ( rcp . amount ) ;
if ( rcp . fSubtractFeeFromAmount )
CoinControlDialog : : fSubtractFeeFromAmount = true ;
}
2013-08-12 17:03:03 +02:00
}
2018-02-01 16:48:56 +00:00
if ( CoinControlDialog : : coinControl ( ) - > HasSelected ( ) )
2013-08-12 17:03:03 +02:00
{
// actual coin control calculation
CoinControlDialog : : updateLabels ( model , this ) ;
// show coin control stats
ui - > labelCoinControlAutomaticallySelected - > hide ( ) ;
ui - > widgetCoinControl - > show ( ) ;
}
else
{
// hide coin control stats
ui - > labelCoinControlAutomaticallySelected - > show ( ) ;
ui - > widgetCoinControl - > hide ( ) ;
ui - > labelCoinControlInsuffFunds - > hide ( ) ;
}
}
2016-05-09 22:26:57 -04:00
2016-09-09 13:43:29 +02:00
SendConfirmationDialog : : SendConfirmationDialog ( const QString & title , const QString & text , int _secDelay ,
2016-05-09 22:26:57 -04:00
QWidget * parent ) :
2016-09-09 13:43:29 +02:00
QMessageBox ( QMessageBox : : Question , title , text , QMessageBox : : Yes | QMessageBox : : Cancel , parent ) , secDelay ( _secDelay )
2016-05-09 22:26:57 -04:00
{
setDefaultButton ( QMessageBox : : Cancel ) ;
yesButton = button ( QMessageBox : : Yes ) ;
updateYesButton ( ) ;
connect ( & countDownTimer , SIGNAL ( timeout ( ) ) , this , SLOT ( countDown ( ) ) ) ;
}
int SendConfirmationDialog : : exec ( )
{
updateYesButton ( ) ;
countDownTimer . start ( 1000 ) ;
return QMessageBox : : exec ( ) ;
}
void SendConfirmationDialog : : countDown ( )
{
secDelay - - ;
updateYesButton ( ) ;
if ( secDelay < = 0 )
{
countDownTimer . stop ( ) ;
}
}
void SendConfirmationDialog : : updateYesButton ( )
{
if ( secDelay > 0 )
{
yesButton - > setEnabled ( false ) ;
yesButton - > setText ( tr ( " Yes " ) + " ( " + QString : : number ( secDelay ) + " ) " ) ;
}
else
{
yesButton - > setEnabled ( true ) ;
yesButton - > setText ( tr ( " Yes " ) ) ;
}
}