feat: Adds AmountsArithmetic utility exposing basic functionality for handling arbitrary precision decimals

Closes: #11484
This commit is contained in:
Michał Cieślak 2023-07-11 10:50:08 +02:00 committed by Michał
parent e1177617b9
commit 4a3e8ca1b2
3 changed files with 124 additions and 0 deletions

View File

@ -0,0 +1,121 @@
pragma Singleton
import QtQuick 2.15
import "big.min.mjs" as Big
/*!
\qmltype AmountsArithmetic
\inherits QtObject
\inqmlmodule StatusQ.Core.Utils
\brief A singleton that provides methods to properly handle cryptocurrency amounts.
A basic set of methods that provide basic operations on cryptocurrency amounts,
such as multiplication or comparison. Correctness is ensured by using numbers
of arbitrary accuracy and decimal arithmetic. Whenever ordinary java script
numbers are used, their textual representation is used to initialize big precision
number, not the exact binary representation.
The big.js library is used internally.
*/
QtObject {
/*!
\qmlmethod AmountsArithmetic::fromNumber(number, multiplier = 0)
\brief Construct an amount expressed in basic units for a currency with a specific divisibility.
The amount is created from a number's `toString` value rather than from
its underlying binary floating point value. This prevents errors such as
those shown in this example:
\qml
console.log(0.07 * 10**18) // 70000000000000010 - incorrect
console.log(AmountsArithmetic.fromNumber(0.07, 18)) // 70000000000000000 - correct
\endqml
The amount is assumed to be an integer, therefore the accuracy after the
decimal point cannot be greater than the indicated multiplier. Otherwise,
a warning will be printed.
*/
function fromNumber(number, multiplier = 0) {
console.assert(!isNaN(number) && Number.isInteger(multiplier)
&& multiplier >= 0)
const amount = new Big.Big(number).times(10 ** multiplier)
console.assert(amount.eq(amount.round()))
return amount
}
/*!
\qmlmethod AmountsArithmetic::toNumber(amount, multiplier = 0)
\brief Converts an amount to a java script number.
This operation may result in loss of precision. Because of that it should
be used only to display a value in the user interface, but requires
further formatting (localization, decimal places adjustment). Other
operations like comparisons or multiplication must be performed directly
on amounts to preserve precision.
\qml
console.log(AmountsArithmetic.toNumber(
AmountsArithmetic.fromString("123456789123456789123"))) // 123456789123456800000
\endqml
*/
function toNumber(amount, multiplier = 0) {
console.assert(Number.isInteger(multiplier) && multiplier >= 0)
return amount.div(10 ** multiplier).toNumber()
}
/*!
\qmlmethod AmountsArithmetic::fromString(numStr)
\brief Construct an amount from a textual representation.
The constructed amount is of arbitrary precision. Using ordinary js
numbers, precision might not be sufficient:
\qml
console.log(1234567891234567891) // 1234567891234568000 - incorrect
console.log(parseInt("1234567891234567891")) // 1234567891234568000 - incorrect
console.log(AmountsArithmetic.fromString("1234567891234567891")) // 1234567891234567891 - correct
\endqml
The obtained amount can be multiplied or compared.
Provided number is assumed to be an amount in basic units, an integer.
*/
function fromString(numStr) {
console.assert(typeof numStr === "string")
const amount = new Big.Big(numStr)
console.assert(amount.eq(amount.round()))
return amount
}
/*!
\qmlmethod AmountsArithmetic::times(amount, multiplier)
\brief Returns an amount whose value is the value of num1 times num2.
\qml
console.log(0.07 * 10**18 * 1000 === 70 * 10**18) // false
console.log(AmountsArithmetic.cmp(
AmountsArithmetic.fromNumber(0.07, 18).times(1000),
AmountsArithmetic.fromNumber(1, 18).times(70)) === 0) // true
\endqml
*/
function times(amount, multiplier) {
console.assert(amount instanceof Big.Big)
console.assert(multiplier instanceof Big.Big || Number.isInteger(multiplier))
return amount.times(multiplier)
}
/*!
\qmlmethod AmountsArithmetic::cmp(amount1, amount2)
\brief Compares two amounts.
Returns 1 if the value of amount1 is greater than the value of amount2.
Returns -1 if the value of amount1 is less than the value of amount2.
Returns 0 if both amounts have the same value.
*/
function cmp(amount1, amount2) {
console.assert(amount1 instanceof Big.Big)
console.assert(amount2 instanceof Big.Big)
return amount1.cmp(amount2)
}
}

View File

@ -9,6 +9,7 @@ ModelsComparator 0.1 ModelsComparator.qml
StackViewStates 0.1 StackViewStates.qml StackViewStates 0.1 StackViewStates.qml
StatesStack 0.1 StatesStack.qml StatesStack 0.1 StatesStack.qml
XSS 1.0 xss.js XSS 1.0 xss.js
singleton AmountsArithmetic 0.1 AmountsArithmetic.qml
singleton Emoji 0.1 Emoji.qml singleton Emoji 0.1 Emoji.qml
singleton ModelUtils 0.1 ModelUtils.qml singleton ModelUtils 0.1 ModelUtils.qml
singleton OperatorsUtils 0.1 OperatorsUtils.qml singleton OperatorsUtils 0.1 OperatorsUtils.qml

View File

@ -220,6 +220,8 @@
<file>StatusQ/Controls/StatusLinkText.qml</file> <file>StatusQ/Controls/StatusLinkText.qml</file>
<file>StatusQ/Core/Utils/ModelChangeGuard.qml</file> <file>StatusQ/Core/Utils/ModelChangeGuard.qml</file>
<file>StatusQ/Core/Utils/StackViewStates.qml</file> <file>StatusQ/Core/Utils/StackViewStates.qml</file>
<file>StatusQ/Core/Utils/AmountsArithmetic.qml</file>
<file>StatusQ/Core/Utils/big.min.mjs</file>
<file>StatusQ/Controls/StatusBlockProgressBar.qml</file> <file>StatusQ/Controls/StatusBlockProgressBar.qml</file>
<file>StatusQ/Components/StatusInfoBoxPanel.qml</file> <file>StatusQ/Components/StatusInfoBoxPanel.qml</file>
</qresource> </qresource>