Vercel/src/utility/number.ts
2025-03-28 15:52:28 -05:00

124 lines
3.4 KiB
TypeScript

export const roundToString = (
number: number,
place = 1,
includeZero?: boolean,
) => {
const precision = Math.pow(10, place);
const result = Math.round(number * precision) / precision;
return includeZero ? result.toFixed(place) : result.toString();
};
export const roundToNumber = (
...args: Parameters<typeof roundToString>
) =>
parseFloat(roundToString(...args));
const gcd = (a: number, b: number): number => {
if (b <= 0.0000001) {
return a;
} else {
return gcd(b, a % b);
}
};
const FRACTION_TOLERANCE = 0.011;
const STICKY_DECIMALS = [0.25, 0.33, 0.5, 0.66, 0.75];
const MAX_FRACTION_LENGTH = 4; // Permit 1/64 but not 1/100
const formatDecimalToFraction = (_decimal: number) => {
// Prevent imprecision which causes numbers such as,
// 0.1 to equal 0.10000000000000009
const decimal = Math.abs(parseFloat(_decimal.toPrecision(8)));
if (Math.abs(decimal - 0.167) <= FRACTION_TOLERANCE) {
return '1/6';
} else if (Math.abs(decimal - 0.333) <= FRACTION_TOLERANCE) {
return '1/3';
} else if (Math.abs(decimal - 0.667) <= FRACTION_TOLERANCE) {
return '2/3';
} else if (Math.abs(decimal - 0.833) <= FRACTION_TOLERANCE) {
return '5/6';
} else {
const length = decimal.toString().length - 2;
let denominator = Math.pow(10, length);
let numerator = decimal * denominator;
const divisor = gcd(numerator, denominator);
numerator /= divisor;
denominator /= divisor;
return `${Math.floor(numerator)}/${Math.floor(denominator)}`;
}
};
export const formatNumberToFraction = (number: number) => {
const sign = number >= 0 ? '+' : '-';
let decimal = (1 - Math.abs(number % 1)) > FRACTION_TOLERANCE
? number % 1
: 0;
if (decimal !== 0) {
for (const stickyDecimal of STICKY_DECIMALS) {
if (Math.abs(Math.abs(decimal) - stickyDecimal) < FRACTION_TOLERANCE) {
decimal = decimal < 0 ? -stickyDecimal : stickyDecimal;
break;
}
}
}
let integer = Math.round(Math.abs(number - decimal));
if (Math.abs(decimal) === 1) {
decimal = 0;
integer += 1;
}
const fraction = decimal !== 0 ? formatDecimalToFraction(decimal) : '';
// Ensure fractions aren't too long
if (!fraction || fraction.length <= MAX_FRACTION_LENGTH) {
const integerString = integer > 0
? fraction ? `${integer} ` : integer
: fraction ? '' : '0';
return `${sign}${integerString}${fraction}`;
} else {
const decimalFormatted = decimal.toPrecision(2).replace(/^-*0+/, '');
return `${sign}${integer}${decimalFormatted}`;
}
};
export const formatBytesToMB = (
bytes: number,
byteSize = 1000,
precision = 1,
) =>
`${(bytes / byteSize / byteSize).toFixed(precision)}MB`;
export const convertNumberToRomanNumeral = (number: number) => {
const romanNumerals = [
{ value: 1000, numeral: 'M' },
{ value: 900, numeral: 'CM' },
{ value: 500, numeral: 'D' },
{ value: 400, numeral: 'CD' },
{ value: 100, numeral: 'C' },
{ value: 90, numeral: 'XC' },
{ value: 50, numeral: 'L' },
{ value: 40, numeral: 'XL' },
{ value: 10, numeral: 'X' },
{ value: 9, numeral: 'IX' },
{ value: 5, numeral: 'V' },
{ value: 4, numeral: 'IV' },
{ value: 1, numeral: 'I' },
];
let result = '';
for (const romanNumeral of romanNumerals) {
while (number >= romanNumeral.value) {
result += romanNumeral.numeral;
number -= romanNumeral.value;
}
}
return result;
};