2019-12-30 22:39:22 +13:00
|
|
|
// Copyright (c) 2011-2019 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/addresstablemodel.h>
|
2013-01-23 21:51:02 +01:00
|
|
|
|
2017-11-10 13:57:53 +13:00
|
|
|
#include <qt/guiutil.h>
|
|
|
|
#include <qt/walletmodel.h>
|
2011-06-26 19:23:24 +02:00
|
|
|
|
2017-09-19 18:12:25 -07:00
|
|
|
#include <key_io.h>
|
2017-11-10 13:57:53 +13:00
|
|
|
#include <wallet/wallet.h>
|
2011-05-09 20:44:46 +02:00
|
|
|
|
2019-08-21 09:44:19 +08:00
|
|
|
#include <algorithm>
|
|
|
|
|
2011-06-02 15:57:23 +02:00
|
|
|
#include <QFont>
|
2013-09-04 11:52:45 +02:00
|
|
|
#include <QDebug>
|
2011-06-02 15:57:23 +02:00
|
|
|
|
2011-05-13 15:58:27 +02:00
|
|
|
const QString AddressTableModel::Send = "S";
|
|
|
|
const QString AddressTableModel::Receive = "R";
|
|
|
|
|
2011-06-01 15:50:09 +02:00
|
|
|
struct AddressTableEntry
|
|
|
|
{
|
|
|
|
enum Type {
|
|
|
|
Sending,
|
2013-08-29 16:19:43 +02:00
|
|
|
Receiving,
|
|
|
|
Hidden /* QSortFilterProxyModel will filter these out */
|
2011-06-01 15:50:09 +02:00
|
|
|
};
|
2011-06-03 15:16:11 +02:00
|
|
|
|
2011-06-01 15:50:09 +02:00
|
|
|
Type type;
|
|
|
|
QString label;
|
|
|
|
QString address;
|
|
|
|
|
|
|
|
AddressTableEntry() {}
|
2016-09-09 13:43:29 +02:00
|
|
|
AddressTableEntry(Type _type, const QString &_label, const QString &_address):
|
|
|
|
type(_type), label(_label), address(_address) {}
|
2011-06-01 15:50:09 +02:00
|
|
|
};
|
|
|
|
|
2012-05-06 22:41:35 +02:00
|
|
|
struct AddressTableEntryLessThan
|
|
|
|
{
|
|
|
|
bool operator()(const AddressTableEntry &a, const AddressTableEntry &b) const
|
|
|
|
{
|
|
|
|
return a.address < b.address;
|
|
|
|
}
|
|
|
|
bool operator()(const AddressTableEntry &a, const QString &b) const
|
|
|
|
{
|
|
|
|
return a.address < b;
|
|
|
|
}
|
|
|
|
bool operator()(const QString &a, const AddressTableEntry &b) const
|
|
|
|
{
|
|
|
|
return a < b.address;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-08-29 16:19:43 +02:00
|
|
|
/* Determine address type from address purpose */
|
|
|
|
static AddressTableEntry::Type translateTransactionType(const QString &strPurpose, bool isMine)
|
|
|
|
{
|
|
|
|
AddressTableEntry::Type addressType = AddressTableEntry::Hidden;
|
|
|
|
// "refund" addresses aren't shown, and change addresses aren't in mapAddressBook at all.
|
|
|
|
if (strPurpose == "send")
|
|
|
|
addressType = AddressTableEntry::Sending;
|
|
|
|
else if (strPurpose == "receive")
|
|
|
|
addressType = AddressTableEntry::Receiving;
|
|
|
|
else if (strPurpose == "unknown" || strPurpose == "") // if purpose not set, guess
|
|
|
|
addressType = (isMine ? AddressTableEntry::Receiving : AddressTableEntry::Sending);
|
|
|
|
return addressType;
|
|
|
|
}
|
|
|
|
|
2011-06-18 11:53:25 +02:00
|
|
|
// Private implementation
|
2012-04-15 12:31:56 +02:00
|
|
|
class AddressTablePriv
|
2011-06-01 15:50:09 +02:00
|
|
|
{
|
2012-04-15 12:31:56 +02:00
|
|
|
public:
|
2011-06-01 15:50:09 +02:00
|
|
|
QList<AddressTableEntry> cachedAddressTable;
|
2012-05-06 22:41:35 +02:00
|
|
|
AddressTableModel *parent;
|
2011-06-01 15:50:09 +02:00
|
|
|
|
2018-07-26 17:15:32 +02:00
|
|
|
explicit AddressTablePriv(AddressTableModel *_parent):
|
2017-04-18 13:01:23 -04:00
|
|
|
parent(_parent) {}
|
2011-06-26 19:23:24 +02:00
|
|
|
|
2018-04-07 03:42:02 -04:00
|
|
|
void refreshAddressTable(interfaces::Wallet& wallet)
|
2011-06-01 15:50:09 +02:00
|
|
|
{
|
|
|
|
cachedAddressTable.clear();
|
|
|
|
{
|
2017-04-18 13:01:23 -04:00
|
|
|
for (const auto& address : wallet.getAddresses())
|
2011-06-01 15:50:09 +02:00
|
|
|
{
|
2013-08-29 16:19:43 +02:00
|
|
|
AddressTableEntry::Type addressType = translateTransactionType(
|
2017-04-18 13:01:23 -04:00
|
|
|
QString::fromStdString(address.purpose), address.is_mine);
|
2013-07-22 16:50:39 +10:00
|
|
|
cachedAddressTable.append(AddressTableEntry(addressType,
|
2017-04-18 13:01:23 -04:00
|
|
|
QString::fromStdString(address.name),
|
|
|
|
QString::fromStdString(EncodeDestination(address.dest))));
|
2011-06-01 15:50:09 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-21 09:49:21 +08:00
|
|
|
// std::lower_bound() and std::upper_bound() require our cachedAddressTable list to be sorted in asc order
|
2013-08-30 08:25:25 +02:00
|
|
|
// Even though the map is already sorted this re-sorting step is needed because the originating map
|
|
|
|
// is sorted by binary address, not by base58() address.
|
2019-08-21 09:44:19 +08:00
|
|
|
std::sort(cachedAddressTable.begin(), cachedAddressTable.end(), AddressTableEntryLessThan());
|
2011-06-01 15:50:09 +02:00
|
|
|
}
|
|
|
|
|
2013-08-29 16:19:43 +02:00
|
|
|
void updateEntry(const QString &address, const QString &label, bool isMine, const QString &purpose, int status)
|
2012-05-06 22:41:35 +02:00
|
|
|
{
|
|
|
|
// Find address / label in model
|
2019-08-21 09:49:21 +08:00
|
|
|
QList<AddressTableEntry>::iterator lower = std::lower_bound(
|
2012-05-06 22:41:35 +02:00
|
|
|
cachedAddressTable.begin(), cachedAddressTable.end(), address, AddressTableEntryLessThan());
|
2019-08-21 09:49:21 +08:00
|
|
|
QList<AddressTableEntry>::iterator upper = std::upper_bound(
|
2012-05-06 22:41:35 +02:00
|
|
|
cachedAddressTable.begin(), cachedAddressTable.end(), address, AddressTableEntryLessThan());
|
|
|
|
int lowerIndex = (lower - cachedAddressTable.begin());
|
|
|
|
int upperIndex = (upper - cachedAddressTable.begin());
|
|
|
|
bool inModel = (lower != upper);
|
2013-08-29 16:19:43 +02:00
|
|
|
AddressTableEntry::Type newEntryType = translateTransactionType(purpose, isMine);
|
2012-05-06 22:41:35 +02:00
|
|
|
|
|
|
|
switch(status)
|
|
|
|
{
|
|
|
|
case CT_NEW:
|
|
|
|
if(inModel)
|
|
|
|
{
|
2015-01-08 11:44:25 +01:00
|
|
|
qWarning() << "AddressTablePriv::updateEntry: Warning: Got CT_NEW, but entry is already in model";
|
2012-05-06 22:41:35 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
parent->beginInsertRows(QModelIndex(), lowerIndex, lowerIndex);
|
|
|
|
cachedAddressTable.insert(lowerIndex, AddressTableEntry(newEntryType, label, address));
|
|
|
|
parent->endInsertRows();
|
|
|
|
break;
|
|
|
|
case CT_UPDATED:
|
|
|
|
if(!inModel)
|
|
|
|
{
|
2015-01-08 11:44:25 +01:00
|
|
|
qWarning() << "AddressTablePriv::updateEntry: Warning: Got CT_UPDATED, but entry is not in model";
|
2012-05-06 22:41:35 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
lower->type = newEntryType;
|
|
|
|
lower->label = label;
|
|
|
|
parent->emitDataChanged(lowerIndex);
|
|
|
|
break;
|
|
|
|
case CT_DELETED:
|
|
|
|
if(!inModel)
|
|
|
|
{
|
2015-01-08 11:44:25 +01:00
|
|
|
qWarning() << "AddressTablePriv::updateEntry: Warning: Got CT_DELETED, but entry is not in model";
|
2012-05-06 22:41:35 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
parent->beginRemoveRows(QModelIndex(), lowerIndex, upperIndex-1);
|
|
|
|
cachedAddressTable.erase(lower, upper);
|
|
|
|
parent->endRemoveRows();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-01 15:50:09 +02:00
|
|
|
int size()
|
|
|
|
{
|
|
|
|
return cachedAddressTable.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
AddressTableEntry *index(int idx)
|
|
|
|
{
|
|
|
|
if(idx >= 0 && idx < cachedAddressTable.size())
|
|
|
|
{
|
|
|
|
return &cachedAddressTable[idx];
|
2011-06-07 18:59:01 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-30 12:37:09 +02:00
|
|
|
return nullptr;
|
2011-06-01 15:50:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-04-18 13:01:23 -04:00
|
|
|
AddressTableModel::AddressTableModel(WalletModel *parent) :
|
2018-04-11 11:56:44 +02:00
|
|
|
QAbstractTableModel(parent), walletModel(parent)
|
2011-05-09 20:44:46 +02:00
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
columns << tr("Label") << tr("Address");
|
2017-04-18 13:01:23 -04:00
|
|
|
priv = new AddressTablePriv(this);
|
|
|
|
priv->refreshAddressTable(parent->wallet());
|
2011-06-01 15:50:09 +02:00
|
|
|
}
|
2011-05-13 15:58:27 +02:00
|
|
|
|
2011-06-01 15:50:09 +02:00
|
|
|
AddressTableModel::~AddressTableModel()
|
|
|
|
{
|
|
|
|
delete priv;
|
2011-05-13 15:58:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int AddressTableModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
2011-05-27 08:20:23 +02:00
|
|
|
Q_UNUSED(parent);
|
2011-06-01 15:50:09 +02:00
|
|
|
return priv->size();
|
2011-05-13 15:58:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int AddressTableModel::columnCount(const QModelIndex &parent) const
|
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
Q_UNUSED(parent);
|
|
|
|
return columns.length();
|
2011-05-13 15:58:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant AddressTableModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if(!index.isValid())
|
|
|
|
return QVariant();
|
|
|
|
|
2011-06-01 15:50:09 +02:00
|
|
|
AddressTableEntry *rec = static_cast<AddressTableEntry*>(index.internalPointer());
|
|
|
|
|
2011-06-02 17:48:45 +02:00
|
|
|
if(role == Qt::DisplayRole || role == Qt::EditRole)
|
2011-05-13 15:58:27 +02:00
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
switch(index.column())
|
|
|
|
{
|
|
|
|
case Label:
|
2011-07-02 17:31:27 +02:00
|
|
|
if(rec->label.isEmpty() && role == Qt::DisplayRole)
|
2011-06-30 21:29:20 +02:00
|
|
|
{
|
|
|
|
return tr("(no label)");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return rec->label;
|
|
|
|
}
|
2011-06-01 15:50:09 +02:00
|
|
|
case Address:
|
|
|
|
return rec->address;
|
|
|
|
}
|
2011-06-07 18:59:01 +02:00
|
|
|
}
|
|
|
|
else if (role == Qt::FontRole)
|
2011-06-01 20:08:21 +02:00
|
|
|
{
|
2011-06-21 19:54:09 +02:00
|
|
|
QFont font;
|
2011-06-01 20:08:21 +02:00
|
|
|
if(index.column() == Address)
|
|
|
|
{
|
2015-10-22 13:33:58 +02:00
|
|
|
font = GUIUtil::fixedPitchFont();
|
2011-06-21 19:54:09 +02:00
|
|
|
}
|
|
|
|
return font;
|
|
|
|
}
|
2011-06-07 18:59:01 +02:00
|
|
|
else if (role == TypeRole)
|
2011-05-13 15:58:27 +02:00
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
switch(rec->type)
|
2011-05-13 15:58:27 +02:00
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
case AddressTableEntry::Sending:
|
|
|
|
return Send;
|
|
|
|
case AddressTableEntry::Receiving:
|
|
|
|
return Receive;
|
|
|
|
default: break;
|
2011-05-13 15:58:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2013-01-08 08:17:58 +01:00
|
|
|
bool AddressTableModel::setData(const QModelIndex &index, const QVariant &value, int role)
|
2011-06-02 17:48:45 +02:00
|
|
|
{
|
|
|
|
if(!index.isValid())
|
|
|
|
return false;
|
2011-06-03 15:16:11 +02:00
|
|
|
AddressTableEntry *rec = static_cast<AddressTableEntry*>(index.internalPointer());
|
2013-07-22 16:50:39 +10:00
|
|
|
std::string strPurpose = (rec->type == AddressTableEntry::Sending ? "send" : "receive");
|
2011-07-16 19:01:05 +02:00
|
|
|
editStatus = OK;
|
|
|
|
|
2011-06-02 17:48:45 +02:00
|
|
|
if(role == Qt::EditRole)
|
|
|
|
{
|
2017-08-22 18:02:33 -07:00
|
|
|
CTxDestination curAddress = DecodeDestination(rec->address.toStdString());
|
2013-12-12 10:12:27 +01:00
|
|
|
if(index.column() == Label)
|
2011-06-02 17:48:45 +02:00
|
|
|
{
|
2013-01-08 08:17:58 +01:00
|
|
|
// Do nothing, if old label == new label
|
|
|
|
if(rec->label == value.toString())
|
|
|
|
{
|
|
|
|
editStatus = NO_CHANGES;
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-18 13:01:23 -04:00
|
|
|
walletModel->wallet().setAddressBook(curAddress, value.toString().toStdString(), strPurpose);
|
2013-12-12 10:12:27 +01:00
|
|
|
} else if(index.column() == Address) {
|
2017-08-22 18:02:33 -07:00
|
|
|
CTxDestination newAddress = DecodeDestination(value.toString().toStdString());
|
2013-12-12 10:12:27 +01:00
|
|
|
// Refuse to set invalid address, set error status and return false
|
|
|
|
if(boost::get<CNoDestination>(&newAddress))
|
2013-01-08 08:17:58 +01:00
|
|
|
{
|
2013-12-12 10:12:27 +01:00
|
|
|
editStatus = INVALID_ADDRESS;
|
2013-01-08 08:17:58 +01:00
|
|
|
return false;
|
|
|
|
}
|
2013-12-12 10:12:27 +01:00
|
|
|
// Do nothing, if old address == new address
|
|
|
|
else if(newAddress == curAddress)
|
2011-07-16 19:01:05 +02:00
|
|
|
{
|
2013-12-12 10:12:27 +01:00
|
|
|
editStatus = NO_CHANGES;
|
2011-07-02 17:31:27 +02:00
|
|
|
return false;
|
2011-07-16 19:01:05 +02:00
|
|
|
}
|
2013-01-08 08:17:58 +01:00
|
|
|
// Check for duplicate addresses to prevent accidental deletion of addresses, if you try
|
|
|
|
// to paste an existing address over another address (with a different label)
|
2018-04-10 11:50:10 -04:00
|
|
|
if (walletModel->wallet().getAddress(
|
|
|
|
newAddress, /* name= */ nullptr, /* is_mine= */ nullptr, /* purpose= */ nullptr))
|
2013-01-08 08:17:58 +01:00
|
|
|
{
|
|
|
|
editStatus = DUPLICATE_ADDRESS;
|
|
|
|
return false;
|
|
|
|
}
|
2011-08-08 17:38:17 +02:00
|
|
|
// Double-check that we're not overwriting a receiving address
|
2013-01-08 08:17:58 +01:00
|
|
|
else if(rec->type == AddressTableEntry::Sending)
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
2013-12-12 10:12:27 +01:00
|
|
|
// Remove old entry
|
2017-04-18 13:01:23 -04:00
|
|
|
walletModel->wallet().delAddressBook(curAddress);
|
2013-12-12 10:12:27 +01:00
|
|
|
// Add new entry with new address
|
2017-04-18 13:01:23 -04:00
|
|
|
walletModel->wallet().setAddressBook(newAddress, value.toString().toStdString(), strPurpose);
|
2011-06-03 15:16:11 +02:00
|
|
|
}
|
2011-06-02 17:48:45 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-05-13 15:58:27 +02:00
|
|
|
QVariant AddressTableModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
if(orientation == Qt::Horizontal)
|
|
|
|
{
|
2013-11-05 18:03:05 +01:00
|
|
|
if(role == Qt::DisplayRole && section < columns.size())
|
2011-06-01 15:50:09 +02:00
|
|
|
{
|
|
|
|
return columns[section];
|
|
|
|
}
|
|
|
|
}
|
2011-05-13 15:58:27 +02:00
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2013-01-08 08:17:58 +01:00
|
|
|
Qt::ItemFlags AddressTableModel::flags(const QModelIndex &index) const
|
2011-07-02 17:31:27 +02:00
|
|
|
{
|
2018-07-31 14:02:34 -04:00
|
|
|
if (!index.isValid()) return Qt::NoItemFlags;
|
|
|
|
|
2011-07-02 17:31:27 +02:00
|
|
|
AddressTableEntry *rec = static_cast<AddressTableEntry*>(index.internalPointer());
|
|
|
|
|
|
|
|
Qt::ItemFlags retval = Qt::ItemIsSelectable | Qt::ItemIsEnabled;
|
|
|
|
// Can edit address and label for sending addresses,
|
|
|
|
// and only label for receiving addresses.
|
|
|
|
if(rec->type == AddressTableEntry::Sending ||
|
|
|
|
(rec->type == AddressTableEntry::Receiving && index.column()==Label))
|
|
|
|
{
|
|
|
|
retval |= Qt::ItemIsEditable;
|
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2013-01-08 08:17:58 +01:00
|
|
|
QModelIndex AddressTableModel::index(int row, int column, const QModelIndex &parent) const
|
2011-05-13 15:58:27 +02:00
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
Q_UNUSED(parent);
|
|
|
|
AddressTableEntry *data = priv->index(row);
|
|
|
|
if(data)
|
|
|
|
{
|
|
|
|
return createIndex(row, column, priv->index(row));
|
2011-06-07 18:59:01 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-06-01 15:50:09 +02:00
|
|
|
return QModelIndex();
|
|
|
|
}
|
2011-05-09 20:44:46 +02:00
|
|
|
}
|
2011-06-01 15:50:09 +02:00
|
|
|
|
2013-08-29 16:19:43 +02:00
|
|
|
void AddressTableModel::updateEntry(const QString &address,
|
|
|
|
const QString &label, bool isMine, const QString &purpose, int status)
|
2011-06-01 20:08:21 +02:00
|
|
|
{
|
2011-08-08 17:38:17 +02:00
|
|
|
// Update address book model from Bitcoin core
|
2013-08-29 16:19:43 +02:00
|
|
|
priv->updateEntry(address, label, isMine, purpose, status);
|
2011-06-01 20:08:21 +02:00
|
|
|
}
|
2011-06-03 15:16:11 +02:00
|
|
|
|
2018-01-16 20:11:40 +00:00
|
|
|
QString AddressTableModel::addRow(const QString &type, const QString &label, const QString &address, const OutputType address_type)
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
|
|
|
std::string strLabel = label.toStdString();
|
|
|
|
std::string strAddress = address.toStdString();
|
|
|
|
|
2011-07-16 19:01:05 +02:00
|
|
|
editStatus = OK;
|
|
|
|
|
2011-06-03 15:16:11 +02:00
|
|
|
if(type == Send)
|
|
|
|
{
|
2011-07-16 19:01:05 +02:00
|
|
|
if(!walletModel->validateAddress(address))
|
|
|
|
{
|
|
|
|
editStatus = INVALID_ADDRESS;
|
|
|
|
return QString();
|
|
|
|
}
|
2011-08-08 17:38:17 +02:00
|
|
|
// Check for duplicate addresses
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
2018-04-10 11:50:10 -04:00
|
|
|
if (walletModel->wallet().getAddress(
|
|
|
|
DecodeDestination(strAddress), /* name= */ nullptr, /* is_mine= */ nullptr, /* purpose= */ nullptr))
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
2011-07-16 19:01:05 +02:00
|
|
|
editStatus = DUPLICATE_ADDRESS;
|
2011-06-03 21:03:20 +02:00
|
|
|
return QString();
|
2011-06-03 15:16:11 +02:00
|
|
|
}
|
|
|
|
}
|
2019-06-18 15:19:13 -04:00
|
|
|
|
|
|
|
// Add entry
|
|
|
|
walletModel->wallet().setAddressBook(DecodeDestination(strAddress), strLabel, "send");
|
2011-06-07 18:59:01 +02:00
|
|
|
}
|
|
|
|
else if(type == Receive)
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
2011-07-26 15:38:31 +02:00
|
|
|
// Generate a new address to associate with given label
|
2019-06-18 15:19:13 -04:00
|
|
|
CTxDestination dest;
|
|
|
|
if(!walletModel->wallet().getNewDestination(address_type, strLabel, dest))
|
2011-09-02 18:02:22 +02:00
|
|
|
{
|
2013-08-15 21:53:21 -07:00
|
|
|
WalletModel::UnlockContext ctx(walletModel->requestUnlock());
|
|
|
|
if(!ctx.isValid())
|
|
|
|
{
|
|
|
|
// Unlock wallet failed or was cancelled
|
|
|
|
editStatus = WALLET_UNLOCK_FAILURE;
|
|
|
|
return QString();
|
|
|
|
}
|
2019-06-18 15:19:13 -04:00
|
|
|
if(!walletModel->wallet().getNewDestination(address_type, strLabel, dest))
|
2013-08-15 21:53:21 -07:00
|
|
|
{
|
|
|
|
editStatus = KEY_GENERATION_FAILURE;
|
|
|
|
return QString();
|
|
|
|
}
|
2011-09-02 18:02:22 +02:00
|
|
|
}
|
2019-06-18 15:19:13 -04:00
|
|
|
strAddress = EncodeDestination(dest);
|
2011-06-07 18:59:01 +02:00
|
|
|
}
|
|
|
|
else
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
2011-06-03 21:03:20 +02:00
|
|
|
return QString();
|
2011-06-03 15:16:11 +02:00
|
|
|
}
|
2011-06-03 21:03:20 +02:00
|
|
|
return QString::fromStdString(strAddress);
|
2011-06-03 15:16:11 +02:00
|
|
|
}
|
|
|
|
|
2013-01-08 08:17:58 +01:00
|
|
|
bool AddressTableModel::removeRows(int row, int count, const QModelIndex &parent)
|
2011-06-03 15:16:11 +02:00
|
|
|
{
|
|
|
|
Q_UNUSED(parent);
|
|
|
|
AddressTableEntry *rec = priv->index(row);
|
|
|
|
if(count != 1 || !rec || rec->type == AddressTableEntry::Receiving)
|
|
|
|
{
|
2011-06-18 11:53:25 +02:00
|
|
|
// Can only remove one row at a time, and cannot remove rows not in model.
|
|
|
|
// Also refuse to remove receiving addresses.
|
2011-06-03 15:16:11 +02:00
|
|
|
return false;
|
|
|
|
}
|
2017-04-18 13:01:23 -04:00
|
|
|
walletModel->wallet().delAddressBook(DecodeDestination(rec->address.toStdString()));
|
2011-06-03 15:16:11 +02:00
|
|
|
return true;
|
|
|
|
}
|
2011-06-21 20:34:43 +02:00
|
|
|
|
2011-07-08 22:27:36 +02:00
|
|
|
QString AddressTableModel::labelForAddress(const QString &address) const
|
|
|
|
{
|
2018-04-10 16:27:40 -04:00
|
|
|
std::string name;
|
|
|
|
if (getAddressData(address, &name, /* purpose= */ nullptr)) {
|
|
|
|
return QString::fromStdString(name);
|
|
|
|
}
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AddressTableModel::purposeForAddress(const QString &address) const
|
|
|
|
{
|
|
|
|
std::string purpose;
|
|
|
|
if (getAddressData(address, /* name= */ nullptr, &purpose)) {
|
|
|
|
return QString::fromStdString(purpose);
|
2011-07-08 22:27:36 +02:00
|
|
|
}
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2018-04-10 16:27:40 -04:00
|
|
|
bool AddressTableModel::getAddressData(const QString &address,
|
|
|
|
std::string* name,
|
|
|
|
std::string* purpose) const {
|
|
|
|
CTxDestination destination = DecodeDestination(address.toStdString());
|
|
|
|
return walletModel->wallet().getAddress(destination, name, /* is_mine= */ nullptr, purpose);
|
|
|
|
}
|
|
|
|
|
2011-07-08 22:27:36 +02:00
|
|
|
int AddressTableModel::lookupAddress(const QString &address) const
|
|
|
|
{
|
|
|
|
QModelIndexList lst = match(index(0, Address, QModelIndex()),
|
|
|
|
Qt::EditRole, address, 1, Qt::MatchExactly);
|
|
|
|
if(lst.isEmpty())
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return lst.at(0).row();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-18 13:01:23 -04:00
|
|
|
OutputType AddressTableModel::GetDefaultAddressType() const { return walletModel->wallet().getDefaultAddressType(); };
|
2018-02-10 21:06:35 -05:00
|
|
|
|
2012-05-06 22:41:35 +02:00
|
|
|
void AddressTableModel::emitDataChanged(int idx)
|
|
|
|
{
|
2015-07-14 13:59:05 +02:00
|
|
|
Q_EMIT dataChanged(index(idx, 0, QModelIndex()), index(idx, columns.length()-1, QModelIndex()));
|
2012-05-06 22:41:35 +02:00
|
|
|
}
|