mirror of
https://github.com/status-im/status-desktop.git
synced 2025-01-09 05:52:41 +00:00
feat: Adds AmountsArithmetic utility exposing basic functionality for handling arbitrary precision decimals
Closes: #11484
This commit is contained in:
parent
e1177617b9
commit
4a3e8ca1b2
121
ui/StatusQ/src/StatusQ/Core/Utils/AmountsArithmetic.qml
Normal file
121
ui/StatusQ/src/StatusQ/Core/Utils/AmountsArithmetic.qml
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
@ -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
|
||||||
|
@ -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>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user