mon-entreprise/source/reducers/reduceSteps.js

201 lines
5.9 KiB
JavaScript

import { path, head, reject, concat, without, length, map } from 'ramda'
import { rules, collectDefaults, rulesFr } from 'Engine/rules'
import {
getNextSteps,
collectMissingVariablesByTarget
} from 'Engine/generateQuestions'
import { analyseMany, parseAll } from 'Engine/traverse'
export default (tracker, flatRules, answerSource) => (state, action) => {
state.flatRules = flatRules
// Optimization - don't parse on each analysis
if (!state.parsedRules) {
state.parsedRules = parseAll(flatRules)
}
// TODO
if (action.type == 'CHANGE_LANG') {
if (action.lang == 'fr') {
flatRules = rulesFr
} else flatRules = rules
return {
...state,
flatRules
}
}
if (
![
'SET_CONVERSATION_TARGETS',
'STEP_ACTION',
'USER_INPUT_UPDATE',
'START_CONVERSATION',
'SET_ACTIVE_TARGET_INPUT'
].includes(action.type)
)
return state
if (path(['form', 'conversation', 'syncErrors'], state)) return state
// Most rules have default values
let rulesDefaults = collectDefaults(flatRules),
situationWithDefaults = assume(answerSource, rulesDefaults)
let analysis = analyseMany(state.parsedRules, state.targetNames)(
situationWithDefaults(state)
)
let userInput = state =>
state.activeTargetInput +
':' +
answerSource(state)(state.activeTargetInput)
if (action.type === 'USER_INPUT_UPDATE') {
tracker.push(['trackEvent', 'input', userInput(state)])
return {
...state,
analysis,
situationGate: situationWithDefaults(state)
}
}
let nextStepsAnalysis = analyseMany(state.parsedRules, state.targetNames)(
answerSource(state)
),
missingVariablesByTarget = collectMissingVariablesByTarget(
nextStepsAnalysis.targets
),
nextSteps = getNextSteps(missingVariablesByTarget),
currentQuestion = head(nextSteps)
let newState = {
...state,
analysis,
situationGate: situationWithDefaults(state),
explainedVariable: null,
nextSteps,
currentQuestion,
foldedSteps:
action.type === 'SET_CONVERSATION_TARGETS' && action.reset
? []
: state.foldedSteps
}
if (action.type === 'SET_ACTIVE_TARGET_INPUT') {
tracker.push(['trackEvent', 'select', state.activeTargetInput])
}
if (action.type === 'START_CONVERSATION') {
tracker.push(['trackEvent', 'refine', userInput(state)])
}
if (
['SET_ACTIVE_TARGET_INPUT', 'START_CONVERSATION'].includes(action.type)
) {
// Si rien n'a été renseigné (stillBlank) on renvoie state et pas newState
// pour éviter que les cases blanches disparaissent, c'est un hack…
let stillBlank =
state.activeTargetInput &&
!answerSource(state)(state.activeTargetInput)
// Il faut recalculer les missingVariablesByTarget à chaque changement d'objectif
// car les variables manquantes du salaire de base calculé par inversion dépendent
// du choix de la variable avec laquelle on fait l'inversion !
// On reste dépendant d'une coincidence: le fait qu'un input soit renseigné
// ou non peut donc également changer la donne.
// Le flux normal est le suivant:
// - SET_ACTIVE_TARGET_INPUT (clic dans une case blanche)
// - USER_INPUT_UPDATE (saisie d'une valeur)
// - START_CONVERSATION (saisie d'une valeur)
// - SET_ACTIVE_TARGET_INPUT (changement d'objectif)
// et dans ce cas ça marche, mais supposons qu'on a ensuite:
// - USER_INPUT_UPDATE (suppression d'une valeur après le début de la conversation)
// si l'input actif est le salaire de base on a un missingVariables incorrect
// puisqu'il ne sait pas avec quelle variable on fait l'inversion, et si on fait
// - SET_ACTIVE_TARGET_INPUT (clic dans une autre case blanche)
// on va se retrouver avec un affichage incohérent, et il ne sera pas corrigé
// lors du USER_INPUT_UPDATE puisqu'on ne recalcule pas lors de cette action
// TODO - corriger ce bug correctement avec des tests auto
// TODO - utiliser le nom qualifié dans analyseMany et qualifier les targetNames
let qualifiedTargets = map(
x => 'contrat salarié . ' + x,
state.targetNames
),
initialAnalysis = analyseMany(state.parsedRules, state.targetNames)(
name =>
qualifiedTargets.includes(name)
? answerSource(state)(name)
: null
),
initialMissingVariablesByTarget = collectMissingVariablesByTarget(
initialAnalysis.targets
)
return {
...(stillBlank ? state : newState),
missingVariablesByTarget: {
initial: initialMissingVariablesByTarget,
current: missingVariablesByTarget
}
}
}
if (action.type == 'STEP_ACTION' && action.name == 'fold') {
tracker.push([
'trackEvent',
'answer:' + action.source,
action.step + ': ' + situationWithDefaults(state)(action.step)
])
if (!newState.currentQuestion) {
tracker.push([
'trackEvent',
'done',
'after ' + length(newState.foldedSteps) + ' questions'
])
}
let foldedSteps = [...state.foldedSteps, state.currentQuestion]
return {
...newState,
foldedSteps,
missingVariablesByTarget: {
...state.missingVariablesByTarget,
current: missingVariablesByTarget
}
}
}
if (action.type == 'STEP_ACTION' && action.name == 'unfold') {
tracker.push(['trackEvent', 'unfold', action.step])
// We are possibly "refolding" a previously open question
let previous = state.currentQuestion,
// we fold it back into foldedSteps if it had been answered
answered = previous && answerSource(state)(previous) != undefined,
rawFoldedSteps = answered
? concat(state.foldedSteps, [previous])
: state.foldedSteps,
foldedSteps = without([action.step], rawFoldedSteps)
return {
...newState,
foldedSteps,
currentQuestion: action.step,
missingVariablesByTarget: {
...state.missingVariablesByTarget,
current: missingVariablesByTarget
}
}
}
}
// assume "wraps" a given situation function with one that overrides its values with
// the given assumptions
export let assume = (evaluator, assumptions) => state => name => {
let userInput = evaluator(state)(name)
return userInput != null ? userInput : assumptions[name]
}