mon-entreprise/source/engine/mecanisms.js

1065 lines
27 KiB
JavaScript
Raw Normal View History

2018-01-08 15:07:26 +00:00
import {
reduce,
path,
mergeWith,
2018-01-08 15:07:26 +00:00
objOf,
dissoc,
add,
find,
pluck,
map,
any,
equals,
is,
keys,
evolve,
curry,
filter,
pipe,
head,
isEmpty,
propEq,
prop,
has,
max,
min,
subtract,
sum,
isNil,
reject,
aperture,
sort,
toPairs,
reduced,
last
2018-01-08 15:07:26 +00:00
} from 'ramda'
import React from 'react'
import { Trans } from 'react-i18next'
2017-10-24 16:25:31 +00:00
import { anyNull, val } from './traverse-common-functions'
import { Node, SimpleRuleLink } from './mecanismViews/common'
2017-10-24 16:25:31 +00:00
import {
makeJsx,
evaluateNode,
rewriteNode,
evaluateArray,
evaluateArrayWithFilter,
evaluateObject,
parseObject,
collectNodeMissing,
mergeAllMissing,
mergeMissing,
bonus
2017-10-24 16:25:31 +00:00
} from './evaluation'
import {
findRuleByName,
disambiguateRuleReference,
findRuleByDottedName
} from './rules'
import 'react-virtualized/styles.css'
import Somme from './mecanismViews/Somme'
import Barème from './mecanismViews/Barème'
import Variations from './mecanismViews/Variations'
2018-02-21 12:46:38 +00:00
import Allègement from './mecanismViews/Allègement'
import Composantes from './mecanismViews/Composantes'
import { trancheValue } from './mecanisms/barème'
import buildSelectionView from './mecanismViews/Selection'
import uniroot from './uniroot'
2017-10-24 16:25:31 +00:00
let constantNode = constant => ({
nodeValue: constant,
// eslint-disable-next-line
2017-10-24 16:25:31 +00:00
jsx: nodeValue => <span className="value">{nodeValue}</span>
})
2017-07-28 12:24:29 +00:00
let decompose = (recurse, k, v) => {
2018-01-08 15:07:26 +00:00
let subProps = dissoc('composantes')(v),
2017-10-24 16:25:31 +00:00
explanation = v.composantes.map(c => ({
...recurse(
2018-01-08 15:07:26 +00:00
objOf(k, {
2017-10-24 16:25:31 +00:00
...subProps,
2018-01-08 15:07:26 +00:00
...dissoc('attributs')(c)
2017-10-24 16:25:31 +00:00
})
),
composante: c.nom ? { nom: c.nom } : c.attributs
}))
let filter = situationGate => c =>
!situationGate('sys.filter') ||
!c.composante ||
((!c.composante['dû par'] ||
c.composante['dû par'] == situationGate('sys.filter')) &&
(!c.composante['impôt sur le revenu'] ||
c.composante['impôt sur le revenu'] == situationGate('sys.filter')))
return {
explanation,
jsx: Composantes,
2018-01-08 15:07:26 +00:00
evaluate: evaluateArrayWithFilter(filter, add, 0),
category: 'mecanism',
name: 'composantes',
type: 'numeric'
}
}
let devariateExplanation = (recurse, mecanismKey, v) => {
2018-08-07 12:36:42 +00:00
let fixedProps = dissoc('variations')(v),
explanation = v.variations.map(({ si, alors, sinon }) => ({
consequence: recurse({
2018-08-07 12:36:42 +00:00
[mecanismKey]: {
...fixedProps,
...(sinon || alors)
2018-08-07 12:36:42 +00:00
}
}),
condition: sinon ? undefined : recurse(si)
2017-10-24 16:25:31 +00:00
}))
2017-07-28 12:24:29 +00:00
return explanation
}
/* @devariate = true => This function will produce variations of a same mecanism (e.g. product) that share some common properties */
export let mecanismVariations = (recurse, k, v, devariate) => {
let explanation = devariate
? devariateExplanation(recurse, k, v)
: v.map(({ si, alors, sinon }) =>
sinon !== undefined
? { consequence: recurse(sinon), condition: undefined }
: { consequence: recurse(alors), condition: recurse(si) }
)
let evaluate = (cache, situationGate, parsedRules, node) => {
let evaluateVariation = map(prop =>
prop === undefined
? undefined
: evaluateNode(cache, situationGate, parsedRules, prop)
),
evaluatedExplanation = map(evaluateVariation, node.explanation),
// mark the satisfied variation if any in the explanation
[, resolvedExplanation] = reduce(
([resolved, result], variation) =>
resolved
? [true, [...result, variation]]
: variation.condition == undefined
? [true, [...result, { ...variation, satisfied: true }]] // We've reached the eventual defaut case
: variation.condition.nodeValue === null
? [true, [...result, variation]] // one case has missing variables => we can't go further
: variation.condition.nodeValue === true
? [true, [...result, { ...variation, satisfied: true }]]
: [false, [...result, variation]],
[false, []]
)(evaluatedExplanation),
satisfiedVariation = resolvedExplanation.find(v => v.satisfied),
nodeValue = satisfiedVariation
? satisfiedVariation.consequence.nodeValue
: null
2017-07-28 12:24:29 +00:00
let leftMissing = mergeAllMissing(
reject(isNil, pluck('condition', evaluatedExplanation))
),
candidateVariations = filter(
node => !node.condition || node.condition.nodeValue !== false,
evaluatedExplanation
),
rightMissing = mergeAllMissing(pluck('consequence', candidateVariations)),
missingVariables = satisfiedVariation
? collectNodeMissing(satisfiedVariation.consequence)
: mergeMissing(bonus(leftMissing), rightMissing)
2017-07-28 12:24:29 +00:00
return rewriteNode(node, nodeValue, resolvedExplanation, missingVariables)
2017-07-28 12:24:29 +00:00
}
// TODO - find an appropriate representation
return {
explanation,
evaluate,
jsx: Variations,
2017-07-28 12:24:29 +00:00
category: 'mecanism',
name: 'variations',
type: 'numeric'
}
}
export let mecanismOneOf = (recurse, k, v) => {
if (!is(Array, v)) throw new Error('should be array')
2018-01-08 15:07:26 +00:00
let explanation = map(recurse, v)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism conditions list"
2017-10-24 16:25:31 +00:00
name="une de ces conditions"
value={nodeValue}
child={
<ul>
2017-10-24 16:25:31 +00:00
{explanation.map(item => (
<li key={item.name || item.text}>{makeJsx(item)}</li>
))}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
let evaluate = (cache, situationGate, parsedRules, node) => {
let evaluateOne = child =>
evaluateNode(cache, situationGate, parsedRules, child),
2018-01-08 15:07:26 +00:00
explanation = map(evaluateOne, node.explanation),
values = pluck('nodeValue', explanation),
nodeValue = any(equals(true), values)
2017-10-24 16:25:31 +00:00
? true
: any(equals(null), values)
? null
: false,
// Unlike most other array merges of missing variables this is a "flat" merge
// because "one of these conditions" tend to be several tests of the same variable
// (e.g. contract type is one of x, y, z)
missingVariables =
nodeValue == null
? reduce(mergeWith(max), {}, map(collectNodeMissing, explanation))
: {}
2017-10-24 16:25:31 +00:00
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
return {
evaluate,
jsx,
explanation,
category: 'mecanism',
name: 'une de ces conditions',
type: 'boolean'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismAllOf = (recurse, k, v) => {
if (!is(Array, v)) throw new Error('should be array')
2018-01-08 15:07:26 +00:00
let explanation = map(recurse, v)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism conditions list"
2017-10-24 16:25:31 +00:00
name="toutes ces conditions"
value={nodeValue}
child={
<ul>
2017-10-24 16:25:31 +00:00
{explanation.map(item => (
<li key={item.name || item.text}>{makeJsx(item)}</li>
))}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
let evaluate = (cache, situationGate, parsedRules, node) => {
let evaluateOne = child =>
evaluateNode(cache, situationGate, parsedRules, child),
2018-01-08 15:07:26 +00:00
explanation = map(evaluateOne, node.explanation),
values = pluck('nodeValue', explanation),
nodeValue = any(equals(false), values)
? false // court-circuit
: any(equals(null), values)
? null
: true,
missingVariables = nodeValue == null ? mergeAllMissing(explanation) : {}
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
return {
evaluate: evaluate,
jsx,
explanation,
category: 'mecanism',
name: 'toutes ces conditions',
type: 'boolean'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismNumericalSwitch = (recurse, k, v) => {
// Si "l'aiguillage" est une constante ou une référence directe à une variable;
// l'utilité de ce cas correspond à un appel récursif au mécanisme
2018-01-08 15:07:26 +00:00
if (is(String, v)) return recurse(v)
2018-01-08 15:07:26 +00:00
if (!is(Object, v) || keys(v).length == 0) {
throw new Error(
'Le mécanisme "aiguillage numérique" et ses sous-logiques doivent contenir au moins une proposition'
)
}
// les termes sont les couples (condition, conséquence) de l'aiguillage numérique
2018-01-08 15:07:26 +00:00
let terms = toPairs(v)
// la conséquence peut être un 'string' ou un autre aiguillage numérique
let parseCondition = ([condition, consequence]) => {
2017-10-24 16:25:31 +00:00
let conditionNode = recurse(condition), // can be a 'comparison', a 'variable', TODO a 'negation'
consequenceNode = mecanismNumericalSwitch(recurse, condition, consequence)
let evaluate = (cache, situationGate, parsedRules, node) => {
2018-01-08 15:07:26 +00:00
let explanation = evolve(
{
condition: curry(evaluateNode)(cache, situationGate, parsedRules),
consequence: curry(evaluateNode)(cache, situationGate, parsedRules)
},
node.explanation
),
leftMissing = explanation.condition.missingVariables,
investigate = explanation.condition.nodeValue !== false,
rightMissing = investigate
? explanation.consequence.missingVariables
: {},
missingVariables = mergeMissing(bonus(leftMissing), rightMissing)
return {
...node,
explanation,
missingVariables,
nodeValue: explanation.consequence.nodeValue,
condValue: explanation.condition.nodeValue
}
}
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, { condition, consequence }) => (
<div className="condition">
2017-10-24 16:25:31 +00:00
{makeJsx(condition)}
<div>{makeJsx(consequence)}</div>
</div>
)
return {
2017-10-24 16:25:31 +00:00
evaluate,
jsx,
explanation: { condition: conditionNode, consequence: consequenceNode },
category: 'condition',
text: condition,
condition: conditionNode,
type: 'boolean'
}
}
let evaluateTerms = (cache, situationGate, parsedRules, node) => {
let evaluateOne = child =>
evaluateNode(cache, situationGate, parsedRules, child),
2018-01-08 15:07:26 +00:00
explanation = map(evaluateOne, node.explanation),
nonFalsyTerms = filter(node => node.condValue !== false, explanation),
getFirst = o =>
pipe(
head,
prop(o)
)(nonFalsyTerms),
nodeValue =
// voilà le "numérique" dans le nom de ce mécanisme : il renvoie zéro si aucune condition n'est vérifiée
2018-01-08 15:07:26 +00:00
isEmpty(nonFalsyTerms)
2017-10-24 16:25:31 +00:00
? 0
: // c'est un 'null', on renvoie null car des variables sont manquantes
getFirst('condValue') == null
? null
: // c'est un true, on renvoie la valeur de la conséquence
getFirst('nodeValue'),
choice = find(node => node.condValue, explanation),
missingVariables = choice
? choice.missingVariables
: mergeAllMissing(explanation)
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
2018-01-08 15:07:26 +00:00
let explanation = map(parseCondition, terms)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism numericalSwitch list"
name="aiguillage numérique"
value={nodeValue}
child={
<ul>
2017-10-24 16:25:31 +00:00
{explanation.map(item => (
<li key={item.name || item.text}>{makeJsx(item)}</li>
))}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
return {
evaluate: evaluateTerms,
jsx,
explanation,
category: 'mecanism',
2017-10-24 16:25:31 +00:00
name: 'aiguillage numérique',
type: 'boolean || numeric' // lol !
}
}
export let findInversion = (situationGate, parsedRules, v, dottedName) => {
let inversions = v.avec
if (!inversions)
throw new Error(
"Une formule d'inversion doit préciser _avec_ quoi on peut inverser la variable"
)
/*
Quelle variable d'inversion possible a sa valeur renseignée dans la situation courante ?
Ex. s'il nous est demandé de calculer le salaire de base, est-ce qu'un candidat à l'inversion, comme
le salaire net, a été renseigné ?
*/
let fixedObjective = inversions
.map(i =>
disambiguateRuleReference(
parsedRules,
parsedRules.find(propEq('dottedName', dottedName)),
i
)
)
.map(name => {
let userInput = situationGate(name) != undefined
let rule = findRuleByDottedName(parsedRules, name)
/* When the fixedObjectiveValue is null, the inversion can't be done : the user needs to set the target's value
* But the objectiveRule can also have an 'alternative' property,
* which must point to a rule whose value either is set by the user,
* or is calculated according to a formula that does not depend on the rule being inversed.
* This alternative's value will be used as a target.
* */
let alternativeRule =
!userInput &&
rule.alternative &&
findRuleByDottedName(parsedRules, rule.alternative)
if (!userInput && !alternativeRule) return null
return {
fixedObjectiveRule: rule,
fixedObjectiveValue: userInput,
alternativeRule
}
})
.find(candidate => candidate != null)
return fixedObjective
}
let doInversion = (oldCache, situationGate, parsedRules, v, dottedName) => {
let inversion = findInversion(situationGate, parsedRules, v, dottedName)
if (!inversion)
return {
missingVariables: { [dottedName]: 1 },
nodeValue: null
}
let { fixedObjectiveValue, fixedObjectiveRule, alternativeRule } = inversion
let evaluatedAlternative =
alternativeRule &&
evaluateNode(oldCache, situationGate, parsedRules, alternativeRule)
if (evaluatedAlternative && evaluatedAlternative.nodeValue == null)
return {
missingVariables: evaluatedAlternative.missingVariables,
nodeValue: null
}
let objectiveValue = evaluatedAlternative
? evaluatedAlternative.nodeValue
: fixedObjectiveValue
let inversionCache = {}
let fx = x => {
inversionCache = { parseLevel: oldCache.parseLevel + 1, op: '<' }
return evaluateNode(
inversionCache, // with an empty cache
n => (dottedName === n ? x : situationGate(n)),
parsedRules,
fixedObjectiveRule
)
}
// si fx renvoie null pour une valeur numérique standard, disons 1000, on peut
// considérer que l'inversion est impossible du fait de variables manquantes
// TODO fx peut être null pour certains x, et valide pour d'autres : on peut implémenter ici le court-circuit
let attempt = fx(1000)
if (attempt.nodeValue == null) {
return attempt
}
let tolerance = 0.1,
// cette fonction détermine la racine d'une fonction sans faire trop d'itérations
nodeValue = uniroot(
x => {
let y = fx(x)
return y.nodeValue - objectiveValue
},
1,
1000000000,
tolerance,
10
)
return {
nodeValue,
missingVariables: {},
inversionCache
}
}
export let mecanismInversion = dottedName => (recurse, k, v) => {
let evaluate = (cache, situationGate, parsedRules, node) => {
let inversion =
// avoid the inversion loop !
situationGate(dottedName) == undefined &&
doInversion(cache, situationGate, parsedRules, v, dottedName),
// TODO - ceci n'est pas vraiment satisfaisant
nodeValue = situationGate(dottedName)
? Number.parseFloat(situationGate(dottedName))
: inversion.nodeValue,
missingVariables = inversion.missingVariables
let evaluatedNode = rewriteNode(node, nodeValue, null, missingVariables)
// TODO - we need this so that ResultsGrid will work, but it's
// just not right
toPairs(inversion.inversionCache).map(([k, v]) => (cache[k] = v))
return evaluatedNode
}
return {
2018-02-06 16:02:13 +00:00
...v,
evaluate,
// eslint-disable-next-line
jsx: nodeValue => (
<Node
classes="mecanism inversion"
name="inversion"
value={nodeValue}
child={
<div>
<div>avec</div>
<ul>
{v.avec.map(recurse).map(el => (
<li key={el.name}>{makeJsx(el)}</li>
))}
</ul>
</div>
}
/>
),
category: 'mecanism',
name: 'inversion',
type: 'numeric'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismSum = (recurse, k, v) => {
let explanation = v.map(recurse)
2018-01-08 15:07:26 +00:00
let evaluate = evaluateArray(add, 0)
return {
evaluate,
// eslint-disable-next-line
2017-10-24 16:25:31 +00:00
jsx: (nodeValue, explanation) => (
<Somme nodeValue={nodeValue} explanation={explanation} />
),
explanation,
category: 'mecanism',
name: 'somme',
type: 'numeric'
}
}
export let mecanismReduction = (recurse, k, v) => {
let objectShape = {
assiette: false,
abattement: constantNode(0),
franchise: constantNode(0)
}
let effect = ({ assiette, abattement, franchise, décote }) => {
let v_assiette = val(assiette)
if (v_assiette == null) return null
let montantFranchiséDécoté =
val(franchise) && v_assiette < val(franchise)
? 0
: décote
? do {
let plafond = val(décote.plafond),
taux = val(décote.taux)
v_assiette > plafond
? v_assiette
: max(0, (1 + taux) * v_assiette - taux * plafond)
}
: v_assiette
return abattement
? val(abattement) == null
? montantFranchiséDécoté === 0
? 0
: null
: abattement.category === 'percentage'
? max(
0,
montantFranchiséDécoté - val(abattement) * montantFranchiséDécoté
)
: max(0, montantFranchiséDécoté - val(abattement))
: montantFranchiséDécoté
}
let base = parseObject(recurse, objectShape, v),
explanation = v.décote
? {
...base,
décote: map(recurse, v.décote)
}
: base,
evaluate = evaluateObject(objectShape, effect)
return {
evaluate,
2018-02-21 12:46:38 +00:00
jsx: Allègement,
explanation,
category: 'mecanism',
name: 'allègement',
type: 'numeric'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismProduct = (recurse, k, v) => {
if (v.composantes) {
//mécanisme de composantes. Voir known-mecanisms.md/composantes
return decompose(recurse, k, v)
}
if (v.variations) {
return mecanismVariations(recurse, k, v, true)
}
let objectShape = {
2017-10-24 16:25:31 +00:00
assiette: false,
taux: constantNode(1),
facteur: constantNode(1),
plafond: constantNode(Infinity)
}
2017-10-24 16:25:31 +00:00
let effect = ({ assiette, taux, facteur, plafond }) => {
let mult = (base, rate, facteur, plafond) =>
Math.min(base, plafond) * rate * facteur
return val(taux) === 0 ||
val(taux) === false ||
val(assiette) === 0 ||
val(facteur) === 0
? 0
: anyNull([taux, assiette, facteur, plafond])
? null
: mult(val(assiette), val(taux), val(facteur), val(plafond))
}
2017-10-24 16:25:31 +00:00
let explanation = parseObject(recurse, objectShape, v),
evaluate = evaluateObject(objectShape, effect)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism multiplication"
name="multiplication"
value={nodeValue}
child={
<ul className="properties">
<li key="assiette">
<span className="key">
<Trans>assiette</Trans>:{' '}
</span>
<span className="value">{makeJsx(explanation.assiette)}</span>
</li>
2017-10-24 16:25:31 +00:00
{(explanation.taux.nodeValue != 1 ||
explanation.taux.category == 'calcExpression') && (
<li key="taux">
<span className="key">
<Trans>taux</Trans>:{' '}
</span>
<span className="value">{makeJsx(explanation.taux)}</span>
</li>
)}
2017-10-24 16:25:31 +00:00
{(explanation.facteur.nodeValue != 1 ||
explanation.facteur.category == 'calcExpression') && (
<li key="facteur">
<span className="key">
<Trans>facteur</Trans>:{' '}
</span>
<span className="value">{makeJsx(explanation.facteur)}</span>
</li>
)}
2017-10-24 16:25:31 +00:00
{explanation.plafond.nodeValue != Infinity && (
<li key="plafond">
<span className="key">
<Trans>plafond</Trans>:{' '}
</span>
2017-10-24 16:25:31 +00:00
<span className="value">{makeJsx(explanation.plafond)}</span>
</li>
)}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
return {
evaluate,
jsx,
explanation,
category: 'mecanism',
name: 'multiplication',
type: 'numeric'
}
}
2018-07-09 10:01:06 +00:00
/* on réécrit en une syntaxe plus bas niveau mais plus régulière les tranches :
`en-dessous de: 1`
devient
```
de: 0
à: 1
```
*/
let desugarScale = recurse => tranches =>
2018-07-09 10:01:06 +00:00
tranches
.map(t =>
has('en-dessous de')(t)
? { ...t, de: 0, à: t['en-dessous de'] }
: has('au-dessus de')(t)
? { ...t, de: t['au-dessus de'], à: Infinity }
: t
)
2018-01-08 15:07:26 +00:00
.map(evolve({ taux: recurse }))
2018-07-09 10:01:06 +00:00
export let mecanismLinearScale = (recurse, k, v) => {
if (v.composantes) {
//mécanisme de composantes. Voir known-mecanisms.md/composantes
return decompose(recurse, k, v)
}
if (v.variations) {
return mecanismVariations(recurse, k, v, true)
}
2018-07-09 10:01:06 +00:00
let tranches = desugarScale(recurse)(v['tranches']),
objectShape = {
assiette: false
}
let effect = ({ assiette, tranches }) => {
if (val(assiette) === null) return null
let roundedAssiette = Math.round(val(assiette))
2018-07-09 10:01:06 +00:00
let matchedTranche = tranches.find(
({ de: min, à: max }) => roundedAssiette >= min && roundedAssiette <= max
2018-07-09 10:01:06 +00:00
)
if (!matchedTranche) return 0
if (matchedTranche.taux)
return matchedTranche.taux.nodeValue * val(assiette)
return matchedTranche.montant
}
2018-07-09 10:01:06 +00:00
let explanation = {
...parseObject(recurse, objectShape, v),
tranches
},
evaluate = evaluateObject(objectShape, effect)
return {
evaluate,
jsx: Barème('linéaire'),
2018-07-09 10:01:06 +00:00
explanation,
category: 'mecanism',
name: 'barème linéaire',
barème: 'en taux',
type: 'numeric'
}
}
export let mecanismScale = (recurse, k, v) => {
// Sous entendu : barème en taux marginaux.
if (v.composantes) {
//mécanisme de composantes. Voir known-mecanisms.md/composantes
return decompose(recurse, k, v)
}
if (v.variations) {
2018-09-05 16:07:58 +00:00
return mecanismVariations(recurse, k, v, true)
2018-07-09 10:01:06 +00:00
}
let tranches = desugarScale(recurse)(v['tranches']),
objectShape = {
assiette: false,
'multiplicateur des tranches': constantNode(1)
}
2017-10-24 16:25:31 +00:00
let effect = ({
assiette,
'multiplicateur des tranches': multiplicateur,
tranches
}) => {
let nulled = val(assiette) == null || val(multiplicateur) == null
2017-10-24 16:25:31 +00:00
return nulled
? null
: sum(tranches.map(trancheValue('marginal')(assiette, multiplicateur)))
2017-10-24 16:25:31 +00:00
}
let explanation = {
2017-10-24 16:25:31 +00:00
...parseObject(recurse, objectShape, v),
tranches
},
evaluate = evaluateObject(objectShape, effect)
return {
evaluate,
jsx: Barème('marginal'),
explanation,
category: 'mecanism',
name: 'barème',
barème: 'en taux marginaux',
type: 'numeric'
}
}
export let mecanismContinuousScale = (recurse, k, v) => {
let objectShape = {
assiette: false,
multiplicateur: constantNode(1)
}
let effect = ({ assiette, multiplicateur, points }) => {
if (anyNull([assiette, multiplicateur])) return null
//We'll build a linear function given the two constraints that must be respected
return pipe(
toPairs,
// we don't rely on the sorting of objects
sort(([k1], [k2]) => k1 - k2),
points => [...points, [Infinity, last(points)[1]]],
aperture(2),
reduce((_, [[lowerLimit, lowerRate], [upperLimit, upperRate]]) => {
let x1 = val(multiplicateur) * lowerLimit,
x2 = val(multiplicateur) * upperLimit,
y1 = val(assiette) * val(recurse(lowerRate)),
y2 = val(assiette) * val(recurse(upperRate))
if (val(assiette) > x1 && val(assiette) <= x2) {
// Outside of these 2 limits, it's a linear function a * x + b
let a = (y2 - y1) / (x2 - x1),
b = y1 - x1 * a
return reduced(a * val(assiette) + b)
}
}, 0)
)(points)
}
let explanation = {
...parseObject(recurse, objectShape, v),
points: v.points
},
evaluate = evaluateObject(objectShape, effect)
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism barèmeContinu"
name="barèmeContinu"
value={nodeValue}
child={<div> Barème continu </div>}
/>
)
return {
evaluate,
jsx,
explanation,
category: 'mecanism',
name: 'barème continu',
type: 'numeric'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismMax = (recurse, k, v) => {
let explanation = v.map(recurse)
2018-01-08 15:07:26 +00:00
let evaluate = evaluateArray(max, Number.NEGATIVE_INFINITY)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism list maximum"
name="le maximum de"
value={nodeValue}
child={
<ul>
2017-10-24 16:25:31 +00:00
{explanation.map((item, i) => (
<li key={i}>
<div className="description">{v[i].description}</div>
{makeJsx(item)}
</li>
))}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
return {
evaluate,
jsx,
explanation,
type: 'numeric',
category: 'mecanism',
name: 'le maximum de'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismMin = (recurse, k, v) => {
let explanation = v.map(recurse)
2018-01-08 15:07:26 +00:00
let evaluate = evaluateArray(min, Infinity)
2017-10-24 16:25:31 +00:00
let jsx = (nodeValue, explanation) => (
<Node
classes="mecanism list minimum"
name="le minimum de"
value={nodeValue}
child={
<ul>
2017-10-24 16:25:31 +00:00
{explanation.map((item, i) => (
<li key={i}>
<div className="description">{v[i].description}</div>
{makeJsx(item)}
</li>
))}
</ul>
}
/>
2017-10-24 16:25:31 +00:00
)
return {
evaluate,
jsx,
explanation,
type: 'numeric',
category: 'mecanism',
name: 'le minimum de'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismComplement = (recurse, k, v) => {
if (v.composantes) {
//mécanisme de composantes. Voir known-mecanisms.md/composantes
return decompose(recurse, k, v)
}
2017-10-24 16:25:31 +00:00
let objectShape = { cible: false, montant: false }
let effect = ({ cible, montant }) => {
let nulled = val(cible) == null
2018-01-08 15:07:26 +00:00
return nulled ? null : subtract(val(montant), min(val(cible), val(montant)))
}
2017-10-24 16:25:31 +00:00
let explanation = parseObject(recurse, objectShape, v)
return {
2017-10-24 16:25:31 +00:00
evaluate: evaluateObject(objectShape, effect),
explanation,
type: 'numeric',
category: 'mecanism',
name: 'complément pour atteindre',
// eslint-disable-next-line
2017-10-24 16:25:31 +00:00
jsx: (nodeValue, explanation) => (
<Node
classes="mecanism list complement"
name="complément"
value={nodeValue}
child={
<ul className="properties">
<li key="cible">
<span className="key">
<Trans>cible</Trans>:{' '}
</span>
2017-10-24 16:25:31 +00:00
<span className="value">{makeJsx(explanation.cible)}</span>
</li>
<li key="mini">
<span className="key">
<Trans>montant à atteindre</Trans>:{' '}
</span>
2017-10-24 16:25:31 +00:00
<span className="value">{makeJsx(explanation.montant)}</span>
</li>
</ul>
}
/>
)
}
}
2017-10-24 16:25:31 +00:00
export let mecanismSelection = (recurse, k, v) => {
if (v.composantes) {
//mécanisme de composantes. Voir known-mecanisms.md/composantes
return decompose(recurse, k, v)
}
let dataSourceName = v['données']
let dataSearchField = v['dans']
let dataTargetName = v['renvoie']
let explanation = recurse(v['cherche'])
let evaluate = (cache, situationGate, parsedRules, node) => {
let explanation = evaluateNode(
cache,
situationGate,
parsedRules,
node.explanation
),
dataSource = findRuleByName(parsedRules, dataSourceName),
data = dataSource ? dataSource['data'] : null,
dataKey = explanation.nodeValue,
found =
2017-10-24 16:25:31 +00:00
data && dataKey && dataSearchField
? find(item => item[dataSearchField] == dataKey, data)
2017-10-24 16:25:31 +00:00
: null,
// return 0 if we found a match for the lookup but not for the specific field,
// so that component sums don't sum to null
nodeValue =
(found &&
found[dataTargetName] &&
Number.parseFloat(found[dataTargetName]) / 100) ||
0,
missingVariables = explanation.missingVariables
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
let SelectionView = buildSelectionView(dataTargetName)
return {
evaluate,
explanation,
// eslint-disable-next-line
jsx: (nodeValue, explanation) => (
<SelectionView nodeValue={nodeValue} explanation={explanation} />
)
}
}
export let mecanismSynchronisation = (recurse, k, v) => {
let evaluate = (cache, situationGate, parsedRules, node) => {
let APIExplanation = evaluateNode(
cache,
situationGate,
parsedRules,
node.explanation.API
)
let nodeValue =
val(APIExplanation) == null
? null
: path(v.chemin.split(' . '))(val(APIExplanation))
let missingVariables =
val(APIExplanation) === null ? { [APIExplanation.dottedName]: 1 } : {}
let explanation = { ...v, API: APIExplanation }
return rewriteNode(node, nodeValue, explanation, missingVariables)
}
return {
explanation: { ...v, API: recurse(v.API) },
evaluate,
jsx: function Synchronisation(nodeValue, explanation) {
return (
<p>
Obtenu à partir de la saisie <SimpleRuleLink rule={explanation.API} />
</p>
)
},
category: 'mecanism',
name: 'synchronisation'
}
}
2017-10-24 16:25:31 +00:00
export let mecanismError = (recurse, k, v) => {
throw new Error("Le mécanisme '" + k + "' est inconnu !" + v)
}