2020-12-31 09:48:25 +01:00
// Copyright (c) 2011-2020 The Bitcoin Core developers
2014-12-13 12:09:33 +08: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.
2017-11-06 19:12:47 +01:00
# if defined(HAVE_CONFIG_H)
# include <config/bitcoin-config.h>
# endif
2017-11-10 13:57:53 +13:00
# include <qt/coincontroldialog.h>
2017-08-15 17:31:26 +02:00
# include <qt/forms/ui_coincontroldialog.h>
2013-08-12 17:03:03 +02:00
2017-11-10 13:57:53 +13:00
# include <qt/addresstablemodel.h>
# include <qt/bitcoinunits.h>
# include <qt/guiutil.h>
# include <qt/optionsmodel.h>
# include <qt/platformstyle.h>
# include <qt/walletmodel.h>
2013-11-16 17:37:31 +01:00
2017-11-10 13:57:53 +13:00
# include <wallet/coincontrol.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 <policy/policy.h>
# 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 <QTreeWidget>
2014-04-22 15:46:19 -07:00
QList < CAmount > CoinControlDialog : : payAmounts ;
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 ) ;
}
2020-05-05 23:56:21 +01:00
CoinControlDialog : : CoinControlDialog ( CCoinControl & coin_control , WalletModel * _model , const PlatformStyle * _platformStyle , QWidget * parent ) :
2020-09-07 19:09:33 +03:00
QDialog ( parent , GUIUtil : : dialog_flags ) ,
2013-08-12 17:03:03 +02:00
ui ( new Ui : : CoinControlDialog ) ,
2020-05-05 23:56:21 +01:00
m_coin_control ( coin_control ) ,
model ( _model ) ,
2016-09-09 13:43:29 +02:00
platformStyle ( _platformStyle )
2013-08-12 17:03:03 +02:00
{
ui - > setupUi ( this ) ;
// context menu
2016-11-18 15:47:20 +01:00
contextMenu = new QMenu ( this ) ;
2021-06-12 19:39:15 +00:00
contextMenu - > addAction ( tr ( " &Copy address " ) , this , & CoinControlDialog : : copyAddress ) ;
contextMenu - > addAction ( tr ( " Copy &label " ) , this , & CoinControlDialog : : copyLabel ) ;
contextMenu - > addAction ( tr ( " Copy &amount " ) , this , & CoinControlDialog : : copyAmount ) ;
copyTransactionHashAction = contextMenu - > addAction ( tr ( " Copy transaction &ID " ) , this , & CoinControlDialog : : copyTransactionHash ) ;
2013-08-12 17:03:03 +02:00
contextMenu - > addSeparator ( ) ;
2021-06-12 19:39:15 +00:00
lockAction = contextMenu - > addAction ( tr ( " L&ock unspent " ) , this , & CoinControlDialog : : lockCoin ) ;
unlockAction = contextMenu - > addAction ( tr ( " &Unlock unspent " ) , this , & CoinControlDialog : : unlockCoin ) ;
2018-06-24 16:18:22 +01:00
connect ( ui - > treeWidget , & QWidget : : customContextMenuRequested , this , & CoinControlDialog : : showMenu ) ;
2013-08-12 17:03:03 +02:00
// 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 ) ;
2018-06-24 16:18:22 +01:00
connect ( clipboardQuantityAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardQuantity ) ;
connect ( clipboardAmountAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardAmount ) ;
connect ( clipboardFeeAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardFee ) ;
connect ( clipboardAfterFeeAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardAfterFee ) ;
connect ( clipboardBytesAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardBytes ) ;
connect ( clipboardLowOutputAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardLowOutput ) ;
connect ( clipboardChangeAction , & QAction : : triggered , this , & CoinControlDialog : : clipboardChange ) ;
2013-08-12 17:03:03 +02:00
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
2018-06-24 16:18:22 +01:00
connect ( ui - > radioTreeMode , & QRadioButton : : toggled , this , & CoinControlDialog : : radioTreeMode ) ;
connect ( ui - > radioListMode , & QRadioButton : : toggled , this , & CoinControlDialog : : radioListMode ) ;
2013-08-12 17:03:03 +02:00
// click on checkbox
2018-06-24 16:18:22 +01:00
connect ( ui - > treeWidget , & QTreeWidget : : itemChanged , this , & CoinControlDialog : : viewItemChanged ) ;
2013-08-12 17:03:03 +02:00
// click on header
2013-11-16 17:37:31 +01:00
ui - > treeWidget - > header ( ) - > setSectionsClickable ( true ) ;
2018-06-24 16:18:22 +01:00
connect ( ui - > treeWidget - > header ( ) , & QHeaderView : : sectionClicked , this , & CoinControlDialog : : headerSectionClicked ) ;
2013-08-12 17:03:03 +02:00
// ok button
2018-06-24 16:18:22 +01:00
connect ( ui - > buttonBox , & QDialogButtonBox : : clicked , this , & CoinControlDialog : : buttonBoxClicked ) ;
2013-08-12 17:03:03 +02:00
// (un)select all
2018-06-24 16:18:22 +01:00
connect ( ui - > pushButtonSelectAll , & QPushButton : : clicked , this , & CoinControlDialog : : buttonSelectAllClicked ) ;
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 ) ;
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 " ) )
2017-06-01 16:13:35 +02:00
sortView ( settings . value ( " nCoinControlSortColumn " ) . toInt ( ) , ( static_cast < Qt : : SortOrder > ( settings . value ( " nCoinControlSortOrder " ) . toInt ( ) ) ) ) ;
2019-04-07 21:33:35 +02:00
GUIUtil : : handleCloseWindowShortcut ( this ) ;
2020-05-05 23:56:21 +01:00
if ( _model - > getOptionsModel ( ) & & _model - > getAddressTableModel ( ) )
{
updateView ( ) ;
updateLabelLocked ( ) ;
CoinControlDialog : : updateLabels ( m_coin_control , _model , this ) ;
}
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 ;
}
// 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 )
2020-05-05 23:56:21 +01:00
m_coin_control . UnSelectAll ( ) ; // just to be sure
CoinControlDialog : : updateLabels ( m_coin_control , model , this ) ;
2013-08-12 17:03:03 +02:00
}
// 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
2018-11-28 12:19:04 +00:00
if ( item - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . length ( ) = = 64 ) // transaction hash is 64 characters (this means it is a child node, so it is not a parent node in tree mode)
2013-08-12 17:03:03 +02:00
{
copyTransactionHashAction - > setEnabled ( true ) ;
2018-11-28 12:19:04 +00:00
if ( model - > wallet ( ) . isLockedCoin ( COutPoint ( uint256S ( item - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . toStdString ( ) ) , item - > data ( COLUMN_ADDRESS , VOutRole ) . 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 ( )
{
2018-11-28 12:19:04 +00:00
GUIUtil : : setClipboard ( contextMenuItem - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) ) ;
2013-08-12 17:03:03 +02:00
}
// context menu action: lock coin
void CoinControlDialog : : lockCoin ( )
{
if ( contextMenuItem - > checkState ( COLUMN_CHECKBOX ) = = Qt : : Checked )
contextMenuItem - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
2018-11-28 12:19:04 +00:00
COutPoint outpt ( uint256S ( contextMenuItem - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . toStdString ( ) ) , contextMenuItem - > data ( COLUMN_ADDRESS , VOutRole ) . toUInt ( ) ) ;
2017-04-17 18:56:44 -04:00
model - > wallet ( ) . lockCoin ( outpt ) ;
2013-08-12 17:03:03 +02:00
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 ( )
{
2018-11-28 12:19:04 +00:00
COutPoint outpt ( uint256S ( contextMenuItem - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . toStdString ( ) ) , contextMenuItem - > data ( COLUMN_ADDRESS , VOutRole ) . toUInt ( ) ) ;
2017-04-17 18:56:44 -04:00
model - > wallet ( ) . unlockCoin ( outpt ) ;
2013-08-12 17:03:03 +02:00
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 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
}
// copy label "After fee" to clipboard
void CoinControlDialog : : clipboardAfterFee ( )
{
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
}
// copy label "Bytes" to clipboard
void CoinControlDialog : : clipboardBytes ( )
{
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
// 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 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
}
// 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 )
{
2018-11-28 12:19:04 +00:00
if ( column = = COLUMN_CHECKBOX & & item - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . length ( ) = = 64 ) // transaction hash is 64 characters (this means it is a child node, so it is not a parent node in tree mode)
2013-08-12 17:03:03 +02:00
{
2018-11-28 12:19:04 +00:00
COutPoint outpt ( uint256S ( item - > data ( COLUMN_ADDRESS , TxHashRole ) . toString ( ) . toStdString ( ) ) , item - > data ( COLUMN_ADDRESS , VOutRole ) . toUInt ( ) ) ;
2013-08-12 17:03:03 +02:00
if ( item - > checkState ( COLUMN_CHECKBOX ) = = Qt : : Unchecked )
2020-05-05 23:56:21 +01:00
m_coin_control . UnSelect ( outpt ) ;
2013-08-12 17:03:03 +02:00
else if ( item - > isDisabled ( ) ) // locked (this happens if "check all" through parent node)
item - > setCheckState ( COLUMN_CHECKBOX , Qt : : Unchecked ) ;
else
2020-05-05 23:56:21 +01:00
m_coin_control . Select ( outpt ) ;
2013-08-12 17:03:03 +02:00
// selection changed -> update labels
if ( ui - > treeWidget - > isEnabled ( ) ) // do not update on every click for (un)select all
2020-05-05 23:56:21 +01:00
CoinControlDialog : : updateLabels ( m_coin_control , model , this ) ;
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 ;
2017-04-17 18:56:44 -04:00
model - > wallet ( ) . listLockedCoins ( vOutpts ) ;
2013-08-12 17:03:03 +02:00
if ( vOutpts . size ( ) > 0 )
{
ui - > labelLocked - > setText ( tr ( " (%1 locked) " ) . arg ( vOutpts . size ( ) ) ) ;
ui - > labelLocked - > setVisible ( true ) ;
}
else ui - > labelLocked - > setVisible ( false ) ;
}
2020-05-05 23:56:21 +01:00
void CoinControlDialog : : updateLabels ( CCoinControl & m_coin_control , WalletModel * model , QDialog * dialog )
2013-08-12 17:03:03 +02:00
{
2014-04-01 10:08:35 +02:00
if ( ! model )
return ;
2013-08-12 17:03:03 +02:00
// nPayAmount
2014-04-22 15:46:19 -07:00
CAmount nPayAmount = 0 ;
2013-08-12 17:03:03 +02:00
bool fDust = false ;
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 )
{
2019-05-31 10:01:15 -04:00
// Assumes a p2pkh script size
CTxOut txout ( amount , CScript ( ) < < std : : vector < unsigned char > ( 24 , 0 ) ) ;
2017-04-17 19:46:08 -04:00
fDust | = IsDust ( txout , model - > node ( ) . getDustRelayFee ( ) ) ;
2013-08-12 17:03:03 +02:00
}
}
2014-04-22 15:46:19 -07: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 ;
2020-05-05 23:56:21 +01:00
m_coin_control . ListSelected ( vCoinControl ) ;
2013-08-12 17:03:03 +02:00
2017-04-17 19:46:08 -04:00
size_t i = 0 ;
for ( const auto & out : model - > wallet ( ) . getCoins ( vCoinControl ) ) {
if ( out . depth_in_main_chain < 0 ) continue ;
2014-02-15 16:38:28 -05:00
// unselect already spent, very unlikely scenario, this could happen
// when selected are spent elsewhere, like rpc or another computer
2017-04-17 19:46:08 -04:00
const COutPoint & outpt = vCoinControl [ i + + ] ;
if ( out . is_spent )
2013-08-12 17:03:03 +02:00
{
2020-05-05 23:56:21 +01:00
m_coin_control . UnSelect ( outpt ) ;
2013-08-12 17:03:03 +02:00
continue ;
}
// Quantity
nQuantity + + ;
// Amount
2017-04-17 19:46:08 -04:00
nAmount + = out . txout . 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 ;
2017-04-17 19:46:08 -04:00
if ( out . txout . scriptPubKey . IsWitnessProgram ( witnessversion , witnessprogram ) )
2015-11-06 01:32:04 +01:00
{
nBytesInputs + = ( 32 + 4 + 1 + ( 107 / WITNESS_SCALE_FACTOR ) + 4 ) ;
fWitness = true ;
}
2017-04-17 19:46:08 -04:00
else if ( ExtractDestination ( out . txout . scriptPubKey , address ) )
2013-08-12 17:03:03 +02:00
{
CPubKey pubkey ;
2021-01-04 11:20:02 +01:00
PKHash * pkhash = std : : get_if < PKHash > ( & address ) ;
2020-01-15 13:40:14 -08:00
if ( pkhash & & model - > wallet ( ) . getPubKey ( out . txout . scriptPubKey , ToKeyID ( * pkhash ) , pubkey ) )
2013-08-12 17:03:03 +02:00
{
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 01:57:31 +03: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
2020-05-05 23:56:21 +01:00
nPayFee = model - > wallet ( ) . getMinimumFee ( nBytes , m_coin_control , nullptr /* returned_target */ , nullptr /* reason */ ) ;
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
{
2019-05-31 10:01:15 -04:00
// Assumes a p2pkh script size
CTxOut txout ( nChange , CScript ( ) < < std : : vector < unsigned char > ( 24 , 0 ) ) ;
2017-04-17 19:46:08 -04:00
if ( IsDust ( txout , model - > node ( ) . getDustRelayFee ( ) ) )
2013-08-12 17:03:03 +02:00
{
2016-12-13 16:38:43 -05: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 22:26:29 +00: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 16:41:53 -04:00
if ( nPayFee > 0 )
2014-06-11 13:58:16 +02:00
{
2015-01-12 22:26:29 +00: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 22:26:29 +00: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-10-15 22:42:01 -07:00
double dFeeVary = ( nBytes ! = 0 ) ? ( double ) nPayFee / nBytes : 0 ;
2017-06-29 11:29:34 -04:00
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 ) ;
}
2021-04-08 21:36:11 +03:00
void CoinControlDialog : : changeEvent ( QEvent * e )
{
# ifdef Q_OS_MACOS
if ( e - > type ( ) = = QEvent : : PaletteChange ) {
updateView ( ) ;
}
# endif
}
2013-08-12 17:03:03 +02:00
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
2017-04-17 19:46:08 -04:00
for ( const auto & coins : model - > wallet ( ) . listCoins ( ) ) {
2019-11-16 23:41:16 +00:00
CCoinControlWidgetItem * itemWalletAddress { nullptr } ;
2017-04-17 19:46:08 -04:00
QString sWalletAddress = QString : : fromStdString ( EncodeDestination ( 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
2019-11-16 23:41:16 +00:00
itemWalletAddress = new CCoinControlWidgetItem ( ui - > treeWidget ) ;
2013-08-12 17:03:03 +02:00
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-22 15:46:19 -07:00
CAmount nSum = 0 ;
2013-08-12 17:03:03 +02:00
int nChildren = 0 ;
2017-04-17 19:46:08 -04:00
for ( const auto & outpair : coins . second ) {
const COutPoint & output = std : : get < 0 > ( outpair ) ;
2018-04-07 03:42:02 -04:00
const interfaces : : WalletTxOut & out = std : : get < 1 > ( outpair ) ;
2017-04-17 19:46:08 -04:00
nSum + = out . txout . 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 = " " ;
2017-04-17 19:46:08 -04:00
if ( ExtractDestination ( out . txout . scriptPubKey , outputAddress ) )
2013-08-12 17:03:03 +02:00
{
2017-08-22 18:02:33 -07: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
2017-04-17 19:46:08 -04:00
itemOutput - > setText ( COLUMN_AMOUNT , BitcoinUnits : : format ( nDisplayUnit , out . txout . nValue ) ) ;
itemOutput - > setData ( COLUMN_AMOUNT , Qt : : UserRole , QVariant ( ( qlonglong ) out . txout . nValue ) ) ; // padding so that sorting works correctly
2013-08-12 17:03:03 +02:00
// date
2017-04-17 19:46:08 -04:00
itemOutput - > setText ( COLUMN_DATE , GUIUtil : : dateTimeStr ( out . time ) ) ;
itemOutput - > setData ( COLUMN_DATE , Qt : : UserRole , QVariant ( ( qlonglong ) out . time ) ) ;
2013-08-12 17:03:03 +02:00
// confirmations
2017-04-17 19:46:08 -04:00
itemOutput - > setText ( COLUMN_CONFIRMATIONS , QString : : number ( out . depth_in_main_chain ) ) ;
itemOutput - > setData ( COLUMN_CONFIRMATIONS , Qt : : UserRole , QVariant ( ( qlonglong ) out . depth_in_main_chain ) ) ;
2013-08-12 17:03:03 +02:00
// transaction hash
2018-11-28 12:19:04 +00:00
itemOutput - > setData ( COLUMN_ADDRESS , TxHashRole , QString : : fromStdString ( output . hash . GetHex ( ) ) ) ;
2013-08-12 17:03:03 +02:00
// vout index
2018-11-28 12:19:04 +00:00
itemOutput - > setData ( COLUMN_ADDRESS , VOutRole , output . n ) ;
2013-08-12 17:03:03 +02:00
// disable locked coins
2017-04-17 19:46:08 -04:00
if ( model - > wallet ( ) . isLockedCoin ( output ) )
2013-08-12 17:03:03 +02:00
{
2020-05-05 23:56:21 +01:00
m_coin_control . UnSelect ( output ) ; // just to be sure
2013-08-12 17:03:03 +02:00
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
2020-05-05 23:56:21 +01:00
if ( m_coin_control . IsSelected ( output ) )
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 ) ;
}