stack_wallet/lib/utilities/amount/amount_unit.dart

211 lines
6 KiB
Dart
Raw Normal View History

2023-04-06 23:27:42 +00:00
import 'dart:math' as math;
import 'package:decimal/decimal.dart';
import 'package:intl/number_symbols.dart';
import 'package:intl/number_symbols_data.dart';
2023-05-29 21:10:55 +00:00
import 'package:stackwallet/models/isar/models/ethereum/eth_contract.dart';
2023-04-06 23:27:42 +00:00
import 'package:stackwallet/utilities/amount/amount.dart';
import 'package:stackwallet/utilities/enums/coin_enum.dart';
// preserve index order as index is used to store value in preferences
2023-04-06 23:27:42 +00:00
enum AmountUnit {
normal(0),
milli(3),
micro(6),
nano(9),
pico(12),
femto(15),
atto(18),
;
const AmountUnit(this.shift);
final int shift;
2023-05-29 22:50:21 +00:00
static List<AmountUnit> valuesForCoin(Coin coin) {
switch (coin) {
case Coin.firo:
case Coin.litecoin:
case Coin.particl:
case Coin.namecoin:
case Coin.bitcoinTestNet:
case Coin.litecoinTestNet:
case Coin.bitcoincashTestnet:
case Coin.dogecoinTestNet:
case Coin.firoTestNet:
case Coin.bitcoin:
case Coin.bitcoincash:
case Coin.dogecoin:
case Coin.eCash:
case Coin.epicCash:
return AmountUnit.values.sublist(0, 4);
case Coin.monero:
case Coin.wownero:
return AmountUnit.values.sublist(0, 5);
case Coin.ethereum:
return AmountUnit.values;
}
}
2023-04-06 23:27:42 +00:00
}
extension AmountUnitExt on AmountUnit {
String unitForCoin(Coin coin) {
switch (this) {
case AmountUnit.normal:
return coin.ticker;
case AmountUnit.milli:
return "m${coin.ticker}";
case AmountUnit.micro:
return "µ${coin.ticker}";
case AmountUnit.nano:
if (coin == Coin.ethereum) {
return "gwei";
} else if (coin == Coin.wownero || coin == Coin.monero) {
return "n${coin.ticker}";
} else {
return "sats";
}
case AmountUnit.pico:
if (coin == Coin.ethereum) {
return "mwei";
} else if (coin == Coin.wownero || coin == Coin.monero) {
return "p${coin.ticker}";
} else {
return "invalid";
}
case AmountUnit.femto:
if (coin == Coin.ethereum) {
return "kwei";
} else {
return "invalid";
}
case AmountUnit.atto:
if (coin == Coin.ethereum) {
return "wei";
} else {
return "invalid";
}
}
}
2023-05-29 21:10:55 +00:00
String unitForContract(EthContract contract) {
switch (this) {
case AmountUnit.normal:
return contract.symbol;
case AmountUnit.milli:
return "m${contract.symbol}";
case AmountUnit.micro:
return "µ${contract.symbol}";
case AmountUnit.nano:
return "gwei";
case AmountUnit.pico:
return "mwei";
case AmountUnit.femto:
return "kwei";
case AmountUnit.atto:
return "wei";
}
}
2023-04-06 23:27:42 +00:00
String displayAmount({
required Amount amount,
required String locale,
required Coin coin,
required int maxDecimalPlaces,
2023-05-29 21:10:55 +00:00
bool withUnitName = true,
String? overrideUnit,
EthContract? tokenContract,
2023-04-06 23:27:42 +00:00
}) {
assert(maxDecimalPlaces >= 0);
2023-05-29 21:10:55 +00:00
2023-04-06 23:27:42 +00:00
// ensure we don't shift past minimum atomic value
final realShift = math.min(shift, amount.fractionDigits);
// shifted to unit
final Decimal shifted = amount.decimal.shift(realShift);
// get shift int value without fractional value
final BigInt wholeNumber = shifted.toBigInt();
// get decimal places to display
final int places = math.max(0, amount.fractionDigits - realShift);
// start building the return value with just the whole value
String returnValue = wholeNumber.toString();
// if true and withUnitName is true, we will show "~" prepended on amount
bool didLosePrecision = false;
2023-04-06 23:27:42 +00:00
// if any decimal places should be shown continue building the return value
if (places > 0) {
// get the fractional value
final Decimal fraction = shifted - shifted.truncate();
2023-05-29 21:10:55 +00:00
// get final decimal based on max precision wanted while ensuring that
// maxDecimalPlaces doesn't exceed the max per coin
final int updatedMax;
if (tokenContract != null) {
updatedMax = maxDecimalPlaces > tokenContract.decimals
? tokenContract.decimals
: maxDecimalPlaces;
} else {
updatedMax =
maxDecimalPlaces > coin.decimals ? coin.decimals : maxDecimalPlaces;
}
final int actualDecimalPlaces = math.min(places, updatedMax);
2023-04-06 23:27:42 +00:00
// get remainder string without the prepending "0."
2023-05-29 21:10:55 +00:00
final fractionString = fraction.toString();
String remainder;
if (fractionString.length > 2) {
remainder = fraction.toString().substring(2);
} else {
remainder = "0";
}
2023-04-06 23:27:42 +00:00
if (remainder.length > actualDecimalPlaces) {
// check for loss of precision
final remainingRemainder =
BigInt.tryParse(remainder.substring(actualDecimalPlaces));
if (remainingRemainder != null) {
didLosePrecision = remainingRemainder > BigInt.zero;
}
2023-04-06 23:27:42 +00:00
// trim unwanted trailing digits
remainder = remainder.substring(0, actualDecimalPlaces);
} else if (remainder.length < actualDecimalPlaces) {
// pad with zeros to achieve requested precision
for (int i = remainder.length; i < actualDecimalPlaces; i++) {
remainder += "0";
}
}
// get decimal separator based on locale
final String separator =
(numberFormatSymbols[locale] as NumberSymbols?)?.DECIMAL_SEP ??
(numberFormatSymbols[locale.substring(0, 2)] as NumberSymbols?)
?.DECIMAL_SEP ??
".";
// append separator and fractional amount
returnValue += "$separator$remainder";
}
2023-05-29 21:10:55 +00:00
if (!withUnitName) {
return returnValue;
}
2023-04-06 23:27:42 +00:00
// return the value with the proper unit symbol
2023-05-29 21:10:55 +00:00
if (tokenContract != null) {
overrideUnit = unitForContract(tokenContract);
}
if (didLosePrecision) {
returnValue = "~$returnValue";
}
2023-05-29 21:10:55 +00:00
return "$returnValue ${overrideUnit ?? unitForCoin(coin)}";
2023-04-06 23:27:42 +00:00
}
}