2016-12-31 19:01:21 +01:00
// Copyright (c) 2011-2016 The Bitcoin Core developers
2014-12-13 05:09:33 +01:00
// Distributed under the MIT software license, see the accompanying
2013-11-16 17:37:31 +01:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2013-08-12 17:03:03 +02:00
# include "coincontroldialog.h"
# include "ui_coincontroldialog.h"
# include "addresstablemodel.h"
2013-11-16 17:37:31 +01:00
# include "bitcoinunits.h"
2013-08-12 17:03:03 +02:00
# include "guiutil.h"
2013-11-16 17:37:31 +01:00
# include "optionsmodel.h"
2015-07-28 15:20:14 +02:00
# include "platformstyle.h"
2015-07-05 14:17:46 +02:00
# include "txmempool.h"
2013-11-16 17:37:31 +01:00
# include "walletmodel.h"
2016-10-21 11:47:10 +02:00
# include "wallet/coincontrol.h"
2015-01-30 10:08:46 +01:00
# include "init.h"
2017-02-15 21:23:34 +01:00
# include "policy/fees.h"
2016-12-13 22:19:17 +01:00
# include "policy/policy.h"
# include "validation.h" // For mempool
2017-08-02 13:19:28 +02:00
# include "wallet/fees.h"
2015-02-03 21:09:47 +01:00
# include "wallet/wallet.h"
2013-08-12 17:03:03 +02:00
# include <QApplication>
# include <QCheckBox>
# include <QCursor>
# include <QDialogButtonBox>
# include <QFlags>
# include <QIcon>
2014-11-02 00:14:47 +01:00
# include <QSettings>
2013-08-12 17:03:03 +02:00
# include <QString>
# include <QTreeWidget>
# include <QTreeWidgetItem>
2014-04-23 00:46:19 +02:00
QList < CAmount > CoinControlDialog : : payAmounts ;
2013-08-12 17:03:03 +02:00
CCoinControl * CoinControlDialog : : coinControl = new CCoinControl ( ) ;
2014-07-23 14:34:36 +02:00
bool CoinControlDialog : : fSubtractFeeFromAmount = false ;
2013-08-12 17:03:03 +02:00
2016-11-18 10:26:38 +01:00
bool CCoinControlWidgetItem : : operator < ( const QTreeWidgetItem & other ) const {
int column = treeWidget ( ) - > sortColumn ( ) ;
2016-11-18 14:33:34 +01:00
if ( column = = CoinControlDialog : : COLUMN_AMOUNT | | column = = CoinControlDialog : : COLUMN_DATE | | column = = CoinControlDialog : : COLUMN_CONFIRMATIONS )
return data ( column , Qt : : UserRole ) . toLongLong ( ) < other . data ( column , Qt : : UserRole ) . toLongLong ( ) ;
2016-11-18 10:26:38 +01:00
return QTreeWidgetItem : : operator < ( other ) ;
}
2016-09-09 13:43:29 +02:00
CoinControlDialog : : CoinControlDialog ( const PlatformStyle * _platformStyle , QWidget * parent ) :
2013-08-12 17:03:03 +02:00
QDialog ( parent ) ,
ui ( new Ui : : CoinControlDialog ) ,
2015-07-28 15:20:14 +02:00
model ( 0 ) ,
2016-09-09 13:43:29 +02:00
platformStyle ( _platformStyle )
2013-08-12 17:03:03 +02:00
{
ui - > setupUi ( this ) ;
// context menu actions
QAction * copyAddressAction = new QAction ( tr ( " Copy address " ) , this ) ;
QAction * copyLabelAction = new QAction ( tr ( " Copy label " ) , this ) ;
QAction * copyAmountAction = new QAction ( tr ( " Copy amount " ) , this ) ;
copyTransactionHashAction = new QAction ( tr ( " Copy transaction ID " ) , this ) ; // we need to enable/disable this
lockAction = new QAction ( tr ( " Lock unspent " ) , this ) ; // we need to enable/disable this
unlockAction = new QAction ( tr ( " Unlock unspent " ) , this ) ; // we need to enable/disable this
// context menu
2016-11-18 15:47:20 +01:00
contextMenu = new QMenu ( this ) ;
2013-08-12 17:03:03 +02:00
contextMenu - > addAction ( copyAddressAction ) ;
contextMenu - > addAction ( copyLabelAction ) ;
contextMenu - > addAction ( copyAmountAction ) ;
contextMenu - > addAction ( copyTransactionHashAction ) ;
contextMenu - > addSeparator ( ) ;
contextMenu - > addAction ( lockAction ) ;
contextMenu - > addAction ( unlockAction ) ;
// context menu signals
connect ( ui - > treeWidget , SIGNAL ( customContextMenuRequested ( QPoint ) ) , this , SLOT ( showMenu ( QPoint ) ) ) ;
connect ( copyAddressAction , SIGNAL ( triggered ( ) ) , this , SLOT ( copyAddress ( ) ) ) ;
connect ( copyLabelAction , SIGNAL ( triggered ( ) ) , this , SLOT ( copyLabel ( ) ) ) ;
connect ( copyAmountAction , SIGNAL ( triggered ( ) ) , this , SLOT ( copyAmount ( ) ) ) ;
connect ( copyTransactionHashAction , SIGNAL ( triggered ( ) ) , this , SLOT ( copyTransactionHash ( ) ) ) ;
connect ( lockAction , SIGNAL ( triggered ( ) ) , this , SLOT ( lockCoin ( ) ) ) ;
connect ( unlockAction , SIGNAL ( triggered ( ) ) , this , SLOT ( unlockCoin ( ) ) ) ;
// 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 ( clipboardQuantity ( ) ) ) ;
connect ( clipboardAmountAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardAmount ( ) ) ) ;
connect ( clipboardFeeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardFee ( ) ) ) ;
connect ( clipboardAfterFeeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardAfterFee ( ) ) ) ;
connect ( clipboardBytesAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardBytes ( ) ) ) ;
connect ( clipboardLowOutputAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardLowOutput ( ) ) ) ;
connect ( clipboardChangeAction , SIGNAL ( triggered ( ) ) , this , SLOT ( clipboardChange ( ) ) ) ;
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 ) ;
// toggle tree/list mode
connect ( ui - > radioTreeMode , SIGNAL ( toggled ( bool ) ) , this , SLOT ( radioTreeMode ( bool ) ) ) ;
connect ( ui - > radioListMode , SIGNAL ( toggled ( bool ) ) , this , SLOT ( radioListMode ( bool ) ) ) ;
// click on checkbox
2014-04-01 10:08:35 +02:00
connect ( ui - > treeWidget , SIGNAL ( itemChanged ( QTreeWidgetItem * , int ) ) , this , SLOT ( viewItemChanged ( QTreeWidgetItem * , int ) ) ) ;
2013-08-12 17:03:03 +02:00
// click on header
2013-11-16 17:37:31 +01:00
# if QT_VERSION < 0x050000
2013-08-12 17:03:03 +02:00
ui - > treeWidget - > header ( ) - > setClickable ( true ) ;
2013-11-16 17:37:31 +01:00
# else
ui - > treeWidget - > header ( ) - > setSectionsClickable ( true ) ;
# endif
2013-08-12 17:03:03 +02:00
connect ( ui - > treeWidget - > header ( ) , SIGNAL ( sectionClicked ( int ) ) , this , SLOT ( headerSectionClicked ( int ) ) ) ;
// ok button
connect ( ui - > buttonBox , SIGNAL ( clicked ( QAbstractButton * ) ) , this , SLOT ( buttonBoxClicked ( QAbstractButton * ) ) ) ;
// (un)select all
connect ( ui - > pushButtonSelectAll , SIGNAL ( clicked ( ) ) , this , SLOT ( buttonSelectAllClicked ( ) ) ) ;
2015-07-06 08:11:34 +02:00
// change coin control first column label due Qt4 bug.
2015-01-28 20:25:11 +01:00
// see https://github.com/bitcoin/bitcoin/issues/5716
ui - > treeWidget - > headerItem ( ) - > setText ( COLUMN_CHECKBOX , QString ( ) ) ;
2013-08-12 17:03:03 +02:00
ui - > treeWidget - > setColumnWidth ( COLUMN_CHECKBOX , 84 ) ;
2016-08-05 17:09:16 +02:00
ui - > treeWidget - > setColumnWidth ( COLUMN_AMOUNT , 110 ) ;
ui - > treeWidget - > setColumnWidth ( COLUMN_LABEL , 190 ) ;
ui - > treeWidget - > setColumnWidth ( COLUMN_ADDRESS , 320 ) ;
ui - > treeWidget - > setColumnWidth ( COLUMN_DATE , 130 ) ;
ui - > treeWidget - > setColumnWidth ( COLUMN_CONFIRMATIONS , 110 ) ;
2016-01-02 22:57:51 +01:00
ui - > treeWidget - > setColumnHidden ( COLUMN_TXHASH , true ) ; // store transaction hash in this column, but don't show it
2015-04-28 16:48:28 +02:00
ui - > treeWidget - > setColumnHidden ( COLUMN_VOUT_INDEX , true ) ; // store vout index in this column, but don't show it
2013-08-12 17:03:03 +02:00
// default view is sorted by amount desc
2016-11-18 14:33:34 +01:00
sortView ( COLUMN_AMOUNT , Qt : : DescendingOrder ) ;
2014-11-02 00:14:47 +01:00
// restore list mode and sortorder as a convenience feature
QSettings settings ;
if ( settings . contains ( " nCoinControlMode " ) & & ! settings . value ( " nCoinControlMode " ) . toBool ( ) )
ui - > radioTreeMode - > click ( ) ;
if ( settings . contains ( " nCoinControlSortColumn " ) & & settings . contains ( " nCoinControlSortOrder " ) )
sortView ( settings . value ( " nCoinControlSortColumn " ) . toInt ( ) , ( ( Qt : : SortOrder ) settings . value ( " nCoinControlSortOrder " ) . toInt ( ) ) ) ;
2013-08-12 17:03:03 +02:00
}
CoinControlDialog : : ~ CoinControlDialog ( )
{
2014-11-02 00:14:47 +01:00
QSettings settings ;
settings . setValue ( " nCoinControlMode " , ui - > radioListMode - > isChecked ( ) ) ;
settings . setValue ( " nCoinControlSortColumn " , sortColumn ) ;
settings . setValue ( " nCoinControlSortOrder " , ( int ) sortOrder ) ;
2013-08-12 17:03:03 +02:00
delete ui ;
}
2016-09-09 13:43:29 +02:00
void CoinControlDialog : : setModel ( WalletModel * _model )
2013-08-12 17:03:03 +02:00
{
2016-09-09 13:43:29 +02:00
this - > model = _model ;
2013-08-12 17:03:03 +02:00
2016-09-09 13:43:29 +02:00
if ( _model & & _model - > getOptionsModel ( ) & & _model - > getAddressTableModel ( ) )
2013-08-12 17:03:03 +02:00
{
updateView ( ) ;
updateLabelLocked ( ) ;
2016-09-09 13:43:29 +02:00
CoinControlDialog : : updateLabels ( _model , this ) ;
2013-08-12 17:03:03 +02:00
}
}
// ok button
void CoinControlDialog : : buttonBoxClicked ( QAbstractButton * button )
{
if ( ui - > buttonBox - > buttonRole ( button ) = = QDialogButtonBox : : AcceptRole )
done ( QDialog : : Accepted ) ; // closes the dialog
}
// (un)select all
void CoinControlDialog : : buttonSelectAllClicked ( )
{
Qt : : CheckState state = Qt : : Checked ;
for ( int i = 0 ; i < ui - > treeWidget - > topLevelItemCount ( ) ; i + + )
{
if ( ui - > treeWidget - > topLevelItem ( i ) - > checkState ( COLUMN_CHECKBOX ) ! = Qt : : Unchecked )
{
state = Qt : : Unchecked ;
break ;
}
}
ui - > treeWidget - > setEnabled ( false ) ;
for ( int i = 0 ; i < ui - > treeWidget - > topLevelItemCount ( ) ; i + + )
if ( ui - > treeWidget - > topLevelItem ( i ) - > checkState ( COLUMN_CHECKBOX ) ! = state )
ui - > treeWidget - > topLevelItem ( i ) - > setCheckState ( COLUMN_CHECKBOX , state ) ;
ui - > treeWidget - > setEnabled ( true ) ;
if ( state = = Qt : : Unchecked )
coinControl - > UnSelectAll ( ) ; // just to be sure
CoinControlDialog : : updateLabels ( model , this ) ;
}
// context menu
void CoinControlDialog : : showMenu ( const QPoint & point )
{
QTreeWidgetItem * item = ui - > treeWidget - > itemAt ( point ) ;
if ( item )
{
contextMenuItem = item ;
// disable some items (like Copy Transaction ID, lock, unlock) for tree roots in context menu
if ( item - > text ( COLUMN_TXHASH ) . length ( ) = = 64 ) // transaction hash is 64 characters (this means its a child node, so its not a parent node in tree mode)
{
copyTransactionHashAction - > setEnabled ( true ) ;
2014-12-16 14:50:05 +01:00
if ( model - > isLockedCoin ( uint256S ( item - > text ( COLUMN_TXHASH ) . toStdString ( ) ) , item - > text ( COLUMN_VOUT_INDEX ) . toUInt ( ) ) )
2013-08-12 17:03:03 +02:00
{
lockAction - > setEnabled ( false ) ;
unlockAction - > setEnabled ( true ) ;
}
else
{
lockAction - > setEnabled ( true ) ;
unlockAction - > setEnabled ( false ) ;
}
}
else // this means click on parent node in tree mode -> disable all
{
copyTransactionHashAction - > setEnabled ( false ) ;
lockAction - > setEnabled ( false ) ;
unlockAction - > setEnabled ( false ) ;
}
// show context menu
contextMenu - > exec ( QCursor : : pos ( ) ) ;
}
}
// context menu action: copy amount
void CoinControlDialog : : copyAmount ( )
{
2014-07-20 04:17:02 +02:00
GUIUtil : : setClipboard ( BitcoinUnits : : removeSpaces ( contextMenuItem - > text ( COLUMN_AMOUNT ) ) ) ;
2013-08-12 17:03:03 +02:00
}
// context menu action: copy label
void CoinControlDialog : : copyLabel ( )
{
if ( ui - > radioTreeMode - > isChecked ( ) & & contextMenuItem - > text ( COLUMN_LABEL ) . length ( ) = = 0 & & contextMenuItem - > parent ( ) )
GUIUtil : : setClipboard ( contextMenuItem - > parent ( ) - > text ( COLUMN_LABEL ) ) ;
else
GUIUtil : : setClipboard ( contextMenuItem - > text ( COLUMN_LABEL ) ) ;
}
// context menu action: copy address
void CoinControlDialog : : copyAddress ( )
{
if ( ui - > radioTreeMode - > isChecked ( ) & & contextMenuItem - > text ( COLUMN_ADDRESS ) . length ( ) = = 0 & & contextMenuItem - > parent ( ) )
GUIUtil : : setClipboard ( contextMenuItem - > parent ( ) - > text ( COLUMN_ADDRESS ) ) ;
else
GUIUtil : : setClipboard ( contextMenuItem - > text ( COLUMN_ADDRESS ) ) ;
}
// context menu action: copy transaction id
void CoinControlDialog : : copyTransactionHash ( )
{
GUIUtil : : setClipboard ( contextMenuItem - > text ( COLUMN_TXHASH ) ) ;
}
// context menu action: lock coin
void CoinControlDialog : : lockCoin ( )
{
if ( contextMenuItem - > checkState ( COLUMN_CHECKBOX ) = = Qt : : Checked )
contextMenuItem - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
2014-12-16 14:50:05 +01:00
COutPoint outpt ( uint256S ( contextMenuItem - > text ( COLUMN_TXHASH ) . toStdString ( ) ) , contextMenuItem - > text ( COLUMN_VOUT_INDEX ) . toUInt ( ) ) ;
2013-08-12 17:03:03 +02:00
model - > lockCoin ( outpt ) ;
contextMenuItem - > setDisabled ( true ) ;
2015-07-28 15:20:14 +02:00
contextMenuItem - > setIcon ( COLUMN_CHECKBOX , platformStyle - > SingleColorIcon ( " :/icons/lock_closed " ) ) ;
2013-08-12 17:03:03 +02:00
updateLabelLocked ( ) ;
}
// context menu action: unlock coin
void CoinControlDialog : : unlockCoin ( )
{
2014-12-16 14:50:05 +01:00
COutPoint outpt ( uint256S ( contextMenuItem - > text ( COLUMN_TXHASH ) . toStdString ( ) ) , contextMenuItem - > text ( COLUMN_VOUT_INDEX ) . toUInt ( ) ) ;
2013-08-12 17:03:03 +02:00
model - > unlockCoin ( outpt ) ;
contextMenuItem - > setDisabled ( false ) ;
contextMenuItem - > setIcon ( COLUMN_CHECKBOX , QIcon ( ) ) ;
updateLabelLocked ( ) ;
}
// copy label "Quantity" to clipboard
void CoinControlDialog : : clipboardQuantity ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlQuantity - > text ( ) ) ;
}
// copy label "Amount" to clipboard
void CoinControlDialog : : clipboardAmount ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlAmount - > text ( ) . left ( ui - > labelCoinControlAmount - > text ( ) . indexOf ( " " ) ) ) ;
}
// copy label "Fee" to clipboard
void CoinControlDialog : : clipboardFee ( )
{
2015-01-12 23:26:29 +01:00
GUIUtil : : setClipboard ( ui - > labelCoinControlFee - > text ( ) . left ( ui - > labelCoinControlFee - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// copy label "After fee" to clipboard
void CoinControlDialog : : clipboardAfterFee ( )
{
2015-01-12 23:26:29 +01:00
GUIUtil : : setClipboard ( ui - > labelCoinControlAfterFee - > text ( ) . left ( ui - > labelCoinControlAfterFee - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// copy label "Bytes" to clipboard
void CoinControlDialog : : clipboardBytes ( )
{
2015-01-12 23:26:29 +01: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
// copy label "Dust" to clipboard
2013-08-12 17:03:03 +02:00
void CoinControlDialog : : clipboardLowOutput ( )
{
GUIUtil : : setClipboard ( ui - > labelCoinControlLowOutput - > text ( ) ) ;
}
// copy label "Change" to clipboard
void CoinControlDialog : : clipboardChange ( )
{
2015-01-12 23:26:29 +01:00
GUIUtil : : setClipboard ( ui - > labelCoinControlChange - > text ( ) . left ( ui - > labelCoinControlChange - > text ( ) . indexOf ( " " ) ) . replace ( ASYMP_UTF8 , " " ) ) ;
2013-08-12 17:03:03 +02:00
}
// treeview: sort
void CoinControlDialog : : sortView ( int column , Qt : : SortOrder order )
{
sortColumn = column ;
sortOrder = order ;
ui - > treeWidget - > sortItems ( column , order ) ;
2016-11-18 14:33:34 +01:00
ui - > treeWidget - > header ( ) - > setSortIndicator ( sortColumn , sortOrder ) ;
2013-08-12 17:03:03 +02:00
}
// treeview: clicked on header
void CoinControlDialog : : headerSectionClicked ( int logicalIndex )
{
if ( logicalIndex = = COLUMN_CHECKBOX ) // click on most left column -> do nothing
{
2016-11-18 14:33:34 +01:00
ui - > treeWidget - > header ( ) - > setSortIndicator ( sortColumn , sortOrder ) ;
2013-08-12 17:03:03 +02:00
}
else
{
if ( sortColumn = = logicalIndex )
sortOrder = ( ( sortOrder = = Qt : : AscendingOrder ) ? Qt : : DescendingOrder : Qt : : AscendingOrder ) ;
else
{
sortColumn = logicalIndex ;
2014-03-03 04:16:42 +01:00
sortOrder = ( ( sortColumn = = COLUMN_LABEL | | sortColumn = = COLUMN_ADDRESS ) ? Qt : : AscendingOrder : Qt : : DescendingOrder ) ; // if label or address then default => asc, else default => desc
2013-08-12 17:03:03 +02:00
}
sortView ( sortColumn , sortOrder ) ;
}
}
// toggle tree mode
void CoinControlDialog : : radioTreeMode ( bool checked )
{
if ( checked & & model )
updateView ( ) ;
}
// toggle list mode
void CoinControlDialog : : radioListMode ( bool checked )
{
if ( checked & & model )
updateView ( ) ;
}
// checkbox clicked by user
void CoinControlDialog : : viewItemChanged ( QTreeWidgetItem * item , int column )
{
if ( column = = COLUMN_CHECKBOX & & item - > text ( COLUMN_TXHASH ) . length ( ) = = 64 ) // transaction hash is 64 characters (this means its a child node, so its not a parent node in tree mode)
{
2014-12-16 14:50:05 +01:00
COutPoint outpt ( uint256S ( item - > text ( COLUMN_TXHASH ) . toStdString ( ) ) , item - > text ( COLUMN_VOUT_INDEX ) . toUInt ( ) ) ;
2013-08-12 17:03:03 +02:00
if ( item - > checkState ( COLUMN_CHECKBOX ) = = Qt : : Unchecked )
coinControl - > UnSelect ( outpt ) ;
else if ( item - > isDisabled ( ) ) // locked (this happens if "check all" through parent node)
item - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
else
coinControl - > Select ( outpt ) ;
// selection changed -> update labels
if ( ui - > treeWidget - > isEnabled ( ) ) // do not update on every click for (un)select all
CoinControlDialog : : updateLabels ( model , this ) ;
}
2014-03-11 13:42:00 +01:00
2016-01-04 16:54:43 +01:00
// TODO: Remove this temporary qt5 fix after Qt5.3 and Qt5.4 are no longer used.
// Fixed in Qt5.5 and above: https://bugreports.qt.io/browse/QTBUG-43473
2014-03-11 13:42:00 +01:00
# if QT_VERSION >= 0x050000
else if ( column = = COLUMN_CHECKBOX & & item - > childCount ( ) > 0 )
{
if ( item - > checkState ( COLUMN_CHECKBOX ) = = Qt : : PartiallyChecked & & item - > child ( 0 ) - > checkState ( COLUMN_CHECKBOX ) = = Qt : : PartiallyChecked )
item - > setCheckState ( COLUMN_CHECKBOX , Qt : : Checked ) ;
}
# endif
2013-08-12 17:03:03 +02:00
}
// shows count of locked unspent outputs
void CoinControlDialog : : updateLabelLocked ( )
{
2015-07-09 13:23:44 +02:00
std : : vector < COutPoint > vOutpts ;
2013-08-12 17:03:03 +02:00
model - > listLockedCoins ( vOutpts ) ;
if ( vOutpts . size ( ) > 0 )
{
ui - > labelLocked - > setText ( tr ( " (%1 locked) " ) . arg ( vOutpts . size ( ) ) ) ;
ui - > labelLocked - > setVisible ( true ) ;
}
else ui - > labelLocked - > setVisible ( false ) ;
}
void CoinControlDialog : : updateLabels ( WalletModel * model , QDialog * dialog )
{
2014-04-01 10:08:35 +02:00
if ( ! model )
return ;
2013-08-12 17:03:03 +02:00
// nPayAmount
2014-04-23 00:46:19 +02:00
CAmount nPayAmount = 0 ;
2013-08-12 17:03:03 +02:00
bool fDust = false ;
2014-06-07 13:53:27 +02:00
CMutableTransaction txDummy ;
2017-06-02 03:25:02 +02:00
for ( const CAmount & amount : CoinControlDialog : : payAmounts )
2013-08-12 17:03:03 +02:00
{
nPayAmount + = amount ;
if ( amount > 0 )
{
2015-07-09 13:23:44 +02:00
CTxOut txout ( amount , ( CScript ) std : : vector < unsigned char > ( 24 , 0 ) ) ;
2013-08-12 17:03:03 +02:00
txDummy . vout . push_back ( txout ) ;
2016-10-14 17:49:05 +02:00
fDust | = IsDust ( txout , : : dustRelayFee ) ;
2013-08-12 17:03:03 +02:00
}
}
2014-04-23 00:46:19 +02:00
CAmount nAmount = 0 ;
CAmount nPayFee = 0 ;
CAmount nAfterFee = 0 ;
CAmount nChange = 0 ;
2013-08-12 17:03:03 +02:00
unsigned int nBytes = 0 ;
unsigned int nBytesInputs = 0 ;
unsigned int nQuantity = 0 ;
2015-11-06 01:32:04 +01:00
bool fWitness = false ;
2013-08-12 17:03:03 +02:00
2015-07-09 13:23:44 +02:00
std : : vector < COutPoint > vCoinControl ;
std : : vector < COutput > vOutputs ;
2013-08-12 17:03:03 +02:00
coinControl - > ListSelected ( vCoinControl ) ;
model - > getOutputs ( vCoinControl , vOutputs ) ;
2017-06-02 03:18:57 +02:00
for ( const COutput & out : vOutputs ) {
2014-02-15 22:38:28 +01:00
// unselect already spent, very unlikely scenario, this could happen
// when selected are spent elsewhere, like rpc or another computer
uint256 txhash = out . tx - > GetHash ( ) ;
COutPoint outpt ( txhash , out . i ) ;
if ( model - > isSpent ( outpt ) )
2013-08-12 17:03:03 +02:00
{
coinControl - > UnSelect ( outpt ) ;
continue ;
}
// Quantity
nQuantity + + ;
// Amount
2016-11-12 01:54:51 +01:00
nAmount + = out . tx - > tx - > vout [ out . i ] . nValue ;
2013-08-12 17:03:03 +02:00
// Bytes
CTxDestination address ;
2015-11-06 01:32:04 +01:00
int witnessversion = 0 ;
std : : vector < unsigned char > witnessprogram ;
2016-11-12 01:54:51 +01:00
if ( out . tx - > tx - > vout [ out . i ] . scriptPubKey . IsWitnessProgram ( witnessversion , witnessprogram ) )
2015-11-06 01:32:04 +01:00
{
nBytesInputs + = ( 32 + 4 + 1 + ( 107 / WITNESS_SCALE_FACTOR ) + 4 ) ;
fWitness = true ;
}
2016-11-12 01:54:51 +01:00
else if ( ExtractDestination ( out . tx - > tx - > vout [ out . i ] . scriptPubKey , address ) )
2013-08-12 17:03:03 +02:00
{
CPubKey pubkey ;
CKeyID * keyid = boost : : get < CKeyID > ( & address ) ;
if ( keyid & & model - > getPubKey ( * keyid , pubkey ) )
{
nBytesInputs + = ( pubkey . IsCompressed ( ) ? 148 : 180 ) ;
}
else
nBytesInputs + = 148 ; // in all error cases, simply assume 148 here
}
else nBytesInputs + = 148 ;
}
// calculation
if ( nQuantity > 0 )
{
// Bytes
nBytes = nBytesInputs + ( ( CoinControlDialog : : payAmounts . size ( ) > 0 ? CoinControlDialog : : payAmounts . size ( ) + 1 : 2 ) * 34 ) + 10 ; // always assume +1 output for change here
2015-11-06 01:32:04 +01:00
if ( fWitness )
{
// there is some fudging in these numbers related to the actual virtual transaction size calculation that will keep this estimate from being exact.
// usually, the result will be an overestimate within a couple of satoshis so that the confirmation dialog ends up displaying a slightly smaller fee.
2017-06-20 00:57:31 +02:00
// also, the witness stack size value is a variable sized integer. usually, the number of stack items will be well under the single byte var int limit.
2015-11-06 01:32:04 +01:00
nBytes + = 2 ; // account for the serialized marker and flag bytes
nBytes + = nQuantity ; // account for the witness byte that holds the number of stack items for each input.
}
2013-08-12 17:03:03 +02:00
2014-07-23 14:34:36 +02:00
// in the subtract fee from amount case, we can tell if zero change already and subtract the bytes, so that fee calculation afterwards is accurate
if ( CoinControlDialog : : fSubtractFeeFromAmount )
if ( nAmount - nPayAmount = = 0 )
nBytes - = 34 ;
2014-11-02 00:14:47 +01:00
// Fee
2017-08-02 13:19:28 +02:00
nPayFee = GetMinimumFee ( nBytes , * coinControl , : : mempool , : : feeEstimator , nullptr /* FeeCalculation */ ) ;
2015-11-25 13:38:44 +01:00
2013-08-12 17:03:03 +02:00
if ( nPayAmount > 0 )
{
2014-07-23 14:34:36 +02:00
nChange = nAmount - nPayAmount ;
if ( ! CoinControlDialog : : fSubtractFeeFromAmount )
nChange - = nPayFee ;
2013-08-12 17:03:03 +02:00
// Never create dust outputs; if we would, just add the dust to the fee.
2015-09-13 23:23:59 +02:00
if ( nChange > 0 & & nChange < MIN_CHANGE )
2013-08-12 17:03:03 +02:00
{
2015-07-09 13:23:44 +02:00
CTxOut txout ( nChange , ( CScript ) std : : vector < unsigned char > ( 24 , 0 ) ) ;
2016-10-14 17:49:05 +02:00
if ( IsDust ( txout , : : dustRelayFee ) )
2013-08-12 17:03:03 +02:00
{
2016-12-13 22:38:43 +01:00
nPayFee + = nChange ;
nChange = 0 ;
if ( CoinControlDialog : : fSubtractFeeFromAmount )
nBytes - = 34 ; // we didn't detect lack of change above
2013-08-12 17:03:03 +02:00
}
}
2014-07-23 14:34:36 +02:00
if ( nChange = = 0 & & ! CoinControlDialog : : fSubtractFeeFromAmount )
2013-08-12 17:03:03 +02:00
nBytes - = 34 ;
}
// after fee
2017-02-03 09:20:54 +01:00
nAfterFee = std : : max < CAmount > ( nAmount - nPayFee , 0 ) ;
2013-08-12 17:03:03 +02:00
}
// actually update labels
int nDisplayUnit = BitcoinUnits : : BTC ;
if ( model & & model - > getOptionsModel ( ) )
nDisplayUnit = model - > getOptionsModel ( ) - > getDisplayUnit ( ) ;
QLabel * l1 = dialog - > findChild < QLabel * > ( " labelCoinControlQuantity " ) ;
QLabel * l2 = dialog - > findChild < QLabel * > ( " labelCoinControlAmount " ) ;
QLabel * l3 = dialog - > findChild < QLabel * > ( " labelCoinControlFee " ) ;
QLabel * l4 = dialog - > findChild < QLabel * > ( " labelCoinControlAfterFee " ) ;
QLabel * l5 = dialog - > findChild < QLabel * > ( " labelCoinControlBytes " ) ;
QLabel * l7 = dialog - > findChild < QLabel * > ( " labelCoinControlLowOutput " ) ;
QLabel * l8 = dialog - > findChild < QLabel * > ( " labelCoinControlChange " ) ;
2014-06-08 01:05:53 +02:00
// enable/disable "dust" and "change"
2013-08-12 17:03:03 +02:00
dialog - > findChild < QLabel * > ( " labelCoinControlLowOutputText " ) - > setEnabled ( nPayAmount > 0 ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlLowOutput " ) - > setEnabled ( nPayAmount > 0 ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlChangeText " ) - > setEnabled ( nPayAmount > 0 ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlChange " ) - > setEnabled ( nPayAmount > 0 ) ;
// stats
l1 - > setText ( QString : : number ( nQuantity ) ) ; // Quantity
l2 - > setText ( BitcoinUnits : : formatWithUnit ( nDisplayUnit , nAmount ) ) ; // Amount
l3 - > setText ( BitcoinUnits : : formatWithUnit ( nDisplayUnit , nPayFee ) ) ; // Fee
l4 - > setText ( BitcoinUnits : : formatWithUnit ( nDisplayUnit , nAfterFee ) ) ; // After Fee
2015-01-12 23:26:29 +01:00
l5 - > setText ( ( ( nBytes > 0 ) ? ASYMP_UTF8 : " " ) + QString : : number ( nBytes ) ) ; // Bytes
2014-06-08 01:05:53 +02:00
l7 - > setText ( fDust ? tr ( " yes " ) : tr ( " no " ) ) ; // Dust
2013-08-12 17:03:03 +02:00
l8 - > setText ( BitcoinUnits : : formatWithUnit ( nDisplayUnit , nChange ) ) ; // Change
2017-05-11 22:41:53 +02:00
if ( nPayFee > 0 )
2014-06-11 13:58:16 +02:00
{
2015-01-12 23:26:29 +01:00
l3 - > setText ( ASYMP_UTF8 + l3 - > text ( ) ) ;
l4 - > setText ( ASYMP_UTF8 + l4 - > text ( ) ) ;
2014-07-23 14:34:36 +02:00
if ( nChange > 0 & & ! CoinControlDialog : : fSubtractFeeFromAmount )
2015-01-12 23:26:29 +01:00
l8 - > setText ( ASYMP_UTF8 + l8 - > text ( ) ) ;
2014-06-11 13:58:16 +02:00
}
2013-08-12 17:03:03 +02:00
2016-08-05 17:09:16 +02:00
// turn label red when dust
l7 - > setStyleSheet ( ( fDust ) ? " color:red; " : " " ) ;
2013-08-12 17:03:03 +02:00
// tool tips
2016-08-05 17:09:16 +02:00
QString toolTipDust = tr ( " This label turns red if any recipient receives an amount smaller than the current dust threshold. " ) ;
2013-12-02 13:36:19 +01:00
2014-06-11 13:58:16 +02:00
// how many satoshis the estimated fee can vary per byte we guess wrong
2017-08-28 09:20:50 +02:00
assert ( nBytes ! = 0 ) ;
2017-06-29 17:29:34 +02:00
double dFeeVary = ( double ) nPayFee / nBytes ;
2014-06-11 13:58:16 +02:00
QString toolTip4 = tr ( " Can vary +/- %1 satoshi(s) per input. " ) . arg ( dFeeVary ) ;
l3 - > setToolTip ( toolTip4 ) ;
l4 - > setToolTip ( toolTip4 ) ;
2016-08-05 17:09:16 +02:00
l7 - > setToolTip ( toolTipDust ) ;
2014-06-11 13:58:16 +02:00
l8 - > setToolTip ( toolTip4 ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlFeeText " ) - > setToolTip ( l3 - > toolTip ( ) ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlAfterFeeText " ) - > setToolTip ( l4 - > toolTip ( ) ) ;
2013-08-12 17:03:03 +02:00
dialog - > findChild < QLabel * > ( " labelCoinControlBytesText " ) - > setToolTip ( l5 - > toolTip ( ) ) ;
dialog - > findChild < QLabel * > ( " labelCoinControlLowOutputText " ) - > setToolTip ( l7 - > toolTip ( ) ) ;
2014-06-11 13:58:16 +02:00
dialog - > findChild < QLabel * > ( " labelCoinControlChangeText " ) - > setToolTip ( l8 - > toolTip ( ) ) ;
2013-08-12 17:03:03 +02:00
// Insufficient funds
QLabel * label = dialog - > findChild < QLabel * > ( " labelCoinControlInsuffFunds " ) ;
if ( label )
label - > setVisible ( nChange < 0 ) ;
}
void CoinControlDialog : : updateView ( )
{
2014-04-01 10:08:35 +02:00
if ( ! model | | ! model - > getOptionsModel ( ) | | ! model - > getAddressTableModel ( ) )
return ;
2013-08-12 17:03:03 +02:00
bool treeMode = ui - > radioTreeMode - > isChecked ( ) ;
ui - > treeWidget - > clear ( ) ;
ui - > treeWidget - > setEnabled ( false ) ; // performance, otherwise updateLabels would be called for every checked checkbox
ui - > treeWidget - > setAlternatingRowColors ( ! treeMode ) ;
2014-04-01 10:08:35 +02:00
QFlags < Qt : : ItemFlag > flgCheckbox = Qt : : ItemIsSelectable | Qt : : ItemIsEnabled | Qt : : ItemIsUserCheckable ;
QFlags < Qt : : ItemFlag > flgTristate = Qt : : ItemIsSelectable | Qt : : ItemIsEnabled | Qt : : ItemIsUserCheckable | Qt : : ItemIsTristate ;
2013-08-12 17:03:03 +02:00
2014-04-01 10:08:35 +02:00
int nDisplayUnit = model - > getOptionsModel ( ) - > getDisplayUnit ( ) ;
2013-08-12 17:03:03 +02:00
2015-07-09 13:23:44 +02:00
std : : map < QString , std : : vector < COutput > > mapCoins ;
2013-08-12 17:03:03 +02:00
model - > listCoins ( mapCoins ) ;
2017-06-02 03:28:42 +02:00
for ( const std : : pair < QString , std : : vector < COutput > > & coins : mapCoins ) {
2016-11-18 10:26:38 +01:00
CCoinControlWidgetItem * itemWalletAddress = new CCoinControlWidgetItem ( ) ;
2014-04-01 10:08:35 +02:00
itemWalletAddress - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
2013-08-12 17:03:03 +02:00
QString sWalletAddress = coins . first ;
2014-04-01 10:08:35 +02:00
QString sWalletLabel = model - > getAddressTableModel ( ) - > labelForAddress ( sWalletAddress ) ;
if ( sWalletLabel . isEmpty ( ) )
2013-08-12 17:03:03 +02:00
sWalletLabel = tr ( " (no label) " ) ;
if ( treeMode )
{
// wallet address
ui - > treeWidget - > addTopLevelItem ( itemWalletAddress ) ;
itemWalletAddress - > setFlags ( flgTristate ) ;
2014-04-01 10:08:35 +02:00
itemWalletAddress - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
2013-08-12 17:03:03 +02:00
// label
itemWalletAddress - > setText ( COLUMN_LABEL , sWalletLabel ) ;
// address
itemWalletAddress - > setText ( COLUMN_ADDRESS , sWalletAddress ) ;
}
2014-04-23 00:46:19 +02:00
CAmount nSum = 0 ;
2013-08-12 17:03:03 +02:00
int nChildren = 0 ;
2017-06-02 03:18:57 +02:00
for ( const COutput & out : coins . second ) {
2016-11-12 01:54:51 +01:00
nSum + = out . tx - > tx - > vout [ out . i ] . nValue ;
2013-08-12 17:03:03 +02:00
nChildren + + ;
2016-11-18 10:26:38 +01:00
CCoinControlWidgetItem * itemOutput ;
if ( treeMode ) itemOutput = new CCoinControlWidgetItem ( itemWalletAddress ) ;
else itemOutput = new CCoinControlWidgetItem ( ui - > treeWidget ) ;
2013-08-12 17:03:03 +02:00
itemOutput - > setFlags ( flgCheckbox ) ;
itemOutput - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
// address
CTxDestination outputAddress ;
QString sAddress = " " ;
2016-11-12 01:54:51 +01:00
if ( ExtractDestination ( out . tx - > tx - > vout [ out . i ] . scriptPubKey , outputAddress ) )
2013-08-12 17:03:03 +02:00
{
2017-08-23 03:02:33 +02:00
sAddress = QString : : fromStdString ( EncodeDestination ( outputAddress ) ) ;
2013-08-12 17:03:03 +02:00
// if listMode or change => show bitcoin address. In tree mode, address is not shown again for direct wallet address outputs
if ( ! treeMode | | ( ! ( sAddress = = sWalletAddress ) ) )
itemOutput - > setText ( COLUMN_ADDRESS , sAddress ) ;
}
// label
if ( ! ( sAddress = = sWalletAddress ) ) // change
{
// tooltip from where the change comes from
itemOutput - > setToolTip ( COLUMN_LABEL , tr ( " change from %1 (%2) " ) . arg ( sWalletLabel ) . arg ( sWalletAddress ) ) ;
itemOutput - > setText ( COLUMN_LABEL , tr ( " (change) " ) ) ;
}
else if ( ! treeMode )
{
2014-04-01 10:08:35 +02:00
QString sLabel = model - > getAddressTableModel ( ) - > labelForAddress ( sAddress ) ;
if ( sLabel . isEmpty ( ) )
2013-08-12 17:03:03 +02:00
sLabel = tr ( " (no label) " ) ;
itemOutput - > setText ( COLUMN_LABEL , sLabel ) ;
}
// amount
2016-11-12 01:54:51 +01:00
itemOutput - > setText ( COLUMN_AMOUNT , BitcoinUnits : : format ( nDisplayUnit , out . tx - > tx - > vout [ out . i ] . nValue ) ) ;
itemOutput - > setData ( COLUMN_AMOUNT , Qt : : UserRole , QVariant ( ( qlonglong ) out . tx - > tx - > vout [ out . i ] . nValue ) ) ; // padding so that sorting works correctly
2013-08-12 17:03:03 +02:00
// date
2013-12-15 15:39:09 +01:00
itemOutput - > setText ( COLUMN_DATE , GUIUtil : : dateTimeStr ( out . tx - > GetTxTime ( ) ) ) ;
2016-11-18 14:33:34 +01:00
itemOutput - > setData ( COLUMN_DATE , Qt : : UserRole , QVariant ( ( qlonglong ) out . tx - > GetTxTime ( ) ) ) ;
2013-08-12 17:03:03 +02:00
// confirmations
2016-11-18 14:33:34 +01:00
itemOutput - > setText ( COLUMN_CONFIRMATIONS , QString : : number ( out . nDepth ) ) ;
itemOutput - > setData ( COLUMN_CONFIRMATIONS , Qt : : UserRole , QVariant ( ( qlonglong ) out . nDepth ) ) ;
2013-08-12 17:03:03 +02:00
// transaction hash
uint256 txhash = out . tx - > GetHash ( ) ;
2014-09-08 12:25:52 +02:00
itemOutput - > setText ( COLUMN_TXHASH , QString : : fromStdString ( txhash . GetHex ( ) ) ) ;
2013-08-12 17:03:03 +02:00
// vout index
itemOutput - > setText ( COLUMN_VOUT_INDEX , QString : : number ( out . i ) ) ;
// disable locked coins
if ( model - > isLockedCoin ( txhash , out . i ) )
{
COutPoint outpt ( txhash , out . i ) ;
coinControl - > UnSelect ( outpt ) ; // just to be sure
itemOutput - > setDisabled ( true ) ;
2015-07-28 15:20:14 +02:00
itemOutput - > setIcon ( COLUMN_CHECKBOX , platformStyle - > SingleColorIcon ( " :/icons/lock_closed " ) ) ;
2013-08-12 17:03:03 +02:00
}
// set checkbox
2016-02-11 02:07:22 +01:00
if ( coinControl - > IsSelected ( COutPoint ( txhash , out . i ) ) )
2014-04-01 10:08:35 +02:00
itemOutput - > setCheckState ( COLUMN_CHECKBOX , Qt : : Checked ) ;
2013-08-12 17:03:03 +02:00
}
// amount
if ( treeMode )
{
itemWalletAddress - > setText ( COLUMN_CHECKBOX , " ( " + QString : : number ( nChildren ) + " ) " ) ;
itemWalletAddress - > setText ( COLUMN_AMOUNT , BitcoinUnits : : format ( nDisplayUnit , nSum ) ) ;
2016-11-18 14:33:34 +01:00
itemWalletAddress - > setData ( COLUMN_AMOUNT , Qt : : UserRole , QVariant ( ( qlonglong ) nSum ) ) ;
2013-08-12 17:03:03 +02:00
}
}
// expand all partially selected
if ( treeMode )
{
for ( int i = 0 ; i < ui - > treeWidget - > topLevelItemCount ( ) ; i + + )
if ( ui - > treeWidget - > topLevelItem ( i ) - > checkState ( COLUMN_CHECKBOX ) = = Qt : : PartiallyChecked )
ui - > treeWidget - > topLevelItem ( i ) - > setExpanded ( true ) ;
}
// sort view
sortView ( sortColumn , sortOrder ) ;
ui - > treeWidget - > setEnabled ( true ) ;
}