feather/src/model/CoinsModel.cpp
2024-01-01 18:07:58 +01:00

271 lines
No EOL
7.7 KiB
C++

// SPDX-License-Identifier: BSD-3-Clause
// SPDX-FileCopyrightText: 2020-2024 The Monero Project
#include "CoinsModel.h"
#include "libwalletqt/rows/CoinsInfo.h"
#include "Coins.h"
#include "constants.h"
#include "utils/ColorScheme.h"
#include "utils/Icons.h"
#include "utils/Utils.h"
#include "libwalletqt/WalletManager.h"
#include <QBrush>
CoinsModel::CoinsModel(QObject *parent, Coins *coins)
: QAbstractTableModel(parent)
, m_coins(coins)
{
connect(m_coins, &Coins::refreshStarted, this, &CoinsModel::startReset);
connect(m_coins, &Coins::refreshFinished, this, &CoinsModel::endReset);
}
void CoinsModel::startReset(){
beginResetModel();
}
void CoinsModel::endReset(){
endResetModel();
}
Coins * CoinsModel::coins() const {
return m_coins;
}
int CoinsModel::rowCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return 0;
} else {
return m_coins->count();
}
}
int CoinsModel::columnCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return 0;
}
return ModelColumn::COUNT;
}
QVariant CoinsModel::data(const QModelIndex &index, int role) const
{
if (!m_coins) {
return QVariant();
}
if (!index.isValid() || index.row() < 0 || static_cast<quint64>(index.row()) >= m_coins->count())
return QVariant();
QVariant result;
bool found = m_coins->coin(index.row(), [this, &index, &result, &role](const CoinsInfo &cInfo) {
bool selected = cInfo.keyImageKnown() && m_selected.contains(cInfo.keyImage());
if(role == Qt::DisplayRole || role == Qt::EditRole || role == Qt::UserRole) {
result = parseTransactionInfo(cInfo, index.column(), role);
}
else if (role == Qt::BackgroundRole) {
if (cInfo.spent()) {
result = QBrush(ColorScheme::RED.asColor(true));
}
else if (cInfo.frozen()) {
result = QBrush(ColorScheme::BLUE.asColor(true));
}
else if (!cInfo.unlocked()) {
result = QBrush(ColorScheme::YELLOW.asColor(true));
}
else if (selected) {
result = QBrush(ColorScheme::GREEN.asColor(true));
}
}
else if (role == Qt::TextAlignmentRole) {
switch (index.column()) {
case Amount:
result = Qt::AlignRight;
}
}
else if (role == Qt::DecorationRole) {
switch (index.column()) {
case KeyImageKnown:
{
if (cInfo.keyImageKnown()) {
result = QVariant(icons()->icon("eye1.png"));
}
else {
result = QVariant(icons()->icon("eye_blind.png"));
}
}
}
}
else if (role == Qt::FontRole) {
switch(index.column()) {
case PubKey:
case TxID:
case Address:
result = Utils::getMonospaceFont();
}
}
else if (role == Qt::ToolTipRole) {
switch(index.column()) {
case KeyImageKnown:
{
if (cInfo.keyImageKnown()) {
result = "Key image known";
} else {
result = "Key image unknown. Outgoing transactions that include this output will not be detected.";
}
}
}
if (cInfo.frozen()) {
result = "Output is frozen.";
}
else if (!cInfo.unlocked()) {
result = "Output is locked (needs more confirmations)";
}
else if (cInfo.spent()) {
result = "Output is spent";
}
else if (selected) {
result = "Coin selected to be spent";
}
}
});
if (!found) {
qCritical("%s: internal error: no transaction info for index %d", __FUNCTION__, index.row());
}
return result;
}
QVariant CoinsModel::headerData(int section, Qt::Orientation orientation, int role) const
{
if (role != Qt::DisplayRole) {
return QVariant();
}
if (orientation == Qt::Horizontal)
{
switch(section) {
case PubKey:
return QString("Pub Key");
case TxID:
return QString("TxID");
case BlockHeight:
return QString("Height");
case Address:
return QString("Address");
case Label:
return QString("Label");
case SpentHeight:
return QString("Spent Height");
case Amount:
return QString("Amount");
case Spent:
return QString("Spent");
case Frozen:
return QString("Frozen");
default:
return QVariant();
}
}
return QVariant();
}
Qt::ItemFlags CoinsModel::flags(const QModelIndex &index) const
{
if (!index.isValid())
return Qt::ItemIsEnabled;
if (index.column() == Label)
return QAbstractTableModel::flags(index) | Qt::ItemIsEditable;
return QAbstractTableModel::flags(index);
}
bool CoinsModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
if (index.isValid() && role == Qt::EditRole) {
const int row = index.row();
QString pubkey;
bool found = m_coins->coin(index.row(), [this, &pubkey](const CoinsInfo &cInfo) {
pubkey = cInfo.pubKey();
});
if (!found) {
return false;
}
switch (index.column()) {
case Label:
m_coins->setDescription(pubkey, m_currentSubaddressAccount, value.toString());
emit descriptionChanged();
break;
default:
return false;
}
emit dataChanged(index, index, {Qt::DisplayRole, Qt::EditRole});
return true;
}
return false;
}
QVariant CoinsModel::parseTransactionInfo(const CoinsInfo &cInfo, int column, int role) const
{
switch (column)
{
case KeyImageKnown:
return "";
case PubKey:
return cInfo.pubKey().mid(0,8);
case TxID:
return cInfo.hash().mid(0, 8) + " ";
case BlockHeight:
return cInfo.blockHeight();
case Address:
return Utils::displayAddress(cInfo.address(), 1, "");
case Label: {
if (!cInfo.description().isEmpty())
return cInfo.description();
if (!cInfo.txNote().isEmpty())
return cInfo.txNote();
return cInfo.addressLabel();
}
case Spent:
return cInfo.spent();
case SpentHeight:
return cInfo.spentHeight();
case Amount:
{
if (role == Qt::UserRole) {
return cInfo.amount();
}
return cInfo.displayAmount();
}
case Frozen:
return cInfo.frozen();
default:
{
qCritical() << "Unimplemented role";
return QVariant();
}
}
}
void CoinsModel::setCurrentSubaddressAccount(quint32 accountIndex) {
m_currentSubaddressAccount = accountIndex;
}
void CoinsModel::setSelected(const QStringList &keyimages) {
m_selected.clear();
for (const auto &ki : keyimages) {
m_selected.insert(ki);
}
emit dataChanged(index(0, 0), index(rowCount() - 1, columnCount() - 1));
}
CoinsInfo* CoinsModel::entryFromIndex(const QModelIndex &index) const {
Q_ASSERT(index.isValid() && index.row() < m_coins->count());
return m_coins->coin(index.row());
}