Commit 3205f3c6 authored by Miguel Rincon's avatar Miguel Rincon

Define easier to use interface for unit formatter

The formatting utils is powerful but hard to use, as it requires
creating a formatter to simply render a number.

This change adds easier to use function to be imported by client
scripts.
parent d54e6dc0
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
import * as Sentry from '@sentry/browser'; import * as Sentry from '@sentry/browser';
import MetricCard from '~/analytics/shared/components/metric_card.vue'; import MetricCard from '~/analytics/shared/components/metric_card.vue';
import { deprecatedCreateFlash as createFlash } from '~/flash'; import { deprecatedCreateFlash as createFlash } from '~/flash';
import { SUPPORTED_FORMATS, getFormatter } from '~/lib/utils/unit_format'; import { number } from '~/lib/utils/unit_format';
import { s__ } from '~/locale'; import { s__ } from '~/locale';
import usageTrendsCountQuery from '../graphql/queries/usage_trends_count.query.graphql'; import usageTrendsCountQuery from '../graphql/queries/usage_trends_count.query.graphql';
...@@ -24,8 +24,7 @@ export default { ...@@ -24,8 +24,7 @@ export default {
update(data) { update(data) {
return Object.entries(data).map(([key, obj]) => { return Object.entries(data).map(([key, obj]) => {
const label = this.$options.i18n.labels[key]; const label = this.$options.i18n.labels[key];
const formatter = getFormatter(SUPPORTED_FORMATS.number); const value = obj.nodes?.length ? number(obj.nodes[0].count, defaultPrecision) : null;
const value = obj.nodes?.length ? formatter(obj.nodes[0].count, defaultPrecision) : null;
return { return {
key, key,
......
import { formatNumber } from '~/locale';
/** /**
* Formats a number as string using `toLocaleString`. * Formats a number as a string using `toLocaleString`.
* *
* @param {Number} number to be converted * @param {Number} number to be converted
* @param {params} Parameters *
* @param {params.fractionDigits} Number of decimal digits * @param {options.maxCharLength} Max output char length at the
* to display, defaults to using `toLocaleString` defaults.
* @param {params.maxLength} Max output char lenght at the
* expense of precision, if the output is longer than this, * expense of precision, if the output is longer than this,
* the formatter switches to using exponential notation. * the formatter switches to using exponential notation.
* @param {params.factor} Value is multiplied by this factor, *
* useful for value normalization. * @param {options.valueFactor} Value is multiplied by this factor,
* @returns Formatted value * useful for value normalization or to alter orders of magnitude.
*
* @param {options} Other options to be passed to
* `formatNumber` such as `valueFactor`, `unit` and `style`.
*
*/ */
function formatNumber( const formatNumberNormalized = (value, { maxCharLength, valueFactor = 1, ...options }) => {
value, const formatted = formatNumber(value * valueFactor, options);
{ fractionDigits = undefined, valueFactor = 1, style = undefined, maxLength = undefined },
) {
if (value === null) {
return '';
}
const locale = document.documentElement.lang || undefined;
const num = value * valueFactor;
const formatted = num.toLocaleString(locale, {
minimumFractionDigits: fractionDigits,
maximumFractionDigits: fractionDigits,
style,
});
if (maxLength !== undefined && formatted.length > maxLength) { if (maxCharLength !== undefined && formatted.length > maxCharLength) {
// 123456 becomes 1.23e+8 // 123456 becomes 1.23e+8
return num.toExponential(2); return value.toExponential(2);
} }
return formatted; return formatted;
} };
/** /**
* Formats a number as a string scaling it up according to units. * Formats a number as a string scaling it up according to units.
...@@ -76,7 +67,10 @@ const scaledFormatter = (units, unitFactor = 1000) => { ...@@ -76,7 +67,10 @@ const scaledFormatter = (units, unitFactor = 1000) => {
const unit = units[scale]; const unit = units[scale];
return `${formatNumber(num, { fractionDigits })}${unit}`; return `${formatNumberNormalized(num, {
maximumFractionDigits: fractionDigits,
minimumFractionDigits: fractionDigits,
})}${unit}`;
}; };
}; };
...@@ -84,8 +78,14 @@ const scaledFormatter = (units, unitFactor = 1000) => { ...@@ -84,8 +78,14 @@ const scaledFormatter = (units, unitFactor = 1000) => {
* Returns a function that formats a number as a string. * Returns a function that formats a number as a string.
*/ */
export const numberFormatter = (style = 'decimal', valueFactor = 1) => { export const numberFormatter = (style = 'decimal', valueFactor = 1) => {
return (value, fractionDigits, maxLength) => { return (value, fractionDigits, maxCharLength) => {
return `${formatNumber(value, { fractionDigits, maxLength, valueFactor, style })}`; return `${formatNumberNormalized(value, {
maxCharLength,
valueFactor,
style,
maximumFractionDigits: fractionDigits,
minimumFractionDigits: fractionDigits,
})}`;
}; };
}; };
...@@ -93,9 +93,15 @@ export const numberFormatter = (style = 'decimal', valueFactor = 1) => { ...@@ -93,9 +93,15 @@ export const numberFormatter = (style = 'decimal', valueFactor = 1) => {
* Returns a function that formats a number as a string with a suffix. * Returns a function that formats a number as a string with a suffix.
*/ */
export const suffixFormatter = (unit = '', valueFactor = 1) => { export const suffixFormatter = (unit = '', valueFactor = 1) => {
return (value, fractionDigits, maxLength) => { return (value, fractionDigits, maxCharLength) => {
const length = maxLength !== undefined ? maxLength - unit.length : undefined; const length = maxCharLength !== undefined ? maxCharLength - unit.length : undefined;
return `${formatNumber(value, { fractionDigits, maxLength: length, valueFactor })}${unit}`;
return `${formatNumberNormalized(value, {
maxCharLength: length,
valueFactor,
maximumFractionDigits: fractionDigits,
minimumFractionDigits: fractionDigits,
})}${unit}`;
}; };
}; };
......
...@@ -58,10 +58,30 @@ const pgettext = (keyOrContext, key) => { ...@@ -58,10 +58,30 @@ const pgettext = (keyOrContext, key) => {
*/ */
const createDateTimeFormat = (formatOptions) => Intl.DateTimeFormat(languageCode(), formatOptions); const createDateTimeFormat = (formatOptions) => Intl.DateTimeFormat(languageCode(), formatOptions);
/**
* Formats a number as a string using `toLocaleString`.
*
* @param {Number} value - number to be converted
* @param {options?} options - options to be passed to
* `toLocaleString` such as `unit` and `style`.
* @param {langCode?} langCode - If set, forces a different
* language code from the one currently in the document.
* @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Intl/NumberFormat/NumberFormat
*
* @returns If value is a number, the formatted value as a string
*/
function formatNumber(value, options = {}, langCode = languageCode()) {
if (typeof value !== 'number' && typeof value !== 'bigint') {
return value;
}
return value.toLocaleString(langCode, options);
}
export { languageCode }; export { languageCode };
export { gettext as __ }; export { gettext as __ };
export { ngettext as n__ }; export { ngettext as n__ };
export { pgettext as s__ }; export { pgettext as s__ };
export { sprintf }; export { sprintf };
export { createDateTimeFormat }; export { createDateTimeFormat };
export { formatNumber };
export default locale; export default locale;
import { setLanguage } from 'helpers/locale_helper'; import { setLanguage } from 'helpers/locale_helper';
import { createDateTimeFormat, languageCode } from '~/locale'; import { createDateTimeFormat, formatNumber, languageCode } from '~/locale';
describe('locale', () => { describe('locale', () => {
afterEach(() => setLanguage(null)); afterEach(() => setLanguage(null));
...@@ -27,4 +27,68 @@ describe('locale', () => { ...@@ -27,4 +27,68 @@ describe('locale', () => {
expect(dateFormat.format(new Date(2015, 6, 3))).toBe('July 3, 2015'); expect(dateFormat.format(new Date(2015, 6, 3))).toBe('July 3, 2015');
}); });
}); });
describe('formatNumber', () => {
it('formats numbers', () => {
expect(formatNumber(1)).toBe('1');
expect(formatNumber(12345)).toBe('12,345');
});
it('formats bigint numbers', () => {
expect(formatNumber(123456789123456789n)).toBe('123,456,789,123,456,789');
});
it('formats numbers with options', () => {
expect(formatNumber(1, { style: 'percent' })).toBe('100%');
expect(formatNumber(1, { style: 'currency', currency: 'USD' })).toBe('$1.00');
});
it('formats localized numbers', () => {
expect(formatNumber(12345, {}, 'es')).toBe('12.345');
});
it('formats NaN', () => {
expect(formatNumber(NaN)).toBe('NaN');
});
it('formats infinity', () => {
expect(formatNumber(Number.POSITIVE_INFINITY)).toBe('');
});
it('formats negative infinity', () => {
expect(formatNumber(Number.NEGATIVE_INFINITY)).toBe('-∞');
});
it('formats EPSILON', () => {
expect(formatNumber(Number.EPSILON)).toBe('0');
});
describe('non-number values should pass through', () => {
it('undefined', () => {
expect(formatNumber(undefined)).toBe(undefined);
});
it('null', () => {
expect(formatNumber(null)).toBe(null);
});
it('arrays', () => {
expect(formatNumber([])).toEqual([]);
});
it('objects', () => {
expect(formatNumber({ a: 'b' })).toEqual({ a: 'b' });
});
});
describe('when in a different locale', () => {
beforeEach(() => {
setLanguage('es');
});
it('formats localized numbers', () => {
expect(formatNumber(12345)).toBe('12.345');
});
});
});
}); });
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment