mon-entreprise/source/engine/units.js

67 lines
1.6 KiB
JavaScript

import { remove, isEmpty, unnest } from 'ramda'
export let parseUnit = string => {
let [a, b = ''] = string.split('/'),
result = {
numerators: a !== '' ? [a] : [],
denominators: b !== '' ? [b] : []
}
return result
}
export let serialiseUnit = ({ numerators = [], denominators = [] }) => {
let n = !isEmpty(numerators)
let d = !isEmpty(denominators)
return !n && !d
? ''
: n && !d
? numerators.join('')
: !n && d
? `/${denominators.join('')}`
: `${numerators.join('')} / ${denominators.join('')}`
}
let noUnit = { numerators: [], denominators: [] }
export let inferUnit = (operator, rawUnits) => {
let 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 null
}
export let removeOnce = element => list => {
let index = list.indexOf(element)
if (index > -1) return remove(index, 1)(list)
else return list
}
let simplify = unit =>
[...unit.numerators, ...unit.denominators].reduce(
({ numerators, denominators }, next) =>
numerators.includes(next) && denominators.includes(next)
? {
numerators: removeOnce(next)(numerators),
denominators: removeOnce(next)(denominators)
}
: { numerators, denominators },
unit
)