2018-01-08 15:07:26 +00:00
|
|
|
import {
|
2018-03-27 19:46:09 +00:00
|
|
|
curry,
|
|
|
|
evolve,
|
2018-01-08 15:07:26 +00:00
|
|
|
path,
|
|
|
|
propEq,
|
|
|
|
pathEq,
|
|
|
|
groupBy,
|
|
|
|
prop,
|
|
|
|
map,
|
2018-03-27 19:46:09 +00:00
|
|
|
mapObjIndexed,
|
2018-01-08 15:07:26 +00:00
|
|
|
sum,
|
|
|
|
filter,
|
|
|
|
concat,
|
|
|
|
pathOr,
|
|
|
|
toPairs,
|
|
|
|
keys,
|
2018-01-15 17:33:36 +00:00
|
|
|
head,
|
|
|
|
find
|
2018-01-08 15:07:26 +00:00
|
|
|
} from 'ramda'
|
2018-01-03 15:54:19 +00:00
|
|
|
import React, { Component } from 'react'
|
2018-03-29 07:58:20 +00:00
|
|
|
import { Trans, translate } from 'react-i18next'
|
2018-03-27 19:46:09 +00:00
|
|
|
import PropTypes from 'prop-types'
|
2018-01-03 15:54:19 +00:00
|
|
|
import { connect } from 'react-redux'
|
|
|
|
import { withRouter } from 'react-router'
|
|
|
|
import { Link } from 'react-router-dom'
|
|
|
|
import { formValueSelector } from 'redux-form'
|
|
|
|
import './Results.css'
|
|
|
|
import '../engine/mecanismViews/Somme.css'
|
2017-12-07 14:19:51 +00:00
|
|
|
|
2018-02-05 16:01:33 +00:00
|
|
|
import { capitalise0, humanFigure } from '../utils'
|
2018-03-27 19:46:09 +00:00
|
|
|
import { nameLeaf, encodeRuleName, rules, findRuleByDottedName } from 'Engine/rules'
|
2017-12-07 14:19:51 +00:00
|
|
|
|
2017-11-28 22:26:04 +00:00
|
|
|
// Filtered variables and rules can't be filtered in a uniform way, for now
|
2018-02-27 15:55:36 +00:00
|
|
|
let paidBy = payer => item =>
|
|
|
|
pathEq(['explanation', item.explanation.type, 'dû par'], payer, item)
|
2018-01-08 15:07:26 +00:00
|
|
|
let filteredBy = pathEq(['cotisation', 'dû par'])
|
|
|
|
export let byName = groupBy(prop('dottedName'))
|
2017-11-28 22:26:04 +00:00
|
|
|
|
|
|
|
export let cell = (branch, payer, analysis) => {
|
2017-12-08 17:31:01 +00:00
|
|
|
let row = byBranch(analysis)[branch],
|
2018-01-08 15:07:26 +00:00
|
|
|
items = filter(item => paidBy(payer)(item) || filteredBy(payer)(item), row),
|
|
|
|
values = map(prop('nodeValue'), items)
|
2017-12-08 17:31:01 +00:00
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
return sum(values)
|
2017-11-28 22:26:04 +00:00
|
|
|
}
|
|
|
|
|
2017-11-29 19:37:24 +00:00
|
|
|
export let subCell = (row, name, payer) => {
|
2017-12-08 17:31:01 +00:00
|
|
|
let cells = row[name],
|
2018-01-08 15:07:26 +00:00
|
|
|
items = filter(
|
2017-12-08 17:31:01 +00:00
|
|
|
item => paidBy(payer)(item) || filteredBy(payer)(item),
|
|
|
|
cells
|
|
|
|
),
|
2018-01-08 15:07:26 +00:00
|
|
|
values = map(prop('nodeValue'), items)
|
2017-12-08 17:31:01 +00:00
|
|
|
|
2018-01-08 15:07:26 +00:00
|
|
|
return sum(values)
|
2017-11-29 19:37:24 +00:00
|
|
|
}
|
|
|
|
|
2017-12-01 11:51:28 +00:00
|
|
|
export let byBranch = analysis => {
|
2018-01-03 15:54:19 +00:00
|
|
|
let sal = analysis.cache['contrat salarié . cotisations salariales']
|
|
|
|
let pat = analysis.cache['contrat salarié . cotisations patronales']
|
2017-12-08 17:31:01 +00:00
|
|
|
|
|
|
|
let l1 = sal ? sal.explanation.formule.explanation.explanation : [],
|
|
|
|
l2 = pat ? pat.explanation.formule.explanation.explanation : [],
|
2018-01-08 15:07:26 +00:00
|
|
|
explanations = concat(l1, l2),
|
|
|
|
result = groupBy(
|
|
|
|
pathOr('autre', ['explanation', 'cotisation', 'branche']),
|
2017-12-08 17:31:01 +00:00
|
|
|
explanations
|
|
|
|
)
|
|
|
|
|
2017-12-12 16:44:59 +00:00
|
|
|
return result
|
2017-11-28 15:00:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@withRouter
|
2017-12-01 11:51:28 +00:00
|
|
|
@connect(state => ({
|
2017-12-08 17:31:01 +00:00
|
|
|
analysis: state.analysis,
|
|
|
|
targetNames: state.targetNames,
|
|
|
|
situationGate: state.situationGate,
|
2018-01-03 15:54:19 +00:00
|
|
|
inversions: formValueSelector('conversation')(state, 'inversions'),
|
2017-12-08 17:31:01 +00:00
|
|
|
done: state.done
|
2017-12-01 11:51:28 +00:00
|
|
|
}))
|
2018-03-29 07:58:20 +00:00
|
|
|
@translate()
|
2017-11-28 15:00:43 +00:00
|
|
|
export default class ResultsGrid extends Component {
|
2017-12-08 17:31:01 +00:00
|
|
|
render() {
|
|
|
|
let { analysis, situationGate, targetNames, inversions, done } = this.props
|
|
|
|
|
|
|
|
if (!done) return null
|
|
|
|
|
|
|
|
if (!analysis) return null
|
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
let extract = x => (typeof x == 'string' ? +x : (x && x.nodeValue) || 0),
|
2018-01-08 15:07:26 +00:00
|
|
|
fromEval = name => find(propEq('dottedName', name), analysis.targets),
|
2017-12-12 19:10:22 +00:00
|
|
|
fromDict = name => analysis.cache[name],
|
2017-12-08 17:31:01 +00:00
|
|
|
get = name =>
|
2017-12-12 10:11:43 +00:00
|
|
|
extract(situationGate(name) || fromEval(name) || fromDict(name))
|
2017-12-08 17:31:01 +00:00
|
|
|
let results = byBranch(analysis),
|
2018-01-03 15:54:19 +00:00
|
|
|
brut = get('contrat salarié . salaire brut'),
|
|
|
|
net = get('contrat salarié . salaire net'),
|
|
|
|
total = get('contrat salarié . salaire total')
|
2018-01-08 15:07:26 +00:00
|
|
|
let inversion = path(['contrat salarié ', ' salaire de base'], inversions),
|
2017-12-08 17:31:01 +00:00
|
|
|
relevantSalaries = new Set(
|
|
|
|
targetNames
|
|
|
|
.concat(inversion ? [nameLeaf(inversion)] : [])
|
2018-01-03 15:54:19 +00:00
|
|
|
.concat(['salaire de base'])
|
2017-12-08 17:31:01 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="somme resultsGrid">
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<td className="element" />
|
|
|
|
<td
|
|
|
|
colSpan={(relevantSalaries.size - 1) * 2}
|
|
|
|
className="element value"
|
|
|
|
id="sommeBase"
|
|
|
|
>
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(brut)}{' '}
|
2018-03-27 19:46:09 +00:00
|
|
|
<span className="annotation"><Trans>Salaire brut</Trans></span>
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
2018-01-08 15:07:26 +00:00
|
|
|
{toPairs(results).map(([branch, values]) => {
|
2018-01-03 15:54:19 +00:00
|
|
|
let props = {
|
|
|
|
key: branch,
|
|
|
|
branch,
|
|
|
|
values,
|
|
|
|
analysis,
|
|
|
|
relevantSalaries
|
|
|
|
}
|
2017-12-08 17:31:01 +00:00
|
|
|
return <Row {...props} />
|
|
|
|
})}
|
2018-01-03 15:54:19 +00:00
|
|
|
<ReductionRow
|
|
|
|
node={fromDict('contrat salarié . réductions de cotisations')}
|
|
|
|
relevantSalaries={relevantSalaries}
|
|
|
|
/>
|
2017-12-08 17:31:01 +00:00
|
|
|
<tr>
|
|
|
|
<td key="blank" className="element" />
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire net') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="netOperator" className="operator">
|
|
|
|
=
|
|
|
|
</td>
|
|
|
|
<td key="net" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(net)}{' '}
|
2018-03-27 19:46:09 +00:00
|
|
|
<span className="annotation"><Trans>Salaire net</Trans></span>
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire total') && [
|
2017-12-08 17:31:01 +00:00
|
|
|
<td key="totalOperator" className="operator">
|
|
|
|
=
|
|
|
|
</td>,
|
|
|
|
<td key="total" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(total)}{' '}
|
2018-03-27 19:46:09 +00:00
|
|
|
<span className="annotation"><Trans>Salaire total</Trans></span>
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
]}
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
2017-11-28 22:26:04 +00:00
|
|
|
}
|
|
|
|
|
2018-03-29 07:58:20 +00:00
|
|
|
@translate()
|
2017-11-28 22:26:04 +00:00
|
|
|
class Row extends Component {
|
2018-03-27 19:46:09 +00:00
|
|
|
static contextTypes = {
|
|
|
|
i18n: PropTypes.object.isRequired
|
|
|
|
}
|
2017-12-08 17:31:01 +00:00
|
|
|
state = {
|
|
|
|
folded: true
|
|
|
|
}
|
|
|
|
render() {
|
|
|
|
let { branch, values, analysis, relevantSalaries } = this.props,
|
2018-03-27 19:46:09 +00:00
|
|
|
detail = byName(values),
|
|
|
|
ruleData = mapObjIndexed((v,k,o) => findRuleByDottedName(rules,k), detail),
|
|
|
|
{ i18n } = this.context
|
2017-12-08 17:31:01 +00:00
|
|
|
|
|
|
|
let title = name => {
|
2018-03-27 19:46:09 +00:00
|
|
|
let node = ruleData[name]
|
|
|
|
return node.title || capitalise0(i18n.t(node.name))
|
2017-12-08 17:31:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let aggregateRow = (
|
|
|
|
<tr
|
|
|
|
key="aggregateRow"
|
|
|
|
onClick={() => this.setState({ folded: !this.state.folded })}
|
|
|
|
>
|
|
|
|
<td key="category" className="element category name">
|
2018-03-27 19:46:09 +00:00
|
|
|
{capitalise0(i18n.t(branch))} <span className="unfoldIndication">
|
|
|
|
{this.state.folded ? i18n.t('déplier')+' >' : i18n.t('replier')}
|
2017-12-08 17:31:01 +00:00
|
|
|
</span>
|
|
|
|
</td>
|
|
|
|
{this.state.folded ? (
|
|
|
|
<>
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire net') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator1" className="operator">
|
|
|
|
-
|
|
|
|
</td>
|
|
|
|
<td key="value1" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(cell(branch, 'salarié', analysis))}
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire total') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator2" className="operator">
|
|
|
|
+
|
|
|
|
</td>
|
|
|
|
<td key="value2" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(cell(branch, 'employeur', analysis))}
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<td key="blank" colSpan="4" />
|
|
|
|
)}
|
|
|
|
</tr>
|
|
|
|
)
|
|
|
|
|
|
|
|
let detailRows = this.state.folded
|
|
|
|
? []
|
2018-01-08 15:07:26 +00:00
|
|
|
: keys(detail).map(subCellName => (
|
2018-01-03 15:54:19 +00:00
|
|
|
<tr key={'detailsRow' + subCellName} className="detailsRow">
|
|
|
|
<td className="element name">
|
2018-01-24 18:07:55 +00:00
|
|
|
<Link to={'/règle/' + encodeRuleName(nameLeaf(subCellName))}>
|
2018-01-03 15:54:19 +00:00
|
|
|
{title(subCellName)}
|
|
|
|
</Link>
|
|
|
|
</td>
|
|
|
|
{relevantSalaries.has('salaire net') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator1" className="operator">
|
|
|
|
-
|
|
|
|
</td>
|
|
|
|
<td key="value1" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(subCell(detail, subCellName, 'salarié'))}
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire total') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator2" className="operator">
|
|
|
|
+
|
|
|
|
</td>
|
|
|
|
<td key="value2" className="element value">
|
2018-01-03 15:54:19 +00:00
|
|
|
{humanFigure(2)(subCell(detail, subCellName, 'employeur'))}
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</tr>
|
|
|
|
))
|
|
|
|
|
|
|
|
// returns an array of <tr>
|
2018-01-08 15:07:26 +00:00
|
|
|
return concat([aggregateRow], detailRows)
|
2017-12-08 17:31:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO Ce code est beaucoup trop spécifique
|
|
|
|
// C'est essentiellement une copie de Row
|
|
|
|
class ReductionRow extends Component {
|
|
|
|
state = {
|
|
|
|
folded: true
|
|
|
|
}
|
|
|
|
render() {
|
2018-01-03 15:54:19 +00:00
|
|
|
let { relevantSalaries, node } = this.props
|
|
|
|
if (!relevantSalaries.has('salaire total')) return null
|
2017-12-12 10:11:43 +00:00
|
|
|
let value = node ? node.nodeValue : 0
|
2017-12-08 17:31:01 +00:00
|
|
|
let aggregateRow = (
|
|
|
|
<tr
|
|
|
|
key="aggregateRowReductions"
|
|
|
|
onClick={() => this.setState({ folded: !this.state.folded })}
|
|
|
|
>
|
|
|
|
<td key="category" className="element category name">
|
|
|
|
Réductions <span className="unfoldIndication">
|
2018-01-03 15:54:19 +00:00
|
|
|
{this.state.folded ? 'déplier >' : 'replier'}
|
2017-12-08 17:31:01 +00:00
|
|
|
</span>
|
|
|
|
</td>
|
|
|
|
{this.state.folded ? (
|
|
|
|
<>
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire net') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator1" className="operator">
|
2017-12-18 17:12:01 +00:00
|
|
|
+
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
<td key="value1" className="element value">
|
|
|
|
{humanFigure(2)(0)}
|
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire total') && (
|
2017-12-08 17:31:01 +00:00
|
|
|
<>
|
|
|
|
<td key="operator2" className="operator">
|
2017-12-18 17:12:01 +00:00
|
|
|
-
|
2017-12-08 17:31:01 +00:00
|
|
|
</td>
|
|
|
|
<td key="value2" className="element value">
|
|
|
|
{humanFigure(2)(value)}
|
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<td key="blank" colSpan="4" />
|
|
|
|
)}
|
|
|
|
</tr>
|
|
|
|
)
|
|
|
|
|
2018-01-03 15:54:19 +00:00
|
|
|
let detailRow = this.state.folded ? null : (
|
2017-12-18 17:20:55 +00:00
|
|
|
<tr key="detailsRowRéductions" className="detailsRow">
|
|
|
|
<td className="element name">
|
2018-01-24 18:07:55 +00:00
|
|
|
<Link to={'/règle/' + encodeRuleName('réductions de cotisations')}>
|
2017-12-18 17:20:55 +00:00
|
|
|
Réductions de cotisations
|
|
|
|
</Link>
|
|
|
|
</td>
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire net') && (
|
2017-12-18 17:20:55 +00:00
|
|
|
<>
|
|
|
|
<td key="operator1" className="operator">
|
|
|
|
+
|
|
|
|
</td>
|
|
|
|
<td key="value1" className="element value">
|
|
|
|
{humanFigure(2)(0)}
|
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
2018-01-03 15:54:19 +00:00
|
|
|
{relevantSalaries.has('salaire total') && (
|
2017-12-18 17:20:55 +00:00
|
|
|
<>
|
|
|
|
<td key="operator2" className="operator">
|
|
|
|
-
|
|
|
|
</td>
|
|
|
|
<td key="value2" className="element value">
|
|
|
|
{humanFigure(2)(value)}
|
|
|
|
</td>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</tr>
|
2018-01-03 15:54:19 +00:00
|
|
|
)
|
2017-12-08 17:31:01 +00:00
|
|
|
|
|
|
|
// returns an array of <tr>
|
|
|
|
return [aggregateRow, detailRow]
|
|
|
|
}
|
2017-11-28 15:00:43 +00:00
|
|
|
}
|