2018-04-26 09:08:36 +00:00
|
|
|
import { path, head, reject, concat, without, length } from 'ramda'
|
2018-04-25 14:23:15 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
|
2018-04-26 10:28:48 +00:00
|
|
|
if (path(['form', 'conversation', 'syncErrors'], state))
|
2018-04-25 14:23:15 +00:00
|
|
|
return state
|
|
|
|
|
|
|
|
// Most rules have default values
|
|
|
|
let rulesDefaults = collectDefaults(flatRules),
|
|
|
|
situationWithDefaults = assume(answerSource, rulesDefaults)
|
|
|
|
|
2018-04-26 10:28:48 +00:00
|
|
|
let analysis = analyseMany(state.parsedRules, state.targetNames)(
|
2018-04-25 14:23:15 +00:00
|
|
|
situationWithDefaults(state)
|
|
|
|
)
|
|
|
|
|
|
|
|
if (action.type === 'USER_INPUT_UPDATE') {
|
2018-04-28 17:11:00 +00:00
|
|
|
tracker.push([
|
|
|
|
'trackEvent',
|
|
|
|
'input:' + state.activeTargetInput,
|
|
|
|
answerSource(state)(state.activeTargetInput)
|
|
|
|
])
|
2018-04-25 14:23:15 +00:00
|
|
|
return { ...state, analysis, situationGate: situationWithDefaults(state) }
|
|
|
|
}
|
|
|
|
|
2018-04-26 10:28:48 +00:00
|
|
|
let nextStepsAnalysis = analyseMany(state.parsedRules, state.targetNames)(
|
2018-04-25 14:23:15 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2018-04-28 17:11:00 +00:00
|
|
|
if (action.type === 'SET_ACTIVE_TARGET_INPUT') {
|
|
|
|
tracker.push([
|
|
|
|
'trackEvent',
|
|
|
|
'select',
|
|
|
|
state.activeTargetInput
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
if (action.type === 'START_CONVERSATION') {
|
|
|
|
tracker.push([
|
|
|
|
'trackEvent',
|
|
|
|
'refine',
|
|
|
|
''
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
2018-04-26 10:28:48 +00:00
|
|
|
// Les nextSteps initiaux ne dépendent que des règles et pourraient être précalculés
|
|
|
|
// TODO: sortir ce calcul du state pour éviter cette "astuce" avec stillBlank
|
|
|
|
if (['START_CONVERSATION', 'SET_ACTIVE_TARGET_INPUT'].includes(action.type)) {
|
|
|
|
// Le premier clic pour sélectionner un input actif permet d'initialiser missingVariablesByTarget
|
2018-04-27 07:50:55 +00:00
|
|
|
let stillBlank = state.activeTargetInput && !answerSource(state)(state.activeTargetInput),
|
|
|
|
initial = path(['missingVariablesByTarget','initial'], state)
|
2018-04-25 14:23:15 +00:00
|
|
|
return {
|
2018-04-26 10:28:48 +00:00
|
|
|
...(stillBlank ? state : newState),
|
2018-04-25 14:23:15 +00:00
|
|
|
missingVariablesByTarget: {
|
2018-04-27 07:50:55 +00:00
|
|
|
initial: initial ? initial : missingVariablesByTarget,
|
2018-04-25 14:23:15 +00:00
|
|
|
current: missingVariablesByTarget
|
|
|
|
}
|
|
|
|
}
|
2018-04-26 10:28:48 +00:00
|
|
|
}
|
2018-04-25 14:23:15 +00:00
|
|
|
|
|
|
|
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',
|
2018-04-28 17:11:00 +00:00
|
|
|
'after ' + length(newState.foldedSteps) + ' questions'
|
2018-04-25 14:23:15 +00:00
|
|
|
])
|
|
|
|
}
|
|
|
|
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]
|
|
|
|
}
|