2017-07-02 17:12:02 +00:00
|
|
|
import R from 'ramda'
|
2016-11-15 18:46:17 +00:00
|
|
|
import { combineReducers } from 'redux'
|
2017-01-10 18:22:44 +00:00
|
|
|
import reduceReducers from 'reduce-reducers'
|
|
|
|
import {reducer as formReducer, formValueSelector} from 'redux-form'
|
|
|
|
|
2017-11-22 09:17:22 +00:00
|
|
|
import {rules, findRuleByName, findRuleByDottedName, collectDefaults} from 'Engine/rules'
|
2017-11-13 14:50:48 +00:00
|
|
|
import {nextSteps} from 'Engine/generateQuestions'
|
2017-07-02 17:12:02 +00:00
|
|
|
import computeThemeColours from 'Components/themeColours'
|
2017-09-28 12:14:34 +00:00
|
|
|
import { STEP_ACTION, START_CONVERSATION, EXPLAIN_VARIABLE, CHANGE_THEME_COLOUR} from './actions'
|
2017-07-08 11:28:50 +00:00
|
|
|
|
2017-11-07 18:46:40 +00:00
|
|
|
import {analyse} from 'Engine/traverse'
|
2017-07-08 11:28:50 +00:00
|
|
|
|
2017-11-13 14:50:48 +00:00
|
|
|
import ReactPiwik from 'Components/Tracker'
|
2017-10-12 14:08:43 +00:00
|
|
|
|
2017-11-09 11:45:26 +00:00
|
|
|
import formValueTypes from 'Components/conversation/formValueTypes'
|
|
|
|
|
|
|
|
let fromConversation = flatRules => state => name => {
|
|
|
|
// Our situationGate retrieves data from the "conversation" form
|
|
|
|
// The search below is to apply input conversions such as replacing "," with "."
|
|
|
|
if (name.startsWith("sys.")) return null
|
|
|
|
|
|
|
|
let rule = findRuleByDottedName(flatRules, name),
|
|
|
|
format = rule ? formValueTypes[rule.format] : null,
|
|
|
|
pre = format && format.validator.pre ? format.validator.pre : R.identity,
|
|
|
|
value = formValueSelector('conversation')(state, name)
|
|
|
|
|
|
|
|
return value && pre(value)
|
|
|
|
}
|
2017-07-08 11:28:50 +00:00
|
|
|
|
2017-09-21 13:46:59 +00:00
|
|
|
// assume "wraps" a given situation function with one that overrides its values with
|
|
|
|
// the given assumptions
|
2017-09-26 15:20:46 +00:00
|
|
|
let assume = (evaluator, assumptions) => state => name => {
|
2017-11-13 14:50:48 +00:00
|
|
|
let userInput = evaluator(state)(name)
|
|
|
|
return userInput != null ? userInput : assumptions[name]
|
|
|
|
}
|
2017-07-07 08:41:06 +00:00
|
|
|
|
2017-10-14 20:11:38 +00:00
|
|
|
export let reduceSteps = (tracker, flatRules, answerSource) => (state, action) => {
|
2017-07-07 08:41:06 +00:00
|
|
|
if (![START_CONVERSATION, STEP_ACTION].includes(action.type))
|
|
|
|
return state
|
|
|
|
|
2017-11-13 14:50:48 +00:00
|
|
|
let targetNames = action.type == START_CONVERSATION ? action.targetNames : state.targetNames
|
2017-07-07 08:41:06 +00:00
|
|
|
|
2017-11-13 14:50:48 +00:00
|
|
|
let sim = targetNames.length === 1 ? findRuleByName(flatRules, targetNames[0]) : {},
|
2017-09-21 13:46:59 +00:00
|
|
|
// Hard assumptions cannot be changed, they are used to specialise a simulator
|
2017-09-28 12:14:34 +00:00
|
|
|
// before the user sees the first question
|
2017-09-21 13:46:59 +00:00
|
|
|
hardAssumptions = R.pathOr({},['simulateur','hypothèses'],sim),
|
2017-10-14 14:50:20 +00:00
|
|
|
intermediateSituation = assume(answerSource, hardAssumptions),
|
2017-11-22 09:57:07 +00:00
|
|
|
rulesDefaults = collectDefaults(flatRules),
|
|
|
|
situationWithDefaults = assume(intermediateSituation, rulesDefaults)
|
2017-09-21 13:46:59 +00:00
|
|
|
|
2017-11-22 09:17:22 +00:00
|
|
|
let situationGate = situationWithDefaults(state),
|
2017-11-22 15:34:23 +00:00
|
|
|
parsedRules = R.path(['analysis', 'parsedRules'], state),
|
|
|
|
analysis = analyse(parsedRules || flatRules, targetNames)(situationGate)
|
2017-09-21 13:46:59 +00:00
|
|
|
|
2017-09-26 15:20:46 +00:00
|
|
|
let newState = {
|
2017-07-07 08:41:06 +00:00
|
|
|
...state,
|
2017-11-13 14:50:48 +00:00
|
|
|
targetNames,
|
2017-11-07 18:46:40 +00:00
|
|
|
analysis,
|
2017-09-28 12:59:10 +00:00
|
|
|
situationGate: situationGate,
|
2017-10-19 10:36:09 +00:00
|
|
|
extraSteps: [],
|
|
|
|
explainedVariable: null
|
2017-07-07 08:41:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (action.type == START_CONVERSATION) {
|
2017-11-07 18:46:40 +00:00
|
|
|
let next = nextSteps(situationGate, flatRules, newState.analysis)
|
2017-07-07 08:41:06 +00:00
|
|
|
return {
|
2017-09-26 15:20:46 +00:00
|
|
|
...newState,
|
2017-07-31 16:16:25 +00:00
|
|
|
foldedSteps: [],
|
2017-11-15 10:12:58 +00:00
|
|
|
currentQuestion: R.head(next)
|
2017-07-07 08:41:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (action.type == STEP_ACTION && action.name == 'fold') {
|
2017-11-13 14:50:48 +00:00
|
|
|
tracker.push(['trackEvent', 'answer', action.step+': '+situationGate(action.step)])
|
2017-10-12 14:08:43 +00:00
|
|
|
|
2017-11-04 14:39:40 +00:00
|
|
|
let foldedSteps = [...state.foldedSteps, state.currentQuestion],
|
2017-11-07 18:46:40 +00:00
|
|
|
next = nextSteps(situationGate, flatRules, newState.analysis),
|
2017-11-22 09:57:07 +00:00
|
|
|
assumptionsMade = !R.isEmpty(rulesDefaults),
|
2017-11-04 14:39:40 +00:00
|
|
|
done = next.length == 0
|
2017-09-21 13:46:59 +00:00
|
|
|
|
|
|
|
// The simulation is "over" - except we can now fill in extra questions
|
2017-09-28 12:14:34 +00:00
|
|
|
// where the answers were previously given default reasonable assumptions
|
2017-10-13 10:06:04 +00:00
|
|
|
if (done && assumptionsMade) {
|
2017-09-21 13:46:59 +00:00
|
|
|
let newSituation = intermediateSituation(state),
|
2017-11-22 15:34:23 +00:00
|
|
|
reanalysis = analyse(analysis.parsedRules, targetNames)(newSituation),
|
2017-11-07 18:46:40 +00:00
|
|
|
extraSteps = nextSteps(newSituation, flatRules, reanalysis)
|
2017-09-21 13:46:59 +00:00
|
|
|
|
2017-11-07 18:46:40 +00:00
|
|
|
tracker.push(['trackEvent', 'done', 'extra questions: '+extraSteps.length])
|
2017-10-13 10:06:04 +00:00
|
|
|
|
2017-09-21 13:46:59 +00:00
|
|
|
return {
|
2017-09-26 15:20:46 +00:00
|
|
|
...newState,
|
2017-09-21 13:46:59 +00:00
|
|
|
foldedSteps,
|
2017-11-22 09:57:07 +00:00
|
|
|
currentQuestion: R.head(extraSteps)
|
2017-09-21 13:46:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-13 10:06:04 +00:00
|
|
|
if (done) {
|
2017-11-13 14:50:48 +00:00
|
|
|
tracker.push(['trackEvent', 'done', 'no more questions'])
|
2017-10-13 10:06:04 +00:00
|
|
|
}
|
|
|
|
|
2017-07-07 08:41:06 +00:00
|
|
|
return {
|
2017-09-26 15:20:46 +00:00
|
|
|
...newState,
|
2017-09-21 13:46:59 +00:00
|
|
|
foldedSteps,
|
2017-11-04 14:39:40 +00:00
|
|
|
currentQuestion: R.head(next)
|
2017-07-07 08:41:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (action.type == STEP_ACTION && action.name == 'unfold') {
|
2017-11-13 14:50:48 +00:00
|
|
|
tracker.push(['trackEvent', 'unfold', action.step])
|
2017-10-12 14:08:43 +00:00
|
|
|
|
2017-11-05 14:19:49 +00:00
|
|
|
// We are possibly "refolding" a previously open question
|
2017-11-04 15:09:13 +00:00
|
|
|
let previous = state.currentQuestion,
|
2017-11-05 14:19:49 +00:00
|
|
|
// we fold it back into foldedSteps if it had been answered
|
|
|
|
answered = previous && answerSource(state)(previous) != undefined,
|
2017-11-22 09:57:07 +00:00
|
|
|
foldedSteps = answered ? R.concat(state.foldedSteps, [previous]) : state.foldedSteps
|
2017-07-07 08:41:06 +00:00
|
|
|
|
|
|
|
return {
|
2017-09-26 15:20:46 +00:00
|
|
|
...newState,
|
2017-11-05 14:19:49 +00:00
|
|
|
foldedSteps: R.without([action.step], foldedSteps),
|
2017-11-04 15:09:13 +00:00
|
|
|
currentQuestion: action.step
|
2017-07-07 08:41:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-10 18:22:44 +00:00
|
|
|
|
|
|
|
function themeColours(state = computeThemeColours(), {type, colour}) {
|
2017-07-08 11:28:50 +00:00
|
|
|
if (type == CHANGE_THEME_COLOUR)
|
2017-01-10 18:22:44 +00:00
|
|
|
return computeThemeColours(colour)
|
|
|
|
else return state
|
|
|
|
}
|
|
|
|
|
2017-02-09 17:38:51 +00:00
|
|
|
function explainedVariable(state = null, {type, variableName=null}) {
|
|
|
|
switch (type) {
|
|
|
|
case EXPLAIN_VARIABLE:
|
2017-02-09 17:15:25 +00:00
|
|
|
return variableName
|
2017-02-09 17:38:51 +00:00
|
|
|
default:
|
|
|
|
return state
|
|
|
|
}
|
2017-02-08 16:50:22 +00:00
|
|
|
}
|
|
|
|
|
2017-03-15 15:26:00 +00:00
|
|
|
|
2017-01-10 18:22:44 +00:00
|
|
|
export default reduceReducers(
|
|
|
|
combineReducers({
|
2017-05-18 14:04:23 +00:00
|
|
|
sessionId: (id = Math.floor(Math.random() * 1000000000000) + '') => id,
|
2017-01-10 18:22:44 +00:00
|
|
|
// this is handled by redux-form, pas touche !
|
|
|
|
form: formReducer,
|
|
|
|
|
|
|
|
/* Have forms been filled or ignored ?
|
|
|
|
false means the user is reconsidering its previous input */
|
2017-04-24 18:03:38 +00:00
|
|
|
foldedSteps: (steps = []) => steps,
|
2017-09-21 13:46:59 +00:00
|
|
|
extraSteps: (steps = []) => steps,
|
2017-11-04 14:39:40 +00:00
|
|
|
currentQuestion: (state = null) => state,
|
2017-01-10 18:22:44 +00:00
|
|
|
|
2017-11-07 18:46:40 +00:00
|
|
|
analysis: (state = null) => state,
|
|
|
|
|
2017-11-13 14:50:48 +00:00
|
|
|
targetNames: (state = null) => state,
|
2017-01-10 18:22:44 +00:00
|
|
|
|
2017-10-05 10:25:35 +00:00
|
|
|
situationGate: (state = name => null) => state,
|
2017-09-21 13:46:59 +00:00
|
|
|
refine: (state = false) => state,
|
|
|
|
|
2017-02-08 16:50:22 +00:00
|
|
|
themeColours,
|
|
|
|
|
2017-09-28 12:14:34 +00:00
|
|
|
explainedVariable
|
2017-03-15 15:26:00 +00:00
|
|
|
|
2017-01-10 18:22:44 +00:00
|
|
|
}),
|
|
|
|
// cross-cutting concerns because here `state` is the whole state tree
|
2017-11-09 11:45:26 +00:00
|
|
|
reduceSteps(ReactPiwik, rules, fromConversation(rules))
|
2017-01-10 18:22:44 +00:00
|
|
|
)
|