2019-07-11 14:29:05 +00:00
|
|
|
import { remove, isEmpty, unnest } from 'ramda'
|
2019-06-05 15:54:52 +00:00
|
|
|
|
|
|
|
export let parseUnit = string => {
|
2019-06-11 10:03:45 +00:00
|
|
|
let [a, b = ''] = string.split('/'),
|
|
|
|
result = {
|
|
|
|
numerators: a !== '' ? [a] : [],
|
|
|
|
denominators: b !== '' ? [b] : []
|
|
|
|
}
|
|
|
|
return result
|
2019-06-05 15:54:52 +00:00
|
|
|
}
|
|
|
|
|
2019-07-11 15:07:34 +00:00
|
|
|
export let serialiseUnit = rawUnit => {
|
|
|
|
let unit = simplify(rawUnit),
|
|
|
|
{ numerators = [], denominators = [] } = unit
|
2019-06-05 15:54:52 +00:00
|
|
|
let n = !isEmpty(numerators)
|
|
|
|
let d = !isEmpty(denominators)
|
2019-07-11 15:07:34 +00:00
|
|
|
let string =
|
|
|
|
!n && !d
|
|
|
|
? ''
|
|
|
|
: n && !d
|
|
|
|
? numerators.join('')
|
|
|
|
: !n && d
|
|
|
|
? `/${denominators.join('')}`
|
|
|
|
: `${numerators.join('')} / ${denominators.join('')}`
|
|
|
|
|
2019-07-11 16:25:08 +00:00
|
|
|
// the unit '%' is only displayed when it is the only unit
|
2019-07-11 15:07:34 +00:00
|
|
|
if (string.length > 1) return string.replace(/%/g, '')
|
|
|
|
return string
|
2019-06-05 15:54:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let noUnit = { numerators: [], denominators: [] }
|
2019-07-11 14:29:05 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
])
|
|
|
|
}
|
2019-06-05 15:54:52 +00:00
|
|
|
|
2019-07-11 14:29:05 +00:00
|
|
|
if (operator === '-' || operator === '+') {
|
|
|
|
return rawUnits.find(u => u)
|
|
|
|
}
|
|
|
|
|
|
|
|
return null
|
|
|
|
}
|
2019-06-05 15:54:52 +00:00
|
|
|
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
|
|
|
|
)
|