-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathNumberDecimal.ts
More file actions
119 lines (92 loc) · 2.5 KB
/
NumberDecimal.ts
File metadata and controls
119 lines (92 loc) · 2.5 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
import type { DecimalClass, ValueType } from './interface';
import { getNumberPrecision, isE, isEmpty, num2str } from './numberUtil';
/**
* We can remove this when IE not support anymore
*/
export default class NumberDecimal implements DecimalClass {
origin: string = '';
number: number;
empty: boolean;
constructor(value: ValueType) {
if (isEmpty(value)) {
this.empty = true;
return;
}
this.origin = String(value);
this.number = Number(value);
}
negate() {
return new NumberDecimal(-this.toNumber());
}
add(value: ValueType) {
if (this.isInvalidate()) {
return new NumberDecimal(value);
}
const target = Number(value);
if (Number.isNaN(target)) {
return this;
}
const number = this.number + target;
// [Legacy] Back to safe integer
if (number > Number.MAX_SAFE_INTEGER) {
return new NumberDecimal(Number.MAX_SAFE_INTEGER);
}
if (number < Number.MIN_SAFE_INTEGER) {
return new NumberDecimal(Number.MIN_SAFE_INTEGER);
}
const maxPrecision = Math.max(
getNumberPrecision(this.number),
getNumberPrecision(target),
);
return new NumberDecimal(number.toFixed(maxPrecision));
}
multi(value: ValueType) {
const target = Number(value);
if (this.isInvalidate() || Number.isNaN(target)) {
return new NumberDecimal(NaN);
}
const number = this.number * target;
// [Legacy] Back to safe integer
if (number > Number.MAX_SAFE_INTEGER) {
return new NumberDecimal(Number.MAX_SAFE_INTEGER);
}
if (number < Number.MIN_SAFE_INTEGER) {
return new NumberDecimal(Number.MIN_SAFE_INTEGER);
}
const maxPrecision = Math.max(
getNumberPrecision(this.number),
getNumberPrecision(target),
);
return new NumberDecimal(number.toFixed(maxPrecision));
}
isEmpty() {
return this.empty;
}
isNaN() {
return Number.isNaN(this.number);
}
isInvalidate() {
return this.isEmpty() || this.isNaN();
}
equals(target: DecimalClass) {
return this.toNumber() === target?.toNumber();
}
lessEquals(target: DecimalClass) {
return this.add(target.negate().toString()).toNumber() <= 0;
}
toNumber() {
return this.number;
}
toString(safe: boolean = true) {
if (!safe) {
return this.origin;
}
if (this.isInvalidate()) {
return '';
}
if (isE(this.number) && getNumberPrecision(this.number) > 100) {
return String(this.number);
}
return num2str(this.number);
}
}