2011-06-20 21:31:42 +02:00
|
|
|
#include "bitcoinamountfield.h"
|
2011-07-16 19:01:05 +02:00
|
|
|
#include "qvalidatedlineedit.h"
|
2011-07-25 21:35:45 +02:00
|
|
|
#include "bitcoinunits.h"
|
2011-06-20 21:31:42 +02:00
|
|
|
|
|
|
|
#include <QLabel>
|
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QRegExpValidator>
|
|
|
|
#include <QHBoxLayout>
|
|
|
|
#include <QKeyEvent>
|
2011-07-26 13:08:34 +02:00
|
|
|
#include <QComboBox>
|
2011-06-20 21:31:42 +02:00
|
|
|
|
|
|
|
BitcoinAmountField::BitcoinAmountField(QWidget *parent):
|
2011-07-26 13:08:34 +02:00
|
|
|
QWidget(parent), amount(0), decimals(0), currentUnit(-1)
|
2011-06-20 21:31:42 +02:00
|
|
|
{
|
2011-07-16 19:01:05 +02:00
|
|
|
amount = new QValidatedLineEdit(this);
|
2011-07-25 21:35:45 +02:00
|
|
|
amount->setValidator(new QRegExpValidator(QRegExp("[0-9]*"), this));
|
2011-06-20 21:31:42 +02:00
|
|
|
amount->setAlignment(Qt::AlignRight|Qt::AlignVCenter);
|
|
|
|
amount->installEventFilter(this);
|
2011-06-25 23:14:10 +02:00
|
|
|
amount->setMaximumWidth(100);
|
2011-07-16 19:01:05 +02:00
|
|
|
decimals = new QValidatedLineEdit(this);
|
2011-06-20 21:31:42 +02:00
|
|
|
decimals->setValidator(new QRegExpValidator(QRegExp("[0-9]+"), this));
|
|
|
|
decimals->setAlignment(Qt::AlignLeft|Qt::AlignVCenter);
|
|
|
|
decimals->setMaximumWidth(75);
|
|
|
|
|
|
|
|
QHBoxLayout *layout = new QHBoxLayout(this);
|
|
|
|
layout->setSpacing(0);
|
|
|
|
layout->addWidget(amount);
|
|
|
|
layout->addWidget(new QLabel(QString(".")));
|
|
|
|
layout->addWidget(decimals);
|
2011-07-26 13:08:34 +02:00
|
|
|
unit = new QComboBox(this);
|
|
|
|
unit->setModel(new BitcoinUnits(this));
|
|
|
|
layout->addWidget(unit);
|
2011-06-20 21:31:42 +02:00
|
|
|
layout->addStretch(1);
|
2011-06-21 07:35:59 +02:00
|
|
|
layout->setContentsMargins(0,0,0,0);
|
2011-06-20 21:31:42 +02:00
|
|
|
|
|
|
|
setLayout(layout);
|
2011-07-16 19:01:05 +02:00
|
|
|
|
|
|
|
setFocusPolicy(Qt::TabFocus);
|
2011-06-20 21:31:42 +02:00
|
|
|
setFocusProxy(amount);
|
|
|
|
|
|
|
|
// If one if the widgets changes, the combined content changes as well
|
|
|
|
connect(amount, SIGNAL(textChanged(QString)), this, SIGNAL(textChanged()));
|
|
|
|
connect(decimals, SIGNAL(textChanged(QString)), this, SIGNAL(textChanged()));
|
2011-07-26 13:08:34 +02:00
|
|
|
connect(unit, SIGNAL(currentIndexChanged(int)), this, SLOT(unitChanged(int)));
|
|
|
|
|
|
|
|
// TODO: set default based on configuration
|
|
|
|
unitChanged(unit->currentIndex());
|
2011-06-20 21:31:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void BitcoinAmountField::setText(const QString &text)
|
|
|
|
{
|
|
|
|
const QStringList parts = text.split(QString("."));
|
|
|
|
if(parts.size() == 2)
|
|
|
|
{
|
|
|
|
amount->setText(parts[0]);
|
|
|
|
decimals->setText(parts[1]);
|
|
|
|
}
|
2011-07-07 17:33:15 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
amount->setText(QString());
|
|
|
|
decimals->setText(QString());
|
|
|
|
}
|
2011-06-20 21:31:42 +02:00
|
|
|
}
|
|
|
|
|
2011-07-22 17:06:37 +02:00
|
|
|
void BitcoinAmountField::clear()
|
|
|
|
{
|
|
|
|
amount->clear();
|
|
|
|
decimals->clear();
|
|
|
|
}
|
|
|
|
|
2011-07-16 19:01:05 +02:00
|
|
|
bool BitcoinAmountField::validate()
|
|
|
|
{
|
|
|
|
bool valid = true;
|
|
|
|
if(decimals->text().isEmpty())
|
|
|
|
{
|
|
|
|
decimals->setValid(false);
|
|
|
|
valid = false;
|
|
|
|
}
|
2011-07-25 21:35:45 +02:00
|
|
|
if(!BitcoinUnits::parse(BitcoinUnits::BTC, text(), 0))
|
|
|
|
{
|
2011-07-26 13:08:34 +02:00
|
|
|
setValid(false);
|
2011-07-25 21:35:45 +02:00
|
|
|
valid = false;
|
|
|
|
}
|
|
|
|
|
2011-07-16 19:01:05 +02:00
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
2011-07-26 13:08:34 +02:00
|
|
|
void BitcoinAmountField::setValid(bool valid)
|
|
|
|
{
|
|
|
|
amount->setValid(valid);
|
|
|
|
decimals->setValid(valid);
|
|
|
|
}
|
|
|
|
|
2011-06-20 21:31:42 +02:00
|
|
|
QString BitcoinAmountField::text() const
|
|
|
|
{
|
2011-07-26 13:08:34 +02:00
|
|
|
if(decimals->text().isEmpty() && amount->text().isEmpty())
|
2011-07-16 19:01:05 +02:00
|
|
|
{
|
2011-06-30 17:32:19 +02:00
|
|
|
return QString();
|
2011-07-16 19:01:05 +02:00
|
|
|
}
|
2011-06-20 21:31:42 +02:00
|
|
|
return amount->text() + QString(".") + decimals->text();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Intercept '.' and ',' keys, if pressed focus a specified widget
|
|
|
|
bool BitcoinAmountField::eventFilter(QObject *object, QEvent *event)
|
|
|
|
{
|
|
|
|
Q_UNUSED(object);
|
|
|
|
if(event->type() == QEvent::KeyPress)
|
|
|
|
{
|
|
|
|
QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
|
|
|
|
if(keyEvent->key() == Qt::Key_Period || keyEvent->key() == Qt::Key_Comma)
|
|
|
|
{
|
|
|
|
decimals->setFocus();
|
2011-06-21 19:14:35 +02:00
|
|
|
decimals->selectAll();
|
2011-06-20 21:31:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2011-07-16 19:01:05 +02:00
|
|
|
|
|
|
|
QWidget *BitcoinAmountField::setupTabChain(QWidget *prev)
|
|
|
|
{
|
|
|
|
QWidget::setTabOrder(prev, amount);
|
|
|
|
QWidget::setTabOrder(amount, decimals);
|
|
|
|
return decimals;
|
|
|
|
}
|
2011-07-26 13:08:34 +02:00
|
|
|
|
|
|
|
qint64 BitcoinAmountField::value(bool *valid_out) const
|
|
|
|
{
|
|
|
|
qint64 val_out = 0;
|
|
|
|
bool valid = BitcoinUnits::parse(currentUnit, text(), &val_out);
|
|
|
|
if(valid_out)
|
|
|
|
{
|
|
|
|
*valid_out = valid;
|
|
|
|
}
|
|
|
|
return val_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BitcoinAmountField::setValue(qint64 value)
|
|
|
|
{
|
|
|
|
setText(BitcoinUnits::format(currentUnit, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
void BitcoinAmountField::unitChanged(int idx)
|
|
|
|
{
|
|
|
|
// Use description tooltip for current unit for the combobox
|
|
|
|
unit->setToolTip(unit->itemData(idx, Qt::ToolTipRole).toString());
|
|
|
|
|
|
|
|
// Determine new unit ID
|
|
|
|
int newUnit = unit->itemData(idx, BitcoinUnits::UnitRole).toInt();
|
|
|
|
|
|
|
|
// Parse current value and convert to new unit
|
|
|
|
bool valid = false;
|
|
|
|
qint64 currentValue = value(&valid);
|
|
|
|
|
|
|
|
currentUnit = newUnit;
|
|
|
|
|
|
|
|
// Set max length after retrieving the value, to prevent truncation
|
|
|
|
amount->setMaxLength(BitcoinUnits::amountDigits(currentUnit));
|
|
|
|
decimals->setMaxLength(BitcoinUnits::decimals(currentUnit));
|
|
|
|
|
|
|
|
if(valid)
|
|
|
|
{
|
|
|
|
setValue(currentValue);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If current value is invalid, just clear field
|
|
|
|
setText("");
|
|
|
|
}
|
|
|
|
setValid(true);
|
|
|
|
|
|
|
|
|
|
|
|
}
|