box-ui-elements-mlh
Version:
94 lines (79 loc) • 3.22 kB
text/typescript
/**
* @file Function to abbreviate a number as a string in a locale-sensitive manner.
* @author Box
*/
import IntlMessageFormat from 'intl-messageformat';
import data, { NumbersData } from 'box-locale-data';
import isNaN from 'lodash/isNaN';
const { languages, numbers } = data;
export enum Lengths {
short = 'short',
long = 'long',
}
export interface NumAbbrOptions {
length?: Lengths;
locale?: string;
numbersData?: NumbersData;
}
function numAbbrWithLocale(input: number, options: NumAbbrOptions, numbersData?: NumbersData, locale?: string): string {
if (!input || !numbersData) {
return '0';
}
let { length }: { length?: Lengths } = options;
length = length || Lengths.short;
let exponent: number = Math.floor(input).toString().length - 1;
if (input < 0) {
exponent -= 1; // take care of the negative sign
}
const formats = numbersData[length];
const digits: number = exponent >= formats.length ? exponent - formats.length + 3 : formats[exponent].digits;
const count: number = Math.round(input / 10 ** (exponent - digits + 1));
const template = new IntlMessageFormat(
formats[exponent > formats.length ? formats.length - 1 : exponent].msg,
locale,
);
return template.format({ count });
}
/**
* Gets the number in abbreviated form in a locale-sensitive manner. This function
* scales the number down to the smallest it can be, taking only up to 4 significant
* digits, and rounding the rest. ie. 12345678 becomes "12M" in English.
*
* The abbreviation words/letters can have the length "short" or "long", specified
* with the "length" property in the options. If "short",
* then this function uses an abbreviation of the bucket such as "12M". If "long",
* then the name of the bucket is written out in full, such as "12 million".
*
* For locales that have complex plurals, such as Russian or Polish, this function
* returns the correctly pluralized suffix/prefix to go along with the scaled number.
*/
function numAbbr(input: unknown, options: NumAbbrOptions = { length: Lengths.short }): string | string[] {
if (!input) return '0';
// languages contains info about the current locale
const { locale = languages.bcp47Tag || 'en-US', numbersData = numbers }: NumAbbrOptions = options;
let value;
switch (typeof input) {
case 'boolean':
input = input ? 1 : 0;
break;
case 'string':
value = parseInt(input, 10);
if (isNaN(value)) {
return '0';
}
break;
case 'object':
// for an array, we format each element of that array. In order to do that safely,
// recursively call numAbbr to make sure each of the elements is converted to a
// number before we call numAbbrWithLocale
return Array.isArray(input)
? input.map((element: unknown) => {
return numAbbr(element, options) as string;
})
: '0';
default:
break;
}
return numAbbrWithLocale(input as number, options, numbersData, locale);
}
export default numAbbr;