bitcoin/gui/src/transactiontablemodel.cpp

328 lines
9.1 KiB
C++
Raw Normal View History

2011-05-12 08:49:42 -04:00
#include "transactiontablemodel.h"
2011-05-27 12:38:30 -04:00
#include "guiutil.h"
#include "transactionrecord.h"
#include "main.h"
2011-05-08 10:30:10 -04:00
2011-05-10 13:03:10 -04:00
#include <QLocale>
2011-05-27 02:20:23 -04:00
#include <QDebug>
#include <QList>
#include <QColor>
2011-05-10 13:03:10 -04:00
const QString TransactionTableModel::Sent = "s";
const QString TransactionTableModel::Received = "r";
2011-05-27 12:38:30 -04:00
const QString TransactionTableModel::Other = "o";
2011-05-28 10:09:23 -04:00
/* Private implementation, no need to pull this into header */
2011-05-27 02:20:23 -04:00
class TransactionTableImpl
{
public:
QList<TransactionRecord> cachedWallet;
/* Update our model of the wallet */
void updateWallet()
{
QList<int> insertedIndices;
QList<int> removedIndices;
cachedWallet.clear();
/* Query wallet from core, and compare with our own
representation.
*/
CRITICAL_BLOCK(cs_mapWallet)
{
for(std::map<uint256, CWalletTx>::iterator it = mapWallet.begin(); it != mapWallet.end(); ++it)
{
/* TODO: Make note of new and removed transactions */
/* insertedIndices */
/* removedIndices */
cachedWallet.append(TransactionRecord::decomposeTransaction(it->second));
2011-05-27 02:20:23 -04:00
}
}
/* beginInsertRows(QModelIndex(), first, last) */
/* endInsertRows */
/* beginRemoveRows(QModelIndex(), first, last) */
/* beginEndRows */
}
int size()
{
return cachedWallet.size();
}
TransactionRecord *index(int idx)
{
if(idx >= 0 && idx < cachedWallet.size())
{
return &cachedWallet[idx];
} else {
return 0;
}
}
};
2011-05-08 16:23:31 -04:00
/* Credit and Debit columns are right-aligned as they contain numbers */
2011-05-10 13:03:10 -04:00
static int column_alignments[] = {
Qt::AlignLeft|Qt::AlignVCenter,
Qt::AlignLeft|Qt::AlignVCenter,
Qt::AlignLeft|Qt::AlignVCenter,
Qt::AlignRight|Qt::AlignVCenter,
Qt::AlignRight|Qt::AlignVCenter,
Qt::AlignLeft|Qt::AlignVCenter
2011-05-08 16:23:31 -04:00
};
2011-05-08 10:30:10 -04:00
TransactionTableModel::TransactionTableModel(QObject *parent):
2011-05-27 02:20:23 -04:00
QAbstractTableModel(parent),
impl(new TransactionTableImpl())
2011-05-08 10:30:10 -04:00
{
columns << tr("Status") << tr("Date") << tr("Description") << tr("Debit") << tr("Credit");
2011-05-27 02:20:23 -04:00
impl->updateWallet();
}
TransactionTableModel::~TransactionTableModel()
{
delete impl;
2011-05-08 10:30:10 -04:00
}
void TransactionTableModel::updateWallet()
{
2011-05-28 10:09:23 -04:00
/* TODO: improve this, way too brute-force at the moment,
only update transactions that actually changed, and add/remove
transactions that were added/removed.
*/
beginResetModel();
impl->updateWallet();
endResetModel();
}
2011-05-27 02:20:23 -04:00
2011-05-08 10:30:10 -04:00
int TransactionTableModel::rowCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
2011-05-27 02:20:23 -04:00
return impl->size();
2011-05-08 10:30:10 -04:00
}
int TransactionTableModel::columnCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return columns.length();
}
2011-05-27 02:20:23 -04:00
QVariant TransactionTableModel::formatTxStatus(const TransactionRecord *wtx) const
{
2011-05-27 12:38:30 -04:00
QString status;
2011-05-27 14:36:58 -04:00
2011-05-27 12:38:30 -04:00
switch(wtx->status.status)
2011-05-27 02:20:23 -04:00
{
2011-05-27 12:38:30 -04:00
case TransactionStatus::OpenUntilBlock:
status = tr("Open for %n block(s)","",wtx->status.open_for);
break;
case TransactionStatus::OpenUntilDate:
status = tr("Open until ") + DateTimeStr(wtx->status.open_for);
break;
case TransactionStatus::Offline:
status = tr("%1/offline").arg(wtx->status.depth);
break;
case TransactionStatus::Unconfirmed:
status = tr("%1/unconfirmed").arg(wtx->status.depth);
break;
case TransactionStatus::HaveConfirmations:
status = tr("%1 confirmations").arg(wtx->status.depth);
break;
2011-05-27 02:20:23 -04:00
}
2011-05-27 12:38:30 -04:00
return QVariant(status);
2011-05-27 02:20:23 -04:00
}
QVariant TransactionTableModel::formatTxDate(const TransactionRecord *wtx) const
{
2011-05-27 12:38:30 -04:00
if(wtx->time)
{
return QVariant(DateTimeStr(wtx->time));
} else {
return QVariant();
}
2011-05-27 02:20:23 -04:00
}
2011-05-27 14:36:58 -04:00
/* Look up address in address book, if found return
address[0:12]... (label)
otherwise just return address
*/
std::string lookupAddress(const std::string &address)
{
std::string description;
CRITICAL_BLOCK(cs_mapAddressBook)
{
std::map<std::string, std::string>::iterator mi = mapAddressBook.find(address);
if (mi != mapAddressBook.end() && !(*mi).second.empty())
{
std::string label = (*mi).second;
description += address.substr(0,12) + "... ";
description += "(" + label + ")";
}
else
description += address;
}
return description;
}
2011-05-27 02:20:23 -04:00
QVariant TransactionTableModel::formatTxDescription(const TransactionRecord *wtx) const
{
QString description;
switch(wtx->type)
{
case TransactionRecord::RecvFromAddress:
2011-05-27 14:36:58 -04:00
description = tr("From: ") + QString::fromStdString(lookupAddress(wtx->address));
break;
case TransactionRecord::RecvFromIP:
description = tr("From IP: ") + QString::fromStdString(wtx->address);
break;
case TransactionRecord::SendToAddress:
2011-05-27 14:36:58 -04:00
description = tr("To: ") + QString::fromStdString(lookupAddress(wtx->address));
break;
case TransactionRecord::SendToIP:
description = tr("To IP: ") + QString::fromStdString(wtx->address);
break;
case TransactionRecord::SendToSelf:
description = tr("Payment to yourself");
break;
case TransactionRecord::Generated:
2011-05-27 16:06:30 -04:00
switch(wtx->status.maturity)
{
case TransactionStatus::Immature:
description = tr("Generated (matures in %n more blocks)", "",
wtx->status.matures_in);
break;
case TransactionStatus::Mature:
description = tr("Generated");
break;
case TransactionStatus::MaturesWarning:
description = tr("Generated - Warning: This block was not received by any other nodes and will probably not be accepted!");
break;
case TransactionStatus::NotAccepted:
description = tr("Generated (not accepted)");
break;
}
break;
}
return QVariant(description);
2011-05-27 02:20:23 -04:00
}
QVariant TransactionTableModel::formatTxDebit(const TransactionRecord *wtx) const
{
2011-05-27 12:38:30 -04:00
if(wtx->debit)
{
QString str = QString::fromStdString(FormatMoney(wtx->debit));
2011-05-28 10:09:23 -04:00
if(!wtx->status.confirmed || wtx->status.maturity != TransactionStatus::Mature)
2011-05-27 12:38:30 -04:00
{
str = QString("[") + str + QString("]");
}
return QVariant(str);
} else {
return QVariant();
}
2011-05-27 02:20:23 -04:00
}
QVariant TransactionTableModel::formatTxCredit(const TransactionRecord *wtx) const
{
2011-05-27 12:38:30 -04:00
if(wtx->credit)
{
QString str = QString::fromStdString(FormatMoney(wtx->credit));
2011-05-28 10:09:23 -04:00
if(!wtx->status.confirmed || wtx->status.maturity != TransactionStatus::Mature)
2011-05-27 12:38:30 -04:00
{
str = QString("[") + str + QString("]");
}
return QVariant(str);
} else {
return QVariant();
}
2011-05-27 02:20:23 -04:00
}
2011-05-08 10:30:10 -04:00
QVariant TransactionTableModel::data(const QModelIndex &index, int role) const
{
if(!index.isValid())
return QVariant();
2011-05-27 02:20:23 -04:00
TransactionRecord *rec = static_cast<TransactionRecord*>(index.internalPointer());
2011-05-08 10:30:10 -04:00
if(role == Qt::DisplayRole)
{
2011-05-28 10:09:23 -04:00
/* Delegate to specific column handlers */
2011-05-27 02:20:23 -04:00
switch(index.column())
{
case Status:
return formatTxStatus(rec);
case Date:
return formatTxDate(rec);
case Description:
return formatTxDescription(rec);
case Debit:
return formatTxDebit(rec);
case Credit:
return formatTxCredit(rec);
}
2011-05-08 16:23:31 -04:00
} else if (role == Qt::TextAlignmentRole)
{
return column_alignments[index.column()];
} else if (role == Qt::ForegroundRole)
{
2011-05-28 10:09:23 -04:00
/* Non-confirmed transactions are grey */
if(rec->status.confirmed)
{
return QColor(0, 0, 0);
} else {
return QColor(128, 128, 128);
}
} else if (role == TypeRole)
2011-05-10 13:03:10 -04:00
{
2011-05-28 10:09:23 -04:00
/* Role for filtering tabs by type */
switch(rec->type)
2011-05-10 13:03:10 -04:00
{
case TransactionRecord::RecvFromAddress:
case TransactionRecord::RecvFromIP:
return TransactionTableModel::Received;
case TransactionRecord::SendToAddress:
case TransactionRecord::SendToIP:
case TransactionRecord::SendToSelf:
return TransactionTableModel::Sent;
default:
return TransactionTableModel::Other;
2011-05-10 13:03:10 -04:00
}
2011-05-08 10:30:10 -04:00
}
return QVariant();
}
QVariant TransactionTableModel::headerData(int section, Qt::Orientation orientation, int role) const
{
2011-05-28 10:09:23 -04:00
if(orientation == Qt::Horizontal)
2011-05-08 16:23:31 -04:00
{
2011-05-28 10:09:23 -04:00
if(role == Qt::DisplayRole)
2011-05-08 16:23:31 -04:00
{
return columns[section];
2011-05-28 10:09:23 -04:00
} else if (role == Qt::TextAlignmentRole)
{
return column_alignments[section];
2011-05-08 16:23:31 -04:00
}
2011-05-08 10:30:10 -04:00
}
return QVariant();
}
Qt::ItemFlags TransactionTableModel::flags(const QModelIndex &index) const
{
return QAbstractTableModel::flags(index);
}
2011-05-27 02:20:23 -04:00
QModelIndex TransactionTableModel::index ( int row, int column, const QModelIndex & parent ) const
{
Q_UNUSED(parent);
TransactionRecord *data = impl->index(row);
if(data)
{
return createIndex(row, column, impl->index(row));
} else {
return QModelIndex();
}
}