import React from 'react' import { Leaf } from './mecanismViews/common' import { findRuleByDottedName, disambiguateRuleReference } from './rules' import { evaluateNode, rewriteNode, makeJsx } from './evaluation' import { evaluateVariable } from './variables' export let treatVariable = (rules, rule, filter) => parseResult => { let evaluate = (cache, situation, parsedRules, node) => { let dottedName = node.dottedName, // On va vérifier dans le cache courant, dict, si la variable n'a pas été déjà évaluée // En effet, l'évaluation dans le cas d'une variable qui a une formule, est coûteuse ! cacheName = dottedName + (filter ? '.' + filter : ''), cached = cache[cacheName] if (cached) { return cached } let variable = findRuleByDottedName(parsedRules, dottedName), variableHasFormula = variable.formule != null, variableHasCond = variable['applicable si'] != null || variable['non applicable si'] != null, situationValue = evaluateVariable(situation, dottedName, variable), needsEvaluation = situationValue == null && (variableHasCond || variableHasFormula), // evaluateVariable renvoit la valeur déduite de la situation courante renseignée par l'utilisateur explanation = needsEvaluation ? evaluateNode(cache, situation, parsedRules, variable) : variable let nodeValue let missingVariables // SITUATION 1 : La variable est directement renseignée if (situationValue != null) { nodeValue = situationValue missingVariables = {} } // SITUATION 2 : La variable est calculée if (situationValue == null && variableHasFormula) { nodeValue = explanation.nodeValue missingVariables = explanation.missingVariables } // SITUATION 3 : La variable est une question sans condition dont la valeur n'a pas été renseignée if (situationValue == null && !variableHasFormula && !variableHasCond) { nodeValue = null missingVariables = { [dottedName]: 1 } } // SITUATION 4 : La variable est une question avec conditions if (situationValue == null && !variableHasFormula && variableHasCond) { // SITUATION 4.1 : La condition est connue et vrai if (explanation.isApplicable) { ;(nodeValue = explanation.nodeValue), (missingVariables = { [dottedName]: 1 }) } // SITUATION 4.2 : La condition est connue et fausse if (explanation.isApplicable === false) { nodeValue = explanation.nodeValue missingVariables = {} } // SITUATION 4.3 : La condition n'est pas connue if (explanation.isApplicable == null) { nodeValue = null missingVariables = explanation.missingVariables } } cache[cacheName] = rewriteNode( node, nodeValue, explanation, missingVariables ) return cache[cacheName] } let { fragments } = parseResult, variablePartialName = fragments.join(' . '), dottedName = disambiguateRuleReference(rules, rule, variablePartialName) return { evaluate, jsx: nodeValue => ( ), name: variablePartialName, category: 'variable', fragments, dottedName, type: 'boolean | numeric' } } // TODO - this is becoming overly specific export let treatFilteredVariable = (rules, rule) => (filter, parseResult) => { let evaluateFiltered = originalEval => ( cache, situation, parsedRules, node ) => { let newSituation = name => (name == 'sys.filter' ? filter : situation(name)) return originalEval(cache, newSituation, parsedRules, node) } let node = treatVariable(rules, rule, filter)(parseResult), // Decorate node with the composante filter (either who is paying, either tax free) cotisation = { ...node.cotisation, 'dû par': filter, 'impôt sur le revenu': filter } return { ...node, cotisation, evaluate: evaluateFiltered(node.evaluate) } } export let treatNegatedVariable = variable => { let evaluate = (cache, situation, parsedRules, node) => { let explanation = evaluateNode( cache, situation, parsedRules, node.explanation ), nodeValue = explanation.nodeValue == null ? null : !explanation.nodeValue, missingVariables = explanation.missingVariables return rewriteNode(node, nodeValue, explanation, missingVariables) } let jsx = (nodeValue, explanation) => ( Non{' '} {makeJsx(explanation)} } /> ) return { evaluate, jsx, category: 'mecanism', name: 'négation', type: 'boolean', explanation: variable } }