/ src / qt / bitcoinunits.cpp
bitcoinunits.cpp
  1  // Copyright (c) 2011-present The Bitcoin Core developers
  2  // Distributed under the MIT software license, see the accompanying
  3  // file COPYING or http://www.opensource.org/licenses/mit-license.php.
  4  
  5  #include <qt/bitcoinunits.h>
  6  
  7  #include <consensus/amount.h>
  8  
  9  #include <QStringList>
 10  
 11  #include <cassert>
 12  
 13  static constexpr auto MAX_DIGITS_BTC = 16;
 14  
 15  BitcoinUnits::BitcoinUnits(QObject *parent):
 16          QAbstractListModel(parent),
 17          unitlist(availableUnits())
 18  {
 19  }
 20  
 21  QList<BitcoinUnit> BitcoinUnits::availableUnits()
 22  {
 23      QList<BitcoinUnit> unitlist;
 24      unitlist.append(Unit::BTC);
 25      unitlist.append(Unit::mBTC);
 26      unitlist.append(Unit::uBTC);
 27      unitlist.append(Unit::SAT);
 28      return unitlist;
 29  }
 30  
 31  QString BitcoinUnits::longName(Unit unit)
 32  {
 33      switch (unit) {
 34      case Unit::BTC: return QString("BTC");
 35      case Unit::mBTC: return QString("mBTC");
 36      case Unit::uBTC: return QString::fromUtf8("µBTC (bits)");
 37      case Unit::SAT: return QString("Satoshi (sat)");
 38      } // no default case, so the compiler can warn about missing cases
 39      assert(false);
 40  }
 41  
 42  QString BitcoinUnits::shortName(Unit unit)
 43  {
 44      switch (unit) {
 45      case Unit::BTC: return longName(unit);
 46      case Unit::mBTC: return longName(unit);
 47      case Unit::uBTC: return QString("bits");
 48      case Unit::SAT: return QString("sat");
 49      } // no default case, so the compiler can warn about missing cases
 50      assert(false);
 51  }
 52  
 53  QString BitcoinUnits::description(Unit unit)
 54  {
 55      switch (unit) {
 56      case Unit::BTC: return QString("Bitcoins");
 57      case Unit::mBTC: return QString("Milli-Bitcoins (1 / 1" THIN_SP_UTF8 "000)");
 58      case Unit::uBTC: return QString("Micro-Bitcoins (bits) (1 / 1" THIN_SP_UTF8 "000" THIN_SP_UTF8 "000)");
 59      case Unit::SAT: return QString("Satoshi (sat) (1 / 100" THIN_SP_UTF8 "000" THIN_SP_UTF8 "000)");
 60      } // no default case, so the compiler can warn about missing cases
 61      assert(false);
 62  }
 63  
 64  qint64 BitcoinUnits::factor(Unit unit)
 65  {
 66      switch (unit) {
 67      case Unit::BTC: return 100'000'000;
 68      case Unit::mBTC: return 100'000;
 69      case Unit::uBTC: return 100;
 70      case Unit::SAT: return 1;
 71      } // no default case, so the compiler can warn about missing cases
 72      assert(false);
 73  }
 74  
 75  int BitcoinUnits::decimals(Unit unit)
 76  {
 77      switch (unit) {
 78      case Unit::BTC: return 8;
 79      case Unit::mBTC: return 5;
 80      case Unit::uBTC: return 2;
 81      case Unit::SAT: return 0;
 82      } // no default case, so the compiler can warn about missing cases
 83      assert(false);
 84  }
 85  
 86  QString BitcoinUnits::format(Unit unit, const CAmount& nIn, bool fPlus, SeparatorStyle separators, bool justify)
 87  {
 88      // Note: not using straight sprintf here because we do NOT want
 89      // localized number formatting.
 90      qint64 n = (qint64)nIn;
 91      qint64 coin = factor(unit);
 92      int num_decimals = decimals(unit);
 93      qint64 n_abs = (n > 0 ? n : -n);
 94      qint64 quotient = n_abs / coin;
 95      QString quotient_str = QString::number(quotient);
 96      if (justify) {
 97          quotient_str = quotient_str.rightJustified(MAX_DIGITS_BTC - num_decimals, ' ');
 98      }
 99  
100      // Use SI-style thin space separators as these are locale independent and can't be
101      // confused with the decimal marker.
102      QChar thin_sp(THIN_SP_CP);
103      int q_size = quotient_str.size();
104      if (separators == SeparatorStyle::ALWAYS || (separators == SeparatorStyle::STANDARD && q_size > 4))
105          for (int i = 3; i < q_size; i += 3)
106              quotient_str.insert(q_size - i, thin_sp);
107  
108      if (n < 0)
109          quotient_str.insert(0, '-');
110      else if (fPlus && n > 0)
111          quotient_str.insert(0, '+');
112  
113      if (num_decimals > 0) {
114          qint64 remainder = n_abs % coin;
115          QString remainder_str = QString::number(remainder).rightJustified(num_decimals, '0');
116          return quotient_str + QString(".") + remainder_str;
117      } else {
118          return quotient_str;
119      }
120  }
121  
122  
123  // NOTE: Using formatWithUnit in an HTML context risks wrapping
124  // quantities at the thousands separator. More subtly, it also results
125  // in a standard space rather than a thin space, due to a bug in Qt's
126  // XML whitespace canonicalisation
127  //
128  // Please take care to use formatHtmlWithUnit instead, when
129  // appropriate.
130  
131  QString BitcoinUnits::formatWithUnit(Unit unit, const CAmount& amount, bool plussign, SeparatorStyle separators)
132  {
133      return format(unit, amount, plussign, separators) + QString(" ") + shortName(unit);
134  }
135  
136  QString BitcoinUnits::formatHtmlWithUnit(Unit unit, const CAmount& amount, bool plussign, SeparatorStyle separators)
137  {
138      QString str(formatWithUnit(unit, amount, plussign, separators));
139      str.replace(QChar(THIN_SP_CP), QString(THIN_SP_HTML));
140      return QString("<span style='white-space: nowrap;'>%1</span>").arg(str);
141  }
142  
143  QString BitcoinUnits::formatWithPrivacy(Unit unit, const CAmount& amount, SeparatorStyle separators, bool privacy)
144  {
145      assert(amount >= 0);
146      QString value;
147      if (privacy) {
148          value = format(unit, 0, false, separators, true).replace('0', '#');
149      } else {
150          value = format(unit, amount, false, separators, true);
151      }
152      return value + QString(" ") + shortName(unit);
153  }
154  
155  bool BitcoinUnits::parse(Unit unit, const QString& value, CAmount* val_out)
156  {
157      if (value.isEmpty()) {
158          return false; // Refuse to parse invalid unit or empty string
159      }
160      int num_decimals = decimals(unit);
161  
162      // Ignore spaces and thin spaces when parsing
163      QStringList parts = removeSpaces(value).split(".");
164  
165      if(parts.size() > 2)
166      {
167          return false; // More than one dot
168      }
169      const QString& whole = parts[0];
170      QString decimals;
171  
172      if(parts.size() > 1)
173      {
174          decimals = parts[1];
175      }
176      if(decimals.size() > num_decimals)
177      {
178          return false; // Exceeds max precision
179      }
180      bool ok = false;
181      QString str = whole + decimals.leftJustified(num_decimals, '0');
182  
183      if(str.size() > 18)
184      {
185          return false; // Longer numbers will exceed 63 bits
186      }
187      CAmount retvalue(str.toLongLong(&ok));
188      if(val_out)
189      {
190          *val_out = retvalue;
191      }
192      return ok;
193  }
194  
195  QString BitcoinUnits::getAmountColumnTitle(Unit unit)
196  {
197      return QObject::tr("Amount") + " (" + shortName(unit) + ")";
198  }
199  
200  int BitcoinUnits::rowCount(const QModelIndex &parent) const
201  {
202      Q_UNUSED(parent);
203      return unitlist.size();
204  }
205  
206  QVariant BitcoinUnits::data(const QModelIndex &index, int role) const
207  {
208      int row = index.row();
209      if(row >= 0 && row < unitlist.size())
210      {
211          Unit unit = unitlist.at(row);
212          switch(role)
213          {
214          case Qt::EditRole:
215          case Qt::DisplayRole:
216              return QVariant(longName(unit));
217          case Qt::ToolTipRole:
218              return QVariant(description(unit));
219          case UnitRole:
220              return QVariant::fromValue(unit);
221          }
222      }
223      return QVariant();
224  }
225  
226  CAmount BitcoinUnits::maxMoney()
227  {
228      return MAX_MONEY;
229  }
230  
231  namespace {
232  qint8 ToQint8(BitcoinUnit unit)
233  {
234      switch (unit) {
235      case BitcoinUnit::BTC: return 0;
236      case BitcoinUnit::mBTC: return 1;
237      case BitcoinUnit::uBTC: return 2;
238      case BitcoinUnit::SAT: return 3;
239      } // no default case, so the compiler can warn about missing cases
240      assert(false);
241  }
242  
243  BitcoinUnit FromQint8(qint8 num)
244  {
245      switch (num) {
246      case 0: return BitcoinUnit::BTC;
247      case 1: return BitcoinUnit::mBTC;
248      case 2: return BitcoinUnit::uBTC;
249      case 3: return BitcoinUnit::SAT;
250      }
251      assert(false);
252  }
253  } // namespace
254  
255  QDataStream& operator<<(QDataStream& out, const BitcoinUnit& unit)
256  {
257      return out << ToQint8(unit);
258  }
259  
260  QDataStream& operator>>(QDataStream& in, BitcoinUnit& unit)
261  {
262      qint8 input;
263      in >> input;
264      unit = FromQint8(input);
265      return in;
266  }