mon-entreprise/source/engine/RuleInput.tsx

137 lines
3.8 KiB
TypeScript
Raw Normal View History

import Input from 'Components/conversation/Input'
import Question from 'Components/conversation/Question'
import SelectGéo from 'Components/conversation/select/SelectGéo'
import SelectAtmp from 'Components/conversation/select/SelectTauxRisque'
import SendButton from 'Components/conversation/SendButton'
import CurrencyInput from 'Components/CurrencyInput/CurrencyInput'
import ToggleSwitch from 'Components/ui/ToggleSwitch'
import { is, prop, unless } from 'ramda'
import React from 'react'
import { useTranslation } from 'react-i18next'
import { DottedName, Rule } from 'Types/rule'
import DateInput from '../components/conversation/DateInput'
import { findRuleByDottedName, queryRule } from './rules'
export const binaryOptionChoices = [
{ value: 'non', label: 'Non' },
{ value: 'oui', label: 'Oui' }
]
type Value = string | number | object | boolean
type Props = {
rules: Array<Rule>
dottedName: DottedName
onChange: (value: Value) => void
useSwitch?: boolean
isTarget?: boolean
value?: Value
className?: string
onSubmit?: (value: Value) => void
}
// This function takes the unknown rule and finds which React component should
// be displayed to get a user input through successive if statements
// That's not great, but we won't invest more time until we have more diverse
// input components and a better type system.
export default function InputComponent({
rules,
dottedName,
onChange,
value,
useSwitch = false,
isTarget = false,
className,
onSubmit
}: Props) {
let rule = findRuleByDottedName(rules, dottedName)
let unit = rule.unit || rule.defaultUnit
let language = useTranslation().i18n.language
let commonProps = {
key: dottedName,
dottedName,
value,
onChange,
onSubmit,
className,
title: rule.title,
question: rule.question,
defaultValue: rule.defaultValue,
suggestions: rule.suggestions
}
if (getVariant(rule)) {
return (
<Question
{...commonProps}
choices={buildVariantTree(rules, dottedName)}
/>
)
}
if (rule.API && rule.API === 'géo')
return <SelectGéo {...{ ...commonProps }} />
if (rule.API) throw new Error("Le seul API implémenté est l'API géo")
if (rule.dottedName == 'contrat salarié . ATMP . taux collectif ATMP')
return <SelectAtmp {...commonProps} />
if (rule.type === 'date') {
return <DateInput {...commonProps} />
}
if (unit == null) {
return useSwitch ? (
<ToggleSwitch
defaultChecked={value === 'oui' || rule.defaultValue === 'oui'}
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
onChange(evt.target.checked ? 'oui' : 'non')
}
/>
) : (
<Question {...commonProps} choices={binaryOptionChoices} />
)
}
if (unit?.numerators.includes('€') && isTarget && typeof value === 'number') {
return (
<>
<CurrencyInput
{...commonProps}
language={language}
debounce={600}
value={value}
name={dottedName}
className="targetInput"
onChange={evt => onChange(evt.target.value)}
/>
{onSubmit && <SendButton disabled={!value} onSubmit={onSubmit} />}
</>
)
}
return <Input {...commonProps} unit={unit} />
}
let getVariant = rule => queryRule(rule)('formule . une possibilité')
export let buildVariantTree = (allRules, path) => {
let rec = path => {
let node = findRuleByDottedName(allRules, path)
if (!node) throw new Error(`La règle ${path} est introuvable`)
let variant = getVariant(node),
variants = variant && unless(is(Array), prop('possibilités'))(variant),
shouldBeExpanded = variant && true, //variants.find( v => relevantPaths.find(rp => contains(path + ' . ' + v)(rp) )),
canGiveUp = variant && !variant['choix obligatoire']
return Object.assign(
node,
shouldBeExpanded
? {
canGiveUp,
children: (variants as any).map(v => rec(path + ' . ' + v))
}
: null
)
}
return rec(path)
}