mon-entreprise/source/components/PaySlip.tsx

175 lines
4.8 KiB
TypeScript
Raw Normal View History

import { useEvaluation, EngineContext } from 'Components/utils/EngineContext'
import Value from 'Components/EngineValue'
import { formatValue } from 'Engine/format'
import React, { Fragment, useContext } from 'react'
import { Trans } from 'react-i18next'
import { DottedName } from 'Rules'
import './PaySlip.css'
import { Line, SalaireBrutSection, SalaireNetSection } from './PaySlipSections'
import RuleLink from './RuleLink'
export const SECTION_ORDER = [
'protection sociale . santé',
'protection sociale . accidents du travail et maladies professionnelles',
'protection sociale . retraite',
'protection sociale . famille',
'protection sociale . assurance chômage',
'protection sociale . formation',
'protection sociale . transport',
'protection sociale . autres'
] as const
type Section = typeof SECTION_ORDER[number]
function getSection(rule): Section {
const section = ('protection sociale . ' +
(rule.cotisation?.branche ?? rule.taxe?.branche)) as Section
if (SECTION_ORDER.includes(section)) {
return section
}
return 'protection sociale . autres'
}
export function getCotisationsBySection(
parsedRules
): Array<[Section, DottedName[]]> {
const cotisations = [
...parsedRules['contrat salarié . cotisations . patronales'].formule
.explanation.explanation,
...parsedRules['contrat salarié . cotisations . salariales'].formule
.explanation.explanation
]
.map(cotisation => cotisation.dottedName)
.filter(Boolean)
.reduce((acc, cotisation) => {
const sectionName = getSection(parsedRules[cotisation])
return {
...acc,
[sectionName]: (acc[sectionName] ?? new Set()).add(cotisation)
}
}, {}) as Record<Section, Set<DottedName>>
return Object.entries(cotisations)
.map(([section, dottedNames]) => [section, [...dottedNames.values()]])
.sort(
([a], [b]) =>
SECTION_ORDER.indexOf(a as Section) -
SECTION_ORDER.indexOf(b as Section)
) as Array<[Section, DottedName[]]>
}
export default function PaySlip() {
const parsedRules = useContext(EngineContext).getParsedRules()
const cotisationsBySection = getCotisationsBySection(parsedRules)
return (
<div
className="payslip__container"
css={`
.value {
display: flex;
align-items: flex-end;
justify-content: flex-end;
padding-right: 0.2em;
}
`}
>
<div className="payslip__salarySection">
<Line
rule="contrat salarié . temps de travail"
displayedUnit="heures/mois"
precision={1}
/>
<Line
rule="contrat salarié . temps de travail . heures supplémentaires"
displayedUnit="heures/mois"
precision={1}
/>
</div>
<SalaireBrutSection />
{/* Section cotisations */}
<div className="payslip__cotisationsSection">
<h4>
<Trans>Cotisations sociales</Trans>
</h4>
<h4>
<Trans>Part employeur</Trans>
</h4>
<h4>
<Trans>Part salarié</Trans>
</h4>
{cotisationsBySection.map(([sectionDottedName, cotisations]) => {
let section = parsedRules[sectionDottedName]
return (
<Fragment key={section.dottedName}>
<h5 className="payslip__cotisationTitle">
<RuleLink {...section} />
</h5>
{cotisations.map(cotisation => (
<Cotisation key={cotisation} dottedName={cotisation} />
))}
</Fragment>
)
})}
{/* Total cotisation */}
<div className="payslip__total">
<Trans>Total des retenues</Trans>
</div>
<Value
expression="contrat salarié . cotisations . patronales"
displayedUnit="€"
className="payslip__total"
/>
<Value
expression="contrat salarié . cotisations . salariales"
displayedUnit="€"
className="payslip__total"
/>
{/* Salaire chargé */}
<Line rule="contrat salarié . rémunération . total" />
<span />
</div>
{/* Section salaire net */}
<SalaireNetSection />
</div>
)
}
function Cotisation({ dottedName }: { dottedName: DottedName }) {
const parsedRules = useContext(EngineContext).getParsedRules()
const partSalariale = useEvaluation(
'contrat salarié . cotisations . salariales'
)?.formule.explanation.explanation.find(
cotisation => cotisation.dottedName === dottedName
)
const partPatronale = useEvaluation(
'contrat salarié . cotisations . patronales'
)?.formule.explanation.explanation.find(
cotisation => cotisation.dottedName === dottedName
)
if (!partPatronale?.nodeValue && !partSalariale?.nodeValue) {
return null
}
return (
<>
<RuleLink
{...parsedRules[dottedName]}
style={{ backgroundColor: 'var(--lightestColor)' }}
/>
<span style={{ backgroundColor: 'var(--lightestColor)' }}>
{partPatronale?.nodeValue
? formatValue({ ...partPatronale, unit: '€' })
: ''}
</span>
<span style={{ backgroundColor: 'var(--lightestColor)' }}>
{partSalariale?.nodeValue
? formatValue({ ...partSalariale, unit: '€' })
: ''}
</span>
</>
)
}