// Copyright (c) 2011-2014 The Bitcoin developers // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include "bitcoinamountfield.h" #include "bitcoinunits.h" #include "guiconstants.h" #include "qvaluecombobox.h" #include #include #include #include #include // for qPow() // QDoubleSpinBox that shows SI-style thin space thousands separators class AmountSpinBox: public QDoubleSpinBox { public: explicit AmountSpinBox(QWidget *parent): QDoubleSpinBox(parent) { } QString textFromValue(double value) const { QStringList parts = QDoubleSpinBox::textFromValue(value).split("."); QString quotient_str = parts[0]; QString remainder_str; if(parts.size() > 1) remainder_str = parts[1]; // Code duplication between here and BitcoinUnits::format // TODO: Figure out how to share this code QChar thin_sp(THIN_SP_CP); int q_size = quotient_str.size(); if (q_size > 4) for (int i = 3; i < q_size; i += 3) quotient_str.insert(q_size - i, thin_sp); int r_size = remainder_str.size(); if (r_size > 4) for (int i = 3, adj = 0; i < r_size; i += 3, adj++) remainder_str.insert(i + adj, thin_sp); if(remainder_str.isEmpty()) return quotient_str; else return quotient_str + QString(".") + remainder_str; } QValidator::State validate (QString &text, int &pos) const { QString s(BitcoinUnits::removeSpaces(text)); return QDoubleSpinBox::validate(s, pos); } double valueFromText(const QString& text) const { return QDoubleSpinBox::valueFromText(BitcoinUnits::removeSpaces(text)); } }; BitcoinAmountField::BitcoinAmountField(QWidget *parent) : QWidget(parent), amount(0), currentUnit(-1) { nSingleStep = 100000; // satoshis amount = new AmountSpinBox(this); amount->setLocale(QLocale::c()); amount->installEventFilter(this); amount->setMaximumWidth(170); QHBoxLayout *layout = new QHBoxLayout(this); layout->addWidget(amount); unit = new QValueComboBox(this); unit->setModel(new BitcoinUnits(this)); layout->addWidget(unit); layout->addStretch(1); layout->setContentsMargins(0,0,0,0); setLayout(layout); setFocusPolicy(Qt::TabFocus); setFocusProxy(amount); // If one if the widgets changes, the combined content changes as well connect(amount, SIGNAL(valueChanged(QString)), this, SIGNAL(textChanged())); connect(unit, SIGNAL(currentIndexChanged(int)), this, SLOT(unitChanged(int))); // Set default based on configuration unitChanged(unit->currentIndex()); } void BitcoinAmountField::setText(const QString &text) { if (text.isEmpty()) amount->clear(); else amount->setValue(BitcoinUnits::removeSpaces(text).toDouble()); } void BitcoinAmountField::clear() { amount->clear(); unit->setCurrentIndex(0); } bool BitcoinAmountField::validate() { bool valid = true; if (amount->value() == 0.0) valid = false; else if (!BitcoinUnits::parse(currentUnit, text(), 0)) valid = false; else if (amount->value() > BitcoinUnits::maxAmount(currentUnit)) valid = false; setValid(valid); return valid; } void BitcoinAmountField::setValid(bool valid) { if (valid) amount->setStyleSheet(""); else amount->setStyleSheet(STYLE_INVALID); } QString BitcoinAmountField::text() const { if (amount->text().isEmpty()) return QString(); else return amount->text(); } bool BitcoinAmountField::eventFilter(QObject *object, QEvent *event) { if (event->type() == QEvent::FocusIn) { // Clear invalid flag on focus setValid(true); } else if (event->type() == QEvent::KeyPress || event->type() == QEvent::KeyRelease) { QKeyEvent *keyEvent = static_cast(event); if (keyEvent->key() == Qt::Key_Comma) { // Translate a comma into a period QKeyEvent periodKeyEvent(event->type(), Qt::Key_Period, keyEvent->modifiers(), ".", keyEvent->isAutoRepeat(), keyEvent->count()); QApplication::sendEvent(object, &periodKeyEvent); return true; } } return QWidget::eventFilter(object, event); } QWidget *BitcoinAmountField::setupTabChain(QWidget *prev) { QWidget::setTabOrder(prev, amount); QWidget::setTabOrder(amount, unit); return unit; } 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::setReadOnly(bool fReadOnly) { amount->setReadOnly(fReadOnly); unit->setEnabled(!fReadOnly); } 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->setDecimals(BitcoinUnits::decimals(currentUnit)); amount->setMaximum(qPow(10, BitcoinUnits::amountDigits(currentUnit)) - qPow(10, -amount->decimals())); amount->setSingleStep((double)nSingleStep / (double)BitcoinUnits::factor(currentUnit)); if (valid) { // If value was valid, re-place it in the widget with the new unit setValue(currentValue); } else { // If current value is invalid, just clear field setText(""); } setValid(true); } void BitcoinAmountField::setDisplayUnit(int newUnit) { unit->setValue(newUnit); } void BitcoinAmountField::setSingleStep(qint64 step) { nSingleStep = step; unitChanged(unit->currentIndex()); }