bitcoin/src/qt/addresstablemodel.cpp

341 lines
8.9 KiB
C++
Raw Normal View History

2011-05-12 08:49:42 -04:00
#include "addresstablemodel.h"
2011-06-01 14:08:21 -04:00
#include "guiutil.h"
2011-07-16 13:01:05 -04:00
#include "walletmodel.h"
#include "headers.h"
2011-05-09 14:44:46 -04:00
#include <QFont>
2011-06-21 13:54:09 -04:00
#include <QColor>
const QString AddressTableModel::Send = "S";
const QString AddressTableModel::Receive = "R";
2011-06-01 09:50:09 -04:00
struct AddressTableEntry
{
enum Type {
Sending,
Receiving
};
2011-06-03 09:16:11 -04:00
2011-06-01 09:50:09 -04:00
Type type;
QString label;
QString address;
AddressTableEntry() {}
AddressTableEntry(Type type, const QString &label, const QString &address):
type(type), label(label), address(address) {}
};
// Private implementation
2011-06-01 09:50:09 -04:00
struct AddressTablePriv
{
CWallet *wallet;
2011-06-01 09:50:09 -04:00
QList<AddressTableEntry> cachedAddressTable;
AddressTablePriv(CWallet *wallet):
wallet(wallet) {}
2011-06-01 09:50:09 -04:00
void refreshAddressTable()
{
cachedAddressTable.clear();
CRITICAL_BLOCK(wallet->cs_wallet)
2011-06-01 09:50:09 -04:00
{
BOOST_FOREACH(const PAIRTYPE(CBitcoinAddress, std::string)& item, wallet->mapAddressBook)
2011-06-01 09:50:09 -04:00
{
const CBitcoinAddress& address = item.first;
const std::string& strName = item.second;
bool fMine = wallet->HaveKey(address);
2011-06-01 09:50:09 -04:00
cachedAddressTable.append(AddressTableEntry(fMine ? AddressTableEntry::Receiving : AddressTableEntry::Sending,
QString::fromStdString(strName),
QString::fromStdString(address.ToString())));
2011-06-01 09:50:09 -04:00
}
}
}
int size()
{
return cachedAddressTable.size();
}
AddressTableEntry *index(int idx)
{
if(idx >= 0 && idx < cachedAddressTable.size())
{
return &cachedAddressTable[idx];
2011-06-07 12:59:01 -04:00
}
else
{
2011-06-01 09:50:09 -04:00
return 0;
}
}
};
2011-07-16 13:01:05 -04:00
AddressTableModel::AddressTableModel(CWallet *wallet, WalletModel *parent) :
QAbstractTableModel(parent),walletModel(parent),wallet(wallet),priv(0)
2011-05-09 14:44:46 -04:00
{
2011-06-01 09:50:09 -04:00
columns << tr("Label") << tr("Address");
priv = new AddressTablePriv(wallet);
2011-06-01 09:50:09 -04:00
priv->refreshAddressTable();
}
2011-06-01 09:50:09 -04:00
AddressTableModel::~AddressTableModel()
{
delete priv;
}
int AddressTableModel::rowCount(const QModelIndex &parent) const
{
2011-05-27 02:20:23 -04:00
Q_UNUSED(parent);
2011-06-01 09:50:09 -04:00
return priv->size();
}
int AddressTableModel::columnCount(const QModelIndex &parent) const
{
2011-06-01 09:50:09 -04:00
Q_UNUSED(parent);
return columns.length();
}
QVariant AddressTableModel::data(const QModelIndex &index, int role) const
{
if(!index.isValid())
return QVariant();
2011-06-01 09:50:09 -04:00
AddressTableEntry *rec = static_cast<AddressTableEntry*>(index.internalPointer());
if(role == Qt::DisplayRole || role == Qt::EditRole)
{
2011-06-01 09:50:09 -04:00
switch(index.column())
{
case Label:
if(rec->label.isEmpty() && role == Qt::DisplayRole)
{
return tr("(no label)");
}
else
{
return rec->label;
}
2011-06-01 09:50:09 -04:00
case Address:
return rec->address;
}
2011-06-07 12:59:01 -04:00
}
else if (role == Qt::FontRole)
2011-06-01 14:08:21 -04:00
{
2011-06-21 13:54:09 -04:00
QFont font;
2011-06-01 14:08:21 -04:00
if(index.column() == Address)
{
2011-06-21 13:54:09 -04:00
font = GUIUtil::bitcoinAddressFont();
}
return font;
}
2011-06-07 12:59:01 -04:00
else if (role == TypeRole)
{
2011-06-01 09:50:09 -04:00
switch(rec->type)
{
2011-06-01 09:50:09 -04:00
case AddressTableEntry::Sending:
return Send;
case AddressTableEntry::Receiving:
return Receive;
default: break;
}
}
return QVariant();
}
bool AddressTableModel::setData(const QModelIndex & index, const QVariant & value, int role)
{
if(!index.isValid())
return false;
2011-06-03 09:16:11 -04:00
AddressTableEntry *rec = static_cast<AddressTableEntry*>(index.internalPointer());
2011-07-16 13:01:05 -04:00
editStatus = OK;
if(role == Qt::EditRole)
{
switch(index.column())
{
case Label:
wallet->SetAddressBookName(rec->address.toStdString(), value.toString().toStdString());
2011-06-03 09:16:11 -04:00
rec->label = value.toString();
break;
case Address:
// Refuse to set invalid address, set error status and return false
2011-07-16 13:01:05 -04:00
if(!walletModel->validateAddress(value.toString()))
{
editStatus = INVALID_ADDRESS;
return false;
2011-07-16 13:01:05 -04:00
}
// Double-check that we're not overwriting a receiving address
2011-06-03 09:16:11 -04:00
if(rec->type == AddressTableEntry::Sending)
{
CRITICAL_BLOCK(wallet->cs_wallet)
{
// Remove old entry
wallet->DelAddressBookName(rec->address.toStdString());
// Add new entry with new address
wallet->SetAddressBookName(value.toString().toStdString(), rec->label.toStdString());
}
2011-06-03 09:16:11 -04:00
rec->address = value.toString();
}
break;
}
2011-06-03 09:16:11 -04:00
emit dataChanged(index, index);
return true;
}
return false;
}
QVariant AddressTableModel::headerData(int section, Qt::Orientation orientation, int role) const
{
2011-06-01 09:50:09 -04:00
if(orientation == Qt::Horizontal)
{
if(role == Qt::DisplayRole)
{
return columns[section];
}
}
return QVariant();
}
Qt::ItemFlags AddressTableModel::flags(const QModelIndex & index) const
{
if(!index.isValid())
return 0;
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;
}
2011-06-01 14:08:21 -04:00
QModelIndex AddressTableModel::index(int row, int column, const QModelIndex & parent) const
{
2011-06-01 09:50:09 -04:00
Q_UNUSED(parent);
AddressTableEntry *data = priv->index(row);
if(data)
{
return createIndex(row, column, priv->index(row));
2011-06-07 12:59:01 -04:00
}
else
{
2011-06-01 09:50:09 -04:00
return QModelIndex();
}
2011-05-09 14:44:46 -04:00
}
2011-06-01 09:50:09 -04:00
void AddressTableModel::update()
2011-06-01 14:08:21 -04:00
{
// Update address book model from Bitcoin core
2011-06-01 14:08:21 -04:00
beginResetModel();
priv->refreshAddressTable();
endResetModel();
}
2011-06-03 09:16:11 -04:00
QString AddressTableModel::addRow(const QString &type, const QString &label, const QString &address)
2011-06-03 09:16:11 -04:00
{
std::string strLabel = label.toStdString();
std::string strAddress = address.toStdString();
2011-07-16 13:01:05 -04:00
editStatus = OK;
2011-06-03 09:16:11 -04:00
if(type == Send)
{
2011-07-16 13:01:05 -04:00
if(!walletModel->validateAddress(address))
{
editStatus = INVALID_ADDRESS;
return QString();
}
// Check for duplicate addresses
CRITICAL_BLOCK(wallet->cs_wallet)
2011-06-03 09:16:11 -04:00
{
if(wallet->mapAddressBook.count(strAddress))
2011-06-03 09:16:11 -04:00
{
2011-07-16 13:01:05 -04:00
editStatus = DUPLICATE_ADDRESS;
return QString();
2011-06-03 09:16:11 -04:00
}
}
2011-06-07 12:59:01 -04:00
}
else if(type == Receive)
2011-06-03 09:16:11 -04:00
{
// Generate a new address to associate with given label
WalletModel::UnlockContext ctx(walletModel->requestUnlock());
if(!ctx.isValid())
{
// Unlock wallet failed or was cancelled
editStatus = WALLET_UNLOCK_FAILURE;
return QString();
}
std::vector<unsigned char> newKey;
if(!wallet->GetKeyFromPool(newKey, true))
{
editStatus = KEY_GENERATION_FAILURE;
return QString();
}
strAddress = CBitcoinAddress(newKey).ToString();
2011-06-07 12:59:01 -04:00
}
else
2011-06-03 09:16:11 -04:00
{
return QString();
2011-06-03 09:16:11 -04:00
}
// Add entry
CRITICAL_BLOCK(wallet->cs_wallet)
wallet->SetAddressBookName(strAddress, strLabel);
return QString::fromStdString(strAddress);
2011-06-03 09:16:11 -04:00
}
bool AddressTableModel::removeRows(int row, int count, const QModelIndex & parent)
{
Q_UNUSED(parent);
AddressTableEntry *rec = priv->index(row);
if(count != 1 || !rec || rec->type == AddressTableEntry::Receiving)
{
// Can only remove one row at a time, and cannot remove rows not in model.
// Also refuse to remove receiving addresses.
2011-06-03 09:16:11 -04:00
return false;
}
CRITICAL_BLOCK(wallet->cs_wallet)
{
wallet->DelAddressBookName(rec->address.toStdString());
}
2011-06-03 09:16:11 -04:00
return true;
}
/* Look up label for address in address book, if not found return empty string.
*/
QString AddressTableModel::labelForAddress(const QString &address) const
{
CRITICAL_BLOCK(wallet->cs_wallet)
{
CBitcoinAddress address_parsed(address.toStdString());
std::map<CBitcoinAddress, std::string>::iterator mi = wallet->mapAddressBook.find(address_parsed);
if (mi != wallet->mapAddressBook.end())
{
return QString::fromStdString(mi->second);
}
}
return QString();
}
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();
}
}