mon-entreprise/publicodes/source/units.ts

288 lines
7.0 KiB
TypeScript

import {
countBy,
equals,
flatten,
isEmpty,
keys,
map,
pipe,
remove,
uniq,
unnest,
without
} from 'ramda'
import i18n from './i18n'
import { Evaluation, Unit } from './types'
//TODO this function does not handle complex units like passenger-kilometer/flight
export const parseUnit = (string: string, lng = 'fr'): Unit => {
const [a, ...b] = string.split('/'),
result = {
numerators: a
.split('.')
.filter(Boolean)
.map(unit => getUnitKey(unit, lng)),
denominators: b.map(unit => getUnitKey(unit, lng))
}
return result
}
const translations = (lng: string) =>
Object.entries(i18n.getResourceBundle(lng, 'units'))
function getUnitKey(unit: string, lng: string): string {
const key = translations(lng)
.find(([, trans]) => trans === unit)?.[0]
.replace(/_plural$/, '')
return key || unit
}
const printUnits = (units: Array<string>, count: number, lng: string): string =>
units
.filter(unit => unit !== '%')
.map(unit => i18n.t(`units:${unit}`, { count, lng }))
.join('.')
const plural = 2
export const serializeUnit = (
rawUnit: Unit | undefined | string,
count: number = plural,
lng = 'fr'
) => {
if (rawUnit === null || typeof rawUnit !== 'object') {
return typeof rawUnit === 'string'
? i18n.t(`units:${rawUnit}`, { count, lng })
: rawUnit
}
const unit = simplify(rawUnit),
{ numerators = [], denominators = [] } = unit
// the unit '%' is only displayed when it is the only unit
const merge = [...numerators, ...denominators]
if (merge.length === 1 && merge[0] === '%') return '%'
const n = !isEmpty(numerators)
const d = !isEmpty(denominators)
const string =
!n && !d
? ''
: n && !d
? printUnits(numerators, count, lng)
: !n && d
? `/${printUnits(denominators, 1, lng)}`
: `${printUnits(numerators, plural, lng)} / ${printUnits(
denominators,
1,
lng
)}`
return string
}
type SupportedOperators = '*' | '/' | '+' | '-'
const noUnit = { numerators: [], denominators: [] }
export const inferUnit = (
operator: SupportedOperators,
rawUnits: Array<Unit | undefined>
): Unit | undefined => {
const units = rawUnits.map(u => u || noUnit)
if (operator === '*')
return simplify({
numerators: unnest(units.map(u => u.numerators)),
denominators: unnest(units.map(u => u.denominators))
})
if (operator === '/') {
if (units.length !== 2)
throw new Error('Infer units of a division with units.length !== 2)')
return inferUnit('*', [
units[0],
{
numerators: units[1].denominators,
denominators: units[1].numerators
}
])
}
if (operator === '-' || operator === '+') {
return rawUnits.find(u => u)
}
return undefined
}
export const removeOnce = <T>(
element: T,
eqFn: (a: T, b: T) => boolean = equals
) => (list: Array<T>): Array<T> => {
const index = list.findIndex(e => eqFn(e, element))
if (index > -1) return remove<T>(index, 1)(list)
else return list
}
const simplify = (
unit: Unit,
eqFn: (a: string, b: string) => boolean = equals
): Unit =>
[...unit.numerators, ...unit.denominators].reduce(
({ numerators, denominators }, next) =>
numerators.find(u => eqFn(next, u)) &&
denominators.find(u => eqFn(next, u))
? {
numerators: removeOnce(next, eqFn)(numerators),
denominators: removeOnce(next, eqFn)(denominators)
}
: { numerators, denominators },
unit
)
const convertTable: { readonly [index: string]: number } = {
'mois/an': 12,
'jour/an': 365,
'jour/mois': 365 / 12,
'trimestre/an': 4,
'mois/trimestre': 3,
'jour/trimestre': (365 / 12) * 3,
'€/k€': 10 ** 3,
'g/kg': 10 ** 3,
'mg/g': 10 ** 3,
'mg/kg': 10 ** 6
}
function singleUnitConversionFactor(
from: string,
to: string
): number | undefined {
return (
convertTable[`${to}/${from}`] ||
(convertTable[`${from}/${to}`] && 1 / convertTable[`${from}/${to}`])
)
}
function unitsConversionFactor(from: string[], to: string[]): number {
let factor =
100 **
// Factor is mutliplied or divided 100 for each '%' in units
(to.filter(unit => unit === '%').length -
from.filter(unit => unit === '%').length)
;[factor] = from.reduce(
([value, toUnits], fromUnit) => {
const index = toUnits.findIndex(
toUnit => !!singleUnitConversionFactor(fromUnit, toUnit)
)
const factor = singleUnitConversionFactor(fromUnit, toUnits[index]) || 1
return [
value * factor,
[...toUnits.slice(0, index + 1), ...toUnits.slice(index + 1)]
]
},
[factor, to]
)
return factor
}
export function convertUnit(
from: Unit | undefined,
to: Unit | undefined,
value: number
): number
export function convertUnit(
from: Unit | undefined,
to: Unit | undefined,
value: Evaluation<number>
): Evaluation<number>
export function convertUnit(
from: Unit | undefined,
to: Unit | undefined,
value: number | Evaluation<number>
) {
if (!areUnitConvertible(from, to)) {
throw new Error(
`Impossible de convertir l'unité '${serializeUnit(
from
)}' en '${serializeUnit(to)}'`
)
}
if (!value) {
return value
}
const [fromSimplified, factorTo] = simplifyUnitWithValue(from || noUnit)
const [toSimplified, factorFrom] = simplifyUnitWithValue(to || noUnit)
return round(
((value * factorTo) / factorFrom) *
unitsConversionFactor(
fromSimplified.numerators,
toSimplified.numerators
) *
unitsConversionFactor(
toSimplified.denominators,
fromSimplified.denominators
)
)
}
const convertibleUnitClasses = [
['mois', 'an', 'jour', 'trimestre'],
['€', 'k€'],
['g', 'kg', 'mg']
]
function areSameClass(a: string, b: string) {
return (
a === b ||
convertibleUnitClasses.some(units => units.includes(a) && units.includes(b))
)
}
function round(value: number) {
return +value.toFixed(16)
}
export function simplifyUnit(unit: Unit): Unit {
const { numerators, denominators } = simplify(unit, areSameClass)
if (numerators.length && numerators.every(symb => symb === '%')) {
return { numerators: ['%'], denominators }
}
return {
numerators: without(['%'], numerators),
denominators: without(['%'], denominators)
}
}
function simplifyUnitWithValue(unit: Unit, value = 1): [Unit, number] {
const { denominators, numerators } = unit
const factor = unitsConversionFactor(numerators, denominators)
return [
simplify(
{
numerators: without(['%'], numerators),
denominators: without(['%'], denominators)
},
areSameClass
),
value ? round(value * factor) : value
]
}
export function areUnitConvertible(a: Unit | undefined, b: Unit | undefined) {
if (a == null || b == null) {
return true
}
const countByUnitClass = countBy((unit: string) => {
const classIndex = convertibleUnitClasses.findIndex(unitClass =>
unitClass.includes(unit)
)
return classIndex === -1 ? unit : '' + classIndex
})
const [numA, denomA, numB, denomB] = [
a.numerators,
a.denominators,
b.numerators,
b.denominators
].map(countByUnitClass)
const unitClasses = pipe(
map(keys),
flatten,
uniq
)([numA, denomA, numB, denomB])
return unitClasses.every(
unitClass =>
(numA[unitClass] || 0) - (denomA[unitClass] || 0) ===
(numB[unitClass] || 0) - (denomB[unitClass] || 0) || unitClass === '%'
)
}