Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a99a12a759 | ||
| 548b5d63a6 | |||
|
|
ed9df4e178 | ||
| 625b4af5ba | |||
|
|
2ec3044cb3 | ||
| f024a6a8de | |||
|
|
a60294a8f7 | ||
| dd7f9ef8a0 | |||
| cfa7d25521 | |||
|
|
5faa0facca | ||
| 04f90afc58 | |||
|
|
e022cfac98 | ||
| e827128392 | |||
| 86cdec50c6 |
@@ -23,7 +23,8 @@
|
|||||||
"Bash(sudo apt-get:*)",
|
"Bash(sudo apt-get:*)",
|
||||||
"Bash(npx xlsx-cli:*)",
|
"Bash(npx xlsx-cli:*)",
|
||||||
"Bash(cat /home/m-tristan/.claude/projects/-home-m-tristan-workspace-SIRH/4b53d9d7-d8ae-451f-a5cc-5d4fd55f2eef/tool-results/toolu_019hng9Cu2m9wiNACuC2Wm3F.json | python3 -c \"import json,sys; data=json.load\\(sys.stdin\\); print\\(data[0]['text']\\)\" 2>/dev/null | head -2000)",
|
"Bash(cat /home/m-tristan/.claude/projects/-home-m-tristan-workspace-SIRH/4b53d9d7-d8ae-451f-a5cc-5d4fd55f2eef/tool-results/toolu_019hng9Cu2m9wiNACuC2Wm3F.json | python3 -c \"import json,sys; data=json.load\\(sys.stdin\\); print\\(data[0]['text']\\)\" 2>/dev/null | head -2000)",
|
||||||
"Bash(pip3 install:*)"
|
"Bash(pip3 install:*)",
|
||||||
|
"Bash(find:*)"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
4
.env
4
.env
@@ -36,6 +36,10 @@ DEFAULT_URI=http://localhost
|
|||||||
DATABASE_URL="postgresql://app:!ChangeMe!@127.0.0.1:5432/app?serverVersion=16&charset=utf8"
|
DATABASE_URL="postgresql://app:!ChangeMe!@127.0.0.1:5432/app?serverVersion=16&charset=utf8"
|
||||||
###< doctrine/doctrine-bundle ###
|
###< doctrine/doctrine-bundle ###
|
||||||
|
|
||||||
|
###> app ###
|
||||||
|
RTT_START_DATE=2026-02-23
|
||||||
|
###< app ###
|
||||||
|
|
||||||
###> nelmio/cors-bundle ###
|
###> nelmio/cors-bundle ###
|
||||||
CORS_ALLOW_ORIGIN='^https?://(localhost|127\.0\.0\.1)(:[0-9]+)?$'
|
CORS_ALLOW_ORIGIN='^https?://(localhost|127\.0\.0\.1)(:[0-9]+)?$'
|
||||||
###< nelmio/cors-bundle ###
|
###< nelmio/cors-bundle ###
|
||||||
|
|||||||
@@ -26,6 +26,14 @@ services:
|
|||||||
arguments:
|
arguments:
|
||||||
$holidayUrl: '%env(HOLIDAY_URL)%'
|
$holidayUrl: '%env(HOLIDAY_URL)%'
|
||||||
|
|
||||||
|
App\Service\Rtt\RttRecoveryComputationService:
|
||||||
|
arguments:
|
||||||
|
$rttStartDate: '%env(RTT_START_DATE)%'
|
||||||
|
|
||||||
|
App\State\EmployeeRttSummaryProvider:
|
||||||
|
arguments:
|
||||||
|
$rttStartDate: '%env(RTT_START_DATE)%'
|
||||||
|
|
||||||
App\Repository\Contract\AbsenceReadRepositoryInterface: '@App\Repository\AbsenceRepository'
|
App\Repository\Contract\AbsenceReadRepositoryInterface: '@App\Repository\AbsenceRepository'
|
||||||
App\Repository\Contract\EmployeeContractPeriodReadRepositoryInterface: '@App\Repository\EmployeeContractPeriodRepository'
|
App\Repository\Contract\EmployeeContractPeriodReadRepositoryInterface: '@App\Repository\EmployeeContractPeriodRepository'
|
||||||
App\Repository\Contract\EmployeeScopedRepositoryInterface: '@App\Repository\EmployeeRepository'
|
App\Repository\Contract\EmployeeScopedRepositoryInterface: '@App\Repository\EmployeeRepository'
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
parameters:
|
parameters:
|
||||||
app.version: '0.1.42'
|
app.version: '0.1.48'
|
||||||
|
|||||||
@@ -116,6 +116,12 @@ Documents complementaires:
|
|||||||
- contrats >= 39h: de 39h à 43h
|
- contrats >= 39h: de 39h à 43h
|
||||||
- Tranche 50%:
|
- Tranche 50%:
|
||||||
- au-delà de 43h
|
- au-delà de 43h
|
||||||
|
- Date de début RTT (`RTT_START_DATE` dans `.env`):
|
||||||
|
- les semaines dont la fin est antérieure à cette date sont ignorées dans le calcul de récupération
|
||||||
|
- permet d'éviter les déficits fictifs avant la mise en service du logiciel
|
||||||
|
- Semaine en déficit (heures travaillées < heures contrat):
|
||||||
|
- le déficit est déduit du cumul RTT : d'abord des heures à 50%, puis des heures à 25%
|
||||||
|
- si aucun solde 50% ni 25%, les heures à 25% deviennent négatives
|
||||||
- Nature `INTERIM`:
|
- Nature `INTERIM`:
|
||||||
- pas de bonus 25%
|
- pas de bonus 25%
|
||||||
- pas de bonus 50%
|
- pas de bonus 50%
|
||||||
@@ -142,7 +148,7 @@ Documents complementaires:
|
|||||||
- Validation: même logique que les heures classiques (`isValid`, `isSiteValid`, bulk)
|
- Validation: même logique que les heures classiques (`isValid`, `isSiteValid`, bulk)
|
||||||
- Vue semaine:
|
- Vue semaine:
|
||||||
- jour/nuit/atelier par jour + indicateurs repas/dîner/nuitée
|
- jour/nuit/atelier par jour + indicateurs repas/dîner/nuitée
|
||||||
- panier de nuit (PN): affiché par jour si nightMinutes > dayMinutes, et total hebdo dans la colonne Jour/Nuit sem.
|
- panier de nuit (PN): affiché par jour si (nightMinutes > dayMinutes) OU (nightMinutes >= 240, soit au moins 4h de travail entre 21h et 6h), et total hebdo dans la colonne Jour/Nuit sem.
|
||||||
- totaux hebdo: jour, nuit, atelier, total, compteurs petit déj/déjeuner/dîner/nuitée
|
- totaux hebdo: jour, nuit, atelier, total, compteurs petit déj/déjeuner/dîner/nuitée
|
||||||
- pas de calcul d'heures supplémentaires pour les conducteurs
|
- pas de calcul d'heures supplémentaires pour les conducteurs
|
||||||
- Le flag `isDriver` est sur `EmployeeContractPeriod` (un employé peut changer de statut chauffeur selon la période)
|
- Le flag `isDriver` est sur `EmployeeContractPeriod` (un employé peut changer de statut chauffeur selon la période)
|
||||||
@@ -278,10 +284,33 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
- `rate`: taux de majoration, valeurs `25` ou `50`
|
- `rate`: taux de majoration, valeurs `25` ou `50`
|
||||||
- les heures payées sont soustraites du disponible RTT (`availableMinutes -= totalPaidMinutes`)
|
- les heures payées sont soustraites du disponible RTT (`availableMinutes -= totalPaidMinutes`)
|
||||||
- affichage: 2 lignes par mois dans le tableau (25% et 50%)
|
- affichage: 2 lignes par mois dans le tableau (25% et 50%)
|
||||||
|
- colonnes Total 25% et Total 50%: somme base + bonus de chaque tranche
|
||||||
|
- ligne Report N-1 (carry rollover): affichée en juin uniquement si carry > 0
|
||||||
|
- ligne Report mois précédent: solde cumulé (carry N-1 + semaines antérieures − paiements antérieurs), affichée à partir de juillet (masquée si nul)
|
||||||
|
- Reste = Report cumulé + Total du mois − Payé du mois (balance courante en fin de mois)
|
||||||
- affichage:
|
- affichage:
|
||||||
- le compteur global RTT est affiché en **heures** (format `Xh00`)
|
- le compteur global RTT est affiché en **heures** (format `Xh00`)
|
||||||
|
|
||||||
## 10) Récapitulatif Salaire (PDF mensuel)
|
## 10) Export récap. congés & RTT (PDF)
|
||||||
|
|
||||||
|
- Accessible depuis la page Employés via le bouton "Export récap. congés" (réservé `ROLE_ADMIN`)
|
||||||
|
- Clic direct (pas de drawer), génère un PDF A4 portrait à la date du jour
|
||||||
|
- Endpoint: `GET /api/leave-recap/print`
|
||||||
|
- Seuls les employés avec contrat actif sont inclus
|
||||||
|
- Données groupées par site
|
||||||
|
|
||||||
|
### Colonnes du tableau
|
||||||
|
|
||||||
|
| Colonne | Logique |
|
||||||
|
|---------|---------|
|
||||||
|
| Nom | lastName + firstName |
|
||||||
|
| Contrat | Contract.name |
|
||||||
|
| CP N-1 restant | CDI/CDD: acquis N-1 − pris sur N-1. Forfait: report N-1 restant |
|
||||||
|
| Samedi restant | CDI/CDD: samedis acquis N-1 − pris. Forfait: `-` |
|
||||||
|
| CP N | Forfait: jours acquis année civile. Non-forfait: en cours d'acquisition |
|
||||||
|
| RTT | Minutes disponibles (report N-1 + acquis N - payés). Format `X h Y m`. Forfait et INTERIM: `-` |
|
||||||
|
|
||||||
|
## 11) Récapitulatif Salaire (PDF mensuel)
|
||||||
|
|
||||||
- Accessible depuis la page Employés via le bouton "Récap. Salaire" (réservé `ROLE_ADMIN`)
|
- Accessible depuis la page Employés via le bouton "Récap. Salaire" (réservé `ROLE_ADMIN`)
|
||||||
- Sélecteur de mois (défaut = mois courant), génère un PDF A3 paysage
|
- Sélecteur de mois (défaut = mois courant), génère un PDF A3 paysage
|
||||||
@@ -296,7 +325,7 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
| Base | Contract.name | Via EmployeeContractResolver pour le mois |
|
| Base | Contract.name | Via EmployeeContractResolver pour le mois |
|
||||||
| Jour de présence Cadre | WorkHour | Uniquement FORFAIT (PRESENCE). Somme isPresentMorning (0.5) + isPresentAfternoon (0.5) |
|
| Jour de présence Cadre | WorkHour | Uniquement FORFAIT (PRESENCE). Somme isPresentMorning (0.5) + isPresentAfternoon (0.5) |
|
||||||
| Heures de nuit | WorkHour | Non-chauffeurs: calcul intervalles nuit (00:00-06:00, 21:00-24:00). Chauffeurs: somme nightHoursMinutes |
|
| Heures de nuit | WorkHour | Non-chauffeurs: calcul intervalles nuit (00:00-06:00, 21:00-24:00). Chauffeurs: somme nightHoursMinutes |
|
||||||
| Panier de nuit | WorkHour | Nombre de jours où nightMinutes > dayMinutes |
|
| Panier de nuit | WorkHour | Nombre de jours où (nightMinutes > dayMinutes) OU (nightMinutes >= 240, soit 4h entre 21h-6h) |
|
||||||
| Heures payés | EmployeeRttPayment | Somme base25Minutes + base50Minutes du mois, convertie en heures |
|
| Heures payés | EmployeeRttPayment | Somme base25Minutes + base50Minutes du mois, convertie en heures |
|
||||||
| Congés - Nombre | Absence code 'C' | Jours (demi-journées = 0.5) |
|
| Congés - Nombre | Absence code 'C' | Jours (demi-journées = 0.5) |
|
||||||
| Congés - Date | Absence code 'C' | Dates formatées dd/mm |
|
| Congés - Date | Absence code 'C' | Dates formatées dd/mm |
|
||||||
@@ -308,7 +337,7 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
| CHAUFFEUR - samedi | WorkHour (samedi) | Samedis travaillés (chauffeurs uniquement) |
|
| CHAUFFEUR - samedi | WorkHour (samedi) | Samedis travaillés (chauffeurs uniquement) |
|
||||||
| Observations | — | Colonne vide pour saisie manuelle |
|
| Observations | — | Colonne vide pour saisie manuelle |
|
||||||
|
|
||||||
## 11) Notifications
|
## 12) Notifications
|
||||||
|
|
||||||
- Icône cloche en topbar:
|
- Icône cloche en topbar:
|
||||||
- badge = nombre de notifications non lues
|
- badge = nombre de notifications non lues
|
||||||
|
|||||||
@@ -40,36 +40,55 @@
|
|||||||
<table class="w-full table-fixed border-collapse text-[18px]">
|
<table class="w-full table-fixed border-collapse text-[18px]">
|
||||||
<colgroup>
|
<colgroup>
|
||||||
<col />
|
<col />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
<col class="w-[14%]" />
|
<col class="w-[11%]" />
|
||||||
|
<col class="w-[11%]" />
|
||||||
|
<col class="w-[11%]" />
|
||||||
</colgroup>
|
</colgroup>
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th class="px-5 py-[10px] text-left font-bold text-primary-500 border border-primary-500">Semaine</th>
|
<th class="px-5 py-[10px] text-left font-bold text-primary-500 border border-primary-500">Semaine</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">Heure</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">Heure</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Base</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Base</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">25%</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">25%</th>
|
||||||
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">Total 25%</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Base</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Base</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">50%</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">50%</th>
|
||||||
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">Total 50%</th>
|
||||||
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Total</th>
|
<th class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">Total</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<!-- Report row (only on June when carry > 0) -->
|
<!-- Report N-1 row (RTT rollover carry, June only) -->
|
||||||
<tr v-if="showReportRow">
|
<tr v-if="showCarryRow" class="bg-tertiary-500">
|
||||||
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Report</td>
|
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Report</td>
|
||||||
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBase25Minutes) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBase25Minutes) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(summary!.carryBonus25Minutes) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBonus25Minutes) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(summary!.carryBase25Minutes + summary!.carryBonus25Minutes) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBase50Minutes) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBase50Minutes) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(summary!.carryBonus50Minutes) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryBonus50Minutes) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(summary!.carryBase50Minutes + summary!.carryBonus50Minutes) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryFromPreviousYearMinutes) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryFromPreviousYearMinutes) }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
<!-- Report mois précédent (cumulated balance from previous months, July+) -->
|
||||||
|
<tr v-if="showMonthReportRow" class="bg-tertiary-500">
|
||||||
|
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Report</td>
|
||||||
|
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.base25) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.bonus25) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(monthReport.total25) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.base50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.bonus50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(monthReport.total50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.total) }}</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
<!-- Week rows (always 5) -->
|
<!-- Week rows (always 5) -->
|
||||||
<tr
|
<tr
|
||||||
v-for="(week, idx) in paddedWeeks"
|
v-for="(week, idx) in paddedWeeks"
|
||||||
@@ -84,19 +103,27 @@
|
|||||||
<span v-else>0 h</span>
|
<span v-else>0 h</span>
|
||||||
</td>
|
</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
||||||
<span v-if="week">{{ formatMinutes(week.base25Minutes) }}</span>
|
<span v-if="week">{{ formatMinutes(week.totalMinutes >= 0 ? week.base25Minutes : 0) }}</span>
|
||||||
<span v-else>0 h</span>
|
|
||||||
</td>
|
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">
|
|
||||||
<span v-if="week">{{ formatMinutes(week.bonus25Minutes) }}</span>
|
|
||||||
<span v-else>0 h</span>
|
<span v-else>0 h</span>
|
||||||
</td>
|
</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
||||||
<span v-if="week">{{ formatMinutes(week.base50Minutes) }}</span>
|
<span v-if="week">{{ formatMinutes(week.totalMinutes >= 0 ? week.bonus25Minutes : 0) }}</span>
|
||||||
<span v-else>0 h</span>
|
<span v-else>0 h</span>
|
||||||
</td>
|
</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">
|
||||||
<span v-if="week">{{ formatMinutes(week.bonus50Minutes) }}</span>
|
<span v-if="week">{{ formatMinutes(week.base25Minutes + week.bonus25Minutes) }}</span>
|
||||||
|
<span v-else>0 h</span>
|
||||||
|
</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
||||||
|
<span v-if="week">{{ formatMinutes(week.totalMinutes >= 0 ? week.base50Minutes : 0) }}</span>
|
||||||
|
<span v-else>0 h</span>
|
||||||
|
</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
||||||
|
<span v-if="week">{{ formatMinutes(week.totalMinutes >= 0 ? week.bonus50Minutes : 0) }}</span>
|
||||||
|
<span v-else>0 h</span>
|
||||||
|
</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">
|
||||||
|
<span v-if="week">{{ formatMinutes(week.base50Minutes + week.bonus50Minutes) }}</span>
|
||||||
<span v-else>0 h</span>
|
<span v-else>0 h</span>
|
||||||
</td>
|
</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">
|
||||||
@@ -110,9 +137,11 @@
|
|||||||
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500 border-t-2">Total</td>
|
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500 border-t-2">Total</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.overtime) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.overtime) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.base25) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.base25) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.bonus25) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.bonus25) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.total25) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.base50) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.base50) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.bonus50) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.bonus50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2 border-t-2">{{ formatMinutes(totals.total50) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.total) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-t-2">{{ formatMinutes(totals.total) }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
@@ -121,9 +150,11 @@
|
|||||||
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Payé</td>
|
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Payé</td>
|
||||||
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBase25Minutes) : '0 h' }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBase25Minutes) : '0 h' }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ currentPayment ? formatMinutes(-currentPayment.paidBonus25Minutes) : '0 h' }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBonus25Minutes) : '0 h' }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ currentPayment ? formatMinutes(-(currentPayment.paidBase25Minutes + currentPayment.paidBonus25Minutes)) : '0 h' }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBase50Minutes) : '0 h' }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBase50Minutes) : '0 h' }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ currentPayment ? formatMinutes(-currentPayment.paidBonus50Minutes) : '0 h' }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ currentPayment ? formatMinutes(-currentPayment.paidBonus50Minutes) : '0 h' }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ currentPayment ? formatMinutes(-(currentPayment.paidBase50Minutes + currentPayment.paidBonus50Minutes)) : '0 h' }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(paidTotal) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(paidTotal) }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
@@ -131,11 +162,13 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Reste</td>
|
<td class="px-5 py-[10px] font-bold text-primary-500 border border-primary-500">Reste</td>
|
||||||
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
<td class="px-4 py-[10px] text-center text-neutral-500 border border-primary-500 border-r-2">-</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(totals.base25 - (currentPayment?.paidBase25Minutes ?? 0)) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.base25) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(totals.bonus25 - (currentPayment?.paidBonus25Minutes ?? 0)) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.bonus25) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(totals.base50 - (currentPayment?.paidBase50Minutes ?? 0)) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(reste.total25) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(totals.bonus50 - (currentPayment?.paidBonus50Minutes ?? 0)) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.base50) }}</td>
|
||||||
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(resteTotal) }}</td>
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.bonus50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(reste.total50) }}</td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.total) }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
@@ -290,44 +323,113 @@ const paddedWeeks = computed((): (EmployeeRttWeekSummary | null)[] => {
|
|||||||
return padded
|
return padded
|
||||||
})
|
})
|
||||||
|
|
||||||
// --- Report row ---
|
// --- Carry row (RTT rollover from previous year, June only) ---
|
||||||
|
|
||||||
const reportMonth = computed(() => {
|
const carryMonth = computed(() => {
|
||||||
if (!props.summary) return 6
|
if (!props.summary) return 6
|
||||||
const carryMonth = props.summary.carryMonth
|
const cm = props.summary.carryMonth
|
||||||
// Report appears in the month AFTER carryMonth (wrapping 12 -> 1)
|
return cm >= 12 ? 1 : cm + 1
|
||||||
return carryMonth >= 12 ? 1 : carryMonth + 1
|
|
||||||
})
|
})
|
||||||
|
|
||||||
const showReportRow = computed(() => {
|
const showCarryRow = computed(() => {
|
||||||
return (
|
if (currentMonth.value !== carryMonth.value) return false
|
||||||
currentMonth.value === reportMonth.value &&
|
if ((props.summary?.carryFromPreviousYearMinutes ?? 0) === 0) return false
|
||||||
(props.summary?.carryFromPreviousYearMinutes ?? 0) > 0
|
|
||||||
)
|
// On the first exercise, hide carry if carry month is before rttStartDate
|
||||||
|
if (props.summary?.rttStartDate) {
|
||||||
|
const startDate = new Date(props.summary.rttStartDate)
|
||||||
|
const viewYear = currentMonth.value >= 6 ? props.summary.year - 1 : props.summary.year
|
||||||
|
const viewDate = new Date(viewYear, currentMonth.value - 1, 1)
|
||||||
|
const startMonthDate = new Date(startDate.getFullYear(), startDate.getMonth(), 1)
|
||||||
|
if (viewDate < startMonthDate) return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
})
|
})
|
||||||
|
|
||||||
// --- Totals ---
|
// --- Month report row (cumulated balance from previous months) ---
|
||||||
|
|
||||||
|
// Months of the exercise in order, starting from the carry month
|
||||||
|
const exerciseMonths = computed((): number[] => {
|
||||||
|
const start = carryMonth.value
|
||||||
|
const startIdx = orderedMonths.indexOf(start as (typeof orderedMonths)[number])
|
||||||
|
if (startIdx === -1) return [...orderedMonths]
|
||||||
|
return [...orderedMonths.slice(startIdx), ...orderedMonths.slice(0, startIdx)]
|
||||||
|
})
|
||||||
|
|
||||||
|
const monthReport = computed(() => {
|
||||||
|
if (!props.summary) return { base25: 0, bonus25: 0, total25: 0, base50: 0, bonus50: 0, total50: 0, total: 0 }
|
||||||
|
|
||||||
|
const cm = currentMonth.value
|
||||||
|
const cmIdx = exerciseMonths.value.indexOf(cm)
|
||||||
|
const previousMonths = exerciseMonths.value.slice(0, cmIdx)
|
||||||
|
|
||||||
|
// Start from carry (included in the cumulation)
|
||||||
|
let base25 = props.summary.carryBase25Minutes
|
||||||
|
let bonus25 = props.summary.carryBonus25Minutes
|
||||||
|
let base50 = props.summary.carryBase50Minutes
|
||||||
|
let bonus50 = props.summary.carryBonus50Minutes
|
||||||
|
let total = props.summary.carryFromPreviousYearMinutes
|
||||||
|
|
||||||
|
// Add weeks from previous months
|
||||||
|
for (const w of props.summary.weeks) {
|
||||||
|
if (previousMonths.includes(w.month)) {
|
||||||
|
base25 += w.base25Minutes
|
||||||
|
bonus25 += w.bonus25Minutes
|
||||||
|
base50 += w.base50Minutes
|
||||||
|
bonus50 += w.bonus50Minutes
|
||||||
|
total += w.totalMinutes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Subtract payments from previous months
|
||||||
|
for (const p of props.summary.monthPayments) {
|
||||||
|
if (previousMonths.includes(p.month)) {
|
||||||
|
base25 -= p.paidBase25Minutes
|
||||||
|
bonus25 -= p.paidBonus25Minutes
|
||||||
|
base50 -= p.paidBase50Minutes
|
||||||
|
bonus50 -= p.paidBonus50Minutes
|
||||||
|
total -= (p.paidBase25Minutes + p.paidBonus25Minutes + p.paidBase50Minutes + p.paidBonus50Minutes)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { base25, bonus25, total25: base25 + bonus25, base50, bonus50, total50: base50 + bonus50, total }
|
||||||
|
})
|
||||||
|
|
||||||
|
const showMonthReportRow = computed(() => {
|
||||||
|
// Not on the carry month — carry row handles that
|
||||||
|
if (currentMonth.value === carryMonth.value) return false
|
||||||
|
|
||||||
|
// On the first exercise (containing rttStartDate), hide report for months before the start date
|
||||||
|
if (props.summary?.rttStartDate) {
|
||||||
|
const startDate = new Date(props.summary.rttStartDate)
|
||||||
|
const startYear = startDate.getFullYear()
|
||||||
|
const startMonth = startDate.getMonth() + 1
|
||||||
|
const viewYear = currentMonth.value >= 6 ? props.summary.year - 1 : props.summary.year
|
||||||
|
const viewDate = new Date(viewYear, currentMonth.value - 1, 1)
|
||||||
|
const startMonthDate = new Date(startYear, startMonth - 1, 1)
|
||||||
|
if (viewDate < startMonthDate) return false
|
||||||
|
}
|
||||||
|
|
||||||
|
const r = monthReport.value
|
||||||
|
return r.total !== 0
|
||||||
|
})
|
||||||
|
|
||||||
|
// --- Totals (current month weeks only) ---
|
||||||
|
|
||||||
const totals = computed(() => {
|
const totals = computed(() => {
|
||||||
const weeks = weeksForCurrentMonth.value
|
const weeks = weeksForCurrentMonth.value
|
||||||
const base = {
|
const positive = weeks.filter((w) => w.totalMinutes >= 0)
|
||||||
|
return {
|
||||||
overtime: weeks.reduce((s, w) => s + w.overtimeMinutes, 0),
|
overtime: weeks.reduce((s, w) => s + w.overtimeMinutes, 0),
|
||||||
base25: weeks.reduce((s, w) => s + w.base25Minutes, 0),
|
base25: positive.reduce((s, w) => s + w.base25Minutes, 0),
|
||||||
bonus25: weeks.reduce((s, w) => s + w.bonus25Minutes, 0),
|
bonus25: positive.reduce((s, w) => s + w.bonus25Minutes, 0),
|
||||||
base50: weeks.reduce((s, w) => s + w.base50Minutes, 0),
|
total25: weeks.reduce((s, w) => s + w.base25Minutes + w.bonus25Minutes, 0),
|
||||||
bonus50: weeks.reduce((s, w) => s + w.bonus50Minutes, 0),
|
base50: positive.reduce((s, w) => s + w.base50Minutes, 0),
|
||||||
|
bonus50: positive.reduce((s, w) => s + w.bonus50Minutes, 0),
|
||||||
|
total50: weeks.reduce((s, w) => s + w.base50Minutes + w.bonus50Minutes, 0),
|
||||||
total: weeks.reduce((s, w) => s + w.totalMinutes, 0),
|
total: weeks.reduce((s, w) => s + w.totalMinutes, 0),
|
||||||
}
|
}
|
||||||
|
|
||||||
if (showReportRow.value && props.summary) {
|
|
||||||
base.base25 += props.summary.carryBase25Minutes
|
|
||||||
base.bonus25 += props.summary.carryBonus25Minutes
|
|
||||||
base.base50 += props.summary.carryBase50Minutes
|
|
||||||
base.bonus50 += props.summary.carryBonus50Minutes
|
|
||||||
base.total += props.summary.carryFromPreviousYearMinutes
|
|
||||||
}
|
|
||||||
|
|
||||||
return base
|
|
||||||
})
|
})
|
||||||
|
|
||||||
const currentPayment = computed(() => {
|
const currentPayment = computed(() => {
|
||||||
@@ -341,8 +443,19 @@ const paidTotal = computed(() => {
|
|||||||
return -(p.paidBase25Minutes + p.paidBonus25Minutes + p.paidBase50Minutes + p.paidBonus50Minutes)
|
return -(p.paidBase25Minutes + p.paidBonus25Minutes + p.paidBase50Minutes + p.paidBonus50Minutes)
|
||||||
})
|
})
|
||||||
|
|
||||||
const resteTotal = computed(() => {
|
const reste = computed(() => {
|
||||||
return totals.value.total + paidTotal.value
|
const total25 = monthReport.value.total25 + totals.value.total25
|
||||||
|
- (currentPayment.value?.paidBase25Minutes ?? 0) - (currentPayment.value?.paidBonus25Minutes ?? 0)
|
||||||
|
const total50 = monthReport.value.total50 + totals.value.total50
|
||||||
|
- (currentPayment.value?.paidBase50Minutes ?? 0) - (currentPayment.value?.paidBonus50Minutes ?? 0)
|
||||||
|
|
||||||
|
const base25 = Math.round(total25 / 1.25)
|
||||||
|
const bonus25 = total25 - base25
|
||||||
|
const base50 = Math.round(total50 / 1.5)
|
||||||
|
const bonus50 = total50 - base50
|
||||||
|
const total = monthReport.value.total + totals.value.total + paidTotal.value
|
||||||
|
|
||||||
|
return { base25, bonus25, total25, base50, bonus50, total50, total }
|
||||||
})
|
})
|
||||||
|
|
||||||
// --- Format ---
|
// --- Format ---
|
||||||
|
|||||||
@@ -4,6 +4,13 @@
|
|||||||
<div class="flex items-center justify-between">
|
<div class="flex items-center justify-between">
|
||||||
<h1 class="text-4xl font-bold text-primary-500">Employés</h1>
|
<h1 class="text-4xl font-bold text-primary-500">Employés</h1>
|
||||||
<div class="flex items-center gap-3">
|
<div class="flex items-center gap-3">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex items-center gap-2 rounded-lg bg-primary-500 px-4 py-2 text-md font-semibold text-white hover:bg-secondary-500"
|
||||||
|
@click="handleLeaveRecapPrint"
|
||||||
|
>
|
||||||
|
Export récap. congés
|
||||||
|
</button>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="flex items-center gap-2 rounded-lg bg-primary-500 px-4 py-2 text-md font-semibold text-white hover:bg-secondary-500"
|
class="flex items-center gap-2 rounded-lg bg-primary-500 px-4 py-2 text-md font-semibold text-white hover:bg-secondary-500"
|
||||||
@@ -27,7 +34,7 @@
|
|||||||
<SiteFilterSelector v-if="sites.length > 0" v-model="selectedSiteIds" :sites="sites"/>
|
<SiteFilterSelector v-if="sites.length > 0" v-model="selectedSiteIds" :sites="sites"/>
|
||||||
<select
|
<select
|
||||||
v-model="contractStatusFilter"
|
v-model="contractStatusFilter"
|
||||||
class="rounded-md border border-primary-500 bg-white px-3 py-2 text-md font-semibold text-primary-500"
|
class="rounded-md border border-primary-500 bg-white px-3 py-2 text-md font-semibold text-primary-500 cursor-pointer"
|
||||||
>
|
>
|
||||||
<option value="active">Avec contrat</option>
|
<option value="active">Avec contrat</option>
|
||||||
<option value="inactive">Sans contrat</option>
|
<option value="inactive">Sans contrat</option>
|
||||||
@@ -530,6 +537,10 @@ const openCreate = () => {
|
|||||||
isDrawerOpen.value = true
|
isDrawerOpen.value = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const handleLeaveRecapPrint = async () => {
|
||||||
|
await printPdf('/leave-recap/print')
|
||||||
|
}
|
||||||
|
|
||||||
const handleSalaryRecapPrint = async (month: string) => {
|
const handleSalaryRecapPrint = async (month: string) => {
|
||||||
await printPdf(`/salary-recap/print?month=${month}`)
|
await printPdf(`/salary-recap/print?month=${month}`)
|
||||||
isSalaryRecapOpen.value = false
|
isSalaryRecapOpen.value = false
|
||||||
|
|||||||
@@ -32,4 +32,5 @@ export type EmployeeRttSummary = {
|
|||||||
availableMinutes: number
|
availableMinutes: number
|
||||||
weeks: EmployeeRttWeekSummary[]
|
weeks: EmployeeRttWeekSummary[]
|
||||||
monthPayments: RttMonthPayment[]
|
monthPayments: RttMonthPayment[]
|
||||||
|
rttStartDate: string | null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ final class EmployeeRttSummary
|
|||||||
public int $currentYearRecoveryMinutes = 0;
|
public int $currentYearRecoveryMinutes = 0;
|
||||||
public int $availableMinutes = 0;
|
public int $availableMinutes = 0;
|
||||||
public int $totalPaidMinutes = 0;
|
public int $totalPaidMinutes = 0;
|
||||||
|
public ?string $rttStartDate = null;
|
||||||
|
|
||||||
/** @var list<RttMonthPayment> */
|
/** @var list<RttMonthPayment> */
|
||||||
public array $monthPayments = [];
|
public array $monthPayments = [];
|
||||||
|
|||||||
20
src/ApiResource/LeaveRecapPrint.php
Normal file
20
src/ApiResource/LeaveRecapPrint.php
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\ApiResource;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\ApiResource;
|
||||||
|
use ApiPlatform\Metadata\Get;
|
||||||
|
use App\State\LeaveRecapPrintProvider;
|
||||||
|
|
||||||
|
#[ApiResource(
|
||||||
|
operations: [
|
||||||
|
new Get(
|
||||||
|
uriTemplate: '/leave-recap/print',
|
||||||
|
provider: LeaveRecapPrintProvider::class,
|
||||||
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)]
|
||||||
|
final class LeaveRecapPrint {}
|
||||||
@@ -24,6 +24,7 @@ use Symfony\Component\Serializer\Normalizer\DateTimeNormalizer;
|
|||||||
paginationEnabled: false,
|
paginationEnabled: false,
|
||||||
security: "is_granted('ROLE_ADMIN')",
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
processor: EmployeeWriteProcessor::class,
|
processor: EmployeeWriteProcessor::class,
|
||||||
|
order: ['site.name' => 'ASC', 'displayOrder' => 'ASC', 'lastName' => 'ASC', 'firstName' => 'ASC'],
|
||||||
)]
|
)]
|
||||||
#[ORM\Entity(repositoryClass: EmployeeRepository::class)]
|
#[ORM\Entity(repositoryClass: EmployeeRepository::class)]
|
||||||
#[ORM\Table(name: 'employees')]
|
#[ORM\Table(name: 'employees')]
|
||||||
|
|||||||
@@ -87,8 +87,7 @@ final class EmployeeRepository extends ServiceEntityRepository implements Employ
|
|||||||
->addSelect('s')
|
->addSelect('s')
|
||||||
->leftJoin('e.contract', 'c')
|
->leftJoin('e.contract', 'c')
|
||||||
->addSelect('c')
|
->addSelect('c')
|
||||||
->orderBy('s.displayOrder', 'ASC')
|
->orderBy('s.name', 'ASC')
|
||||||
->addOrderBy('s.name', 'ASC')
|
|
||||||
->addOrderBy('e.displayOrder', 'ASC')
|
->addOrderBy('e.displayOrder', 'ASC')
|
||||||
->addOrderBy('e.lastName', 'ASC')
|
->addOrderBy('e.lastName', 'ASC')
|
||||||
->addOrderBy('e.firstName', 'ASC')
|
->addOrderBy('e.firstName', 'ASC')
|
||||||
|
|||||||
@@ -21,13 +21,18 @@ use DateTimeImmutable;
|
|||||||
|
|
||||||
final readonly class RttRecoveryComputationService
|
final readonly class RttRecoveryComputationService
|
||||||
{
|
{
|
||||||
|
private ?DateTimeImmutable $rttStartDate;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private WorkHourRepository $workHourRepository,
|
private WorkHourRepository $workHourRepository,
|
||||||
private AbsenceRepository $absenceRepository,
|
private AbsenceRepository $absenceRepository,
|
||||||
private AbsenceSegmentsResolver $absenceSegmentsResolver,
|
private AbsenceSegmentsResolver $absenceSegmentsResolver,
|
||||||
private WorkedHoursCreditPolicy $workedHoursCreditPolicy,
|
private WorkedHoursCreditPolicy $workedHoursCreditPolicy,
|
||||||
private EmployeeContractResolver $contractResolver,
|
private EmployeeContractResolver $contractResolver,
|
||||||
) {}
|
string $rttStartDate = '',
|
||||||
|
) {
|
||||||
|
$this->rttStartDate = '' !== $rttStartDate ? new DateTimeImmutable($rttStartDate) : null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return array{DateTimeImmutable, DateTimeImmutable}
|
* @return array{DateTimeImmutable, DateTimeImmutable}
|
||||||
@@ -71,7 +76,7 @@ final readonly class RttRecoveryComputationService
|
|||||||
return $weeks;
|
return $weeks;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function computeTotalRecoveryForExercise(Employee $employee, int $exerciseYear): WeekRecoveryDetail
|
public function computeTotalRecoveryForExercise(Employee $employee, int $exerciseYear, ?DateTimeImmutable $limitDate = null): WeekRecoveryDetail
|
||||||
{
|
{
|
||||||
[$from, $to] = $this->resolveExerciseBounds($exerciseYear);
|
[$from, $to] = $this->resolveExerciseBounds($exerciseYear);
|
||||||
$weeks = $this->buildWeeksForExercise($from, $to);
|
$weeks = $this->buildWeeksForExercise($from, $to);
|
||||||
@@ -85,7 +90,7 @@ final readonly class RttRecoveryComputationService
|
|||||||
$weeks
|
$weeks
|
||||||
);
|
);
|
||||||
|
|
||||||
$byWeek = $this->computeRecoveryByWeek($employee, $weekRanges, $from, $to, null);
|
$byWeek = $this->computeRecoveryByWeek($employee, $weekRanges, $from, $to, $limitDate);
|
||||||
|
|
||||||
$total = new WeekRecoveryDetail();
|
$total = new WeekRecoveryDetail();
|
||||||
foreach ($byWeek as $detail) {
|
foreach ($byWeek as $detail) {
|
||||||
@@ -172,6 +177,12 @@ final readonly class RttRecoveryComputationService
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($this->rttStartDate instanceof DateTimeImmutable && $effectiveEnd < $this->rttStartDate) {
|
||||||
|
$results[$weekKey] = new WeekRecoveryDetail();
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
$weekDays = [];
|
$weekDays = [];
|
||||||
for ($cursor = $effectiveStart; $cursor <= $effectiveEnd; $cursor = $cursor->modify('+1 day')) {
|
for ($cursor = $effectiveStart; $cursor <= $effectiveEnd; $cursor = $cursor->modify('+1 day')) {
|
||||||
$weekDays[] = $cursor->format('Y-m-d');
|
$weekDays[] = $cursor->format('Y-m-d');
|
||||||
@@ -203,7 +214,7 @@ final readonly class RttRecoveryComputationService
|
|||||||
$overtime25StartMinutes = $this->computeWeeklyOvertime25StartMinutes($weekDays, $employeeContractsByDate);
|
$overtime25StartMinutes = $this->computeWeeklyOvertime25StartMinutes($weekDays, $employeeContractsByDate);
|
||||||
$weeklyOvertimeTotalMinutes = $isWeekPresenceTracking
|
$weeklyOvertimeTotalMinutes = $isWeekPresenceTracking
|
||||||
? 0
|
? 0
|
||||||
: max(0, $weeklyTotalMinutes - $overtimeReferenceMinutes);
|
: $weeklyTotalMinutes - $overtimeReferenceMinutes;
|
||||||
|
|
||||||
$base25 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : max(0, min($weeklyTotalMinutes, 43 * 60) - $overtime25StartMinutes);
|
$base25 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : max(0, min($weeklyTotalMinutes, 43 * 60) - $overtime25StartMinutes);
|
||||||
$bonus25 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : (int) round($base25 * 0.25);
|
$bonus25 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : (int) round($base25 * 0.25);
|
||||||
|
|||||||
@@ -126,9 +126,9 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
* previousYearRemainingDays: float
|
* previousYearRemainingDays: float
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
private function computeYearSummary(Employee $employee, int $targetYear): ?array
|
public function computeYearSummary(Employee $employee, int $targetYear): ?array
|
||||||
{
|
{
|
||||||
$firstYear = $this->resolveFirstComputationYear($employee);
|
$firstYear = max($this->resolveFirstComputationYear($employee), $targetYear - 1);
|
||||||
if ($targetYear < $firstYear) {
|
if ($targetYear < $firstYear) {
|
||||||
$targetYear = $firstYear;
|
$targetYear = $firstYear;
|
||||||
}
|
}
|
||||||
@@ -286,6 +286,16 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
return $targetSummary;
|
return $targetSummary;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function resolveLeaveYearForToday(Employee $employee): int
|
||||||
|
{
|
||||||
|
$today = new DateTimeImmutable('today');
|
||||||
|
if (ContractType::FORFAIT === $employee->getContract()?->getType()) {
|
||||||
|
return (int) $today->format('Y');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->resolveCurrentLeaveYear($today);
|
||||||
|
}
|
||||||
|
|
||||||
private function resolveEffectivePeriodStart(
|
private function resolveEffectivePeriodStart(
|
||||||
Employee $employee,
|
Employee $employee,
|
||||||
DateTimeImmutable $from,
|
DateTimeImmutable $from,
|
||||||
|
|||||||
@@ -26,6 +26,8 @@ use Symfony\Component\HttpKernel\Exception\UnprocessableEntityHttpException;
|
|||||||
|
|
||||||
final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
||||||
{
|
{
|
||||||
|
private ?string $rttStartDate;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private Security $security,
|
private Security $security,
|
||||||
private RequestStack $requestStack,
|
private RequestStack $requestStack,
|
||||||
@@ -34,7 +36,10 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
private EmployeeRttBalanceRepository $rttBalanceRepository,
|
private EmployeeRttBalanceRepository $rttBalanceRepository,
|
||||||
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
||||||
private RttRecoveryComputationService $rttRecoveryService,
|
private RttRecoveryComputationService $rttRecoveryService,
|
||||||
) {}
|
string $rttStartDate = '',
|
||||||
|
) {
|
||||||
|
$this->rttStartDate = '' !== $rttStartDate ? $rttStartDate : null;
|
||||||
|
}
|
||||||
|
|
||||||
public function provide(Operation $operation, array $uriVariables = [], array $context = []): EmployeeRttSummary
|
public function provide(Operation $operation, array $uriVariables = [], array $context = []): EmployeeRttSummary
|
||||||
{
|
{
|
||||||
@@ -72,9 +77,12 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
$weeks
|
$weeks
|
||||||
);
|
);
|
||||||
|
|
||||||
$limitDate = null;
|
|
||||||
if ($year > $currentExerciseYear) {
|
if ($year > $currentExerciseYear) {
|
||||||
$limitDate = $periodFrom->modify('-1 day');
|
$limitDate = $periodFrom->modify('-1 day');
|
||||||
|
} else {
|
||||||
|
// Exclude the current (incomplete) week: limit to last Sunday
|
||||||
|
$isoDay = (int) $today->format('N'); // 1=Monday .. 7=Sunday
|
||||||
|
$limitDate = 7 === $isoDay ? $today : $today->modify('last sunday');
|
||||||
}
|
}
|
||||||
|
|
||||||
$currentByWeekStart = $this->rttRecoveryService->computeRecoveryByWeek($employee, $weekRanges, $periodFrom, $periodTo, $limitDate);
|
$currentByWeekStart = $this->rttRecoveryService->computeRecoveryByWeek($employee, $weekRanges, $periodFrom, $periodTo, $limitDate);
|
||||||
@@ -90,7 +98,15 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
$summary->carryBonus50Minutes = $carry->bonus50Minutes;
|
$summary->carryBonus50Minutes = $carry->bonus50Minutes;
|
||||||
$summary->currentYearRecoveryMinutes = array_sum(array_map(static fn ($d) => $d->totalMinutes, $currentByWeekStart));
|
$summary->currentYearRecoveryMinutes = array_sum(array_map(static fn ($d) => $d->totalMinutes, $currentByWeekStart));
|
||||||
$summary->availableMinutes = $summary->carryFromPreviousYearMinutes + $summary->currentYearRecoveryMinutes;
|
$summary->availableMinutes = $summary->carryFromPreviousYearMinutes + $summary->currentYearRecoveryMinutes;
|
||||||
$summary->weeks = array_map(
|
|
||||||
|
// Pass rttStartDate only if it falls within this exercise
|
||||||
|
if (null !== $this->rttStartDate) {
|
||||||
|
$startDate = new DateTimeImmutable($this->rttStartDate);
|
||||||
|
if ($startDate >= $periodFrom && $startDate <= $periodTo) {
|
||||||
|
$summary->rttStartDate = $this->rttStartDate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$summary->weeks = array_map(
|
||||||
static function (array $week) use ($currentByWeekStart) {
|
static function (array $week) use ($currentByWeekStart) {
|
||||||
$detail = $currentByWeekStart[$week['start']->format('Y-m-d')] ?? new WeekRecoveryDetail();
|
$detail = $currentByWeekStart[$week['start']->format('Y-m-d')] ?? new WeekRecoveryDetail();
|
||||||
|
|
||||||
@@ -110,6 +126,37 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
$weekRanges
|
$weekRanges
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Post-process: distribute deficit weeks across cumulative balance (50% first, then 25%)
|
||||||
|
$cumulative50 = $carry->base50Minutes + $carry->bonus50Minutes;
|
||||||
|
$cumulative25 = $carry->base25Minutes + $carry->bonus25Minutes;
|
||||||
|
|
||||||
|
foreach ($summary->weeks as $i => $week) {
|
||||||
|
if ($week->totalMinutes >= 0) {
|
||||||
|
$cumulative50 += $week->base50Minutes + $week->bonus50Minutes;
|
||||||
|
$cumulative25 += $week->base25Minutes + $week->bonus25Minutes;
|
||||||
|
} else {
|
||||||
|
$deficit = -$week->totalMinutes;
|
||||||
|
$from50 = min($deficit, max(0, $cumulative50));
|
||||||
|
$from25 = $deficit - $from50;
|
||||||
|
|
||||||
|
$cumulative50 -= $from50;
|
||||||
|
$cumulative25 -= $from25;
|
||||||
|
|
||||||
|
$summary->weeks[$i] = new EmployeeRttWeekSummary(
|
||||||
|
month: $week->month,
|
||||||
|
weekNumber: $week->weekNumber,
|
||||||
|
weekStart: $week->weekStart,
|
||||||
|
weekEnd: $week->weekEnd,
|
||||||
|
overtimeMinutes: $week->overtimeMinutes,
|
||||||
|
base25Minutes: $from25 > 0 ? -$from25 : 0,
|
||||||
|
bonus25Minutes: 0,
|
||||||
|
base50Minutes: $from50 > 0 ? -$from50 : 0,
|
||||||
|
bonus50Minutes: 0,
|
||||||
|
totalMinutes: $week->totalMinutes,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$payments = $this->rttPaymentRepository->findByEmployeeAndYear($employee, $year);
|
$payments = $this->rttPaymentRepository->findByEmployeeAndYear($employee, $year);
|
||||||
$monthBuckets = [];
|
$monthBuckets = [];
|
||||||
|
|
||||||
|
|||||||
181
src/State/LeaveRecapPrintProvider.php
Normal file
181
src/State/LeaveRecapPrintProvider.php
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\State;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\Operation;
|
||||||
|
use ApiPlatform\State\ProviderInterface;
|
||||||
|
use App\Entity\Employee;
|
||||||
|
use App\Enum\ContractNature;
|
||||||
|
use App\Enum\ContractType;
|
||||||
|
use App\Enum\TrackingMode;
|
||||||
|
use App\Repository\EmployeeRepository;
|
||||||
|
use App\Repository\EmployeeRttBalanceRepository;
|
||||||
|
use App\Repository\EmployeeRttPaymentRepository;
|
||||||
|
use App\Service\Rtt\RttRecoveryComputationService;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Dompdf\Dompdf;
|
||||||
|
use Dompdf\Options;
|
||||||
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Throwable;
|
||||||
|
use Twig\Environment;
|
||||||
|
|
||||||
|
class LeaveRecapPrintProvider implements ProviderInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private Environment $twig,
|
||||||
|
private EmployeeRepository $employeeRepository,
|
||||||
|
private EmployeeLeaveSummaryProvider $leaveSummaryProvider,
|
||||||
|
private RttRecoveryComputationService $rttRecoveryService,
|
||||||
|
private EmployeeRttBalanceRepository $rttBalanceRepository,
|
||||||
|
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
||||||
|
private EntityManagerInterface $entityManager,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function provide(Operation $operation, array $uriVariables = [], array $context = []): Response
|
||||||
|
{
|
||||||
|
$today = new DateTimeImmutable('today');
|
||||||
|
$employees = $this->employeeRepository->findForPrintBySiteIds([]);
|
||||||
|
|
||||||
|
$siteGroups = [];
|
||||||
|
|
||||||
|
foreach ($employees as $employee) {
|
||||||
|
if (!$employee->getHasActiveContract()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$site = $employee->getSite();
|
||||||
|
$siteId = $site ? $site->getId() : 0;
|
||||||
|
|
||||||
|
if (!isset($siteGroups[$siteId])) {
|
||||||
|
$siteGroups[$siteId] = [
|
||||||
|
'name' => $site ? $site->getName() : 'Sans site',
|
||||||
|
'color' => $site?->getColor() ?? '#ffd7d7',
|
||||||
|
'employees' => [],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
$siteGroups[$siteId]['employees'][] = $this->buildEmployeeRow($employee, $today);
|
||||||
|
$this->entityManager->clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Re-load Twig environment after clear
|
||||||
|
$options = new Options();
|
||||||
|
$options->set('isRemoteEnabled', true);
|
||||||
|
|
||||||
|
$dompdf = new Dompdf($options);
|
||||||
|
$html = $this->twig->render('leave-recap/print.html.twig', [
|
||||||
|
'today' => $today,
|
||||||
|
'siteGroups' => $siteGroups,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$dompdf->loadHtml($html);
|
||||||
|
$dompdf->setPaper('A4', 'portrait');
|
||||||
|
$dompdf->render();
|
||||||
|
|
||||||
|
$filename = sprintf('recap_conges_%s.pdf', $today->format('Y-m-d'));
|
||||||
|
|
||||||
|
return new Response($dompdf->output(), Response::HTTP_OK, [
|
||||||
|
'Content-Type' => 'application/pdf',
|
||||||
|
'Content-Disposition' => 'inline; filename="'.$filename.'"',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function buildEmployeeRow(Employee $employee, DateTimeImmutable $today): array
|
||||||
|
{
|
||||||
|
$contract = $employee->getContract();
|
||||||
|
$contractName = $contract?->getName();
|
||||||
|
$isForfait = ContractType::FORFAIT === $contract?->getType();
|
||||||
|
$nature = ContractNature::tryFrom($employee->getCurrentContractNature());
|
||||||
|
$isInterim = ContractNature::INTERIM === $nature;
|
||||||
|
|
||||||
|
$cpN1Remaining = 0.0;
|
||||||
|
$cpN = '-';
|
||||||
|
$acquiredSaturdays = '-';
|
||||||
|
$rtt = '-';
|
||||||
|
|
||||||
|
if (!$isInterim) {
|
||||||
|
$leaveYear = $this->leaveSummaryProvider->resolveLeaveYearForToday($employee);
|
||||||
|
$yearSummary = $this->leaveSummaryProvider->computeYearSummary($employee, $leaveYear);
|
||||||
|
|
||||||
|
if (null !== $yearSummary) {
|
||||||
|
if ($isForfait) {
|
||||||
|
$cpN1Remaining = round($yearSummary['previousYearRemainingDays'], 2);
|
||||||
|
$cpN = (string) round($yearSummary['acquiredDays'], 2);
|
||||||
|
$acquiredSaturdays = '-';
|
||||||
|
} else {
|
||||||
|
$cpN1Remaining = round($yearSummary['remainingDays'], 2);
|
||||||
|
$cpN = (string) round($yearSummary['accruingDays'], 2);
|
||||||
|
$acquiredSaturdays = (string) round($yearSummary['remainingSaturdays'], 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$isForfait && TrackingMode::PRESENCE->value !== $contract?->getTrackingMode()) {
|
||||||
|
try {
|
||||||
|
$rtt = $this->formatMinutes($this->computeAvailableRttMinutes($employee, $today));
|
||||||
|
} catch (Throwable) {
|
||||||
|
$rtt = '-';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'lastName' => $employee->getLastName(),
|
||||||
|
'firstName' => $employee->getFirstName(),
|
||||||
|
'contractName' => $contractName,
|
||||||
|
'cpN1Remaining' => $cpN1Remaining,
|
||||||
|
'cpN' => $cpN,
|
||||||
|
'acquiredSaturdays' => $acquiredSaturdays,
|
||||||
|
'rtt' => $rtt,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function computeAvailableRttMinutes(Employee $employee, DateTimeImmutable $today): int
|
||||||
|
{
|
||||||
|
$month = (int) $today->format('n');
|
||||||
|
$year = (int) $today->format('Y');
|
||||||
|
$exerciseYear = $month >= 6 ? $year + 1 : $year;
|
||||||
|
|
||||||
|
// Exclude incomplete current week: limit to last Sunday
|
||||||
|
$isoDay = (int) $today->format('N');
|
||||||
|
$limitDate = 7 === $isoDay ? $today : $today->modify('last sunday');
|
||||||
|
|
||||||
|
// Carry from previous exercise
|
||||||
|
$carry = 0;
|
||||||
|
$balance = $this->rttBalanceRepository->findOneByEmployeeAndYear($employee, $exerciseYear);
|
||||||
|
if (null !== $balance) {
|
||||||
|
$carry = $balance->getTotalOpeningMinutes();
|
||||||
|
} else {
|
||||||
|
$previousTotal = $this->rttRecoveryService->computeTotalRecoveryForExercise($employee, $exerciseYear - 1);
|
||||||
|
$carry = $previousTotal->totalMinutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Current exercise (limited to completed weeks)
|
||||||
|
$current = $this->rttRecoveryService->computeTotalRecoveryForExercise($employee, $exerciseYear, $limitDate);
|
||||||
|
|
||||||
|
// Paid RTT
|
||||||
|
$paid = 0;
|
||||||
|
$payments = $this->rttPaymentRepository->findByEmployeeAndYear($employee, $exerciseYear);
|
||||||
|
foreach ($payments as $payment) {
|
||||||
|
$paid += $payment->getBase25Minutes() + $payment->getBase50Minutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $carry + $current->totalMinutes - $paid;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function formatMinutes(int $minutes): string
|
||||||
|
{
|
||||||
|
if (0 === $minutes) {
|
||||||
|
return '0 h';
|
||||||
|
}
|
||||||
|
|
||||||
|
$sign = $minutes < 0 ? '- ' : '';
|
||||||
|
$abs = abs($minutes);
|
||||||
|
$h = intdiv($abs, 60);
|
||||||
|
$m = $abs % 60;
|
||||||
|
|
||||||
|
return 0 === $m ? "{$sign}{$h} h" : "{$sign}{$h} h {$m} m";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -298,7 +298,7 @@ class SalaryRecapPrintProvider implements ProviderInterface
|
|||||||
$nightMinutesTotal += $wh->getNightHoursMinutes() ?? 0;
|
$nightMinutesTotal += $wh->getNightHoursMinutes() ?? 0;
|
||||||
$dayMin = $wh->getDayHoursMinutes() ?? 0;
|
$dayMin = $wh->getDayHoursMinutes() ?? 0;
|
||||||
$nightMin = $wh->getNightHoursMinutes() ?? 0;
|
$nightMin = $wh->getNightHoursMinutes() ?? 0;
|
||||||
if ($nightMin > $dayMin && $nightMin > 0) {
|
if (($nightMin > $dayMin && $nightMin > 0) || $nightMin >= 240) {
|
||||||
++$nightBasketCount;
|
++$nightBasketCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -322,7 +322,7 @@ class SalaryRecapPrintProvider implements ProviderInterface
|
|||||||
} else {
|
} else {
|
||||||
$metrics = $this->computeNightMinutes($wh);
|
$metrics = $this->computeNightMinutes($wh);
|
||||||
$nightMinutesTotal += $metrics['nightMinutes'];
|
$nightMinutesTotal += $metrics['nightMinutes'];
|
||||||
if ($metrics['nightMinutes'] > $metrics['dayMinutes'] && $metrics['nightMinutes'] > 0) {
|
if (($metrics['nightMinutes'] > $metrics['dayMinutes'] && $metrics['nightMinutes'] > 0) || $metrics['nightMinutes'] >= 240) {
|
||||||
++$nightBasketCount;
|
++$nightBasketCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -271,7 +271,7 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
$present = min(1.0, $morning + $afternoon + $creditedPresence);
|
$present = min(1.0, $morning + $afternoon + $creditedPresence);
|
||||||
}
|
}
|
||||||
|
|
||||||
$hasNightBasket = $nightMinutes > $dayMinutes && $nightMinutes > 0;
|
$hasNightBasket = ($nightMinutes > $dayMinutes && $nightMinutes > 0) || $nightMinutes >= 240;
|
||||||
if ($hasNightBasket) {
|
if ($hasNightBasket) {
|
||||||
++$weeklyNightBasketCount;
|
++$weeklyNightBasketCount;
|
||||||
}
|
}
|
||||||
|
|||||||
124
templates/leave-recap/print.html.twig
Normal file
124
templates/leave-recap/print.html.twig
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="fr">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>Récapitulatif Congés & RTT</title>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
@page { size: A4 portrait; margin: 4mm; }
|
||||||
|
|
||||||
|
html, body {
|
||||||
|
margin: 0;
|
||||||
|
padding: 2mm;
|
||||||
|
font-family: Helvetica, sans-serif;
|
||||||
|
font-size: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-bar {
|
||||||
|
position: relative;
|
||||||
|
margin: 0 0 6mm 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 18px;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-box {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
border: 2px solid #000;
|
||||||
|
padding: 4px 12px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.recap {
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
table-layout: auto;
|
||||||
|
border: 4px solid #0a0a0a;
|
||||||
|
}
|
||||||
|
|
||||||
|
th, td {
|
||||||
|
border: 2px solid #0a0a0a;
|
||||||
|
padding: 3px 5px;
|
||||||
|
vertical-align: middle;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.site-header td {
|
||||||
|
font-weight: 700;
|
||||||
|
font-size: 12px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
thead th {
|
||||||
|
text-align: center;
|
||||||
|
font-weight: 700;
|
||||||
|
font-size: 10px;
|
||||||
|
white-space: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.name {
|
||||||
|
text-align: left;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
td.base { text-align: center; }
|
||||||
|
td.num { text-align: center; }
|
||||||
|
td.obs { min-width: 40mm; }
|
||||||
|
|
||||||
|
tbody td { font-size: 10px; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<div class="title-bar">
|
||||||
|
<h1>RECAPITULATIF CONGES & RTT</h1>
|
||||||
|
<div class="date-box">{{ today|date('d/m/Y') }}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table class="recap">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th style="text-align: left;">Nom</th>
|
||||||
|
<th>Contrat</th>
|
||||||
|
<th>CP N-1<br>restant</th>
|
||||||
|
<th>Samedi<br>restant</th>
|
||||||
|
<th>CP<br>N</th>
|
||||||
|
<th>RTT</th>
|
||||||
|
<th style="width: 40mm;">Observations</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{% for siteId, group in siteGroups %}
|
||||||
|
{% set siteColor = group.color ?? '#B3E5FC' %}
|
||||||
|
<tr class="site-header">
|
||||||
|
<td style="background: {{ siteColor }}; text-align: left;" colspan="7">
|
||||||
|
{{ group.name }}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% for row in group.employees %}
|
||||||
|
<tr>
|
||||||
|
<td class="name">{{ row.lastName }} {{ row.firstName }}</td>
|
||||||
|
<td class="base">{{ row.contractName ?? '' }}</td>
|
||||||
|
<td class="num">{{ row.cpN1Remaining }}</td>
|
||||||
|
<td class="num">{{ row.acquiredSaturdays }}</td>
|
||||||
|
<td class="num">{{ row.cpN }}</td>
|
||||||
|
<td class="num">{{ row.rtt }}</td>
|
||||||
|
<td class="obs"></td>
|
||||||
|
</tr>
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan="7">Aucun employé.</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
Reference in New Issue
Block a user