Bitcoin Core  28.1.0
P2P Digital Currency
bitcoinunits.cpp
Go to the documentation of this file.
1 // Copyright (c) 2011-2021 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 
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 
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 
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 
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  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 
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 
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 }
static QList< Unit > availableUnits()
Get list of units, for drop-down box.
Unit
Bitcoin units.
Definition: bitcoinunits.h:42
assert(!tx.IsCoinBase())
QList< Unit > unitlist
Definition: bitcoinunits.h:109
static QString description(Unit unit)
Longer description.
#define THIN_SP_HTML
Definition: bitcoinunits.h:27
static bool parse(Unit unit, const QString &value, CAmount *val_out)
Parse string to coin amount.
QDataStream & operator<<(QDataStream &out, const BitcoinUnit &unit)
QDataStream & operator>>(QDataStream &in, BitcoinUnit &unit)
static QString formatWithUnit(Unit unit, const CAmount &amount, bool plussign=false, SeparatorStyle separators=SeparatorStyle::STANDARD)
Format as string (with unit)
int64_t CAmount
Amount in satoshis (Can be negative)
Definition: amount.h:12
static QString formatWithPrivacy(Unit unit, const CAmount &amount, SeparatorStyle separators, bool privacy)
Format as string (with unit) of fixed length to preserve privacy, if it is set.
BitcoinUnits(QObject *parent)
static QString format(Unit unit, const CAmount &amount, bool plussign=false, SeparatorStyle separators=SeparatorStyle::STANDARD, bool justify=false)
Format as string.
static constexpr auto MAX_DIGITS_BTC
#define THIN_SP_UTF8
Definition: bitcoinunits.h:26
Unit identifier.
Definition: bitcoinunits.h:92
static int decimals(Unit unit)
Number of decimals left.
int rowCount(const QModelIndex &parent) const override
static QString removeSpaces(QString text)
Definition: bitcoinunits.h:98
QVariant data(const QModelIndex &index, int role) const override
static constexpr CAmount MAX_MONEY
No amount larger than this (in satoshi) is valid.
Definition: amount.h:26
static QString getAmountColumnTitle(Unit unit)
Gets title for amount column including current display unit if optionsModel reference available */...
static QString formatHtmlWithUnit(Unit unit, const CAmount &amount, bool plussign=false, SeparatorStyle separators=SeparatorStyle::STANDARD)
Format as HTML string (with unit)
static QString shortName(Unit unit)
Short name.
static QString longName(Unit unit)
Long name.
#define THIN_SP_CP
Definition: bitcoinunits.h:25
static CAmount maxMoney()
Return maximum number of base units (Satoshis)
static qint64 factor(Unit unit)
Number of Satoshis (1e-8) per unit.