2022-04-19 11:29:36 +05:30
|
|
|
import { Fyo } from 'fyo';
|
|
|
|
import { DocValue } from 'fyo/core/types';
|
2022-04-24 12:18:44 +05:30
|
|
|
import { Doc } from 'fyo/model/doc';
|
2022-04-18 13:31:41 +05:30
|
|
|
import { DateTime } from 'luxon';
|
|
|
|
import Money from 'pesa/dist/types/src/money';
|
|
|
|
import { Field, FieldType, FieldTypeEnum } from 'schemas/types';
|
|
|
|
import { getIsNullOrUndef } from 'utils';
|
|
|
|
import {
|
|
|
|
DEFAULT_CURRENCY,
|
|
|
|
DEFAULT_DATE_FORMAT,
|
|
|
|
DEFAULT_DISPLAY_PRECISION,
|
|
|
|
DEFAULT_LOCALE,
|
|
|
|
} from './consts';
|
|
|
|
|
|
|
|
export function format(
|
|
|
|
value: DocValue,
|
2022-04-18 16:59:20 +05:30
|
|
|
df: string | Field | null,
|
|
|
|
doc: Doc | null,
|
2022-04-19 11:29:36 +05:30
|
|
|
fyo: Fyo
|
2022-04-18 13:31:41 +05:30
|
|
|
): string {
|
|
|
|
if (!df) {
|
|
|
|
return String(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
const field: Field = getField(df);
|
|
|
|
|
|
|
|
if (field.fieldtype === FieldTypeEnum.Currency) {
|
2022-04-19 11:29:36 +05:30
|
|
|
return formatCurrency(value, field, doc, fyo);
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (field.fieldtype === FieldTypeEnum.Date) {
|
2022-04-19 11:29:36 +05:30
|
|
|
return formatDate(value, fyo);
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (field.fieldtype === FieldTypeEnum.Check) {
|
|
|
|
return Boolean(value).toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getIsNullOrUndef(value)) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return String(value);
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
function formatDate(value: DocValue, fyo: Fyo): string {
|
2022-04-18 13:31:41 +05:30
|
|
|
const dateFormat =
|
2022-04-19 11:29:36 +05:30
|
|
|
(fyo.singles.SystemSettings?.dateFormat as string) ?? DEFAULT_DATE_FORMAT;
|
2022-04-18 13:31:41 +05:30
|
|
|
|
|
|
|
let dateValue: DateTime;
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
dateValue = DateTime.fromISO(value);
|
|
|
|
} else if (value instanceof Date) {
|
|
|
|
dateValue = DateTime.fromJSDate(value);
|
|
|
|
} else {
|
|
|
|
dateValue = DateTime.fromSeconds(value as number);
|
|
|
|
}
|
|
|
|
|
|
|
|
const formattedDate = dateValue.toFormat(dateFormat);
|
|
|
|
if (value === 'Invalid DateTime') {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return formattedDate;
|
|
|
|
}
|
|
|
|
|
2022-04-18 16:59:20 +05:30
|
|
|
function formatCurrency(
|
|
|
|
value: DocValue,
|
|
|
|
field: Field,
|
|
|
|
doc: Doc | null,
|
2022-04-19 11:29:36 +05:30
|
|
|
fyo: Fyo
|
2022-04-18 16:59:20 +05:30
|
|
|
): string {
|
2022-04-19 11:29:36 +05:30
|
|
|
const currency = getCurrency(field, doc, fyo);
|
2022-04-18 13:31:41 +05:30
|
|
|
|
|
|
|
let valueString;
|
|
|
|
try {
|
2022-04-19 11:29:36 +05:30
|
|
|
valueString = formatNumber(value, fyo);
|
2022-04-18 13:31:41 +05:30
|
|
|
} catch (err) {
|
|
|
|
(err as Error).message += ` value: '${value}', type: ${typeof value}`;
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
const currencySymbol = fyo.currencySymbols[currency];
|
2022-04-18 13:31:41 +05:30
|
|
|
if (currencySymbol !== undefined) {
|
|
|
|
return currencySymbol + ' ' + valueString;
|
|
|
|
}
|
|
|
|
|
|
|
|
return valueString;
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
function formatNumber(value: DocValue, fyo: Fyo): string {
|
|
|
|
const numberFormatter = getNumberFormatter(fyo);
|
2022-04-18 13:31:41 +05:30
|
|
|
if (typeof value === 'number') {
|
|
|
|
return numberFormatter.format(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((value as Money).round) {
|
|
|
|
const floatValue = parseFloat((value as Money).round());
|
|
|
|
return numberFormatter.format(floatValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
const floatValue = parseFloat(value as string);
|
|
|
|
const formattedNumber = numberFormatter.format(floatValue);
|
|
|
|
|
|
|
|
if (formattedNumber === 'NaN') {
|
|
|
|
throw Error(
|
|
|
|
`invalid value passed to formatNumber: '${value}' of type ${typeof value}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return formattedNumber;
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
function getNumberFormatter(fyo: Fyo) {
|
|
|
|
if (fyo.currencyFormatter) {
|
|
|
|
return fyo.currencyFormatter;
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
const locale =
|
2022-04-19 11:29:36 +05:30
|
|
|
(fyo.singles.SystemSettings?.locale as string) ?? DEFAULT_LOCALE;
|
2022-04-18 13:31:41 +05:30
|
|
|
const display =
|
2022-04-19 11:29:36 +05:30
|
|
|
(fyo.singles.SystemSettings?.displayPrecision as number) ??
|
2022-04-18 13:31:41 +05:30
|
|
|
DEFAULT_DISPLAY_PRECISION;
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
return (fyo.currencyFormatter = Intl.NumberFormat(locale, {
|
2022-04-18 13:31:41 +05:30
|
|
|
style: 'decimal',
|
|
|
|
minimumFractionDigits: display,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
function getCurrency(field: Field, doc: Doc | null, fyo: Fyo): string {
|
2022-04-28 18:07:07 +05:30
|
|
|
let getCurrency = doc?.getCurrencies?.[field.fieldname];
|
2022-04-23 14:53:44 +05:30
|
|
|
if (getCurrency !== undefined) {
|
|
|
|
return getCurrency();
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
getCurrency = doc?.parentdoc?.getCurrencies[field.fieldname];
|
|
|
|
if (getCurrency !== undefined) {
|
|
|
|
return getCurrency();
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
return (fyo.singles.SystemSettings?.currency as string) ?? DEFAULT_CURRENCY;
|
2022-04-18 13:31:41 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
function getField(df: string | Field): Field {
|
|
|
|
if (typeof df === 'string') {
|
|
|
|
return {
|
|
|
|
label: '',
|
|
|
|
fieldname: '',
|
|
|
|
fieldtype: df as FieldType,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return df;
|
|
|
|
}
|