Compare commits
52 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
13743738fd | ||
| 085fe0c150 | |||
|
|
a1110069b5 | ||
| 4901c58ebf | |||
| 4de891579c | |||
|
|
a17d6a67cf | ||
| 29db3b5025 | |||
|
|
6df9110187 | ||
| f0dfb30566 | |||
| 049e64288e | |||
|
|
9577a70ea3 | ||
| e85f7b6f4c | |||
|
|
834b4cb695 | ||
| 17f871e82d | |||
|
|
3ec1e1f10d | ||
| 24b7512c8a | |||
| f047e3ed4b | |||
|
|
1feedd0381 | ||
| f9cd5a0143 | |||
|
|
ede7decaa7 | ||
| 2cfb05e5de | |||
|
|
0a8399a950 | ||
| 6a64cb4c58 | |||
|
|
facded4c55 | ||
| 9787231052 | |||
|
|
8563ddb08c | ||
| 353d4d9d2b | |||
|
|
8745e5e425 | ||
| 4d8c850a77 | |||
| 1974ace1f2 | |||
|
|
a99a12a759 | ||
| 548b5d63a6 | |||
|
|
ed9df4e178 | ||
| 625b4af5ba | |||
|
|
2ec3044cb3 | ||
| f024a6a8de | |||
|
|
a60294a8f7 | ||
| dd7f9ef8a0 | |||
| cfa7d25521 | |||
|
|
5faa0facca | ||
| 04f90afc58 | |||
|
|
e022cfac98 | ||
| e827128392 | |||
| 86cdec50c6 | |||
|
|
443ed1e003 | ||
| cef364fcec | |||
|
|
d4884bc489 | ||
| b93c4bf3e9 | |||
|
|
f0ee489c26 | ||
| 01f8058f56 | |||
|
|
3d26d6b50f | ||
| 339d650b41 |
@@ -21,7 +21,10 @@
|
|||||||
"Bash(ls /home/m-tristan/workspace/SIRH/docker* /home/m-tristan/workspace/SIRH/Makefile 2>/dev/null; cat /home/m-tristan/workspace/SIRH/Makefile 2>/dev/null | grep -E \"\\(phpunit|test|php\\)\" | head -20)",
|
"Bash(ls /home/m-tristan/workspace/SIRH/docker* /home/m-tristan/workspace/SIRH/Makefile 2>/dev/null; cat /home/m-tristan/workspace/SIRH/Makefile 2>/dev/null | grep -E \"\\(phpunit|test|php\\)\" | head -20)",
|
||||||
"Bash(which python3:*)",
|
"Bash(which python3:*)",
|
||||||
"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(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 ###
|
||||||
|
|||||||
6
.idea/sqldialects.xml
generated
Normal file
6
.idea/sqldialects.xml
generated
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="SqlDialectMappings">
|
||||||
|
<file url="file://$PROJECT_DIR$/sirh.sql" dialect="GenericSQL" />
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
@@ -32,6 +32,7 @@
|
|||||||
- Employee contract history: `employee_contract_periods`, resolved by `EmployeeContractResolver`
|
- Employee contract history: `employee_contract_periods`, resolved by `EmployeeContractResolver`
|
||||||
- Absences: stored per day (auto-split), AM/PM/full day, clear corresponding hour slots
|
- Absences: stored per day (auto-split), AM/PM/full day, clear corresponding hour slots
|
||||||
- Absences with `countAsWorkedHours=true`: credit minutes (TIME) or nothing (PRESENCE)
|
- Absences with `countAsWorkedHours=true`: credit minutes (TIME) or nothing (PRESENCE)
|
||||||
|
- Driver periods (`isDriver=true` on `EmployeeContractPeriod`): separate screen `/driver-hours`, uses `dayHoursMinutes`/`nightHoursMinutes` + meal/overnight flags on `WorkHour`
|
||||||
|
|
||||||
## Validation Rules
|
## Validation Rules
|
||||||
- `isValid` (RH): locks line for everyone (admin can only untoggle validation)
|
- `isValid` (RH): locks line for everyone (admin can only untoggle validation)
|
||||||
@@ -42,7 +43,15 @@
|
|||||||
## Overtime Rules
|
## Overtime Rules
|
||||||
- Contracts <= 35h: +25% from 35h to 43h, +50% beyond
|
- Contracts <= 35h: +25% from 35h to 43h, +50% beyond
|
||||||
- Contracts >= 39h: +25% from 39h to 43h, +50% beyond
|
- Contracts >= 39h: +25% from 39h to 43h, +50% beyond
|
||||||
|
- CUSTOM contracts (weeklyHours ≠ 35 and ≠ 39, not INTERIM/FORFAIT): reference = actual contractual hours, no 25%/50% bonuses (1h overtime = 1h recovery), deficit doesn't impact balance
|
||||||
- INTERIM: no overtime bonuses, no recovery time
|
- INTERIM: no overtime bonuses, no recovery time
|
||||||
|
- Driver contracts: RTT uses `dayHoursMinutes + nightHoursMinutes + workshopHoursMinutes` instead of morning/afternoon/evening time ranges
|
||||||
|
- FORFAIT weekend/holiday bonus: each weekend or public holiday day worked gives bonus leave (full day if morning+afternoon, 0.5 if only one). Added to acquired days, no cap. PRESENCE mode only.
|
||||||
|
|
||||||
|
## Frais (MileageAllowance)
|
||||||
|
- Onglet "Frais" (anciennement "Frais Kms") sur la fiche employé
|
||||||
|
- Validation: mois obligatoire + au moins `kilometers > 0` ou `amount > 0`
|
||||||
|
- Les deux champs km et montant sont optionnels individuellement mais au moins un requis
|
||||||
|
|
||||||
## Frontend Patterns
|
## Frontend Patterns
|
||||||
|
|
||||||
|
|||||||
@@ -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.38'
|
app.version: '0.1.61'
|
||||||
|
|||||||
@@ -40,6 +40,10 @@ Documents complementaires:
|
|||||||
|
|
||||||
## 3) Heures (vue jour)
|
## 3) Heures (vue jour)
|
||||||
|
|
||||||
|
- Visibilité des employés:
|
||||||
|
- vue jour: un employé sans contrat à la date sélectionnée est masqué
|
||||||
|
- vue semaine: un employé sans contrat sur aucun jour de la semaine est masqué
|
||||||
|
- même règle pour les heures classiques et les heures conducteurs
|
||||||
- Saisie par salarié et par date:
|
- Saisie par salarié et par date:
|
||||||
- matin / après-midi / soir
|
- matin / après-midi / soir
|
||||||
- pour `PRESENCE`: demi-journées matin/après-midi
|
- pour `PRESENCE`: demi-journées matin/après-midi
|
||||||
@@ -112,11 +116,48 @@ 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
|
||||||
|
- Contrats CUSTOM (heures hebdo ≠ 35h et ≠ 39h, hors INTERIM/FORFAIT):
|
||||||
|
- référence heures sup = heures contractuelles réelles (ex: 4h → référence 4h)
|
||||||
|
- pas de bonus 25% ni 50% : 1 heure sup = 1 heure de récupération
|
||||||
|
- le déficit (travail < contrat) ne génère pas de récup mais n'impacte pas le solde
|
||||||
- Nature `INTERIM`:
|
- Nature `INTERIM`:
|
||||||
- pas de bonus 25%
|
- pas de bonus 25%
|
||||||
- pas de bonus 50%
|
- pas de bonus 50%
|
||||||
- pas de total récup
|
- pas de total récup
|
||||||
|
|
||||||
|
## 6bis) Heures Conducteurs
|
||||||
|
|
||||||
|
- Écran dédié `/driver-hours` pour les employés dont le contrat est marqué `isDriver = true`
|
||||||
|
- Les conducteurs sont exclus de l'écran `/hours` classique
|
||||||
|
- Colonnes spécifiques (vue jour):
|
||||||
|
- Heure de jour (durée HH:MM via TimeSelect)
|
||||||
|
- Heure de nuit (durée HH:MM via TimeSelect)
|
||||||
|
- Heure atelier (durée HH:MM via TimeSelect)
|
||||||
|
- Total (somme jour + nuit + atelier, calculé)
|
||||||
|
- Petit déjeuner (checkbox)
|
||||||
|
- Déjeuner (checkbox)
|
||||||
|
- Dîner (checkbox)
|
||||||
|
- Nuitée (checkbox)
|
||||||
|
- Stockage backend:
|
||||||
|
- `dayHoursMinutes`, `nightHoursMinutes` et `workshopHoursMinutes` (entiers, minutes) sur `WorkHour`
|
||||||
|
- `hasBreakfast`, `hasLunch`, `hasDinner`, `hasOvernight` (booleans) sur `WorkHour`
|
||||||
|
- les champs time classiques (morning/afternoon/evening) sont mis à null pour les chauffeurs
|
||||||
|
- Absences `countAsWorkedHours=true`: les minutes créditées sont ajoutées aux heures de jour (vue jour et vue semaine), même logique que les employés classiques
|
||||||
|
- Validation: même logique que les heures classiques (`isValid`, `isSiteValid`, bulk)
|
||||||
|
- Vue semaine:
|
||||||
|
- jour/nuit/atelier par jour + indicateurs repas/dîner/nuitée
|
||||||
|
- 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
|
||||||
|
- les conducteurs utilisent `dayHoursMinutes + nightHoursMinutes + workshopHoursMinutes` pour le calcul RTT (au lieu des créneaux morning/afternoon/evening)
|
||||||
|
- Le flag `isDriver` est sur `EmployeeContractPeriod` (un employé peut changer de statut chauffeur selon la période)
|
||||||
|
- Exposé en API via un getter virtuel sur `Employee` (`employee:read`) qui résout depuis la période active
|
||||||
|
|
||||||
## 7) Fériés
|
## 7) Fériés
|
||||||
|
|
||||||
- Les jours fériés sont identifiés et affichés
|
- Les jours fériés sont identifiés et affichés
|
||||||
@@ -145,6 +186,11 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
- Modification employé:
|
- Modification employé:
|
||||||
- uniquement prénom, nom, site
|
- uniquement prénom, nom, site
|
||||||
- pas de modification de contrat depuis ce drawer
|
- pas de modification de contrat depuis ce drawer
|
||||||
|
- Liste employés — filtre par statut de contrat:
|
||||||
|
- 3 options: "Avec contrat" (défaut), "Sans contrat", "Tous"
|
||||||
|
- "Avec contrat": employés ayant une période de contrat active à la date du jour
|
||||||
|
- "Sans contrat": employés sans période de contrat active
|
||||||
|
- "Tous": aucun filtrage sur le contrat
|
||||||
- Détail employé:
|
- Détail employé:
|
||||||
- onglet `Suivi contrat` avec affichage de l'historique des périodes de contrat
|
- onglet `Suivi contrat` avec affichage de l'historique des périodes de contrat
|
||||||
- chaque ligne expose: nature (`CDI`/`CDD`/`INTERIM`), contrat/temps de travail, date de début, date de fin (ou "En cours")
|
- chaque ligne expose: nature (`CDI`/`CDD`/`INTERIM`), contrat/temps de travail, date de début, date de fin (ou "En cours")
|
||||||
@@ -170,14 +216,22 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
- en cours d'acquisition jours: `25/12 = 2,08` jours/mois
|
- en cours d'acquisition jours: `25/12 = 2,08` jours/mois
|
||||||
- en cours d'acquisition samedis: `5/12 = 0,42` samedi/mois (non detaille en UI)
|
- en cours d'acquisition samedis: `5/12 = 0,42` samedi/mois (non detaille en UI)
|
||||||
- en cas de début/fin en cours de mois, l'acquisition est proratisée au nombre de jours calendaires couverts dans le mois
|
- en cas de début/fin en cours de mois, l'acquisition est proratisée au nombre de jours calendaires couverts dans le mois
|
||||||
|
- en cas de suspension en cours de mois, l'acquisition est proratisée en jours ouvrés (lun-ven hors fériés) travaillés / 22 (standard mensuel)
|
||||||
|
- arrêt maladie long (absences continues de type `M` > 1 mois):
|
||||||
|
- premier mois de maladie (date début + 1 mois calendaire): acquisition normale (`2,50`/mois)
|
||||||
|
- après le premier mois: acquisition réduite à `2,00`/mois (facteur `0,80` appliqué aux deux taux jours et samedis)
|
||||||
|
- en cas de mois partiellement couvert par la période réduite, le prorata est calculé en jours calendaires (jours normaux × taux normal + jours réduits × taux réduit)
|
||||||
|
- la détection est automatique à partir des absences MALADIE consécutives en base (tolérance de gap ≤ 3 jours)
|
||||||
- samedis acquis affiches: uniquement `opening_saturdays` (report N-1)
|
- samedis acquis affiches: uniquement `opening_saturdays` (report N-1)
|
||||||
- contrat `4h`:
|
- contrat `4h`:
|
||||||
- acquis annuel CP: `10`
|
- acquis annuel CP: `10`
|
||||||
- acquis annuel samedi: `0`
|
- acquis annuel samedi: `0`
|
||||||
- en cours d'acquisition: `0.83` jour/mois
|
- en cours d'acquisition: `0.83` jour/mois
|
||||||
- en cas de début/fin en cours de mois, l'acquisition est proratisée au nombre de jours calendaires couverts dans le mois
|
- en cas de début/fin en cours de mois, l'acquisition est proratisée au nombre de jours calendaires couverts dans le mois
|
||||||
|
- en cas de suspension en cours de mois, l'acquisition est proratisée en jours ouvrés (lun-ven hors fériés) travaillés / 22
|
||||||
- contrat `FORFAIT`:
|
- contrat `FORFAIT`:
|
||||||
- base annuelle: `jours ouvrés de l'exercice (lundi-vendredi, hors jours fériés métropole) - 218`
|
- base annuelle: `jours ouvrés de l'exercice (lundi-vendredi, hors jours fériés métropole) - 218`
|
||||||
|
- bonus weekend/férié: chaque jour travaillé un weekend ou jour férié donne 1 jour de congé supplémentaire (journée ≥ 5h = 1.0 jour, demi-journée > 0h et < 5h = 0.5 jour), sans plafond
|
||||||
- prorata: en cas de démarrage/fin de contrat en cours d'année civile, le calcul ne couvre que l'intervalle actif du contrat dans l'année
|
- prorata: en cas de démarrage/fin de contrat en cours d'année civile, le calcul ne couvre que l'intervalle actif du contrat dans l'année
|
||||||
- reste à prendre: `acquis - absences` (toutes absences, demi-journées incluses)
|
- reste à prendre: `acquis - absences` (toutes absences, demi-journées incluses)
|
||||||
- pas de samedi (`0`)
|
- pas de samedi (`0`)
|
||||||
@@ -225,6 +279,7 @@ Tous les filtres checkbox sont cochés par défaut à l'ouverture du drawer.
|
|||||||
- base identique aux calculs d'heures supplémentaires de la vue semaine Heures
|
- base identique aux calculs d'heures supplémentaires de la vue semaine Heures
|
||||||
- minutes de récupération hebdomadaires = `HS totales + bonus 25% + bonus 50%`
|
- minutes de récupération hebdomadaires = `HS totales + bonus 25% + bonus 50%`
|
||||||
- contrats `INTERIM` et suivi `PRESENCE`: récupération à `0`
|
- contrats `INTERIM` et suivi `PRESENCE`: récupération à `0`
|
||||||
|
- date limite de calcul: uniquement les semaines terminées (jusqu'au dernier dimanche), **ou** la semaine en cours si tous les jours existants sont validés RH (`isValid = true`). En cas de fin de contrat en milieu de semaine, seuls les jours jusqu'à la date de fin sont vérifiés.
|
||||||
- compteur global:
|
- compteur global:
|
||||||
- affiché en **jours** (1 jour = 7h = 420 minutes)
|
- affiché en **jours** (1 jour = 7h = 420 minutes)
|
||||||
- report:
|
- report:
|
||||||
@@ -242,10 +297,81 @@ 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) Notifications
|
## 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`)
|
||||||
|
- Sélecteur de mois (défaut = mois courant), génère un PDF A3 paysage
|
||||||
|
- Endpoint: `GET /api/salary-recap/print?month=YYYY-MM`
|
||||||
|
- Données groupées par site, un en-tête par site
|
||||||
|
|
||||||
|
### Colonnes du tableau
|
||||||
|
|
||||||
|
| Colonne | Source | Logique |
|
||||||
|
|---------|--------|---------|
|
||||||
|
| Nom | Employee | firstName + lastName |
|
||||||
|
| Base | Contract.name | Via EmployeeContractResolver pour le mois |
|
||||||
|
| 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 |
|
||||||
|
| 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 |
|
||||||
|
| Congés - Nombre | Absence code 'C' | Jours (demi-journées = 0.5) |
|
||||||
|
| Congés - Date | Absence code 'C' | Dates formatées dd/mm |
|
||||||
|
| Maladie - Nombre | Absence code 'M' ou 'AT' | Jours (demi-journées = 0.5) |
|
||||||
|
| Maladie - Date | Absence code 'M' ou 'AT' | Dates formatées dd/mm |
|
||||||
|
| CHAUFFEUR - PDJ | WorkHour.hasBreakfast | Comptage mois (chauffeurs uniquement) |
|
||||||
|
| CHAUFFEUR - REPAS | WorkHour.hasLunch + hasDinner | Comptage mois (chauffeurs uniquement) |
|
||||||
|
| CHAUFFEUR - NUITEE | WorkHour.hasOvernight | Comptage mois (chauffeurs uniquement) |
|
||||||
|
| CHAUFFEUR - samedi | WorkHour (samedi) | Samedis travaillés (chauffeurs uniquement) |
|
||||||
|
| Observations | — | Colonne vide pour saisie manuelle |
|
||||||
|
|
||||||
|
## 12) Frais
|
||||||
|
|
||||||
|
- Onglet "Frais" sur la fiche employé (icône `mdi:account-cash-outline`)
|
||||||
|
- Entité `MileageAllowance` (table `mileage_allowances`)
|
||||||
|
- Champs:
|
||||||
|
- `month` (mois, obligatoire)
|
||||||
|
- `kilometers` (nombre de km, optionnel)
|
||||||
|
- `amount` (montant en €, optionnel)
|
||||||
|
- `comment` (commentaire, optionnel)
|
||||||
|
- `receiptPath` / `receiptName` (justificatif Km, PDF)
|
||||||
|
- `amountReceiptPath` / `amountReceiptName` (justificatif Montant, PDF)
|
||||||
|
- Règle de validation:
|
||||||
|
- le mois est obligatoire
|
||||||
|
- au moins un des deux champs `kilometers` ou `amount` doit être > 0
|
||||||
|
- les deux peuvent être remplis simultanément
|
||||||
|
- Tableau: colonnes Mois, Nombre de Km, Montant €, Commentaire, Justif. Km, Justif. Montant
|
||||||
|
- Deux justificatifs distincts (upload PDF uniquement):
|
||||||
|
- Justificatif Km : upload via `/mileage_allowances/{id}/receipt`, téléchargement via GET même URL
|
||||||
|
- Justificatif Montant : upload via `/mileage_allowances/{id}/amount-receipt`, téléchargement via GET même URL
|
||||||
|
- La suppression d'un frais supprime les deux fichiers justificatifs du disque
|
||||||
|
|
||||||
|
## 13) Notifications
|
||||||
|
|
||||||
- Icône cloche en topbar:
|
- Icône cloche en topbar:
|
||||||
- badge = nombre de notifications non lues
|
- badge = nombre de notifications non lues
|
||||||
|
|||||||
87
frontend/components/SalaryRecapDrawer.vue
Normal file
87
frontend/components/SalaryRecapDrawer.vue
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
<template>
|
||||||
|
<AppDrawer v-model="drawerOpen" title="Récapitulatif Salaire">
|
||||||
|
<form class="space-y-4" @submit.prevent="handleSubmit">
|
||||||
|
<div>
|
||||||
|
<label class="text-md font-semibold text-neutral-700" for="salary-recap-month">
|
||||||
|
Mois <span class="text-red-600">*</span>
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
id="salary-recap-month"
|
||||||
|
v-model="selectedMonth"
|
||||||
|
type="month"
|
||||||
|
:class="monthFieldClass"
|
||||||
|
/>
|
||||||
|
<p v-if="showMonthError" class="mt-1 text-sm text-red-600">
|
||||||
|
Le mois est obligatoire.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex justify-center pt-2">
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
class="flex w-[200px] items-center justify-center gap-2 rounded-md bg-primary-500 px-4 py-2 text-md font-semibold text-white hover:bg-secondary-500"
|
||||||
|
:class="submitButtonClass"
|
||||||
|
>
|
||||||
|
Imprimer
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</AppDrawer>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, ref, watch } from 'vue'
|
||||||
|
import AppDrawer from '~/components/AppDrawer.vue'
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
modelValue: boolean
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(event: 'update:modelValue', value: boolean): void
|
||||||
|
(event: 'submit', month: string): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const drawerOpen = computed({
|
||||||
|
get: () => props.modelValue,
|
||||||
|
set: (value: boolean) => emit('update:modelValue', value)
|
||||||
|
})
|
||||||
|
|
||||||
|
const now = new Date()
|
||||||
|
const defaultMonth = `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, '0')}`
|
||||||
|
const selectedMonth = ref(defaultMonth)
|
||||||
|
const validationTouched = ref(false)
|
||||||
|
|
||||||
|
const isMonthValid = computed(() => selectedMonth.value.trim() !== '')
|
||||||
|
const showMonthError = computed(() => validationTouched.value && !isMonthValid.value)
|
||||||
|
|
||||||
|
const baseInputClass = 'mt-2 w-full rounded-md border px-3 py-2 text-md text-neutral-900'
|
||||||
|
const monthFieldClass = computed(() => {
|
||||||
|
if (showMonthError.value) {
|
||||||
|
return `${baseInputClass} border-red-500`
|
||||||
|
}
|
||||||
|
return `${baseInputClass} border-neutral-300`
|
||||||
|
})
|
||||||
|
|
||||||
|
const submitButtonClass = computed(() => {
|
||||||
|
if (!isMonthValid.value) {
|
||||||
|
return 'opacity-50 cursor-not-allowed'
|
||||||
|
}
|
||||||
|
return ''
|
||||||
|
})
|
||||||
|
|
||||||
|
const handleSubmit = () => {
|
||||||
|
validationTouched.value = true
|
||||||
|
if (!isMonthValid.value) return
|
||||||
|
emit('submit', selectedMonth.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.modelValue,
|
||||||
|
(isOpen) => {
|
||||||
|
if (!isOpen) {
|
||||||
|
validationTouched.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
</script>
|
||||||
241
frontend/components/driver-hours/DriverHoursDayView.vue
Normal file
241
frontend/components/driver-hours/DriverHoursDayView.vue
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
<template>
|
||||||
|
<div class="bg-white overflow-hidden flex min-h-0 flex-col">
|
||||||
|
<div class="overflow-y-auto min-h-0">
|
||||||
|
<div
|
||||||
|
class="grid w-full min-w-0 gap-1 border border-black bg-tertiary-500 px-4 py-3 text-sm font-semibold text-black rounded-t-md sticky top-0 z-10"
|
||||||
|
:style="{ gridTemplateColumns: dayGridCols }"
|
||||||
|
>
|
||||||
|
<span>Nom</span>
|
||||||
|
<span class="pl-2">Absence</span>
|
||||||
|
<span class="pl-4">Heure de jour</span>
|
||||||
|
<span class="pl-2">Heure de nuit</span>
|
||||||
|
<span class="pl-2">Heure atelier</span>
|
||||||
|
<span class="pl-2">Total</span>
|
||||||
|
<span>Petit déj.</span>
|
||||||
|
<span>Déjeuner</span>
|
||||||
|
<span>Dîner</span>
|
||||||
|
<span>Nuitée</span>
|
||||||
|
<span v-if="isAdmin" class="flex justify-between items-center">
|
||||||
|
<span>Valider</span>
|
||||||
|
<input
|
||||||
|
ref="bulkValidationInput"
|
||||||
|
:checked="isBulkValidationChecked"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4 cursor-pointer"
|
||||||
|
@change="onBulkValidationChange"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
<span v-else-if="isSiteManager" class="inline-flex items-center gap-2">
|
||||||
|
<span>Site</span>
|
||||||
|
<input
|
||||||
|
ref="bulkSiteValidationInput"
|
||||||
|
:checked="isBulkSiteValidationChecked"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4"
|
||||||
|
:class="canBulkToggleSiteValidation ? 'cursor-pointer' : 'cursor-not-allowed opacity-50'"
|
||||||
|
:disabled="!canBulkToggleSiteValidation"
|
||||||
|
@change="onBulkSiteValidationChange"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
<span v-else>Site <Icon name="mdi:check-bold" class="ml-1"/></span>
|
||||||
|
<span v-if="!isAdmin">RH <Icon name="mdi:check-bold" class="ml-1"/></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="border-x border-b border-primary-500 rounded-b-md">
|
||||||
|
<div
|
||||||
|
v-for="employee in employees"
|
||||||
|
:key="employee.id"
|
||||||
|
class="grid w-full min-w-0 items-center gap-1 border-b border-primary-500 px-4 py-2 text-sm font-bold text-primary-500 last:border-b-0 hover:bg-tertiary-500"
|
||||||
|
:style="{ gridTemplateColumns: dayGridCols }"
|
||||||
|
>
|
||||||
|
<div class="text-neutral-900 min-w-0">
|
||||||
|
<p class="font-semibold truncate">
|
||||||
|
{{ employee.firstName }} {{ employee.lastName }}
|
||||||
|
<span class="font-normal text-neutral-600">({{ contractLabel(employee) }})</span>
|
||||||
|
</p>
|
||||||
|
<p class="text-neutral-500 truncate inline-flex items-center gap-2">
|
||||||
|
<span>{{ employee.site?.name ?? 'Sans site' }}</span>
|
||||||
|
<span
|
||||||
|
v-if="isAdmin && (rows[employee.id]?.isSiteValid ?? false)"
|
||||||
|
class="rounded-full bg-green-500 flex justify-center item-center text-white p-0.5"
|
||||||
|
title="Validation site"
|
||||||
|
>
|
||||||
|
<Icon name="mdi:check"/>
|
||||||
|
</span>
|
||||||
|
</p>
|
||||||
|
<p v-if="isAdmin && getRowUpdatedAt(employee.id)" class="text-neutral-400 text-xs truncate">
|
||||||
|
Modifié le {{ getRowUpdatedAt(employee.id) }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="pl-2 min-w-0 self-stretch flex flex-col gap-1 justify-between py-0.5">
|
||||||
|
<p
|
||||||
|
class="w-full min-w-0 rounded-md px-2 py-1 text-xs truncate"
|
||||||
|
:class="getRowAbsenceLabel(employee.id) ? 'text-white' : 'invisible'"
|
||||||
|
:title="getRowAbsenceLabel(employee.id) || ''"
|
||||||
|
:style="getRowAbsenceStyle(employee.id)"
|
||||||
|
>
|
||||||
|
{{ getRowAbsenceLabel(employee.id) || '—' }}
|
||||||
|
</p>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="self-start text-left text-xs font-semibold underline"
|
||||||
|
:class="isHoliday || isRowLocked(employee.id) || !hasContractAtSelectedDate(employee.id) ? 'text-neutral-400 cursor-not-allowed' : 'text-primary-500 cursor-pointer'"
|
||||||
|
:disabled="isHoliday || isRowLocked(employee.id) || !hasContractAtSelectedDate(employee.id)"
|
||||||
|
@click="onAbsenceClick(employee.id)"
|
||||||
|
>
|
||||||
|
Modifier
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="pl-4">
|
||||||
|
<TimeSelect
|
||||||
|
v-model="rows[employee.id].dayHours"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="pl-2">
|
||||||
|
<TimeSelect
|
||||||
|
v-model="rows[employee.id].nightHours"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="pl-2">
|
||||||
|
<TimeSelect
|
||||||
|
v-model="rows[employee.id].workshopHours"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="pl-2 text-sm font-semibold">
|
||||||
|
{{ formatMinutes(getRowMetrics(employee.id).totalMinutes) }}
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<input
|
||||||
|
v-model="rows[employee.id].hasBreakfast"
|
||||||
|
type="checkbox"
|
||||||
|
class="cursor-pointer h-4 w-4"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<input
|
||||||
|
v-model="rows[employee.id].hasLunch"
|
||||||
|
type="checkbox"
|
||||||
|
class="cursor-pointer h-4 w-4"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<input
|
||||||
|
v-model="rows[employee.id].hasDinner"
|
||||||
|
type="checkbox"
|
||||||
|
class="cursor-pointer h-4 w-4"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<input
|
||||||
|
v-model="rows[employee.id].hasOvernight"
|
||||||
|
type="checkbox"
|
||||||
|
class="cursor-pointer h-4 w-4"
|
||||||
|
:disabled="!hasContractAtSelectedDate(employee.id) || isRowLocked(employee.id)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div v-if="isAdmin" class="text-right">
|
||||||
|
<input
|
||||||
|
:checked="rows[employee.id]?.isValid ?? false"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4 cursor-pointer"
|
||||||
|
@change="onToggleValidation(employee.id, ($event.target as HTMLInputElement).checked)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div v-else class="text-right p-5">
|
||||||
|
<input
|
||||||
|
v-if="isSiteManager"
|
||||||
|
:checked="rows[employee.id]?.isSiteValid ?? false"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4 cursor-pointer"
|
||||||
|
:disabled="(!canToggleSiteValidation(employee.id) && !canCreateSiteValidationRowFromAbsence(employee.id)) || isSiteValidationPending(employee.id)"
|
||||||
|
@change="onToggleSiteValidation(employee.id, ($event.target as HTMLInputElement).checked)"
|
||||||
|
/>
|
||||||
|
<span v-else-if="rows[employee.id]?.isSiteValid" class="text-xs font-semibold text-neutral-700">Validé</span>
|
||||||
|
<span v-else class="text-xs text-neutral-500">-</span>
|
||||||
|
</div>
|
||||||
|
<div v-if="!isAdmin">
|
||||||
|
<span v-if="rows[employee.id]?.isValid" class="text-xs font-semibold text-neutral-700">Validé</span>
|
||||||
|
<span v-else class="text-xs text-neutral-500">-</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import type { Employee } from '~/services/dto/employee'
|
||||||
|
import TimeSelect from '~/components/ui/TimeSelect.vue'
|
||||||
|
import type { DriverHourRow } from '~/services/dto/work-hour'
|
||||||
|
|
||||||
|
const rows = defineModel<Record<number, DriverHourRow>>('rows', { required: true })
|
||||||
|
const bulkValidationInput = ref<HTMLInputElement | null>(null)
|
||||||
|
const bulkSiteValidationInput = ref<HTMLInputElement | null>(null)
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
employees: Employee[]
|
||||||
|
isAdmin: boolean
|
||||||
|
isSiteManager: boolean
|
||||||
|
dayGridCols: string
|
||||||
|
isHoliday: boolean
|
||||||
|
contractLabel: (employee: Employee) => string
|
||||||
|
isRowLocked: (employeeId: number) => boolean
|
||||||
|
hasContractAtSelectedDate: (employeeId: number) => boolean
|
||||||
|
isValidationPending: (employeeId: number) => boolean
|
||||||
|
isSiteValidationPending: (employeeId: number) => boolean
|
||||||
|
canToggleValidation: (employeeId: number) => boolean
|
||||||
|
canToggleSiteValidation: (employeeId: number) => boolean
|
||||||
|
canCreateSiteValidationRowFromAbsence: (employeeId: number) => boolean
|
||||||
|
isBulkValidationChecked: boolean
|
||||||
|
isBulkValidationIndeterminate: boolean
|
||||||
|
isBulkSiteValidationChecked: boolean
|
||||||
|
isBulkSiteValidationIndeterminate: boolean
|
||||||
|
canBulkToggleSiteValidation: boolean
|
||||||
|
onToggleValidation: (employeeId: number, checked: boolean) => void
|
||||||
|
onToggleSiteValidation: (employeeId: number, checked: boolean) => void
|
||||||
|
onToggleValidationBulk: (checked: boolean) => Promise<void> | void
|
||||||
|
onToggleSiteValidationBulk: (checked: boolean) => Promise<void> | void
|
||||||
|
getRowMetrics: (employeeId: number) => { dayMinutes: number; nightMinutes: number; totalMinutes: number }
|
||||||
|
getRowAbsenceLabel: (employeeId: number) => string
|
||||||
|
getRowAbsenceStyle: (employeeId: number) => { backgroundColor: string } | undefined
|
||||||
|
getRowUpdatedAt: (employeeId: number) => string
|
||||||
|
onAbsenceClick: (employeeId: number) => void
|
||||||
|
formatMinutes: (minutes: number) => string
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const onBulkValidationChange = (event: Event) => {
|
||||||
|
props.onToggleValidationBulk((event.target as HTMLInputElement).checked)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onBulkSiteValidationChange = (event: Event) => {
|
||||||
|
props.onToggleSiteValidationBulk((event.target as HTMLInputElement).checked)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onToggleSiteValidation = (employeeId: number, checked: boolean) => {
|
||||||
|
props.onToggleSiteValidation(employeeId, checked)
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.isBulkValidationIndeterminate,
|
||||||
|
(isIndeterminate) => {
|
||||||
|
if (!bulkValidationInput.value) return
|
||||||
|
bulkValidationInput.value.indeterminate = isIndeterminate
|
||||||
|
},
|
||||||
|
{ immediate: true }
|
||||||
|
)
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.isBulkSiteValidationIndeterminate,
|
||||||
|
(isIndeterminate) => {
|
||||||
|
if (!bulkSiteValidationInput.value) return
|
||||||
|
bulkSiteValidationInput.value.indeterminate = isIndeterminate
|
||||||
|
},
|
||||||
|
{ immediate: true }
|
||||||
|
)
|
||||||
|
</script>
|
||||||
108
frontend/components/driver-hours/DriverHoursWeekView.vue
Normal file
108
frontend/components/driver-hours/DriverHoursWeekView.vue
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
<template>
|
||||||
|
<div class="bg-white overflow-hidden flex min-h-0 flex-col">
|
||||||
|
<div v-if="isWeekLoading" class="p-6 text-md text-neutral-600">Chargement de la semaine...</div>
|
||||||
|
<div v-else class="overflow-y-auto min-h-0">
|
||||||
|
<div
|
||||||
|
class="grid w-full min-w-0 gap-1 border border-black bg-tertiary-500 px-4 py-3 text-sm font-semibold text-black rounded-t-md sticky top-0 z-10"
|
||||||
|
:style="{ gridTemplateColumns: weekGridCols }"
|
||||||
|
>
|
||||||
|
<span>Nom</span>
|
||||||
|
<span v-for="day in weekDayHeaders" :key="day.date" class="text-left">{{ day.weekday }}<br>{{ day.dayDate }}</span>
|
||||||
|
<span>Jour/Nuit <br>sem.</span>
|
||||||
|
<span>Atelier <br>sem.</span>
|
||||||
|
<span>Total <br>sem.</span>
|
||||||
|
<span>Total <br>h. supp.</span>
|
||||||
|
<span>+25%</span>
|
||||||
|
<span>+50%</span>
|
||||||
|
<span>Total <br>récup.</span>
|
||||||
|
<span>Petit <br>déj.</span>
|
||||||
|
<span>Déj.</span>
|
||||||
|
<span>Dîner</span>
|
||||||
|
<span>Nuit.</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="border-x border-b border-primary-500 rounded-b-md">
|
||||||
|
<div
|
||||||
|
v-for="row in weeklySummary?.rows ?? []"
|
||||||
|
:key="row.employeeId"
|
||||||
|
class="grid w-full min-w-0 items-center gap-1 border-b border-primary-500 px-4 py-2 text-sm font-bold text-primary-500 last:border-b-0 hover:bg-tertiary-500"
|
||||||
|
:style="{ gridTemplateColumns: weekGridCols }"
|
||||||
|
>
|
||||||
|
<div class="text-neutral-900 min-w-0">
|
||||||
|
<p class="font-semibold truncate">
|
||||||
|
{{ row.firstName }} {{ row.lastName }}
|
||||||
|
<span class="font-normal text-neutral-600">({{ row.contractName ?? '-' }})</span>
|
||||||
|
</p>
|
||||||
|
<p class="text-[11px] text-neutral-500 truncate">{{ row.siteName ?? 'Sans site' }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-for="daily in row.daily"
|
||||||
|
:key="daily.date"
|
||||||
|
class="text-left leading-4 rounded-md px-2 py-1"
|
||||||
|
:class="daily.hasAbsence ? 'text-white' : ''"
|
||||||
|
:style="getDailyCellStyle(daily)"
|
||||||
|
:title="daily.absenceLabel ?? ''"
|
||||||
|
>
|
||||||
|
<div>J {{ formatMinutes(daily.dayMinutes) }}</div>
|
||||||
|
<div>N {{ formatMinutes(daily.nightMinutes) }}</div>
|
||||||
|
<div v-if="daily.workshopMinutes">A {{ formatMinutes(daily.workshopMinutes) }}</div>
|
||||||
|
<div v-if="daily.hasBreakfast || daily.hasLunch || daily.hasDinner || daily.hasOvernight" class="text-[10px] flex gap-1 mt-0.5">
|
||||||
|
<span v-if="daily.hasBreakfast" title="Petit déjeuner">PD</span>
|
||||||
|
<span v-if="daily.hasLunch" title="Déjeuner">DJ</span>
|
||||||
|
<span v-if="daily.hasDinner" title="Dîner">DI</span>
|
||||||
|
<span v-if="daily.hasOvernight" title="Nuitée">NU</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="font-semibold leading-4">
|
||||||
|
<div>J {{ formatMinutes(row.weeklyDayMinutes) }}</div>
|
||||||
|
<div>N {{ formatMinutes(row.weeklyNightMinutes) }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyWorkshopMinutes ?? 0) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyTotalMinutes) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyOvertimeTotalMinutes ?? 0) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyOvertime25Minutes ?? 0) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyOvertime50Minutes ?? 0) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ formatMinutes(row.weeklyRecoveryMinutes ?? 0) }}
|
||||||
|
</div>
|
||||||
|
<div class="font-semibold">{{ row.weeklyBreakfastCount ?? 0 }}</div>
|
||||||
|
<div class="font-semibold">{{ row.weeklyLunchCount ?? 0 }}</div>
|
||||||
|
<div class="font-semibold">{{ row.weeklyDinnerCount ?? 0 }}</div>
|
||||||
|
<div class="font-semibold">{{ row.weeklyOvernightCount ?? 0 }}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import type { WeeklyWorkHourSummary } from '~/services/dto/work-hour'
|
||||||
|
|
||||||
|
const getDailyCellStyle = (daily: {
|
||||||
|
hasAbsence?: boolean
|
||||||
|
absenceColor?: string | null
|
||||||
|
}) => {
|
||||||
|
if (!daily.hasAbsence) return undefined
|
||||||
|
return { backgroundColor: daily.absenceColor || '#dc2626' }
|
||||||
|
}
|
||||||
|
|
||||||
|
defineProps<{
|
||||||
|
isWeekLoading: boolean
|
||||||
|
weekGridCols: string
|
||||||
|
weeklySummary: WeeklyWorkHourSummary | null
|
||||||
|
weekDayHeaders: Array<{ date: string; weekday: string; dayDate: string }>
|
||||||
|
formatMinutes: (minutes: number) => string
|
||||||
|
}>()
|
||||||
|
</script>
|
||||||
@@ -240,6 +240,18 @@
|
|||||||
<input id="create-contract-end-date" v-model="createContractForm.endDate" type="date" :class="createContractEndDateFieldClass" />
|
<input id="create-contract-end-date" v-model="createContractForm.endDate" type="date" :class="createContractEndDateFieldClass" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-md border border-neutral-200 bg-neutral-50 p-3">
|
||||||
|
<label class="inline-flex items-center gap-2 text-md font-semibold text-neutral-700" for="create-contract-is-driver">
|
||||||
|
<input
|
||||||
|
id="create-contract-is-driver"
|
||||||
|
v-model="createContractForm.isDriver"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4 rounded border-neutral-300 text-primary-500 focus:ring-primary-500"
|
||||||
|
/>
|
||||||
|
Chauffeur
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="flex justify-center pt-2">
|
<div class="flex justify-center pt-2">
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
@@ -281,6 +293,7 @@ type CreateContractForm = {
|
|||||||
contractNature: 'CDI' | 'CDD' | 'INTERIM'
|
contractNature: 'CDI' | 'CDD' | 'INTERIM'
|
||||||
startDate: string
|
startDate: string
|
||||||
endDate: string
|
endDate: string
|
||||||
|
isDriver: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
|
|||||||
@@ -1,35 +1,35 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="flex h-full min-h-0 flex-col overflow-hidden pt-8">
|
<section class="flex h-full min-h-0 flex-col overflow-hidden pt-8">
|
||||||
<div class="grid grid-cols-4 rounded-md bg-tertiary-500 text-primary-500 text-[18px] border border-primary-500">
|
<div class="grid grid-cols-4 rounded-md bg-tertiary-500 text-primary-500 text-[18px] border border-primary-500">
|
||||||
<p class="col-start-1 p-[10px] border-b border-b-black"><strong class="uppercase font-semibold">Année acquis :</strong> {{
|
<p class="col-start-1 p-[10px] border-b border-r border-primary-500"><strong class="uppercase font-semibold">Année acquis :</strong> {{
|
||||||
formatCount(summary?.acquiredDays)
|
formatCount(summary?.acquiredDays)
|
||||||
}} Jours
|
}} Jours
|
||||||
</p>
|
</p>
|
||||||
<p class="col-start-2 p-[10px] border-b border-b-black"><strong class="uppercase font-semibold">Pris :</strong>
|
<p class="col-start-2 p-[10px] border-b border-r border-primary-500"><strong class="uppercase font-semibold">Pris :</strong>
|
||||||
{{ formatCount(isForfaitRule ? currentYearTakenDays : summary?.takenDays) }} Jours
|
{{ formatCount(isForfaitRule ? currentYearTakenDays : summary?.takenDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p class="col-start-3 p-[10px] border-b border-b-black"><strong class="uppercase font-semibold">Reste à prendre :</strong>
|
<p class="col-start-3 p-[10px] border-b border-r border-b-white border-r-primary-500 bg-primary-500 text-white"><strong class="uppercase font-semibold">Reste à prendre :</strong>
|
||||||
{{ formatCount(summary?.remainingDays) }} Jours
|
{{ formatCount(summary?.remainingDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p class="col-start-4 p-[10px] border-b border-b-black"><strong class="uppercase font-semibold">En cours d'acquisition :</strong>
|
<p class="col-start-4 p-[10px] border-b border-primary-500"><strong class="uppercase font-semibold">En cours d'acquisition :</strong>
|
||||||
{{ formatCount(summary?.accruingDays) }} Jours
|
{{ formatCount(summary?.accruingDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-if="!isForfaitRule" class="col-start-1 p-[10px]"><span class="uppercase font-semibold">Samedi acquis :</span>
|
<p v-if="!isForfaitRule" class="col-start-1 p-[10px] border-r border-primary-500"><span class="uppercase font-semibold">Samedi acquis :</span>
|
||||||
{{ formatCount(summary?.acquiredSaturdays) }} Jours
|
{{ formatCount(summary?.acquiredSaturdays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-else class="col-start-1 p-[10px]"><span class="uppercase font-semibold">Année N-1 acquis :</span>
|
<p v-else class="col-start-1 p-[10px] border-r border-primary-500"><span class="uppercase font-semibold">Année N-1 acquis :</span>
|
||||||
{{ formatCount(summary?.previousYearAcquiredDays) }} Jours
|
{{ formatCount(summary?.previousYearAcquiredDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-if="!isForfaitRule" class="col-start-2 p-[10px]"><span class="uppercase font-semibold">Pris :</span>
|
<p v-if="!isForfaitRule" class="col-start-2 p-[10px] border-r border-primary-500"><span class="uppercase font-semibold">Pris :</span>
|
||||||
{{ formatCount(summary?.takenSaturdays) }} Jours
|
{{ formatCount(summary?.takenSaturdays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-if="!isForfaitRule" class="col-start-3 p-[10px]"><span class="uppercase font-semibold">Reste à prendre :</span>
|
<p v-if="!isForfaitRule" class="col-start-3 p-[10px] border-r border-r-primary-500 bg-primary-500 text-white"><span class="uppercase font-semibold">Reste à prendre :</span>
|
||||||
{{ formatCount(summary?.remainingSaturdays) }} Jours
|
{{ formatCount(summary?.remainingSaturdays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-else class="col-start-2 p-[10px]"><span class="uppercase font-semibold">Pris :</span>
|
<p v-else class="col-start-2 p-[10px] border-r border-primary-500"><span class="uppercase font-semibold">Pris :</span>
|
||||||
{{ formatCount(summary?.previousYearTakenDays) }} Jours
|
{{ formatCount(summary?.previousYearTakenDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<p v-if="isForfaitRule" class="col-start-3 p-[10px]"><span class="uppercase font-semibold">Reste à prendre :</span>
|
<p v-if="isForfaitRule" class="col-start-3 p-[10px] border-r-primary-500 bg-primary-500 text-white"><span class="uppercase font-semibold">Reste à prendre :</span>
|
||||||
{{ formatCount(summary?.previousYearRemainingDays) }} Jours
|
{{ formatCount(summary?.previousYearRemainingDays) }} Jours
|
||||||
</p>
|
</p>
|
||||||
<div v-if="!isForfaitRule" class="col-start-4 p-[10px] flex gap-7 items-center">
|
<div v-if="!isForfaitRule" class="col-start-4 p-[10px] flex gap-7 items-center">
|
||||||
|
|||||||
@@ -2,11 +2,13 @@
|
|||||||
<section class="mt-8">
|
<section class="mt-8">
|
||||||
<div class="overflow-hidden bg-white">
|
<div class="overflow-hidden bg-white">
|
||||||
<div
|
<div
|
||||||
class="grid grid-cols-4 border border-black bg-tertiary-500 px-6 py-3 text-[20px] font-semibold text-black rounded-t-md">
|
class="grid grid-cols-6 border border-black bg-tertiary-500 px-6 py-3 text-[20px] font-semibold text-black rounded-t-md">
|
||||||
<p>Mois</p>
|
<p>Mois</p>
|
||||||
<p>Nombre de Km</p>
|
<p>Nombre de Km</p>
|
||||||
|
<p>Montant €</p>
|
||||||
<p>Commentaire</p>
|
<p>Commentaire</p>
|
||||||
<p>Justificatif</p>
|
<p>Justif. Km</p>
|
||||||
|
<p>Justif. Montant</p>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="allowances.length === 0" class="px-6 py-4 text-[20px] font-bold text-primary-500 border-x border-b border-primary-500 rounded-b-md">
|
<div v-if="allowances.length === 0" class="px-6 py-4 text-[20px] font-bold text-primary-500 border-x border-b border-primary-500 rounded-b-md">
|
||||||
Aucun frais kilométrique.
|
Aucun frais kilométrique.
|
||||||
@@ -15,22 +17,36 @@
|
|||||||
<div
|
<div
|
||||||
v-for="item in allowances"
|
v-for="item in allowances"
|
||||||
:key="item.id"
|
:key="item.id"
|
||||||
class="grid grid-cols-4 border-b border-primary-500 px-6 py-3 text-md font-bold text-primary-500 last:border-b-0 cursor-pointer hover:bg-tertiary-500"
|
class="grid grid-cols-6 border-b border-primary-500 px-6 py-3 text-md font-bold text-primary-500 last:border-b-0 cursor-pointer hover:bg-tertiary-500"
|
||||||
@click="onOpenEditDrawer(item)"
|
@click="onOpenEditDrawer(item)"
|
||||||
>
|
>
|
||||||
<p>{{ formatMonth(item.month) }}</p>
|
<p>{{ formatMonth(item.month) }}</p>
|
||||||
<p>{{ item.kilometers }}</p>
|
<p>{{ item.kilometers }}</p>
|
||||||
|
<p>{{ item.amount ? item.amount + ' €' : '-' }}</p>
|
||||||
<p>{{ item.comment ?? '-' }}</p>
|
<p>{{ item.comment ?? '-' }}</p>
|
||||||
<p>
|
<p class="min-w-0">
|
||||||
<a
|
<a
|
||||||
v-if="item.receiptPath"
|
v-if="item.receiptPath"
|
||||||
:href="getReceiptUrl(props.apiBase, item.id)"
|
:href="getKmReceiptUrl(props.apiBase, item.id)"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
class="text-primary-500 hover:text-secondary-500 flex gap-2 items-center"
|
class="text-primary-500 hover:text-secondary-500 flex gap-2 items-center"
|
||||||
@click.stop
|
@click.stop
|
||||||
>
|
>
|
||||||
<Icon name="mdi:file-download-outline" size="20"/>
|
<Icon name="mdi:file-download-outline" size="20" class="shrink-0"/>
|
||||||
<span>{{ item.receiptName ?? 'Télécharger' }}</span>
|
<span class="truncate">{{ item.receiptName ?? 'Télécharger' }}</span>
|
||||||
|
</a>
|
||||||
|
<span v-else>-</span>
|
||||||
|
</p>
|
||||||
|
<p class="min-w-0">
|
||||||
|
<a
|
||||||
|
v-if="item.amountReceiptPath"
|
||||||
|
:href="getAmountReceiptUrl(props.apiBase, item.id)"
|
||||||
|
target="_blank"
|
||||||
|
class="text-primary-500 hover:text-secondary-500 flex gap-2 items-center"
|
||||||
|
@click.stop
|
||||||
|
>
|
||||||
|
<Icon name="mdi:file-download-outline" size="20" class="shrink-0"/>
|
||||||
|
<span class="truncate">{{ item.amountReceiptName ?? 'Télécharger' }}</span>
|
||||||
</a>
|
</a>
|
||||||
<span v-else>-</span>
|
<span v-else>-</span>
|
||||||
</p>
|
</p>
|
||||||
@@ -48,7 +64,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
<AppDrawer v-model="isDrawerOpen" title="Frais Kms">
|
<AppDrawer v-model="isDrawerOpen" title="Frais">
|
||||||
<form class="space-y-4" @submit.prevent="onSubmit">
|
<form class="space-y-4" @submit.prevent="onSubmit">
|
||||||
<div>
|
<div>
|
||||||
<label class="text-md font-semibold text-neutral-700" for="mileage-month">
|
<label class="text-md font-semibold text-neutral-700" for="mileage-month">
|
||||||
@@ -64,7 +80,7 @@
|
|||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label class="text-md font-semibold text-neutral-700" for="mileage-kilometers">
|
<label class="text-md font-semibold text-neutral-700" for="mileage-kilometers">
|
||||||
Nombre de Km <span class="text-red-600">*</span>
|
Nombre de Km
|
||||||
</label>
|
</label>
|
||||||
<input
|
<input
|
||||||
id="mileage-kilometers"
|
id="mileage-kilometers"
|
||||||
@@ -77,20 +93,53 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label class="text-md font-semibold text-neutral-700" for="mileage-receipt">
|
<label class="text-md font-semibold text-neutral-700" for="mileage-amount">
|
||||||
Justificatif
|
Montant (€)
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
id="mileage-amount"
|
||||||
|
v-model.number="form.amount"
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
min="0"
|
||||||
|
class="mt-2 w-full rounded-md border border-neutral-300 px-3 py-2 text-base text-neutral-900 focus:border-primary-500 focus:outline-none focus:ring-2 focus:ring-secondary-500/20"
|
||||||
|
/>
|
||||||
|
<p class="mt-1 text-sm text-neutral-500">Au moins un des deux champs doit être rempli</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<label class="text-md font-semibold text-neutral-700" for="mileage-km-receipt">
|
||||||
|
Justificatif Km
|
||||||
</label>
|
</label>
|
||||||
<div v-if="isEditing && editingItem?.receiptName" class="mt-1 text-sm text-neutral-500">
|
<div v-if="isEditing && editingItem?.receiptName" class="mt-1 text-sm text-neutral-500">
|
||||||
Fichier actuel : {{ editingItem.receiptName }}
|
Fichier actuel : {{ editingItem.receiptName }}
|
||||||
</div>
|
</div>
|
||||||
<input
|
<input
|
||||||
id="mileage-receipt"
|
id="mileage-km-receipt"
|
||||||
ref="fileInput"
|
ref="kmFileInput"
|
||||||
type="file"
|
type="file"
|
||||||
class="mt-2 w-full rounded-md border border-neutral-300 px-3 py-2 text-base text-neutral-900 file:mr-3 file:rounded file:border-0 file:bg-primary-500 file:px-3 file:py-1 file:text-sm file:text-white"
|
class="mt-2 w-full rounded-md border border-neutral-300 px-3 py-2 text-base text-neutral-900 file:mr-3 file:rounded file:border-0 file:bg-primary-500 file:px-3 file:py-1 file:text-sm file:text-white"
|
||||||
@change="onFileChange"
|
@change="onKmFileChange"
|
||||||
/>
|
/>
|
||||||
<p v-if="fileError" class="mt-1 text-sm text-red-600">{{ fileError }}</p>
|
<p v-if="kmFileError" class="mt-1 text-sm text-red-600">{{ kmFileError }}</p>
|
||||||
|
<p v-else class="mt-1 text-sm text-neutral-500">Fichier au format pdf</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<label class="text-md font-semibold text-neutral-700" for="mileage-amount-receipt">
|
||||||
|
Justificatif Montant
|
||||||
|
</label>
|
||||||
|
<div v-if="isEditing && editingItem?.amountReceiptName" class="mt-1 text-sm text-neutral-500">
|
||||||
|
Fichier actuel : {{ editingItem.amountReceiptName }}
|
||||||
|
</div>
|
||||||
|
<input
|
||||||
|
id="mileage-amount-receipt"
|
||||||
|
ref="amountFileInput"
|
||||||
|
type="file"
|
||||||
|
class="mt-2 w-full rounded-md border border-neutral-300 px-3 py-2 text-base text-neutral-900 file:mr-3 file:rounded file:border-0 file:bg-primary-500 file:px-3 file:py-1 file:text-sm file:text-white"
|
||||||
|
@change="onAmountFileChange"
|
||||||
|
/>
|
||||||
|
<p v-if="amountFileError" class="mt-1 text-sm text-red-600">{{ amountFileError }}</p>
|
||||||
<p v-else class="mt-1 text-sm text-neutral-500">Fichier au format pdf</p>
|
<p v-else class="mt-1 text-sm text-neutral-500">Fichier au format pdf</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -139,7 +188,7 @@
|
|||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import type {MileageAllowance} from '~/services/dto/mileage-allowance'
|
import type {MileageAllowance} from '~/services/dto/mileage-allowance'
|
||||||
import {getReceiptUrl} from '~/services/mileage-allowances'
|
import {getKmReceiptUrl, getAmountReceiptUrl} from '~/services/mileage-allowances'
|
||||||
import AppDrawer from '~/components/AppDrawer.vue'
|
import AppDrawer from '~/components/AppDrawer.vue'
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
@@ -148,17 +197,20 @@ const props = defineProps<{
|
|||||||
}>()
|
}>()
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(event: 'create', data: { month: string; kilometers: number; comment?: string }, file?: File): void
|
(event: 'create', data: { month: string; kilometers: number; amount: number; comment?: string }, kmFile?: File, amountFile?: File): void
|
||||||
(event: 'update', id: number, data: { month: string; kilometers: number; comment?: string }, file?: File): void
|
(event: 'update', id: number, data: { month: string; kilometers: number; amount: number; comment?: string }, kmFile?: File, amountFile?: File): void
|
||||||
(event: 'delete', id: number): void
|
(event: 'delete', id: number): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const isDrawerOpen = ref(false)
|
const isDrawerOpen = ref(false)
|
||||||
const isEditing = ref(false)
|
const isEditing = ref(false)
|
||||||
const editingItem = ref<MileageAllowance | null>(null)
|
const editingItem = ref<MileageAllowance | null>(null)
|
||||||
const selectedFile = ref<File | undefined>(undefined)
|
const selectedKmFile = ref<File | undefined>(undefined)
|
||||||
const fileInput = ref<HTMLInputElement | null>(null)
|
const selectedAmountFile = ref<File | undefined>(undefined)
|
||||||
const fileError = ref('')
|
const kmFileInput = ref<HTMLInputElement | null>(null)
|
||||||
|
const amountFileInput = ref<HTMLInputElement | null>(null)
|
||||||
|
const kmFileError = ref('')
|
||||||
|
const amountFileError = ref('')
|
||||||
|
|
||||||
const currentYearMonth = () => {
|
const currentYearMonth = () => {
|
||||||
const now = new Date()
|
const now = new Date()
|
||||||
@@ -168,11 +220,12 @@ const currentYearMonth = () => {
|
|||||||
const form = reactive({
|
const form = reactive({
|
||||||
month: currentYearMonth(),
|
month: currentYearMonth(),
|
||||||
kilometers: 0,
|
kilometers: 0,
|
||||||
|
amount: 0,
|
||||||
comment: ''
|
comment: ''
|
||||||
})
|
})
|
||||||
|
|
||||||
const isFormValid = computed(() => {
|
const isFormValid = computed(() => {
|
||||||
return form.month && form.kilometers > 0 && !fileError.value
|
return form.month && (form.kilometers > 0 || form.amount > 0) && !kmFileError.value && !amountFileError.value
|
||||||
})
|
})
|
||||||
|
|
||||||
const monthLabels: Record<number, string> = {
|
const monthLabels: Record<number, string> = {
|
||||||
@@ -201,11 +254,17 @@ const formatMonth = (dateStr: string): string => {
|
|||||||
const resetForm = () => {
|
const resetForm = () => {
|
||||||
form.month = currentYearMonth()
|
form.month = currentYearMonth()
|
||||||
form.kilometers = 0
|
form.kilometers = 0
|
||||||
|
form.amount = 0
|
||||||
form.comment = ''
|
form.comment = ''
|
||||||
selectedFile.value = undefined
|
selectedKmFile.value = undefined
|
||||||
fileError.value = ''
|
selectedAmountFile.value = undefined
|
||||||
if (fileInput.value) {
|
kmFileError.value = ''
|
||||||
fileInput.value.value = ''
|
amountFileError.value = ''
|
||||||
|
if (kmFileInput.value) {
|
||||||
|
kmFileInput.value.value = ''
|
||||||
|
}
|
||||||
|
if (amountFileInput.value) {
|
||||||
|
amountFileInput.value.value = ''
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,38 +281,57 @@ const onOpenEditDrawer = (item: MileageAllowance) => {
|
|||||||
// Extract YYYY-MM from YYYY-MM-DD
|
// Extract YYYY-MM from YYYY-MM-DD
|
||||||
form.month = item.month.substring(0, 7)
|
form.month = item.month.substring(0, 7)
|
||||||
form.kilometers = item.kilometers
|
form.kilometers = item.kilometers
|
||||||
|
form.amount = item.amount
|
||||||
form.comment = item.comment ?? ''
|
form.comment = item.comment ?? ''
|
||||||
selectedFile.value = undefined
|
selectedKmFile.value = undefined
|
||||||
if (fileInput.value) {
|
selectedAmountFile.value = undefined
|
||||||
fileInput.value.value = ''
|
if (kmFileInput.value) {
|
||||||
|
kmFileInput.value.value = ''
|
||||||
|
}
|
||||||
|
if (amountFileInput.value) {
|
||||||
|
amountFileInput.value.value = ''
|
||||||
}
|
}
|
||||||
isDrawerOpen.value = true
|
isDrawerOpen.value = true
|
||||||
}
|
}
|
||||||
|
|
||||||
const onFileChange = (event: Event) => {
|
const onKmFileChange = (event: Event) => {
|
||||||
const target = event.target as HTMLInputElement
|
const target = event.target as HTMLInputElement
|
||||||
const file = target.files?.[0]
|
const file = target.files?.[0]
|
||||||
if (file && file.type !== 'application/pdf') {
|
if (file && file.type !== 'application/pdf') {
|
||||||
fileError.value = 'Seuls les fichiers PDF sont acceptés.'
|
kmFileError.value = 'Seuls les fichiers PDF sont acceptés.'
|
||||||
selectedFile.value = undefined
|
selectedKmFile.value = undefined
|
||||||
target.value = ''
|
target.value = ''
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fileError.value = ''
|
kmFileError.value = ''
|
||||||
selectedFile.value = file ?? undefined
|
selectedKmFile.value = file ?? undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
const onAmountFileChange = (event: Event) => {
|
||||||
|
const target = event.target as HTMLInputElement
|
||||||
|
const file = target.files?.[0]
|
||||||
|
if (file && file.type !== 'application/pdf') {
|
||||||
|
amountFileError.value = 'Seuls les fichiers PDF sont acceptés.'
|
||||||
|
selectedAmountFile.value = undefined
|
||||||
|
target.value = ''
|
||||||
|
return
|
||||||
|
}
|
||||||
|
amountFileError.value = ''
|
||||||
|
selectedAmountFile.value = file ?? undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
const onSubmit = () => {
|
const onSubmit = () => {
|
||||||
const data = {
|
const data = {
|
||||||
month: `${form.month}-01`,
|
month: `${form.month}-01`,
|
||||||
kilometers: form.kilometers,
|
kilometers: form.kilometers,
|
||||||
|
amount: form.amount,
|
||||||
comment: form.comment || undefined
|
comment: form.comment || undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isEditing.value && editingItem.value) {
|
if (isEditing.value && editingItem.value) {
|
||||||
emit('update', editingItem.value.id, data, selectedFile.value)
|
emit('update', editingItem.value.id, data, selectedKmFile.value, selectedAmountFile.value)
|
||||||
} else {
|
} else {
|
||||||
emit('create', data, selectedFile.value)
|
emit('create', data, selectedKmFile.value, selectedAmountFile.value)
|
||||||
}
|
}
|
||||||
isDrawerOpen.value = false
|
isDrawerOpen.value = false
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,8 +22,8 @@
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<p class="text-[16px]">
|
<p class="text-[16px]">
|
||||||
<span class="font-bold">RTT À LA DATE DU JOUR :</span>
|
<span class="font-bold">RTT À LA SEMAINE {{ lastCompleteWeek }} : </span>
|
||||||
{{ formatMinutes(summary?.availableMinutes ?? 0) }}
|
<span class="font-bold">{{ formatMinutes(summary?.availableMinutes ?? 0) }}</span>
|
||||||
</p>
|
</p>
|
||||||
<div class="flex justify-center">
|
<div class="flex justify-center">
|
||||||
<button
|
<button
|
||||||
@@ -40,34 +40,53 @@
|
|||||||
<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) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBase25Minutes) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBonus25Minutes) }}</span></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!.carryBase25Minutes + summary!.carryBonus25Minutes) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBase25Minutes + summary!.carryBonus25Minutes) }}</span></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!.carryBase50Minutes) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBase50Minutes) }}</span></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!.carryBonus50Minutes) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBonus50Minutes) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryBase50Minutes + summary!.carryBonus50Minutes) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(summary!.carryFromPreviousYearMinutes) }} <span class="text-neutral-400">/ {{ formatCentiemes(summary!.carryFromPreviousYearMinutes) }}</span></td>
|
||||||
|
</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) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.base25) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.bonus25) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.bonus25) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(monthReport.total25) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.total25) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.base50) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.base50) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.bonus50) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.bonus50) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(monthReport.total50) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.total50) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(monthReport.total) }} <span class="text-neutral-400">/ {{ formatCentiemes(monthReport.total) }}</span></td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- Week rows (always 5) -->
|
<!-- Week rows (always 5) -->
|
||||||
@@ -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) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.base25) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.bonus25) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.total25) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.base50) }}</span></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) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.bonus50) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500 border-r-2">{{ formatMinutes(reste.total50) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.total50) }}</span></td>
|
||||||
|
<td class="px-4 py-[10px] text-center font-bold text-primary-500 border border-primary-500">{{ formatMinutes(reste.total) }} <span class="text-neutral-400">/ {{ formatCentiemes(reste.total) }}</span></td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
@@ -225,6 +258,17 @@ const emit = defineEmits<{
|
|||||||
(event: 'submit-rtt-payment', month: number, base25Minutes: number, bonus25Minutes: number, base50Minutes: number, bonus50Minutes: number): void
|
(event: 'submit-rtt-payment', month: number, base25Minutes: number, bonus25Minutes: number, base50Minutes: number, bonus50Minutes: number): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
|
// --- Last complete week number ---
|
||||||
|
|
||||||
|
const lastCompleteWeek = computed(() => {
|
||||||
|
const now = new Date()
|
||||||
|
const startOfYear = new Date(now.getFullYear(), 0, 1)
|
||||||
|
const dayOfYear = Math.floor((now.getTime() - startOfYear.getTime()) / 86400000) + 1
|
||||||
|
const dayOfWeek = now.getDay() || 7 // Monday = 1, Sunday = 7
|
||||||
|
const currentWeek = Math.ceil((dayOfYear - dayOfWeek + 10) / 7)
|
||||||
|
return currentWeek - 1
|
||||||
|
})
|
||||||
|
|
||||||
// --- Month navigation ---
|
// --- Month navigation ---
|
||||||
|
|
||||||
const orderedMonths = [6, 7, 8, 9, 10, 11, 12, 1, 2, 3, 4, 5] as const
|
const orderedMonths = [6, 7, 8, 9, 10, 11, 12, 1, 2, 3, 4, 5] as const
|
||||||
@@ -290,44 +334,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 +454,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 ---
|
||||||
@@ -357,6 +481,11 @@ const formatMinutes = (minutes: number): string => {
|
|||||||
return `${sign}${hours} h ${rest} m`
|
return `${sign}${hours} h ${rest} m`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const formatCentiemes = (minutes: number): string => {
|
||||||
|
const value = minutes / 60
|
||||||
|
return value.toFixed(2).replace('.', ',')
|
||||||
|
}
|
||||||
|
|
||||||
// --- Payment drawer ---
|
// --- Payment drawer ---
|
||||||
|
|
||||||
const isPaymentDrawerOpen = ref(false)
|
const isPaymentDrawerOpen = ref(false)
|
||||||
|
|||||||
@@ -14,6 +14,7 @@
|
|||||||
<span>+25%</span>
|
<span>+25%</span>
|
||||||
<span>+50%</span>
|
<span>+50%</span>
|
||||||
<span>Total <br>récup.</span>
|
<span>Total <br>récup.</span>
|
||||||
|
<span>Panier <br>nuit</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="border-x border-b border-primary-500 rounded-b-md">
|
<div class="border-x border-b border-primary-500 rounded-b-md">
|
||||||
@@ -68,6 +69,9 @@
|
|||||||
<div class="font-semibold">
|
<div class="font-semibold">
|
||||||
{{ row.trackingMode === 'PRESENCE' || isInterimContract(row.contractType) ? '-' : formatMinutes(row.weeklyRecoveryMinutes ?? 0) }}
|
{{ row.trackingMode === 'PRESENCE' || isInterimContract(row.contractType) ? '-' : formatMinutes(row.weeklyRecoveryMinutes ?? 0) }}
|
||||||
</div>
|
</div>
|
||||||
|
<div class="font-semibold">
|
||||||
|
{{ (row.weeklyNightBasketCount ?? 0) > 0 ? row.weeklyNightBasketCount : '-' }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
983
frontend/composables/useDriverHoursPage.ts
Normal file
983
frontend/composables/useDriverHoursPage.ts
Normal file
@@ -0,0 +1,983 @@
|
|||||||
|
import { computed, onMounted, ref, watch } from 'vue'
|
||||||
|
import type { Employee } from '~/services/dto/employee'
|
||||||
|
import type { Site } from '~/services/dto/site'
|
||||||
|
import type { WorkHour, WorkHourDayContext, WeeklyWorkHourSummary } from '~/services/dto/work-hour'
|
||||||
|
import type { AbsenceType } from '~/services/dto/absence-type'
|
||||||
|
import type { Absence } from '~/services/dto/absence'
|
||||||
|
import type { HalfDay } from '~/services/dto/half-day'
|
||||||
|
import type { DriverHourRow } from '~/services/dto/work-hour'
|
||||||
|
import { listScopedEmployees } from '~/services/employees'
|
||||||
|
import { listAbsenceTypes } from '~/services/absence-types'
|
||||||
|
import { createAbsence, deleteAbsence, listAbsences, updateAbsence } from '~/services/absences'
|
||||||
|
import { listPublicHolidays } from '~/services/public-holidays'
|
||||||
|
import {
|
||||||
|
bulkUpdateWorkHourSiteValidation,
|
||||||
|
bulkUpdateWorkHourValidation,
|
||||||
|
bulkUpsertWorkHours,
|
||||||
|
getWorkHourDayContext,
|
||||||
|
getWeeklyWorkHourSummary,
|
||||||
|
listWorkHoursByDate,
|
||||||
|
updateWorkHourSiteValidation,
|
||||||
|
updateWorkHourValidation
|
||||||
|
} from '~/services/work-hours'
|
||||||
|
import {
|
||||||
|
formatDateLongFr,
|
||||||
|
formatWeekRangeFr,
|
||||||
|
getIsoWeekNumber,
|
||||||
|
getOffsetFromTodayYmd,
|
||||||
|
getWeekStartDate,
|
||||||
|
getTodayYmd,
|
||||||
|
parseYmd,
|
||||||
|
shiftYmd
|
||||||
|
} from '~/utils/date'
|
||||||
|
import { sortEmployeesBySiteAndOrder } from '~/utils/employee'
|
||||||
|
|
||||||
|
export const useDriverHoursPage = () => {
|
||||||
|
const auth = useAuthStore()
|
||||||
|
const toast = useToast()
|
||||||
|
const isAdmin = computed(() => auth.user?.roles?.includes('ROLE_ADMIN') ?? false)
|
||||||
|
const isSelfUser = computed(() => auth.user?.roles?.includes('ROLE_SELF') ?? false)
|
||||||
|
const isSiteManager = computed(() => !isAdmin.value && !isSelfUser.value)
|
||||||
|
const viewMode = ref<'day' | 'week'>('day')
|
||||||
|
|
||||||
|
const selectedDate = ref(getTodayYmd())
|
||||||
|
const employees = ref<Employee[]>([])
|
||||||
|
const employeeFilter = ref('')
|
||||||
|
const selectedSiteIds = ref<number[]>([])
|
||||||
|
const sitesInitialized = ref(false)
|
||||||
|
const rows = ref<Record<number, DriverHourRow>>({})
|
||||||
|
const dayContext = ref<WorkHourDayContext | null>(null)
|
||||||
|
const weeklySummary = ref<WeeklyWorkHourSummary | null>(null)
|
||||||
|
const absenceTypes = ref<AbsenceType[]>([])
|
||||||
|
const absences = ref<Absence[]>([])
|
||||||
|
const publicHolidaysByYear = ref<Record<number, Record<string, string>>>({})
|
||||||
|
const isAbsenceDrawerOpen = ref(false)
|
||||||
|
const isAbsenceSubmitting = ref(false)
|
||||||
|
const editingAbsence = ref<Absence | null>(null)
|
||||||
|
const absenceForm = ref({
|
||||||
|
employeeId: '' as number | '',
|
||||||
|
typeId: '' as number | '',
|
||||||
|
startDate: '',
|
||||||
|
startHalf: 'AM' as HalfDay,
|
||||||
|
endDate: '',
|
||||||
|
endHalf: 'PM' as HalfDay,
|
||||||
|
comment: ''
|
||||||
|
})
|
||||||
|
const isLoading = ref(false)
|
||||||
|
const isWeekLoading = ref(false)
|
||||||
|
const isSubmitting = ref(false)
|
||||||
|
const validatingRowIds = ref<number[]>([])
|
||||||
|
const siteValidatingRowIds = ref<number[]>([])
|
||||||
|
|
||||||
|
const dayGridCols = computed(() => {
|
||||||
|
const metricCol = '0.4fr'
|
||||||
|
const validationCols = isAdmin.value ? `${metricCol}` : `${metricCol} ${metricCol}`
|
||||||
|
return `1.2fr 0.6fr 0.8fr 0.8fr 0.8fr ${metricCol} ${metricCol} ${metricCol} ${metricCol} ${metricCol} ${validationCols}`
|
||||||
|
})
|
||||||
|
|
||||||
|
const weekGridCols = '1.6fr repeat(7, 0.6fr) repeat(7, 0.6fr) repeat(4, 0.4fr)'
|
||||||
|
|
||||||
|
const sites = computed<Site[]>(() => {
|
||||||
|
const siteMap = new Map<number, Site>()
|
||||||
|
for (const employee of employees.value) {
|
||||||
|
if (employee.site) {
|
||||||
|
siteMap.set(employee.site.id, employee.site)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Array.from(siteMap.values()).sort((siteA, siteB) => {
|
||||||
|
const orderA = siteA.displayOrder ?? 0
|
||||||
|
const orderB = siteB.displayOrder ?? 0
|
||||||
|
if (orderA !== orderB) return orderA - orderB
|
||||||
|
return siteA.name.localeCompare(siteB.name, 'fr')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const visibleEmployees = computed(() => {
|
||||||
|
if (selectedSiteIds.value.length === 0) return []
|
||||||
|
const filter = employeeFilter.value.trim().toLowerCase()
|
||||||
|
return employees.value.filter((employee) => {
|
||||||
|
if (employee.isDriver !== true) return false
|
||||||
|
const siteId = employee.site?.id
|
||||||
|
if (!siteId || !selectedSiteIds.value.includes(siteId)) return false
|
||||||
|
if (!filter) return true
|
||||||
|
const firstName = employee.firstName?.toLowerCase() ?? ''
|
||||||
|
const lastName = employee.lastName?.toLowerCase() ?? ''
|
||||||
|
return firstName.includes(filter) || lastName.includes(filter)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const displayedEmployees = computed(() => {
|
||||||
|
return visibleEmployees.value.filter((employee) => hasContractAtSelectedDate(employee.id))
|
||||||
|
})
|
||||||
|
|
||||||
|
const visibleEmployeeIdSet = computed(() => new Set(visibleEmployees.value.map((employee) => employee.id)))
|
||||||
|
|
||||||
|
const filteredWeeklySummary = computed<WeeklyWorkHourSummary | null>(() => {
|
||||||
|
if (!weeklySummary.value) return null
|
||||||
|
return {
|
||||||
|
...weeklySummary.value,
|
||||||
|
rows: weeklySummary.value.rows.filter((row) =>
|
||||||
|
visibleEmployeeIdSet.value.has(row.employeeId) && row.hasContractForWeek !== false
|
||||||
|
)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const saveButtonClass = computed(() => {
|
||||||
|
if (isSubmitting.value || employees.value.length === 0) {
|
||||||
|
return 'opacity-50 cursor-not-allowed'
|
||||||
|
}
|
||||||
|
return ''
|
||||||
|
})
|
||||||
|
|
||||||
|
const isValidationPending = (employeeId: number) => validatingRowIds.value.includes(employeeId)
|
||||||
|
const isSiteValidationPending = (employeeId: number) => siteValidatingRowIds.value.includes(employeeId)
|
||||||
|
const canToggleValidation = (employeeId: number) => !!rows.value[employeeId]?.workHourId
|
||||||
|
const canToggleSiteValidation = (employeeId: number) => {
|
||||||
|
if (!isSiteManager.value) return false
|
||||||
|
const row = rows.value[employeeId]
|
||||||
|
if (!row?.workHourId) return false
|
||||||
|
if (row.isValid) return false
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
const canCreateEmptyValidationRow = (employeeId: number) => {
|
||||||
|
const row = rows.value[employeeId]
|
||||||
|
if (row?.workHourId) return false
|
||||||
|
if (!hasContractAtSelectedDate(employeeId)) return false
|
||||||
|
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
||||||
|
return !!dayRow?.absenceLabel
|
||||||
|
}
|
||||||
|
|
||||||
|
const canCreateValidationRowFromAbsence = (employeeId: number) => canCreateEmptyValidationRow(employeeId)
|
||||||
|
const canCreateSiteValidationRowFromAbsence = (employeeId: number) => canCreateEmptyValidationRow(employeeId)
|
||||||
|
|
||||||
|
const bulkValidatableEmployeeIds = computed(() => {
|
||||||
|
return visibleEmployees.value
|
||||||
|
.map((employee) => employee.id)
|
||||||
|
.filter((employeeId) => canToggleValidation(employeeId) || canCreateValidationRowFromAbsence(employeeId))
|
||||||
|
})
|
||||||
|
|
||||||
|
const isBulkValidationChecked = computed(() => {
|
||||||
|
const ids = bulkValidatableEmployeeIds.value
|
||||||
|
if (ids.length === 0) return false
|
||||||
|
return ids.every((employeeId) => rows.value[employeeId]?.isValid ?? false)
|
||||||
|
})
|
||||||
|
|
||||||
|
const isBulkValidationIndeterminate = computed(() => {
|
||||||
|
const ids = bulkValidatableEmployeeIds.value
|
||||||
|
if (ids.length === 0) return false
|
||||||
|
const checkedCount = ids.filter((employeeId) => rows.value[employeeId]?.isValid ?? false).length
|
||||||
|
return checkedCount > 0 && checkedCount < ids.length
|
||||||
|
})
|
||||||
|
|
||||||
|
const bulkSiteValidatableEmployeeIds = computed(() => {
|
||||||
|
if (!isSiteManager.value) return []
|
||||||
|
return visibleEmployees.value
|
||||||
|
.map((employee) => employee.id)
|
||||||
|
.filter((employeeId) => canToggleSiteValidation(employeeId) || canCreateSiteValidationRowFromAbsence(employeeId))
|
||||||
|
})
|
||||||
|
|
||||||
|
const isBulkSiteValidationChecked = computed(() => {
|
||||||
|
const ids = bulkSiteValidatableEmployeeIds.value
|
||||||
|
if (ids.length === 0) return false
|
||||||
|
return ids.every((employeeId) => rows.value[employeeId]?.isSiteValid ?? false)
|
||||||
|
})
|
||||||
|
|
||||||
|
const isBulkSiteValidationIndeterminate = computed(() => {
|
||||||
|
const ids = bulkSiteValidatableEmployeeIds.value
|
||||||
|
if (ids.length === 0) return false
|
||||||
|
const checkedCount = ids.filter((employeeId) => rows.value[employeeId]?.isSiteValid ?? false).length
|
||||||
|
return checkedCount > 0 && checkedCount < ids.length
|
||||||
|
})
|
||||||
|
|
||||||
|
const canBulkToggleSiteValidation = computed(() => bulkSiteValidatableEmployeeIds.value.length > 0)
|
||||||
|
|
||||||
|
const dayContextByEmployeeId = computed(() => {
|
||||||
|
const map = new Map<number, WorkHourDayContext['rows'][number]>()
|
||||||
|
for (const row of dayContext.value?.rows ?? []) {
|
||||||
|
map.set(row.employeeId, row)
|
||||||
|
}
|
||||||
|
return map
|
||||||
|
})
|
||||||
|
|
||||||
|
const shortcutButtonClass = (target: 'yesterday' | 'today' | 'tomorrow') => {
|
||||||
|
const targetDate = target === 'yesterday'
|
||||||
|
? getOffsetFromTodayYmd(-1)
|
||||||
|
: target === 'tomorrow'
|
||||||
|
? getOffsetFromTodayYmd(1)
|
||||||
|
: getTodayYmd()
|
||||||
|
|
||||||
|
if (selectedDate.value === targetDate) {
|
||||||
|
return 'bg-primary-500 text-white'
|
||||||
|
}
|
||||||
|
return 'bg-white text-primary-500 hover:bg-tertiary-500'
|
||||||
|
}
|
||||||
|
|
||||||
|
const weekShortcutButtonClass = (target: 'previousWeek' | 'thisWeek' | 'nextWeek') => {
|
||||||
|
const selected = parseYmd(selectedDate.value)
|
||||||
|
if (!selected) {
|
||||||
|
return 'bg-white text-primary-500 hover:bg-tertiary-500'
|
||||||
|
}
|
||||||
|
|
||||||
|
const today = new Date()
|
||||||
|
const targetDate = new Date(today)
|
||||||
|
if (target === 'previousWeek') targetDate.setDate(today.getDate() - 7)
|
||||||
|
if (target === 'nextWeek') targetDate.setDate(today.getDate() + 7)
|
||||||
|
|
||||||
|
const selectedWeekStart = getWeekStartDate(selected)
|
||||||
|
const targetWeekStart = getWeekStartDate(targetDate)
|
||||||
|
const isActive = selectedWeekStart.getTime() === targetWeekStart.getTime()
|
||||||
|
|
||||||
|
if (isActive) {
|
||||||
|
return 'bg-primary-500 text-white'
|
||||||
|
}
|
||||||
|
return 'bg-white text-primary-500 hover:bg-tertiary-500'
|
||||||
|
}
|
||||||
|
|
||||||
|
const getWeekShortcutLabel = (target: 'previousWeek' | 'thisWeek' | 'nextWeek') => {
|
||||||
|
const today = new Date()
|
||||||
|
if (target === 'previousWeek') today.setDate(today.getDate() - 7)
|
||||||
|
if (target === 'nextWeek') today.setDate(today.getDate() + 7)
|
||||||
|
|
||||||
|
const weekNumber = getIsoWeekNumber(today)
|
||||||
|
return `Sem. S${weekNumber}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const formattedSelectedDate = computed(() => {
|
||||||
|
const parsed = parseYmd(selectedDate.value)
|
||||||
|
if (!parsed) return selectedDate.value
|
||||||
|
|
||||||
|
if (viewMode.value === 'week') {
|
||||||
|
return formatWeekRangeFr(parsed)
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatDateLongFr(parsed)
|
||||||
|
})
|
||||||
|
|
||||||
|
const selectedYear = computed(() => {
|
||||||
|
const parsed = parseYmd(selectedDate.value)
|
||||||
|
return parsed ? parsed.getFullYear() : null
|
||||||
|
})
|
||||||
|
|
||||||
|
const selectedHolidayLabel = computed(() => {
|
||||||
|
const year = selectedYear.value
|
||||||
|
if (!year) return ''
|
||||||
|
return publicHolidaysByYear.value[year]?.[selectedDate.value] ?? ''
|
||||||
|
})
|
||||||
|
|
||||||
|
const isSelectedDateHoliday = computed(() => selectedHolidayLabel.value !== '')
|
||||||
|
|
||||||
|
const weekDayHeaders = computed(() => {
|
||||||
|
const days = weeklySummary.value?.days ?? []
|
||||||
|
return days.map((date) => {
|
||||||
|
const parsed = parseYmd(date)
|
||||||
|
if (!parsed) return { date, weekday: '', dayDate: '' }
|
||||||
|
const weekday = new Intl.DateTimeFormat('fr-FR', { weekday: 'short' }).format(parsed)
|
||||||
|
const dayDate = new Intl.DateTimeFormat('fr-FR', { day: '2-digit', month: '2-digit' }).format(parsed)
|
||||||
|
return { date, weekday, dayDate }
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const shiftDate = (steps: number) => {
|
||||||
|
const offset = viewMode.value === 'week' ? (steps * 7) : steps
|
||||||
|
const next = shiftYmd(selectedDate.value, offset)
|
||||||
|
if (!next) return
|
||||||
|
selectedDate.value = next
|
||||||
|
}
|
||||||
|
|
||||||
|
const setToday = () => { selectedDate.value = getTodayYmd() }
|
||||||
|
const setYesterday = () => { setToday(); shiftDate(-1) }
|
||||||
|
const setTomorrow = () => { setToday(); shiftDate(1) }
|
||||||
|
const setThisWeek = () => { selectedDate.value = getTodayYmd() }
|
||||||
|
const setPreviousWeek = () => {
|
||||||
|
const previousWeek = shiftYmd(getTodayYmd(), -7)
|
||||||
|
if (!previousWeek) return
|
||||||
|
selectedDate.value = previousWeek
|
||||||
|
}
|
||||||
|
const setNextWeek = () => {
|
||||||
|
const nextWeek = shiftYmd(getTodayYmd(), 7)
|
||||||
|
if (!nextWeek) return
|
||||||
|
selectedDate.value = nextWeek
|
||||||
|
}
|
||||||
|
|
||||||
|
const resetAbsenceForm = () => {
|
||||||
|
absenceForm.value = {
|
||||||
|
employeeId: '',
|
||||||
|
typeId: '',
|
||||||
|
startDate: '',
|
||||||
|
startHalf: 'AM',
|
||||||
|
endDate: '',
|
||||||
|
endHalf: 'PM',
|
||||||
|
comment: ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const closeAbsenceDrawer = () => {
|
||||||
|
isAbsenceDrawerOpen.value = false
|
||||||
|
editingAbsence.value = null
|
||||||
|
resetAbsenceForm()
|
||||||
|
}
|
||||||
|
|
||||||
|
const toMinutes = (time: string): number => {
|
||||||
|
if (!time) return 0
|
||||||
|
const [hours, minutes] = time.split(':').map(Number)
|
||||||
|
if (!Number.isInteger(hours) || !Number.isInteger(minutes)) return 0
|
||||||
|
return (hours * 60) + minutes
|
||||||
|
}
|
||||||
|
|
||||||
|
const formatMinutes = (minutes: number) => {
|
||||||
|
const safeMinutes = Math.max(0, minutes)
|
||||||
|
const hours = Math.floor(safeMinutes / 60)
|
||||||
|
const rest = safeMinutes % 60
|
||||||
|
return `${String(hours).padStart(2, '0')}:${String(rest).padStart(2, '0')}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const minutesToTimeString = (minutes: number | null | undefined): string => {
|
||||||
|
if (minutes === null || minutes === undefined || minutes === 0) return ''
|
||||||
|
return formatMinutes(minutes)
|
||||||
|
}
|
||||||
|
|
||||||
|
const emptyRow = (): DriverHourRow => ({
|
||||||
|
workHourId: null,
|
||||||
|
dayHours: '',
|
||||||
|
nightHours: '',
|
||||||
|
workshopHours: '',
|
||||||
|
hasBreakfast: false,
|
||||||
|
hasLunch: false,
|
||||||
|
hasDinner: false,
|
||||||
|
hasOvernight: false,
|
||||||
|
isSiteValid: false,
|
||||||
|
isValid: false,
|
||||||
|
updatedAt: null
|
||||||
|
})
|
||||||
|
|
||||||
|
const isRowLocked = (employeeId: number) => {
|
||||||
|
const row = rows.value[employeeId]
|
||||||
|
if (!row) return false
|
||||||
|
if (row.isValid) return true
|
||||||
|
if (!isAdmin.value && row.isSiteValid) return true
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
const contractLabel = (employee: Employee) => {
|
||||||
|
const contract = employee.contract
|
||||||
|
if (!contract) return '-'
|
||||||
|
return contract.name
|
||||||
|
}
|
||||||
|
|
||||||
|
const getRowMetrics = (employeeId: number) => {
|
||||||
|
const row = rows.value[employeeId] ?? emptyRow()
|
||||||
|
const credited = dayContextByEmployeeId.value.get(employeeId)?.creditedMinutes ?? 0
|
||||||
|
const dayMinutes = toMinutes(row.dayHours) + credited
|
||||||
|
const nightMinutes = toMinutes(row.nightHours)
|
||||||
|
const workshopMinutes = toMinutes(row.workshopHours)
|
||||||
|
const totalMinutes = dayMinutes + nightMinutes + workshopMinutes
|
||||||
|
return { dayMinutes, nightMinutes, workshopMinutes, totalMinutes }
|
||||||
|
}
|
||||||
|
|
||||||
|
const getRowAbsenceLabel = (employeeId: number) => {
|
||||||
|
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
||||||
|
if (dayRow && dayRow.hasContractAtDate === false) {
|
||||||
|
return 'Contrat non démarré'
|
||||||
|
}
|
||||||
|
if (isSelectedDateHoliday.value) return 'Férié'
|
||||||
|
if (!dayRow?.absenceLabel) return ''
|
||||||
|
if (dayRow.absenceHalf === 'AM' || dayRow.absenceHalf === 'PM') {
|
||||||
|
const halfLabel = dayRow.absenceHalf === 'AM' ? 'Matin' : 'ap.-m.'
|
||||||
|
return `${dayRow.absenceLabel} (${halfLabel})`
|
||||||
|
}
|
||||||
|
return `${dayRow.absenceLabel} (journée)`
|
||||||
|
}
|
||||||
|
|
||||||
|
const getRowAbsenceStyle = (employeeId: number) => {
|
||||||
|
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
||||||
|
if (dayRow && dayRow.hasContractAtDate === false) {
|
||||||
|
return { backgroundColor: '#6b7280' }
|
||||||
|
}
|
||||||
|
if (!dayRow?.absenceLabel) return undefined
|
||||||
|
return { backgroundColor: dayRow.absenceColor || '#dc2626' }
|
||||||
|
}
|
||||||
|
|
||||||
|
const getRowUpdatedAt = (employeeId: number): string => {
|
||||||
|
const raw = rows.value[employeeId]?.updatedAt
|
||||||
|
if (!raw) return ''
|
||||||
|
const date = new Date(raw)
|
||||||
|
if (Number.isNaN(date.getTime())) return ''
|
||||||
|
return date.toLocaleDateString('fr-FR', { day: '2-digit', month: '2-digit', year: 'numeric', hour: '2-digit', minute: '2-digit' })
|
||||||
|
}
|
||||||
|
|
||||||
|
const hasContractAtSelectedDate = (employeeId: number) => {
|
||||||
|
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
||||||
|
if (!dayRow) return true
|
||||||
|
return dayRow.hasContractAtDate !== false
|
||||||
|
}
|
||||||
|
|
||||||
|
const hydrateRows = (workHours: WorkHour[]) => {
|
||||||
|
const byEmployeeId = new Map<number, WorkHour>()
|
||||||
|
for (const workHour of workHours) {
|
||||||
|
byEmployeeId.set(workHour.employee.id, workHour)
|
||||||
|
}
|
||||||
|
|
||||||
|
const nextRows: Record<number, DriverHourRow> = {}
|
||||||
|
for (const employee of employees.value) {
|
||||||
|
if (employee.isDriver !== true) continue
|
||||||
|
const workHour = byEmployeeId.get(employee.id)
|
||||||
|
nextRows[employee.id] = {
|
||||||
|
workHourId: workHour?.id ?? null,
|
||||||
|
dayHours: minutesToTimeString(workHour?.dayHoursMinutes),
|
||||||
|
nightHours: minutesToTimeString(workHour?.nightHoursMinutes),
|
||||||
|
workshopHours: minutesToTimeString(workHour?.workshopHoursMinutes),
|
||||||
|
hasBreakfast: workHour?.hasBreakfast ?? false,
|
||||||
|
hasLunch: workHour?.hasLunch ?? false,
|
||||||
|
hasDinner: workHour?.hasDinner ?? false,
|
||||||
|
hasOvernight: workHour?.hasOvernight ?? false,
|
||||||
|
isSiteValid: workHour?.isSiteValid ?? false,
|
||||||
|
isValid: workHour?.isValid ?? false,
|
||||||
|
updatedAt: workHour?.updatedAt ?? null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rows.value = nextRows
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadAbsenceTypes = async () => {
|
||||||
|
absenceTypes.value = await listAbsenceTypes()
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadPublicHolidaysForSelectedYear = async () => {
|
||||||
|
const year = selectedYear.value
|
||||||
|
if (!year) return
|
||||||
|
if (publicHolidaysByYear.value[year]) return
|
||||||
|
|
||||||
|
const holidays = await listPublicHolidays('metropole', year)
|
||||||
|
publicHolidaysByYear.value = {
|
||||||
|
...publicHolidaysByYear.value,
|
||||||
|
[year]: holidays
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadAbsences = async () => {
|
||||||
|
absences.value = await listAbsences({
|
||||||
|
from: selectedDate.value,
|
||||||
|
to: selectedDate.value,
|
||||||
|
siteIds: isAdmin.value ? selectedSiteIds.value : undefined
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const openAbsenceDrawer = (employeeId: number) => {
|
||||||
|
if (!hasContractAtSelectedDate(employeeId)) return
|
||||||
|
if (isSelectedDateHoliday.value) return
|
||||||
|
|
||||||
|
const existing = absences.value.find((absence) => {
|
||||||
|
if (absence.employee?.id !== employeeId) return false
|
||||||
|
const start = absence.startDate.slice(0, 10)
|
||||||
|
const end = absence.endDate.slice(0, 10)
|
||||||
|
return selectedDate.value >= start && selectedDate.value <= end
|
||||||
|
}) ?? null
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
editingAbsence.value = existing
|
||||||
|
absenceForm.value = {
|
||||||
|
employeeId,
|
||||||
|
typeId: existing.type?.id ?? '',
|
||||||
|
startDate: existing.startDate.slice(0, 10),
|
||||||
|
startHalf: existing.startHalf ?? 'AM',
|
||||||
|
endDate: existing.endDate.slice(0, 10),
|
||||||
|
endHalf: existing.endHalf ?? 'PM',
|
||||||
|
comment: existing.comment ?? ''
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
editingAbsence.value = null
|
||||||
|
absenceForm.value = {
|
||||||
|
employeeId,
|
||||||
|
typeId: '',
|
||||||
|
startDate: selectedDate.value,
|
||||||
|
startHalf: 'AM',
|
||||||
|
endDate: selectedDate.value,
|
||||||
|
endHalf: 'PM',
|
||||||
|
comment: ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
isAbsenceDrawerOpen.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const refreshAfterAbsenceChange = async () => {
|
||||||
|
if (isAdmin.value) {
|
||||||
|
await Promise.all([loadWeeklySummary(), loadDayContext(), loadAbsences()])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
weeklySummary.value = null
|
||||||
|
await Promise.all([loadDayContext(), loadAbsences()])
|
||||||
|
}
|
||||||
|
|
||||||
|
const submitAbsence = async () => {
|
||||||
|
const form = absenceForm.value
|
||||||
|
if (isAbsenceSubmitting.value || form.employeeId === '' || form.typeId === '') return
|
||||||
|
|
||||||
|
isAbsenceSubmitting.value = true
|
||||||
|
try {
|
||||||
|
if (editingAbsence.value) {
|
||||||
|
await updateAbsence({
|
||||||
|
id: editingAbsence.value.id,
|
||||||
|
employeeId: Number(form.employeeId),
|
||||||
|
typeId: Number(form.typeId),
|
||||||
|
startDate: form.startDate,
|
||||||
|
startHalf: form.startHalf,
|
||||||
|
endDate: form.endDate,
|
||||||
|
endHalf: form.endHalf,
|
||||||
|
comment: editingAbsence.value.comment ?? ''
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
await createAbsence({
|
||||||
|
employeeId: Number(form.employeeId),
|
||||||
|
typeId: Number(form.typeId),
|
||||||
|
startDate: form.startDate,
|
||||||
|
startHalf: form.startHalf,
|
||||||
|
endDate: form.endDate,
|
||||||
|
endHalf: form.endHalf,
|
||||||
|
comment: ''
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
closeAbsenceDrawer()
|
||||||
|
await refreshAfterAbsenceChange()
|
||||||
|
} finally {
|
||||||
|
isAbsenceSubmitting.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const deleteAbsenceFromDrawer = async () => {
|
||||||
|
if (!editingAbsence.value || isAbsenceSubmitting.value) return
|
||||||
|
|
||||||
|
isAbsenceSubmitting.value = true
|
||||||
|
try {
|
||||||
|
await deleteAbsence(editingAbsence.value.id)
|
||||||
|
closeAbsenceDrawer()
|
||||||
|
await refreshAfterAbsenceChange()
|
||||||
|
} finally {
|
||||||
|
isAbsenceSubmitting.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const buildEmptyDriverEntry = (employeeId: number) => ({
|
||||||
|
employeeId,
|
||||||
|
morningFrom: null,
|
||||||
|
morningTo: null,
|
||||||
|
afternoonFrom: null,
|
||||||
|
afternoonTo: null,
|
||||||
|
eveningFrom: null,
|
||||||
|
eveningTo: null,
|
||||||
|
isPresentMorning: false,
|
||||||
|
isPresentAfternoon: false,
|
||||||
|
dayHoursMinutes: null,
|
||||||
|
nightHoursMinutes: null,
|
||||||
|
workshopHoursMinutes: null,
|
||||||
|
hasBreakfast: false,
|
||||||
|
hasLunch: false,
|
||||||
|
hasDinner: false,
|
||||||
|
hasOvernight: false
|
||||||
|
})
|
||||||
|
|
||||||
|
const toggleValidation = async (
|
||||||
|
employeeId: number,
|
||||||
|
checked: boolean,
|
||||||
|
options: { toast?: boolean } = {}
|
||||||
|
) => {
|
||||||
|
const row = rows.value[employeeId]
|
||||||
|
if (!row?.workHourId && checked) {
|
||||||
|
if (canCreateEmptyValidationRow(employeeId)) {
|
||||||
|
await bulkUpsertWorkHours({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
entries: [buildEmptyDriverEntry(employeeId)]
|
||||||
|
}, { toast: false })
|
||||||
|
await loadWorkHours()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatedRow = rows.value[employeeId]
|
||||||
|
if (!updatedRow?.workHourId) {
|
||||||
|
if (options.toast !== false) {
|
||||||
|
toast.error({
|
||||||
|
title: 'Validation impossible',
|
||||||
|
message: 'La ligne doit contenir des heures ou une absence.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isValidationPending(employeeId)) return
|
||||||
|
|
||||||
|
validatingRowIds.value = [...validatingRowIds.value, employeeId]
|
||||||
|
try {
|
||||||
|
await updateWorkHourValidation(updatedRow.workHourId, checked, { toast: options.toast })
|
||||||
|
updatedRow.isValid = checked
|
||||||
|
} finally {
|
||||||
|
validatingRowIds.value = validatingRowIds.value.filter((id) => id !== employeeId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleSiteValidation = async (
|
||||||
|
employeeId: number,
|
||||||
|
checked: boolean,
|
||||||
|
options: { toast?: boolean } = {}
|
||||||
|
) => {
|
||||||
|
const row = rows.value[employeeId]
|
||||||
|
if (!row?.workHourId && checked) {
|
||||||
|
if (canCreateEmptyValidationRow(employeeId)) {
|
||||||
|
await bulkUpsertWorkHours({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
entries: [buildEmptyDriverEntry(employeeId)]
|
||||||
|
}, { toast: false })
|
||||||
|
await loadWorkHours()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatedRow = rows.value[employeeId]
|
||||||
|
if (!updatedRow?.workHourId) {
|
||||||
|
if (options.toast !== false) {
|
||||||
|
toast.error({
|
||||||
|
title: 'Validation impossible',
|
||||||
|
message: 'La ligne doit contenir des heures ou une absence.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isSiteValidationPending(employeeId)) return
|
||||||
|
if (!canToggleSiteValidation(employeeId)) return
|
||||||
|
|
||||||
|
siteValidatingRowIds.value = [...siteValidatingRowIds.value, employeeId]
|
||||||
|
try {
|
||||||
|
await updateWorkHourSiteValidation(updatedRow.workHourId, checked, { toast: options.toast })
|
||||||
|
updatedRow.isSiteValid = checked
|
||||||
|
} finally {
|
||||||
|
siteValidatingRowIds.value = siteValidatingRowIds.value.filter((id) => id !== employeeId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleValidationBulk = async (checked: boolean) => {
|
||||||
|
const employeeIds = bulkValidatableEmployeeIds.value
|
||||||
|
if (employeeIds.length === 0) return
|
||||||
|
|
||||||
|
const pendingIds = new Set(validatingRowIds.value)
|
||||||
|
const availableEmployeeIds = employeeIds.filter((employeeId) => !pendingIds.has(employeeId))
|
||||||
|
if (availableEmployeeIds.length === 0) return
|
||||||
|
|
||||||
|
if (checked) {
|
||||||
|
const toCreateIds = availableEmployeeIds.filter((employeeId) => canCreateValidationRowFromAbsence(employeeId))
|
||||||
|
if (toCreateIds.length > 0) {
|
||||||
|
await bulkUpsertWorkHours({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
entries: toCreateIds.map((employeeId) => buildEmptyDriverEntry(employeeId))
|
||||||
|
}, { toast: false })
|
||||||
|
await loadWorkHours()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetEmployeeIds = availableEmployeeIds.filter((employeeId) => canToggleValidation(employeeId))
|
||||||
|
if (targetEmployeeIds.length === 0) {
|
||||||
|
toast.error({
|
||||||
|
title: 'Validation impossible',
|
||||||
|
message: 'Aucune ligne ne peut être validée.'
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
validatingRowIds.value = Array.from(new Set([...validatingRowIds.value, ...targetEmployeeIds]))
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await bulkUpdateWorkHourValidation({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
isValid: checked,
|
||||||
|
employeeIds: targetEmployeeIds
|
||||||
|
}, { toast: false })
|
||||||
|
|
||||||
|
await loadWorkHours()
|
||||||
|
|
||||||
|
if (result.updated === 0) {
|
||||||
|
toast.error({ title: 'Erreur', message: 'Aucune ligne mise à jour.' })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.skipped > 0) {
|
||||||
|
toast.success({
|
||||||
|
title: 'Succès partiel',
|
||||||
|
message: `${result.updated} mise(s) à jour, ${result.skipped} ignorée(s).`
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.success({
|
||||||
|
title: 'Succès',
|
||||||
|
message: checked
|
||||||
|
? `${result.updated} ligne(s) validée(s).`
|
||||||
|
: `${result.updated} validation(s) retirée(s).`
|
||||||
|
})
|
||||||
|
} catch {
|
||||||
|
toast.error({ title: 'Erreur', message: 'Impossible de mettre à jour les validations.' })
|
||||||
|
} finally {
|
||||||
|
validatingRowIds.value = validatingRowIds.value.filter((id) => !targetEmployeeIds.includes(id))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleSiteValidationBulk = async (checked: boolean) => {
|
||||||
|
if (!isSiteManager.value) return
|
||||||
|
|
||||||
|
const employeeIds = bulkSiteValidatableEmployeeIds.value
|
||||||
|
if (employeeIds.length === 0) return
|
||||||
|
|
||||||
|
const pendingIds = new Set(siteValidatingRowIds.value)
|
||||||
|
const availableEmployeeIds = employeeIds.filter((employeeId) => !pendingIds.has(employeeId))
|
||||||
|
if (availableEmployeeIds.length === 0) return
|
||||||
|
|
||||||
|
if (checked) {
|
||||||
|
const toCreateIds = availableEmployeeIds.filter((employeeId) => canCreateSiteValidationRowFromAbsence(employeeId))
|
||||||
|
if (toCreateIds.length > 0) {
|
||||||
|
await bulkUpsertWorkHours({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
entries: toCreateIds.map((employeeId) => buildEmptyDriverEntry(employeeId))
|
||||||
|
}, { toast: false })
|
||||||
|
await loadWorkHours()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetEmployeeIds = availableEmployeeIds.filter((employeeId) => canToggleSiteValidation(employeeId))
|
||||||
|
if (targetEmployeeIds.length === 0) {
|
||||||
|
toast.error({
|
||||||
|
title: 'Validation impossible',
|
||||||
|
message: 'Aucune ligne ne peut être validée côté site.'
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
siteValidatingRowIds.value = Array.from(new Set([...siteValidatingRowIds.value, ...targetEmployeeIds]))
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await bulkUpdateWorkHourSiteValidation({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
isSiteValid: checked,
|
||||||
|
employeeIds: targetEmployeeIds
|
||||||
|
}, { toast: false })
|
||||||
|
|
||||||
|
await loadWorkHours()
|
||||||
|
|
||||||
|
if (result.updated === 0) {
|
||||||
|
toast.error({ title: 'Erreur', message: 'Aucune ligne site mise à jour.' })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.skipped > 0) {
|
||||||
|
toast.success({
|
||||||
|
title: 'Succès partiel',
|
||||||
|
message: `${result.updated} mise(s) à jour, ${result.skipped} ignorée(s).`
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.success({
|
||||||
|
title: 'Succès',
|
||||||
|
message: checked
|
||||||
|
? `${result.updated} validation(s) site enregistrée(s).`
|
||||||
|
: `${result.updated} validation(s) site retirée(s).`
|
||||||
|
})
|
||||||
|
} catch {
|
||||||
|
toast.error({ title: 'Erreur', message: 'Impossible de mettre à jour les validations site.' })
|
||||||
|
} finally {
|
||||||
|
siteValidatingRowIds.value = siteValidatingRowIds.value.filter((id) => !targetEmployeeIds.includes(id))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadEmployees = async () => {
|
||||||
|
const scopedEmployees = await listScopedEmployees()
|
||||||
|
employees.value = sortEmployeesBySiteAndOrder(scopedEmployees)
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadWorkHours = async () => {
|
||||||
|
const workHours = await listWorkHoursByDate(selectedDate.value)
|
||||||
|
hydrateRows(workHours)
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadWeeklySummary = async () => {
|
||||||
|
isWeekLoading.value = true
|
||||||
|
try {
|
||||||
|
weeklySummary.value = await getWeeklyWorkHourSummary(selectedDate.value)
|
||||||
|
} finally {
|
||||||
|
isWeekLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadDayContext = async () => {
|
||||||
|
dayContext.value = await getWorkHourDayContext(selectedDate.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
const refreshByDate = async () => {
|
||||||
|
if (isAdmin.value) {
|
||||||
|
await Promise.all([loadWorkHours(), loadWeeklySummary(), loadDayContext(), loadAbsences()])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
weeklySummary.value = null
|
||||||
|
await Promise.all([loadWorkHours(), loadDayContext(), loadAbsences()])
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadPage = async () => {
|
||||||
|
isLoading.value = true
|
||||||
|
try {
|
||||||
|
await loadPublicHolidaysForSelectedYear()
|
||||||
|
await loadEmployees()
|
||||||
|
await loadAbsenceTypes()
|
||||||
|
await refreshByDate()
|
||||||
|
} finally {
|
||||||
|
isLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(loadPage)
|
||||||
|
|
||||||
|
watch(sites, (nextSites) => {
|
||||||
|
const currentSiteIds = nextSites.map((site) => site.id)
|
||||||
|
|
||||||
|
if (!sitesInitialized.value) {
|
||||||
|
if (currentSiteIds.length === 0) return
|
||||||
|
selectedSiteIds.value = currentSiteIds
|
||||||
|
sitesInitialized.value = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
selectedSiteIds.value = selectedSiteIds.value.filter((siteId) => currentSiteIds.includes(siteId))
|
||||||
|
}, { immediate: true })
|
||||||
|
|
||||||
|
watch(isAdmin, async (admin) => {
|
||||||
|
if (!admin) {
|
||||||
|
viewMode.value = 'day'
|
||||||
|
weeklySummary.value = null
|
||||||
|
await Promise.all([loadAbsenceTypes(), loadAbsences()])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
await loadAbsenceTypes()
|
||||||
|
await loadAbsences()
|
||||||
|
}, { immediate: true })
|
||||||
|
|
||||||
|
watch(selectedDate, async () => {
|
||||||
|
await loadPublicHolidaysForSelectedYear()
|
||||||
|
await refreshByDate()
|
||||||
|
})
|
||||||
|
|
||||||
|
const handleSave = async () => {
|
||||||
|
if (isSubmitting.value || employees.value.length === 0) return
|
||||||
|
|
||||||
|
isSubmitting.value = true
|
||||||
|
try {
|
||||||
|
const driverEmployees = employees.value.filter(
|
||||||
|
(e) => e.isDriver === true && hasContractAtSelectedDate(e.id)
|
||||||
|
)
|
||||||
|
|
||||||
|
const entries = driverEmployees.map((employee) => {
|
||||||
|
const employeeId = employee.id
|
||||||
|
const row = rows.value[employeeId] ?? emptyRow()
|
||||||
|
const dayMin = toMinutes(row.dayHours)
|
||||||
|
const nightMin = toMinutes(row.nightHours)
|
||||||
|
const workshopMin = toMinutes(row.workshopHours)
|
||||||
|
|
||||||
|
return {
|
||||||
|
employeeId,
|
||||||
|
morningFrom: null,
|
||||||
|
morningTo: null,
|
||||||
|
afternoonFrom: null,
|
||||||
|
afternoonTo: null,
|
||||||
|
eveningFrom: null,
|
||||||
|
eveningTo: null,
|
||||||
|
isPresentMorning: false,
|
||||||
|
isPresentAfternoon: false,
|
||||||
|
dayHoursMinutes: dayMin || null,
|
||||||
|
nightHoursMinutes: nightMin || null,
|
||||||
|
workshopHoursMinutes: workshopMin || null,
|
||||||
|
hasBreakfast: row.hasBreakfast,
|
||||||
|
hasLunch: row.hasLunch,
|
||||||
|
hasDinner: row.hasDinner,
|
||||||
|
hasOvernight: row.hasOvernight
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (entries.length === 0) return
|
||||||
|
|
||||||
|
await bulkUpsertWorkHours({
|
||||||
|
workDate: selectedDate.value,
|
||||||
|
entries
|
||||||
|
})
|
||||||
|
|
||||||
|
await refreshByDate()
|
||||||
|
} finally {
|
||||||
|
isSubmitting.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
isAdmin,
|
||||||
|
isSelfUser,
|
||||||
|
isSiteManager,
|
||||||
|
viewMode,
|
||||||
|
selectedDate,
|
||||||
|
employeeFilter,
|
||||||
|
sites,
|
||||||
|
selectedSiteIds,
|
||||||
|
employees,
|
||||||
|
visibleEmployees,
|
||||||
|
displayedEmployees,
|
||||||
|
rows,
|
||||||
|
absenceTypes,
|
||||||
|
absenceForm,
|
||||||
|
isAbsenceDrawerOpen,
|
||||||
|
isAbsenceSubmitting,
|
||||||
|
editingAbsence,
|
||||||
|
weeklySummary,
|
||||||
|
filteredWeeklySummary,
|
||||||
|
isLoading,
|
||||||
|
isWeekLoading,
|
||||||
|
isSubmitting,
|
||||||
|
dayGridCols,
|
||||||
|
weekGridCols,
|
||||||
|
saveButtonClass,
|
||||||
|
formattedSelectedDate,
|
||||||
|
isSelectedDateHoliday,
|
||||||
|
weekDayHeaders,
|
||||||
|
shortcutButtonClass,
|
||||||
|
weekShortcutButtonClass,
|
||||||
|
getWeekShortcutLabel,
|
||||||
|
setToday,
|
||||||
|
setYesterday,
|
||||||
|
setTomorrow,
|
||||||
|
setThisWeek,
|
||||||
|
setPreviousWeek,
|
||||||
|
setNextWeek,
|
||||||
|
shiftDate,
|
||||||
|
contractLabel,
|
||||||
|
isRowLocked,
|
||||||
|
hasContractAtSelectedDate,
|
||||||
|
isValidationPending,
|
||||||
|
isSiteValidationPending,
|
||||||
|
canToggleValidation,
|
||||||
|
canToggleSiteValidation,
|
||||||
|
canCreateSiteValidationRowFromAbsence,
|
||||||
|
isBulkValidationChecked,
|
||||||
|
isBulkValidationIndeterminate,
|
||||||
|
isBulkSiteValidationChecked,
|
||||||
|
isBulkSiteValidationIndeterminate,
|
||||||
|
canBulkToggleSiteValidation,
|
||||||
|
toggleValidation,
|
||||||
|
toggleSiteValidation,
|
||||||
|
toggleValidationBulk,
|
||||||
|
toggleSiteValidationBulk,
|
||||||
|
getRowMetrics,
|
||||||
|
getRowAbsenceLabel,
|
||||||
|
getRowAbsenceStyle,
|
||||||
|
getRowUpdatedAt,
|
||||||
|
openAbsenceDrawer,
|
||||||
|
submitAbsence,
|
||||||
|
deleteAbsenceFromDrawer,
|
||||||
|
closeAbsenceDrawer,
|
||||||
|
formatMinutes,
|
||||||
|
handleSave
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -43,7 +43,8 @@ export const useEmployeeContract = (employee: Ref<Employee | null>, reloadEmploy
|
|||||||
contractId: '' as number | '',
|
contractId: '' as number | '',
|
||||||
contractNature: 'CDI' as 'CDI' | 'CDD' | 'INTERIM',
|
contractNature: 'CDI' as 'CDI' | 'CDD' | 'INTERIM',
|
||||||
startDate: '',
|
startDate: '',
|
||||||
endDate: ''
|
endDate: '',
|
||||||
|
isDriver: false
|
||||||
})
|
})
|
||||||
|
|
||||||
const createValidationTouched = reactive({
|
const createValidationTouched = reactive({
|
||||||
@@ -171,6 +172,7 @@ export const useEmployeeContract = (employee: Ref<Employee | null>, reloadEmploy
|
|||||||
createContractForm.contractId = ''
|
createContractForm.contractId = ''
|
||||||
createContractForm.contractNature = 'CDI'
|
createContractForm.contractNature = 'CDI'
|
||||||
createContractForm.endDate = ''
|
createContractForm.endDate = ''
|
||||||
|
createContractForm.isDriver = false
|
||||||
createContractForm.startDate = currentActiveContractPeriod.value?.endDate
|
createContractForm.startDate = currentActiveContractPeriod.value?.endDate
|
||||||
? (shiftYmd(currentActiveContractPeriod.value.endDate, 1) ?? currentActiveContractPeriod.value.endDate)
|
? (shiftYmd(currentActiveContractPeriod.value.endDate, 1) ?? currentActiveContractPeriod.value.endDate)
|
||||||
: getTodayYmd()
|
: getTodayYmd()
|
||||||
@@ -244,7 +246,8 @@ export const useEmployeeContract = (employee: Ref<Employee | null>, reloadEmploy
|
|||||||
contractId: Number(createContractForm.contractId),
|
contractId: Number(createContractForm.contractId),
|
||||||
contractNature: createContractForm.contractNature,
|
contractNature: createContractForm.contractNature,
|
||||||
contractStartDate: createContractForm.startDate,
|
contractStartDate: createContractForm.startDate,
|
||||||
contractEndDate: createContractForm.endDate || null
|
contractEndDate: createContractForm.endDate || null,
|
||||||
|
isDriverInput: createContractForm.isDriver
|
||||||
})
|
})
|
||||||
isCreateContractDrawerOpen.value = false
|
isCreateContractDrawerOpen.value = false
|
||||||
await reloadEmployee()
|
await reloadEmployee()
|
||||||
|
|||||||
@@ -6,7 +6,8 @@ import {
|
|||||||
createMileageAllowance,
|
createMileageAllowance,
|
||||||
updateMileageAllowance,
|
updateMileageAllowance,
|
||||||
deleteMileageAllowance,
|
deleteMileageAllowance,
|
||||||
uploadReceipt
|
uploadKmReceipt,
|
||||||
|
uploadAmountReceipt
|
||||||
} from '~/services/mileage-allowances'
|
} from '~/services/mileage-allowances'
|
||||||
|
|
||||||
export const useEmployeeMileage = (employee: Ref<Employee | null>, reloadEmployee: () => Promise<void>) => {
|
export const useEmployeeMileage = (employee: Ref<Employee | null>, reloadEmployee: () => Promise<void>) => {
|
||||||
@@ -32,24 +33,33 @@ export const useEmployeeMileage = (employee: Ref<Employee | null>, reloadEmploye
|
|||||||
mileageDataLoaded.value = false
|
mileageDataLoaded.value = false
|
||||||
}
|
}
|
||||||
|
|
||||||
const submitCreateMileage = async (data: { month: string; kilometers: number; comment?: string }, file?: File) => {
|
const submitCreateMileage = async (data: { month: string; kilometers: number; amount: number; comment?: string }, kmFile?: File, amountFile?: File) => {
|
||||||
if (!employee.value) return
|
if (!employee.value) return
|
||||||
const result = await createMileageAllowance({
|
const result = await createMileageAllowance({
|
||||||
employeeId: employee.value.id,
|
employeeId: employee.value.id,
|
||||||
month: data.month,
|
month: data.month,
|
||||||
kilometers: data.kilometers,
|
kilometers: data.kilometers,
|
||||||
|
amount: data.amount,
|
||||||
comment: data.comment
|
comment: data.comment
|
||||||
})
|
})
|
||||||
if (file && result?.id) {
|
if (result?.id) {
|
||||||
await uploadReceipt(apiBase, result.id, file)
|
if (kmFile) {
|
||||||
|
await uploadKmReceipt(apiBase, result.id, kmFile)
|
||||||
|
}
|
||||||
|
if (amountFile) {
|
||||||
|
await uploadAmountReceipt(apiBase, result.id, amountFile)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
await reloadEmployee()
|
await reloadEmployee()
|
||||||
}
|
}
|
||||||
|
|
||||||
const submitUpdateMileage = async (id: number, data: { month: string; kilometers: number; comment?: string }, file?: File) => {
|
const submitUpdateMileage = async (id: number, data: { month: string; kilometers: number; amount: number; comment?: string }, kmFile?: File, amountFile?: File) => {
|
||||||
await updateMileageAllowance(id, data)
|
await updateMileageAllowance(id, data)
|
||||||
if (file) {
|
if (kmFile) {
|
||||||
await uploadReceipt(apiBase, id, file)
|
await uploadKmReceipt(apiBase, id, kmFile)
|
||||||
|
}
|
||||||
|
if (amountFile) {
|
||||||
|
await uploadAmountReceipt(apiBase, id, amountFile)
|
||||||
}
|
}
|
||||||
await reloadEmployee()
|
await reloadEmployee()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ export const useHoursPage = () => {
|
|||||||
return `1.2fr 0.6fr repeat(6, 0.8fr) ${metricCol} ${metricCol} ${metricCol} ${validationCols}`
|
return `1.2fr 0.6fr repeat(6, 0.8fr) ${metricCol} ${metricCol} ${metricCol} ${validationCols}`
|
||||||
})
|
})
|
||||||
|
|
||||||
const weekGridCols = '1.6fr repeat(7, 1fr) repeat(6, 0.6fr)'
|
const weekGridCols = '1.6fr repeat(7, 1fr) repeat(6, 0.6fr) 0.3fr'
|
||||||
|
|
||||||
const sites = computed<Site[]>(() => {
|
const sites = computed<Site[]>(() => {
|
||||||
const siteMap = new Map<number, Site>()
|
const siteMap = new Map<number, Site>()
|
||||||
@@ -99,6 +99,7 @@ export const useHoursPage = () => {
|
|||||||
if (selectedSiteIds.value.length === 0) return []
|
if (selectedSiteIds.value.length === 0) return []
|
||||||
const filter = employeeFilter.value.trim().toLowerCase()
|
const filter = employeeFilter.value.trim().toLowerCase()
|
||||||
return employees.value.filter((employee) => {
|
return employees.value.filter((employee) => {
|
||||||
|
if (employee.isDriver === true) return false
|
||||||
const siteId = employee.site?.id
|
const siteId = employee.site?.id
|
||||||
if (!siteId || !selectedSiteIds.value.includes(siteId)) return false
|
if (!siteId || !selectedSiteIds.value.includes(siteId)) return false
|
||||||
if (!filter) return true
|
if (!filter) return true
|
||||||
@@ -108,13 +109,19 @@ export const useHoursPage = () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const displayedEmployees = computed(() => {
|
||||||
|
return visibleEmployees.value.filter((employee) => hasContractAtSelectedDate(employee.id))
|
||||||
|
})
|
||||||
|
|
||||||
const visibleEmployeeIdSet = computed(() => new Set(visibleEmployees.value.map((employee) => employee.id)))
|
const visibleEmployeeIdSet = computed(() => new Set(visibleEmployees.value.map((employee) => employee.id)))
|
||||||
|
|
||||||
const filteredWeeklySummary = computed<WeeklyWorkHourSummary | null>(() => {
|
const filteredWeeklySummary = computed<WeeklyWorkHourSummary | null>(() => {
|
||||||
if (!weeklySummary.value) return null
|
if (!weeklySummary.value) return null
|
||||||
return {
|
return {
|
||||||
...weeklySummary.value,
|
...weeklySummary.value,
|
||||||
rows: weeklySummary.value.rows.filter((row) => visibleEmployeeIdSet.value.has(row.employeeId))
|
rows: weeklySummary.value.rows.filter((row) =>
|
||||||
|
visibleEmployeeIdSet.value.has(row.employeeId) && row.hasContractForWeek !== false
|
||||||
|
)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -462,6 +469,9 @@ export const useHoursPage = () => {
|
|||||||
|
|
||||||
const getRowAbsenceStyle = (employeeId: number) => {
|
const getRowAbsenceStyle = (employeeId: number) => {
|
||||||
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
const dayRow = dayContextByEmployeeId.value.get(employeeId)
|
||||||
|
if (dayRow && dayRow.hasContractAtDate === false) {
|
||||||
|
return { backgroundColor: '#6b7280' }
|
||||||
|
}
|
||||||
if (!dayRow?.absenceLabel) return undefined
|
if (!dayRow?.absenceLabel) return undefined
|
||||||
return { backgroundColor: dayRow.absenceColor || '#dc2626' }
|
return { backgroundColor: dayRow.absenceColor || '#dc2626' }
|
||||||
}
|
}
|
||||||
@@ -1035,7 +1045,7 @@ export const useHoursPage = () => {
|
|||||||
isSubmitting.value = true
|
isSubmitting.value = true
|
||||||
try {
|
try {
|
||||||
const entries = employees.value
|
const entries = employees.value
|
||||||
.filter((employee) => hasContractAtSelectedDate(employee.id))
|
.filter((employee) => hasContractAtSelectedDate(employee.id) && !isRowLocked(employee.id))
|
||||||
.map((employee) => {
|
.map((employee) => {
|
||||||
const employeeId = employee.id
|
const employeeId = employee.id
|
||||||
const row = rows.value[employeeId] ?? emptyRow()
|
const row = rows.value[employeeId] ?? emptyRow()
|
||||||
@@ -1092,6 +1102,7 @@ export const useHoursPage = () => {
|
|||||||
selectedSiteIds,
|
selectedSiteIds,
|
||||||
employees,
|
employees,
|
||||||
visibleEmployees,
|
visibleEmployees,
|
||||||
|
displayedEmployees,
|
||||||
rows,
|
rows,
|
||||||
absenceTypes,
|
absenceTypes,
|
||||||
absenceForm,
|
absenceForm,
|
||||||
|
|||||||
@@ -21,13 +21,25 @@
|
|||||||
<NuxtLink
|
<NuxtLink
|
||||||
to="/hours"
|
to="/hours"
|
||||||
class="flex items-center gap-2 py-2 text-md text-black hover:bg-tertiary-500 hover:text-primary-500"
|
class="flex items-center gap-2 py-2 text-md text-black hover:bg-tertiary-500 hover:text-primary-500"
|
||||||
:class="route.path.startsWith('/hours')
|
:class="[
|
||||||
? 'bg-tertiary-500 text-primary-500 font-bold'
|
route.path.startsWith('/hours') ? 'bg-tertiary-500 text-primary-500 font-bold' : '',
|
||||||
: ''"
|
!isAdmin ? 'border-t border-secondary-500 pt-3' : ''
|
||||||
|
]"
|
||||||
>
|
>
|
||||||
<Icon name="mdi:clock-time-four-outline" size="24"/>
|
<Icon name="mdi:clock-time-four-outline" size="24"/>
|
||||||
<p>Heures</p>
|
<p>Heures</p>
|
||||||
</NuxtLink>
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
v-if="isAdmin"
|
||||||
|
to="/driver-hours"
|
||||||
|
class="flex items-center gap-2 py-2 text-md text-black hover:bg-tertiary-500 hover:text-primary-500"
|
||||||
|
:class="route.path.startsWith('/driver-hours')
|
||||||
|
? 'bg-tertiary-500 text-primary-500 font-bold'
|
||||||
|
: ''"
|
||||||
|
>
|
||||||
|
<Icon name="mdi:truck-outline" size="24"/>
|
||||||
|
<p>Heures Conducteurs</p>
|
||||||
|
</NuxtLink>
|
||||||
<template v-if="isAdmin">
|
<template v-if="isAdmin">
|
||||||
<NuxtLink
|
<NuxtLink
|
||||||
to="/employees"
|
to="/employees"
|
||||||
|
|||||||
183
frontend/pages/driver-hours.vue
Normal file
183
frontend/pages/driver-hours.vue
Normal file
@@ -0,0 +1,183 @@
|
|||||||
|
<template>
|
||||||
|
<div class="h-full overflow-hidden flex flex-col">
|
||||||
|
<div class="flex flex-wrap items-center justify-between gap-4">
|
||||||
|
<h1 class="text-4xl font-bold text-primary-500">Heures Conducteurs</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<HoursToolbar
|
||||||
|
v-model:selected-date="selectedDate"
|
||||||
|
v-model:view-mode="viewMode"
|
||||||
|
v-model:selected-site-ids="selectedSiteIds"
|
||||||
|
v-model:employee-filter="employeeFilter"
|
||||||
|
:is-admin="isAdmin"
|
||||||
|
:sites="sites"
|
||||||
|
:absence-types="absenceTypes"
|
||||||
|
:formatted-selected-date="formattedSelectedDate"
|
||||||
|
:shortcut-button-class="shortcutButtonClass"
|
||||||
|
:week-shortcut-button-class="weekShortcutButtonClass"
|
||||||
|
:get-week-shortcut-label="getWeekShortcutLabel"
|
||||||
|
@set-yesterday="setYesterday"
|
||||||
|
@set-today="setToday"
|
||||||
|
@set-tomorrow="setTomorrow"
|
||||||
|
@set-previous-week="setPreviousWeek"
|
||||||
|
@set-this-week="setThisWeek"
|
||||||
|
@set-next-week="setNextWeek"
|
||||||
|
@shift-date="shiftDate"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div v-if="isLoading" class="rounded-lg border border-neutral-200 bg-white p-6 text-md text-neutral-600">
|
||||||
|
Chargement...
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else-if="visibleEmployees.length === 0" class="rounded-lg border border-neutral-200 bg-white p-6 text-md text-neutral-600">
|
||||||
|
Aucun conducteur accessible.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="flex min-h-0 flex-col gap-4">
|
||||||
|
<div class="min-h-0 flex flex-col max-h-[calc(100vh-300px)]">
|
||||||
|
<DriverHoursDayView
|
||||||
|
v-if="viewMode === 'day'"
|
||||||
|
v-model:rows="rows"
|
||||||
|
:employees="displayedEmployees"
|
||||||
|
:is-admin="isAdmin"
|
||||||
|
:is-site-manager="isSiteManager"
|
||||||
|
:day-grid-cols="dayGridCols"
|
||||||
|
:is-holiday="isSelectedDateHoliday"
|
||||||
|
:contract-label="contractLabel"
|
||||||
|
:is-row-locked="isRowLocked"
|
||||||
|
:has-contract-at-selected-date="hasContractAtSelectedDate"
|
||||||
|
:is-validation-pending="isValidationPending"
|
||||||
|
:is-site-validation-pending="isSiteValidationPending"
|
||||||
|
:can-toggle-validation="canToggleValidation"
|
||||||
|
:can-toggle-site-validation="canToggleSiteValidation"
|
||||||
|
:can-create-site-validation-row-from-absence="canCreateSiteValidationRowFromAbsence"
|
||||||
|
:is-bulk-validation-checked="isBulkValidationChecked"
|
||||||
|
:is-bulk-validation-indeterminate="isBulkValidationIndeterminate"
|
||||||
|
:is-bulk-site-validation-checked="isBulkSiteValidationChecked"
|
||||||
|
:is-bulk-site-validation-indeterminate="isBulkSiteValidationIndeterminate"
|
||||||
|
:can-bulk-toggle-site-validation="canBulkToggleSiteValidation"
|
||||||
|
:on-toggle-validation="toggleValidation"
|
||||||
|
:on-toggle-site-validation="toggleSiteValidation"
|
||||||
|
:on-toggle-validation-bulk="toggleValidationBulk"
|
||||||
|
:on-toggle-site-validation-bulk="toggleSiteValidationBulk"
|
||||||
|
:get-row-metrics="getRowMetrics"
|
||||||
|
:get-row-absence-label="getRowAbsenceLabel"
|
||||||
|
:get-row-absence-style="getRowAbsenceStyle"
|
||||||
|
:get-row-updated-at="getRowUpdatedAt"
|
||||||
|
:on-absence-click="openAbsenceDrawer"
|
||||||
|
:format-minutes="formatMinutes"
|
||||||
|
class="max-h-[calc(100vh-300px)]"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<DriverHoursWeekView
|
||||||
|
v-else-if="isAdmin && viewMode === 'week'"
|
||||||
|
:is-week-loading="isWeekLoading"
|
||||||
|
:week-grid-cols="weekGridCols"
|
||||||
|
:weekly-summary="filteredWeeklySummary"
|
||||||
|
:week-day-headers="weekDayHeaders"
|
||||||
|
:format-minutes="formatMinutes"
|
||||||
|
class="max-h-[calc(100vh-300px)]"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="viewMode === 'day'" class="shrink-0 px-4 pt-4 flex justify-center">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded-lg bg-primary-500 px-6 py-2 text-md font-semibold text-white hover:bg-secondary-500"
|
||||||
|
:class="saveButtonClass"
|
||||||
|
:disabled="isSubmitting || visibleEmployees.length === 0"
|
||||||
|
@click="handleSave"
|
||||||
|
>
|
||||||
|
Enregistrer
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<AbsenceFormDrawer
|
||||||
|
v-model="isAbsenceDrawerOpen"
|
||||||
|
:employees="employees"
|
||||||
|
:absence-types="absenceTypes"
|
||||||
|
:form="absenceForm"
|
||||||
|
:editing-absence="editingAbsence"
|
||||||
|
:is-submitting="isAbsenceSubmitting"
|
||||||
|
:lock-employee="true"
|
||||||
|
:lock-dates="true"
|
||||||
|
:show-comment="false"
|
||||||
|
@submit="submitAbsence"
|
||||||
|
@delete="deleteAbsenceFromDrawer"
|
||||||
|
@cancel="closeAbsenceDrawer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const {
|
||||||
|
isAdmin,
|
||||||
|
isSiteManager,
|
||||||
|
viewMode,
|
||||||
|
selectedDate,
|
||||||
|
employeeFilter,
|
||||||
|
sites,
|
||||||
|
selectedSiteIds,
|
||||||
|
employees,
|
||||||
|
visibleEmployees,
|
||||||
|
displayedEmployees,
|
||||||
|
rows,
|
||||||
|
absenceTypes,
|
||||||
|
absenceForm,
|
||||||
|
isAbsenceDrawerOpen,
|
||||||
|
isAbsenceSubmitting,
|
||||||
|
editingAbsence,
|
||||||
|
filteredWeeklySummary,
|
||||||
|
isLoading,
|
||||||
|
isWeekLoading,
|
||||||
|
isSubmitting,
|
||||||
|
dayGridCols,
|
||||||
|
weekGridCols,
|
||||||
|
saveButtonClass,
|
||||||
|
formattedSelectedDate,
|
||||||
|
weekDayHeaders,
|
||||||
|
shortcutButtonClass,
|
||||||
|
weekShortcutButtonClass,
|
||||||
|
getWeekShortcutLabel,
|
||||||
|
setToday,
|
||||||
|
setYesterday,
|
||||||
|
setTomorrow,
|
||||||
|
setThisWeek,
|
||||||
|
setPreviousWeek,
|
||||||
|
setNextWeek,
|
||||||
|
shiftDate,
|
||||||
|
contractLabel,
|
||||||
|
isRowLocked,
|
||||||
|
hasContractAtSelectedDate,
|
||||||
|
isValidationPending,
|
||||||
|
isSiteValidationPending,
|
||||||
|
canToggleValidation,
|
||||||
|
canToggleSiteValidation,
|
||||||
|
canCreateSiteValidationRowFromAbsence,
|
||||||
|
isBulkValidationChecked,
|
||||||
|
isBulkValidationIndeterminate,
|
||||||
|
isBulkSiteValidationChecked,
|
||||||
|
isBulkSiteValidationIndeterminate,
|
||||||
|
canBulkToggleSiteValidation,
|
||||||
|
toggleValidation,
|
||||||
|
toggleSiteValidation,
|
||||||
|
toggleValidationBulk,
|
||||||
|
toggleSiteValidationBulk,
|
||||||
|
getRowMetrics,
|
||||||
|
getRowAbsenceLabel,
|
||||||
|
getRowAbsenceStyle,
|
||||||
|
getRowUpdatedAt,
|
||||||
|
openAbsenceDrawer,
|
||||||
|
submitAbsence,
|
||||||
|
deleteAbsenceFromDrawer,
|
||||||
|
closeAbsenceDrawer,
|
||||||
|
formatMinutes,
|
||||||
|
isSelectedDateHoliday,
|
||||||
|
handleSave
|
||||||
|
} = useDriverHoursPage()
|
||||||
|
|
||||||
|
useHead({
|
||||||
|
title: 'Heures Conducteurs'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -62,8 +62,8 @@
|
|||||||
: 'border-transparent text-primary-500/50 hover:text-primary-500'"
|
: 'border-transparent text-primary-500/50 hover:text-primary-500'"
|
||||||
@click="activeTab = 'mileage'"
|
@click="activeTab = 'mileage'"
|
||||||
>
|
>
|
||||||
<Icon name="mdi:car-outline" size="24" class="align-self"/>
|
<Icon name="mdi:account-cash-outline" size="24" class="align-self"/>
|
||||||
Frais Kms
|
Frais
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
class="pb-2 border-b-2 flex items-center gap-3"
|
class="pb-2 border-b-2 flex items-center gap-3"
|
||||||
|
|||||||
@@ -3,19 +3,43 @@
|
|||||||
<div class="shrink-0">
|
<div class="shrink-0">
|
||||||
<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>
|
||||||
<button
|
<div class="flex items-center gap-3">
|
||||||
type="button"
|
<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"
|
type="button"
|
||||||
@click="openCreate"
|
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"
|
||||||
+ Ajouter un employé
|
>
|
||||||
</button>
|
Export récap. congés
|
||||||
|
</button>
|
||||||
|
<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="isSalaryRecapOpen = true"
|
||||||
|
>
|
||||||
|
Export récap. salaire
|
||||||
|
</button>
|
||||||
|
<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="openCreate"
|
||||||
|
>
|
||||||
|
+ Ajouter un employé
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex gap-10 py-7">
|
<div class="flex gap-3 py-7">
|
||||||
<div class="w-80">
|
<div class="w-80">
|
||||||
<EmployeeNameFilterInput v-model="employeeFilter"/>
|
<EmployeeNameFilterInput v-model="employeeFilter"/>
|
||||||
</div>
|
</div>
|
||||||
<SiteFilterSelector v-if="sites.length > 0" v-model="selectedSiteIds" :sites="sites"/>
|
<SiteFilterSelector v-if="sites.length > 0" v-model="selectedSiteIds" :sites="sites"/>
|
||||||
|
<select
|
||||||
|
v-model="contractStatusFilter"
|
||||||
|
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="inactive">Sans contrat</option>
|
||||||
|
<option value="all">Tous</option>
|
||||||
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -40,7 +64,7 @@
|
|||||||
<div class="text-center text-[20px]">
|
<div class="text-center text-[20px]">
|
||||||
<p class="text-primary-500 font-bold">{{ employee.firstName }} {{ employee.lastName }}</p>
|
<p class="text-primary-500 font-bold">{{ employee.firstName }} {{ employee.lastName }}</p>
|
||||||
<p>Nom du poste occupé</p>
|
<p>Nom du poste occupé</p>
|
||||||
<p>Site ({{ employee.site?.name ?? '-' }})</p>
|
<p>{{ employee.site?.name ?? '-' }}</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -170,6 +194,17 @@
|
|||||||
La date de fin est obligatoire pour un CDD.
|
La date de fin est obligatoire pour un CDD.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="rounded-md border border-neutral-200 bg-neutral-50 p-3">
|
||||||
|
<label class="inline-flex items-center gap-2 text-md font-semibold text-neutral-700" for="is-driver">
|
||||||
|
<input
|
||||||
|
id="is-driver"
|
||||||
|
v-model="form.isDriver"
|
||||||
|
type="checkbox"
|
||||||
|
class="h-4 w-4 rounded border-neutral-300 text-primary-500 focus:ring-primary-500"
|
||||||
|
/>
|
||||||
|
Chauffeur
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<div class="flex justify-end gap-3 pt-2">
|
<div class="flex justify-end gap-3 pt-2">
|
||||||
<button
|
<button
|
||||||
@@ -189,6 +224,11 @@
|
|||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</AppDrawer>
|
</AppDrawer>
|
||||||
|
|
||||||
|
<SalaryRecapDrawer
|
||||||
|
v-model="isSalaryRecapOpen"
|
||||||
|
@submit="handleSalaryRecapPrint"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -200,7 +240,9 @@ import {listContracts} from '~/services/contracts'
|
|||||||
import {createEmployee, deleteEmployee, listEmployees, updateEmployee} from '~/services/employees'
|
import {createEmployee, deleteEmployee, listEmployees, updateEmployee} from '~/services/employees'
|
||||||
import {listSites} from '~/services/sites'
|
import {listSites} from '~/services/sites'
|
||||||
import SiteFilterSelector from '~/components/SiteFilterSelector.vue'
|
import SiteFilterSelector from '~/components/SiteFilterSelector.vue'
|
||||||
|
import SalaryRecapDrawer from '~/components/SalaryRecapDrawer.vue'
|
||||||
import {contractNatureLabel, isContractNature, requiresContractEndDate, showsContractEndDate} from '~/utils/contract'
|
import {contractNatureLabel, isContractNature, requiresContractEndDate, showsContractEndDate} from '~/utils/contract'
|
||||||
|
import {usePdfPrinter} from '~/composables/usePdfPrinter'
|
||||||
|
|
||||||
useHead({
|
useHead({
|
||||||
title: 'Employés'
|
title: 'Employés'
|
||||||
@@ -209,6 +251,8 @@ useHead({
|
|||||||
const isDrawerOpen = ref(false)
|
const isDrawerOpen = ref(false)
|
||||||
const isSubmitting = ref(false)
|
const isSubmitting = ref(false)
|
||||||
const isLoading = ref(false)
|
const isLoading = ref(false)
|
||||||
|
const isSalaryRecapOpen = ref(false)
|
||||||
|
const { printPdf } = usePdfPrinter()
|
||||||
const sitesInitialized = ref(false)
|
const sitesInitialized = ref(false)
|
||||||
const editingEmployee = ref<Employee | null>(null)
|
const editingEmployee = ref<Employee | null>(null)
|
||||||
const drawerTitle = computed(() =>
|
const drawerTitle = computed(() =>
|
||||||
@@ -219,20 +263,21 @@ const employees = ref<Employee[]>([])
|
|||||||
const sites = ref<Site[]>([])
|
const sites = ref<Site[]>([])
|
||||||
const contracts = ref<Contract[]>([])
|
const contracts = ref<Contract[]>([])
|
||||||
const employeeFilter = ref('')
|
const employeeFilter = ref('')
|
||||||
|
const contractStatusFilter = ref<'active' | 'inactive' | 'all'>('active')
|
||||||
const selectedSiteIds = ref<number[]>([])
|
const selectedSiteIds = ref<number[]>([])
|
||||||
|
|
||||||
const filteredEmployees = computed<Employee[]>(() => {
|
const filteredEmployees = computed<Employee[]>(() => {
|
||||||
if (selectedSiteIds.value.length === 0) return []
|
if (selectedSiteIds.value.length === 0) return []
|
||||||
|
|
||||||
const filter = employeeFilter.value.trim().toLowerCase()
|
const filter = employeeFilter.value.trim().toLowerCase()
|
||||||
const bySite = employees.value.filter((employee) => {
|
return employees.value.filter((employee) => {
|
||||||
const siteId = employee.site?.id
|
const siteId = employee.site?.id
|
||||||
return !!siteId && selectedSiteIds.value.includes(siteId)
|
if (!siteId || !selectedSiteIds.value.includes(siteId)) return false
|
||||||
})
|
|
||||||
|
|
||||||
if (!filter) return bySite
|
if (contractStatusFilter.value === 'active' && !employee.hasActiveContract) return false
|
||||||
|
if (contractStatusFilter.value === 'inactive' && employee.hasActiveContract) return false
|
||||||
|
|
||||||
return bySite.filter((employee) => {
|
if (!filter) return true
|
||||||
const firstName = employee.firstName?.toLowerCase() ?? ''
|
const firstName = employee.firstName?.toLowerCase() ?? ''
|
||||||
const lastName = employee.lastName?.toLowerCase() ?? ''
|
const lastName = employee.lastName?.toLowerCase() ?? ''
|
||||||
return firstName.includes(filter) || lastName.includes(filter)
|
return firstName.includes(filter) || lastName.includes(filter)
|
||||||
@@ -246,7 +291,8 @@ const form = reactive({
|
|||||||
contractId: '' as number | '',
|
contractId: '' as number | '',
|
||||||
contractNature: 'CDI' as 'CDI' | 'CDD' | 'INTERIM',
|
contractNature: 'CDI' as 'CDI' | 'CDD' | 'INTERIM',
|
||||||
contractStartDate: '',
|
contractStartDate: '',
|
||||||
contractEndDate: ''
|
contractEndDate: '',
|
||||||
|
isDriver: false
|
||||||
})
|
})
|
||||||
|
|
||||||
const validationTouched = reactive({
|
const validationTouched = reactive({
|
||||||
@@ -431,7 +477,8 @@ const handleSubmit = async () => {
|
|||||||
contractId: Number(form.contractId),
|
contractId: Number(form.contractId),
|
||||||
contractNature: form.contractNature,
|
contractNature: form.contractNature,
|
||||||
contractStartDate: form.contractStartDate,
|
contractStartDate: form.contractStartDate,
|
||||||
contractEndDate: form.contractEndDate || null
|
contractEndDate: form.contractEndDate || null,
|
||||||
|
isDriverInput: form.isDriver
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -442,6 +489,7 @@ const handleSubmit = async () => {
|
|||||||
form.contractNature = 'CDI'
|
form.contractNature = 'CDI'
|
||||||
form.contractStartDate = new Date().toISOString().slice(0, 10)
|
form.contractStartDate = new Date().toISOString().slice(0, 10)
|
||||||
form.contractEndDate = ''
|
form.contractEndDate = ''
|
||||||
|
form.isDriver = false
|
||||||
editingEmployee.value = null
|
editingEmployee.value = null
|
||||||
isDrawerOpen.value = false
|
isDrawerOpen.value = false
|
||||||
await loadEmployees()
|
await loadEmployees()
|
||||||
@@ -485,9 +533,19 @@ const openCreate = () => {
|
|||||||
form.contractNature = 'CDI'
|
form.contractNature = 'CDI'
|
||||||
form.contractStartDate = new Date().toISOString().slice(0, 10)
|
form.contractStartDate = new Date().toISOString().slice(0, 10)
|
||||||
form.contractEndDate = ''
|
form.contractEndDate = ''
|
||||||
|
form.isDriver = false
|
||||||
isDrawerOpen.value = true
|
isDrawerOpen.value = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const handleLeaveRecapPrint = async () => {
|
||||||
|
await printPdf('/leave-recap/print')
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleSalaryRecapPrint = async (month: string) => {
|
||||||
|
await printPdf(`/salary-recap/print?month=${month}`)
|
||||||
|
isSalaryRecapOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
const confirmDelete = async (employee: Employee) => {
|
const confirmDelete = async (employee: Employee) => {
|
||||||
const ok = window.confirm(`Supprimer ${employee.firstName} ${employee.lastName} ?`)
|
const ok = window.confirm(`Supprimer ${employee.firstName} ${employee.lastName} ?`)
|
||||||
if (!ok) return
|
if (!ok) return
|
||||||
|
|||||||
@@ -38,7 +38,7 @@
|
|||||||
<HoursDayView
|
<HoursDayView
|
||||||
v-if="viewMode === 'day'"
|
v-if="viewMode === 'day'"
|
||||||
v-model:rows="rows"
|
v-model:rows="rows"
|
||||||
:employees="visibleEmployees"
|
:employees="displayedEmployees"
|
||||||
:is-admin="isAdmin"
|
:is-admin="isAdmin"
|
||||||
:is-site-manager="isSiteManager"
|
:is-site-manager="isSiteManager"
|
||||||
:day-grid-cols="dayGridCols"
|
:day-grid-cols="dayGridCols"
|
||||||
@@ -126,6 +126,7 @@ const {
|
|||||||
selectedSiteIds,
|
selectedSiteIds,
|
||||||
employees,
|
employees,
|
||||||
visibleEmployees,
|
visibleEmployees,
|
||||||
|
displayedEmployees,
|
||||||
rows,
|
rows,
|
||||||
absenceTypes,
|
absenceTypes,
|
||||||
absenceForm,
|
absenceForm,
|
||||||
|
|||||||
@@ -68,7 +68,8 @@ const handleSubmit = async () => {
|
|||||||
try {
|
try {
|
||||||
await auth.login(username.value, password.value)
|
await auth.login(username.value, password.value)
|
||||||
|
|
||||||
await router.push('/calendar')
|
const isAdmin = auth.user?.roles?.includes('ROLE_ADMIN')
|
||||||
|
await router.push(isAdmin ? '/calendar' : '/hours')
|
||||||
} finally {
|
} finally {
|
||||||
isSubmitting.value = false
|
isSubmitting.value = false
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,4 +32,5 @@ export type EmployeeRttSummary = {
|
|||||||
availableMinutes: number
|
availableMinutes: number
|
||||||
weeks: EmployeeRttWeekSummary[]
|
weeks: EmployeeRttWeekSummary[]
|
||||||
monthPayments: RttMonthPayment[]
|
monthPayments: RttMonthPayment[]
|
||||||
|
rttStartDate: string | null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ export type ContractHistoryItem = {
|
|||||||
comment?: string | null
|
comment?: string | null
|
||||||
periodId?: number | null
|
periodId?: number | null
|
||||||
suspensions?: ContractSuspension[]
|
suspensions?: ContractSuspension[]
|
||||||
|
isDriver?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Employee = {
|
export type Employee = {
|
||||||
@@ -26,6 +27,8 @@ export type Employee = {
|
|||||||
lastName: string
|
lastName: string
|
||||||
site: Site
|
site: Site
|
||||||
contract?: Contract | null
|
contract?: Contract | null
|
||||||
|
hasActiveContract?: boolean
|
||||||
|
isDriver?: boolean
|
||||||
currentContractNature?: 'CDI' | 'CDD' | 'INTERIM'
|
currentContractNature?: 'CDI' | 'CDD' | 'INTERIM'
|
||||||
currentContractStartDate?: string | null
|
currentContractStartDate?: string | null
|
||||||
currentContractEndDate?: string | null
|
currentContractEndDate?: string | null
|
||||||
|
|||||||
@@ -2,8 +2,11 @@ export type MileageAllowance = {
|
|||||||
id: number
|
id: number
|
||||||
month: string
|
month: string
|
||||||
kilometers: number
|
kilometers: number
|
||||||
|
amount: number
|
||||||
comment: string | null
|
comment: string | null
|
||||||
receiptPath: string | null
|
receiptPath: string | null
|
||||||
receiptName: string | null
|
receiptName: string | null
|
||||||
|
amountReceiptPath: string | null
|
||||||
|
amountReceiptName: string | null
|
||||||
createdAt: string
|
createdAt: string
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,13 @@ export type WorkHour = {
|
|||||||
eveningTo?: string | null
|
eveningTo?: string | null
|
||||||
isPresentMorning?: boolean
|
isPresentMorning?: boolean
|
||||||
isPresentAfternoon?: boolean
|
isPresentAfternoon?: boolean
|
||||||
|
dayHoursMinutes?: number | null
|
||||||
|
nightHoursMinutes?: number | null
|
||||||
|
workshopHoursMinutes?: number | null
|
||||||
|
hasBreakfast?: boolean
|
||||||
|
hasLunch?: boolean
|
||||||
|
hasDinner?: boolean
|
||||||
|
hasOvernight?: boolean
|
||||||
isSiteValid?: boolean
|
isSiteValid?: boolean
|
||||||
isValid?: boolean
|
isValid?: boolean
|
||||||
updatedAt?: string | null
|
updatedAt?: string | null
|
||||||
@@ -28,17 +35,30 @@ export type WorkHourEntryPayload = {
|
|||||||
eveningTo?: string | null
|
eveningTo?: string | null
|
||||||
isPresentMorning?: boolean
|
isPresentMorning?: boolean
|
||||||
isPresentAfternoon?: boolean
|
isPresentAfternoon?: boolean
|
||||||
|
dayHoursMinutes?: number | null
|
||||||
|
nightHoursMinutes?: number | null
|
||||||
|
workshopHoursMinutes?: number | null
|
||||||
|
hasBreakfast?: boolean
|
||||||
|
hasLunch?: boolean
|
||||||
|
hasDinner?: boolean
|
||||||
|
hasOvernight?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export type WeeklyWorkHourDailySummary = {
|
export type WeeklyWorkHourDailySummary = {
|
||||||
date: string
|
date: string
|
||||||
dayMinutes: number
|
dayMinutes: number
|
||||||
nightMinutes: number
|
nightMinutes: number
|
||||||
|
workshopMinutes?: number
|
||||||
totalMinutes: number
|
totalMinutes: number
|
||||||
present?: number | null
|
present?: number | null
|
||||||
hasAbsence?: boolean
|
hasAbsence?: boolean
|
||||||
absenceLabel?: string | null
|
absenceLabel?: string | null
|
||||||
absenceColor?: string | null
|
absenceColor?: string | null
|
||||||
|
hasNightBasket?: boolean
|
||||||
|
hasBreakfast?: boolean
|
||||||
|
hasLunch?: boolean
|
||||||
|
hasDinner?: boolean
|
||||||
|
hasOvernight?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export type WeeklyWorkHourRowSummary = {
|
export type WeeklyWorkHourRowSummary = {
|
||||||
@@ -52,12 +72,20 @@ export type WeeklyWorkHourRowSummary = {
|
|||||||
daily: WeeklyWorkHourDailySummary[]
|
daily: WeeklyWorkHourDailySummary[]
|
||||||
weeklyDayMinutes: number
|
weeklyDayMinutes: number
|
||||||
weeklyNightMinutes: number
|
weeklyNightMinutes: number
|
||||||
|
weeklyWorkshopMinutes?: number
|
||||||
weeklyTotalMinutes: number
|
weeklyTotalMinutes: number
|
||||||
weeklyPresenceCount?: number
|
weeklyPresenceCount?: number
|
||||||
weeklyOvertimeTotalMinutes?: number
|
weeklyOvertimeTotalMinutes?: number
|
||||||
weeklyOvertime25Minutes?: number
|
weeklyOvertime25Minutes?: number
|
||||||
weeklyOvertime50Minutes?: number
|
weeklyOvertime50Minutes?: number
|
||||||
weeklyRecoveryMinutes?: number
|
weeklyRecoveryMinutes?: number
|
||||||
|
weeklyNightBasketCount?: number
|
||||||
|
isDriver?: boolean
|
||||||
|
weeklyBreakfastCount?: number
|
||||||
|
weeklyLunchCount?: number
|
||||||
|
weeklyDinnerCount?: number
|
||||||
|
weeklyOvernightCount?: number
|
||||||
|
hasContractForWeek?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export type WeeklyWorkHourSummary = {
|
export type WeeklyWorkHourSummary = {
|
||||||
@@ -77,9 +105,24 @@ export type WorkHourDayContextRow = {
|
|||||||
absentAfternoon: boolean
|
absentAfternoon: boolean
|
||||||
creditedMinutes: number
|
creditedMinutes: number
|
||||||
creditedPresenceUnits: number
|
creditedPresenceUnits: number
|
||||||
|
isDriverContract?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export type WorkHourDayContext = {
|
export type WorkHourDayContext = {
|
||||||
workDate: string
|
workDate: string
|
||||||
rows: WorkHourDayContextRow[]
|
rows: WorkHourDayContextRow[]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type DriverHourRow = {
|
||||||
|
workHourId: number | null
|
||||||
|
dayHours: string
|
||||||
|
nightHours: string
|
||||||
|
workshopHours: string
|
||||||
|
hasBreakfast: boolean
|
||||||
|
hasLunch: boolean
|
||||||
|
hasDinner: boolean
|
||||||
|
hasOvernight: boolean
|
||||||
|
isSiteValid: boolean
|
||||||
|
isValid: boolean
|
||||||
|
updatedAt: string | null
|
||||||
|
}
|
||||||
|
|||||||
@@ -34,6 +34,7 @@ export const createEmployee = async (payload: {
|
|||||||
contractNature?: 'CDI' | 'CDD' | 'INTERIM'
|
contractNature?: 'CDI' | 'CDD' | 'INTERIM'
|
||||||
contractStartDate?: string
|
contractStartDate?: string
|
||||||
contractEndDate?: string | null
|
contractEndDate?: string | null
|
||||||
|
isDriverInput?: boolean
|
||||||
}) => {
|
}) => {
|
||||||
const api = useApi()
|
const api = useApi()
|
||||||
return api.post<Employee>('/employees', {
|
return api.post<Employee>('/employees', {
|
||||||
@@ -43,7 +44,8 @@ export const createEmployee = async (payload: {
|
|||||||
contract: `/api/contracts/${payload.contractId}`,
|
contract: `/api/contracts/${payload.contractId}`,
|
||||||
contractNature: payload.contractNature,
|
contractNature: payload.contractNature,
|
||||||
contractStartDate: payload.contractStartDate,
|
contractStartDate: payload.contractStartDate,
|
||||||
contractEndDate: payload.contractEndDate ?? null
|
contractEndDate: payload.contractEndDate ?? null,
|
||||||
|
isDriverInput: payload.isDriverInput ?? false
|
||||||
}, {
|
}, {
|
||||||
toastSuccessKey: 'success.employee.create',
|
toastSuccessKey: 'success.employee.create',
|
||||||
toastErrorKey: 'errors.employee.create'
|
toastErrorKey: 'errors.employee.create'
|
||||||
@@ -63,6 +65,7 @@ export const updateEmployee = async (
|
|||||||
contractPaidLeaveSettled?: boolean
|
contractPaidLeaveSettled?: boolean
|
||||||
contractComment?: string | null
|
contractComment?: string | null
|
||||||
displayOrder?: number
|
displayOrder?: number
|
||||||
|
isDriverInput?: boolean
|
||||||
}
|
}
|
||||||
) => {
|
) => {
|
||||||
const api = useApi()
|
const api = useApi()
|
||||||
@@ -91,6 +94,9 @@ export const updateEmployee = async (
|
|||||||
if (payload.contractComment !== undefined) {
|
if (payload.contractComment !== undefined) {
|
||||||
body.contractComment = payload.contractComment ?? null
|
body.contractComment = payload.contractComment ?? null
|
||||||
}
|
}
|
||||||
|
if (payload.isDriverInput !== undefined) {
|
||||||
|
body.isDriverInput = payload.isDriverInput
|
||||||
|
}
|
||||||
|
|
||||||
return api.patch<Employee>(`/employees/${id}`, body, {
|
return api.patch<Employee>(`/employees/${id}`, body, {
|
||||||
toastSuccessKey: 'success.employee.update',
|
toastSuccessKey: 'success.employee.update',
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ export const createMileageAllowance = async (data: {
|
|||||||
employeeId: number
|
employeeId: number
|
||||||
month: string
|
month: string
|
||||||
kilometers: number
|
kilometers: number
|
||||||
|
amount: number
|
||||||
comment?: string
|
comment?: string
|
||||||
}) => {
|
}) => {
|
||||||
const api = useApi()
|
const api = useApi()
|
||||||
@@ -23,6 +24,7 @@ export const createMileageAllowance = async (data: {
|
|||||||
employee: `/api/employees/${data.employeeId}`,
|
employee: `/api/employees/${data.employeeId}`,
|
||||||
month: data.month,
|
month: data.month,
|
||||||
kilometers: data.kilometers,
|
kilometers: data.kilometers,
|
||||||
|
amount: data.amount,
|
||||||
comment: data.comment
|
comment: data.comment
|
||||||
}, {
|
}, {
|
||||||
toastSuccessKey: 'success.mileage.create',
|
toastSuccessKey: 'success.mileage.create',
|
||||||
@@ -33,12 +35,14 @@ export const createMileageAllowance = async (data: {
|
|||||||
export const updateMileageAllowance = async (id: number, data: {
|
export const updateMileageAllowance = async (id: number, data: {
|
||||||
month: string
|
month: string
|
||||||
kilometers: number
|
kilometers: number
|
||||||
|
amount: number
|
||||||
comment?: string
|
comment?: string
|
||||||
}) => {
|
}) => {
|
||||||
const api = useApi()
|
const api = useApi()
|
||||||
return api.patch<MileageAllowance>(`/mileage_allowances/${id}`, {
|
return api.patch<MileageAllowance>(`/mileage_allowances/${id}`, {
|
||||||
month: data.month,
|
month: data.month,
|
||||||
kilometers: data.kilometers,
|
kilometers: data.kilometers,
|
||||||
|
amount: data.amount,
|
||||||
comment: data.comment
|
comment: data.comment
|
||||||
}, {
|
}, {
|
||||||
toastSuccessKey: 'success.mileage.update',
|
toastSuccessKey: 'success.mileage.update',
|
||||||
@@ -54,7 +58,7 @@ export const deleteMileageAllowance = async (id: number) => {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const uploadReceipt = async (baseURL: string, id: number, file: File) => {
|
export const uploadKmReceipt = async (baseURL: string, id: number, file: File) => {
|
||||||
const formData = new FormData()
|
const formData = new FormData()
|
||||||
formData.append('file', file)
|
formData.append('file', file)
|
||||||
return $fetch(`${baseURL}/mileage_allowances/${id}/receipt`, {
|
return $fetch(`${baseURL}/mileage_allowances/${id}/receipt`, {
|
||||||
@@ -64,6 +68,20 @@ export const uploadReceipt = async (baseURL: string, id: number, file: File) =>
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getReceiptUrl = (baseURL: string, id: number): string => {
|
export const uploadAmountReceipt = async (baseURL: string, id: number, file: File) => {
|
||||||
|
const formData = new FormData()
|
||||||
|
formData.append('file', file)
|
||||||
|
return $fetch(`${baseURL}/mileage_allowances/${id}/amount-receipt`, {
|
||||||
|
method: 'POST',
|
||||||
|
body: formData,
|
||||||
|
credentials: 'include'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getKmReceiptUrl = (baseURL: string, id: number): string => {
|
||||||
return `${baseURL}/mileage_allowances/${id}/receipt`
|
return `${baseURL}/mileage_allowances/${id}/receipt`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const getAmountReceiptUrl = (baseURL: string, id: number): string => {
|
||||||
|
return `${baseURL}/mileage_allowances/${id}/amount-receipt`
|
||||||
|
}
|
||||||
|
|||||||
26
migrations/Version20260315100000.php
Normal file
26
migrations/Version20260315100000.php
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260315100000 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add is_driver flag to employee_contract_periods';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE employee_contract_periods ADD is_driver BOOLEAN DEFAULT FALSE NOT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE employee_contract_periods DROP COLUMN is_driver');
|
||||||
|
}
|
||||||
|
}
|
||||||
34
migrations/Version20260315100100.php
Normal file
34
migrations/Version20260315100100.php
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260315100100 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add driver-specific fields to work_hours';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD day_hours_minutes INTEGER DEFAULT NULL');
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD night_hours_minutes INTEGER DEFAULT NULL');
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD has_breakfast BOOLEAN DEFAULT FALSE NOT NULL');
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD has_lunch BOOLEAN DEFAULT FALSE NOT NULL');
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD has_overnight BOOLEAN DEFAULT FALSE NOT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN day_hours_minutes');
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN night_hours_minutes');
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN has_breakfast');
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN has_lunch');
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN has_overnight');
|
||||||
|
}
|
||||||
|
}
|
||||||
26
migrations/Version20260316100000.php
Normal file
26
migrations/Version20260316100000.php
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260316100000 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add has_dinner column to work_hours';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD has_dinner BOOLEAN DEFAULT FALSE NOT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN has_dinner');
|
||||||
|
}
|
||||||
|
}
|
||||||
26
migrations/Version20260316100100.php
Normal file
26
migrations/Version20260316100100.php
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260316100100 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add workshop_hours_minutes column to work_hours';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours ADD workshop_hours_minutes INTEGER DEFAULT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE work_hours DROP COLUMN workshop_hours_minutes');
|
||||||
|
}
|
||||||
|
}
|
||||||
26
migrations/Version20260318143503.php
Normal file
26
migrations/Version20260318143503.php
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260318143503 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add amount column to mileage_allowances';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances ADD COLUMN amount DOUBLE PRECISION DEFAULT 0 NOT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances DROP COLUMN amount');
|
||||||
|
}
|
||||||
|
}
|
||||||
28
migrations/Version20260319100000.php
Normal file
28
migrations/Version20260319100000.php
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20260319100000 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Add amount receipt fields to mileage_allowances';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances ADD amount_receipt_path VARCHAR(255) DEFAULT NULL');
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances ADD amount_receipt_name VARCHAR(255) DEFAULT NULL');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances DROP COLUMN amount_receipt_path');
|
||||||
|
$this->addSql('ALTER TABLE mileage_allowances DROP COLUMN amount_receipt_name');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -12,7 +12,7 @@ use App\State\EmployeeLeaveSummaryProvider;
|
|||||||
operations: [
|
operations: [
|
||||||
new Get(
|
new Get(
|
||||||
uriTemplate: '/employees/{id}/leave-summary',
|
uriTemplate: '/employees/{id}/leave-summary',
|
||||||
security: "is_granted('ROLE_USER')",
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
provider: EmployeeLeaveSummaryProvider::class
|
provider: EmployeeLeaveSummaryProvider::class
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ use App\State\EmployeeRttSummaryProvider;
|
|||||||
operations: [
|
operations: [
|
||||||
new Get(
|
new Get(
|
||||||
uriTemplate: '/employees/{id}/rtt-summary',
|
uriTemplate: '/employees/{id}/rtt-summary',
|
||||||
security: "is_granted('ROLE_USER')",
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
provider: EmployeeRttSummaryProvider::class
|
provider: EmployeeRttSummaryProvider::class
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -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
src/ApiResource/SalaryRecapPrint.php
Normal file
24
src/ApiResource/SalaryRecapPrint.php
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\ApiResource;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\ApiResource;
|
||||||
|
use ApiPlatform\Metadata\Get;
|
||||||
|
use ApiPlatform\Metadata\QueryParameter;
|
||||||
|
use App\State\SalaryRecapPrintProvider;
|
||||||
|
|
||||||
|
#[ApiResource(
|
||||||
|
operations: [
|
||||||
|
new Get(
|
||||||
|
uriTemplate: '/salary-recap/print',
|
||||||
|
provider: SalaryRecapPrintProvider::class,
|
||||||
|
parameters: [
|
||||||
|
new QueryParameter(key: 'month', required: true),
|
||||||
|
],
|
||||||
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)]
|
||||||
|
final class SalaryRecapPrint {}
|
||||||
@@ -32,7 +32,12 @@ final class WorkHourBulkUpsert
|
|||||||
* eveningFrom?:?string,
|
* eveningFrom?:?string,
|
||||||
* eveningTo?:?string,
|
* eveningTo?:?string,
|
||||||
* isPresentMorning?:bool,
|
* isPresentMorning?:bool,
|
||||||
* isPresentAfternoon?:bool
|
* isPresentAfternoon?:bool,
|
||||||
|
* dayHoursMinutes?:?int,
|
||||||
|
* nightHoursMinutes?:?int,
|
||||||
|
* hasBreakfast?:bool,
|
||||||
|
* hasLunch?:bool,
|
||||||
|
* hasOvernight?:bool
|
||||||
* }>
|
* }>
|
||||||
*/
|
*/
|
||||||
public array $entries = [];
|
public array $entries = [];
|
||||||
|
|||||||
@@ -27,5 +27,7 @@ final class ContractHistoryItem
|
|||||||
public ?int $periodId = null,
|
public ?int $periodId = null,
|
||||||
#[Groups(['employee:read'])]
|
#[Groups(['employee:read'])]
|
||||||
public array $suspensions = [],
|
public array $suspensions = [],
|
||||||
|
#[Groups(['employee:read'])]
|
||||||
|
public bool $isDriver = false,
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ final class DayContextRow
|
|||||||
public bool $absentAfternoon = false,
|
public bool $absentAfternoon = false,
|
||||||
public int $creditedMinutes = 0,
|
public int $creditedMinutes = 0,
|
||||||
public float $creditedPresenceUnits = 0.0,
|
public float $creditedPresenceUnits = 0.0,
|
||||||
|
public bool $isDriverContract = false,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
public function addAbsence(
|
public function addAbsence(
|
||||||
@@ -78,6 +79,7 @@ final class DayContextRow
|
|||||||
'absentAfternoon' => $this->absentAfternoon,
|
'absentAfternoon' => $this->absentAfternoon,
|
||||||
'creditedMinutes' => $this->creditedMinutes,
|
'creditedMinutes' => $this->creditedMinutes,
|
||||||
'creditedPresenceUnits' => $this->creditedPresenceUnits,
|
'creditedPresenceUnits' => $this->creditedPresenceUnits,
|
||||||
|
'isDriverContract' => $this->isDriverContract,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,10 +10,16 @@ final class WeeklyDaySummary
|
|||||||
public string $date,
|
public string $date,
|
||||||
public int $dayMinutes,
|
public int $dayMinutes,
|
||||||
public int $nightMinutes,
|
public int $nightMinutes,
|
||||||
|
public int $workshopMinutes,
|
||||||
public int $totalMinutes,
|
public int $totalMinutes,
|
||||||
public ?float $present = null,
|
public ?float $present = null,
|
||||||
public bool $hasAbsence = false,
|
public bool $hasAbsence = false,
|
||||||
public ?string $absenceLabel = null,
|
public ?string $absenceLabel = null,
|
||||||
public ?string $absenceColor = null,
|
public ?string $absenceColor = null,
|
||||||
|
public bool $hasNightBasket = false,
|
||||||
|
public bool $hasBreakfast = false,
|
||||||
|
public bool $hasLunch = false,
|
||||||
|
public bool $hasDinner = false,
|
||||||
|
public bool $hasOvernight = false,
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,11 +20,19 @@ final class WeeklySummaryRow
|
|||||||
public array $daily,
|
public array $daily,
|
||||||
public int $weeklyDayMinutes,
|
public int $weeklyDayMinutes,
|
||||||
public int $weeklyNightMinutes,
|
public int $weeklyNightMinutes,
|
||||||
|
public int $weeklyWorkshopMinutes,
|
||||||
public int $weeklyTotalMinutes,
|
public int $weeklyTotalMinutes,
|
||||||
public float $weeklyPresenceCount,
|
public float $weeklyPresenceCount,
|
||||||
public int $weeklyOvertimeTotalMinutes,
|
public int $weeklyOvertimeTotalMinutes,
|
||||||
public int $weeklyOvertime25Minutes,
|
public int $weeklyOvertime25Minutes,
|
||||||
public int $weeklyOvertime50Minutes,
|
public int $weeklyOvertime50Minutes,
|
||||||
public int $weeklyRecoveryMinutes,
|
public int $weeklyRecoveryMinutes,
|
||||||
|
public int $weeklyNightBasketCount = 0,
|
||||||
|
public bool $isDriver = false,
|
||||||
|
public int $weeklyBreakfastCount = 0,
|
||||||
|
public int $weeklyLunchCount = 0,
|
||||||
|
public int $weeklyDinnerCount = 0,
|
||||||
|
public int $weeklyOvernightCount = 0,
|
||||||
|
public bool $hasContractForWeek = true,
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,10 +21,10 @@ use Symfony\Component\Serializer\Attribute\Groups;
|
|||||||
#[ApiResource(
|
#[ApiResource(
|
||||||
operations: [
|
operations: [
|
||||||
new Get(
|
new Get(
|
||||||
security: "is_granted('ROLE_USER')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
),
|
),
|
||||||
new GetCollection(
|
new GetCollection(
|
||||||
security: "is_granted('ROLE_USER')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
),
|
),
|
||||||
new Post(
|
new Post(
|
||||||
security: "is_granted('ROLE_ADMIN')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
|
|||||||
@@ -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')]
|
||||||
@@ -88,6 +89,9 @@ class Employee
|
|||||||
#[Groups(['employee:write'])]
|
#[Groups(['employee:write'])]
|
||||||
private ?string $contractComment = null;
|
private ?string $contractComment = null;
|
||||||
|
|
||||||
|
#[Groups(['employee:write'])]
|
||||||
|
private ?bool $isDriverInput = null;
|
||||||
|
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
$this->createdAt = new DateTimeImmutable();
|
$this->createdAt = new DateTimeImmutable();
|
||||||
@@ -245,6 +249,30 @@ class Employee
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getIsDriverInput(): ?bool
|
||||||
|
{
|
||||||
|
return $this->isDriverInput;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setIsDriverInput(?bool $isDriverInput): self
|
||||||
|
{
|
||||||
|
$this->isDriverInput = $isDriverInput;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Groups(['employee:read'])]
|
||||||
|
public function getHasActiveContract(): bool
|
||||||
|
{
|
||||||
|
return null !== $this->resolveCurrentContractPeriod();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Groups(['employee:read'])]
|
||||||
|
public function getIsDriver(): bool
|
||||||
|
{
|
||||||
|
return $this->resolveCurrentContractPeriod()?->getIsDriver() ?? false;
|
||||||
|
}
|
||||||
|
|
||||||
#[Groups(['employee:read'])]
|
#[Groups(['employee:read'])]
|
||||||
public function getCurrentContractNature(): string
|
public function getCurrentContractNature(): string
|
||||||
{
|
{
|
||||||
@@ -329,6 +357,7 @@ class Employee
|
|||||||
comment: $period->getComment(),
|
comment: $period->getComment(),
|
||||||
periodId: $period->getId(),
|
periodId: $period->getId(),
|
||||||
suspensions: $suspensionData,
|
suspensions: $suspensionData,
|
||||||
|
isDriver: $period->getIsDriver(),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
$periods
|
$periods
|
||||||
|
|||||||
@@ -39,6 +39,9 @@ class EmployeeContractPeriod
|
|||||||
#[ORM\Column(type: 'string', length: 20, options: ['default' => ContractNature::CDI->value])]
|
#[ORM\Column(type: 'string', length: 20, options: ['default' => ContractNature::CDI->value])]
|
||||||
private string $contractNature = ContractNature::CDI->value;
|
private string $contractNature = ContractNature::CDI->value;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
|
private bool $isDriver = false;
|
||||||
|
|
||||||
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
private bool $paidLeaveSettled = false;
|
private bool $paidLeaveSettled = false;
|
||||||
|
|
||||||
@@ -137,6 +140,18 @@ class EmployeeContractPeriod
|
|||||||
return $this->createdAt;
|
return $this->createdAt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getIsDriver(): bool
|
||||||
|
{
|
||||||
|
return $this->isDriver;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setIsDriver(bool $isDriver): self
|
||||||
|
{
|
||||||
|
$this->isDriver = $isDriver;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function isPaidLeaveSettled(): bool
|
public function isPaidLeaveSettled(): bool
|
||||||
{
|
{
|
||||||
return $this->paidLeaveSettled;
|
return $this->paidLeaveSettled;
|
||||||
|
|||||||
@@ -14,6 +14,8 @@ use ApiPlatform\Metadata\GetCollection;
|
|||||||
use ApiPlatform\Metadata\Patch;
|
use ApiPlatform\Metadata\Patch;
|
||||||
use ApiPlatform\Metadata\Post;
|
use ApiPlatform\Metadata\Post;
|
||||||
use App\Repository\MileageAllowanceRepository;
|
use App\Repository\MileageAllowanceRepository;
|
||||||
|
use App\State\MileageAllowanceAmountReceiptDownloadProvider;
|
||||||
|
use App\State\MileageAllowanceAmountReceiptUploadProcessor;
|
||||||
use App\State\MileageAllowanceDeleteProcessor;
|
use App\State\MileageAllowanceDeleteProcessor;
|
||||||
use App\State\MileageAllowanceReceiptDownloadProvider;
|
use App\State\MileageAllowanceReceiptDownloadProvider;
|
||||||
use App\State\MileageAllowanceReceiptUploadProcessor;
|
use App\State\MileageAllowanceReceiptUploadProcessor;
|
||||||
@@ -24,10 +26,10 @@ use Symfony\Component\Serializer\Attribute\Groups;
|
|||||||
#[ApiResource(
|
#[ApiResource(
|
||||||
operations: [
|
operations: [
|
||||||
new Get(
|
new Get(
|
||||||
security: "is_granted('ROLE_USER')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
),
|
),
|
||||||
new GetCollection(
|
new GetCollection(
|
||||||
security: "is_granted('ROLE_USER')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
),
|
),
|
||||||
new Post(
|
new Post(
|
||||||
security: "is_granted('ROLE_ADMIN')"
|
security: "is_granted('ROLE_ADMIN')"
|
||||||
@@ -47,9 +49,20 @@ use Symfony\Component\Serializer\Attribute\Groups;
|
|||||||
),
|
),
|
||||||
new Get(
|
new Get(
|
||||||
uriTemplate: '/mileage_allowances/{id}/receipt',
|
uriTemplate: '/mileage_allowances/{id}/receipt',
|
||||||
security: "is_granted('ROLE_USER')",
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
provider: MileageAllowanceReceiptDownloadProvider::class,
|
provider: MileageAllowanceReceiptDownloadProvider::class,
|
||||||
),
|
),
|
||||||
|
new Post(
|
||||||
|
uriTemplate: '/mileage_allowances/{id}/amount-receipt',
|
||||||
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
|
deserialize: false,
|
||||||
|
processor: MileageAllowanceAmountReceiptUploadProcessor::class,
|
||||||
|
),
|
||||||
|
new Get(
|
||||||
|
uriTemplate: '/mileage_allowances/{id}/amount-receipt',
|
||||||
|
security: "is_granted('ROLE_ADMIN')",
|
||||||
|
provider: MileageAllowanceAmountReceiptDownloadProvider::class,
|
||||||
|
),
|
||||||
],
|
],
|
||||||
normalizationContext: [
|
normalizationContext: [
|
||||||
'groups' => ['mileage_allowance:read', 'employee:read'],
|
'groups' => ['mileage_allowance:read', 'employee:read'],
|
||||||
@@ -87,6 +100,10 @@ class MileageAllowance
|
|||||||
#[Groups(['mileage_allowance:read', 'mileage_allowance:write'])]
|
#[Groups(['mileage_allowance:read', 'mileage_allowance:write'])]
|
||||||
private float $kilometers = 0;
|
private float $kilometers = 0;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'float', options: ['default' => 0])]
|
||||||
|
#[Groups(['mileage_allowance:read', 'mileage_allowance:write'])]
|
||||||
|
private float $amount = 0;
|
||||||
|
|
||||||
#[ORM\Column(type: 'text', nullable: true)]
|
#[ORM\Column(type: 'text', nullable: true)]
|
||||||
#[Groups(['mileage_allowance:read', 'mileage_allowance:write'])]
|
#[Groups(['mileage_allowance:read', 'mileage_allowance:write'])]
|
||||||
private ?string $comment = null;
|
private ?string $comment = null;
|
||||||
@@ -99,6 +116,14 @@ class MileageAllowance
|
|||||||
#[Groups(['mileage_allowance:read'])]
|
#[Groups(['mileage_allowance:read'])]
|
||||||
private ?string $receiptName = null;
|
private ?string $receiptName = null;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'string', length: 255, nullable: true)]
|
||||||
|
#[Groups(['mileage_allowance:read'])]
|
||||||
|
private ?string $amountReceiptPath = null;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'string', length: 255, nullable: true)]
|
||||||
|
#[Groups(['mileage_allowance:read'])]
|
||||||
|
private ?string $amountReceiptName = null;
|
||||||
|
|
||||||
#[ORM\Column(type: 'datetime_immutable')]
|
#[ORM\Column(type: 'datetime_immutable')]
|
||||||
#[Groups(['mileage_allowance:read'])]
|
#[Groups(['mileage_allowance:read'])]
|
||||||
private DateTimeImmutable $createdAt;
|
private DateTimeImmutable $createdAt;
|
||||||
@@ -149,6 +174,18 @@ class MileageAllowance
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getAmount(): float
|
||||||
|
{
|
||||||
|
return $this->amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setAmount(float $amount): self
|
||||||
|
{
|
||||||
|
$this->amount = $amount;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function getComment(): ?string
|
public function getComment(): ?string
|
||||||
{
|
{
|
||||||
return $this->comment;
|
return $this->comment;
|
||||||
@@ -185,6 +222,30 @@ class MileageAllowance
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getAmountReceiptPath(): ?string
|
||||||
|
{
|
||||||
|
return $this->amountReceiptPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setAmountReceiptPath(?string $amountReceiptPath): self
|
||||||
|
{
|
||||||
|
$this->amountReceiptPath = $amountReceiptPath;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAmountReceiptName(): ?string
|
||||||
|
{
|
||||||
|
return $this->amountReceiptName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setAmountReceiptName(?string $amountReceiptName): self
|
||||||
|
{
|
||||||
|
$this->amountReceiptName = $amountReceiptName;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function getCreatedAt(): DateTimeImmutable
|
public function getCreatedAt(): DateTimeImmutable
|
||||||
{
|
{
|
||||||
return $this->createdAt;
|
return $this->createdAt;
|
||||||
|
|||||||
@@ -99,6 +99,34 @@ class WorkHour
|
|||||||
#[Groups(['work_hour:read'])]
|
#[Groups(['work_hour:read'])]
|
||||||
private bool $isPresentAfternoon = false;
|
private bool $isPresentAfternoon = false;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'integer', nullable: true)]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private ?int $dayHoursMinutes = null;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'integer', nullable: true)]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private ?int $nightHoursMinutes = null;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'integer', nullable: true)]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private ?int $workshopHoursMinutes = null;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private bool $hasBreakfast = false;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private bool $hasLunch = false;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private bool $hasDinner = false;
|
||||||
|
|
||||||
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
|
#[Groups(['work_hour:read'])]
|
||||||
|
private bool $hasOvernight = false;
|
||||||
|
|
||||||
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
#[ORM\Column(type: 'boolean', options: ['default' => false])]
|
||||||
#[Groups(['work_hour:read', 'work_hour:validate'])]
|
#[Groups(['work_hour:read', 'work_hour:validate'])]
|
||||||
private bool $isValid = false;
|
private bool $isValid = false;
|
||||||
@@ -212,6 +240,90 @@ class WorkHour
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getDayHoursMinutes(): ?int
|
||||||
|
{
|
||||||
|
return $this->dayHoursMinutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setDayHoursMinutes(?int $dayHoursMinutes): self
|
||||||
|
{
|
||||||
|
$this->dayHoursMinutes = $dayHoursMinutes;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getNightHoursMinutes(): ?int
|
||||||
|
{
|
||||||
|
return $this->nightHoursMinutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setNightHoursMinutes(?int $nightHoursMinutes): self
|
||||||
|
{
|
||||||
|
$this->nightHoursMinutes = $nightHoursMinutes;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getWorkshopHoursMinutes(): ?int
|
||||||
|
{
|
||||||
|
return $this->workshopHoursMinutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setWorkshopHoursMinutes(?int $workshopHoursMinutes): self
|
||||||
|
{
|
||||||
|
$this->workshopHoursMinutes = $workshopHoursMinutes;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHasBreakfast(): bool
|
||||||
|
{
|
||||||
|
return $this->hasBreakfast;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setHasBreakfast(bool $hasBreakfast): self
|
||||||
|
{
|
||||||
|
$this->hasBreakfast = $hasBreakfast;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHasLunch(): bool
|
||||||
|
{
|
||||||
|
return $this->hasLunch;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setHasLunch(bool $hasLunch): self
|
||||||
|
{
|
||||||
|
$this->hasLunch = $hasLunch;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHasDinner(): bool
|
||||||
|
{
|
||||||
|
return $this->hasDinner;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setHasDinner(bool $hasDinner): self
|
||||||
|
{
|
||||||
|
$this->hasDinner = $hasDinner;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHasOvernight(): bool
|
||||||
|
{
|
||||||
|
return $this->hasOvernight;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setHasOvernight(bool $hasOvernight): self
|
||||||
|
{
|
||||||
|
$this->hasOvernight = $hasOvernight;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
public function isPresentMorning(): bool
|
public function isPresentMorning(): bool
|
||||||
{
|
{
|
||||||
return $this->isPresentMorning;
|
return $this->isPresentMorning;
|
||||||
|
|||||||
@@ -100,6 +100,38 @@ final class AbsenceRepository extends ServiceEntityRepository implements Absence
|
|||||||
return $qb->getQuery()->getResult();
|
return $qb->getQuery()->getResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<DateTimeImmutable> sorted maladie dates
|
||||||
|
*/
|
||||||
|
public function findMaladieDatesByEmployee(
|
||||||
|
Employee $employee,
|
||||||
|
DateTimeImmutable $from,
|
||||||
|
DateTimeImmutable $to
|
||||||
|
): array {
|
||||||
|
$results = $this->createQueryBuilder('a')
|
||||||
|
->select('a.startDate')
|
||||||
|
->join('a.type', 't')
|
||||||
|
->andWhere('a.employee = :employee')
|
||||||
|
->andWhere('t.code = :code')
|
||||||
|
->andWhere('a.startDate >= :from')
|
||||||
|
->andWhere('a.startDate <= :to')
|
||||||
|
->setParameter('employee', $employee)
|
||||||
|
->setParameter('code', 'M')
|
||||||
|
->setParameter('from', $from)
|
||||||
|
->setParameter('to', $to)
|
||||||
|
->orderBy('a.startDate', 'ASC')
|
||||||
|
->getQuery()
|
||||||
|
->getArrayResult()
|
||||||
|
;
|
||||||
|
|
||||||
|
return array_map(
|
||||||
|
static fn (array $row): DateTimeImmutable => $row['startDate'] instanceof DateTimeImmutable
|
||||||
|
? $row['startDate']
|
||||||
|
: DateTimeImmutable::createFromInterface($row['startDate']),
|
||||||
|
$results
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return list<Absence>
|
* @return list<Absence>
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace App\Repository;
|
namespace App\Repository;
|
||||||
|
|
||||||
use App\Entity\Bonus;
|
use App\Entity\Bonus;
|
||||||
|
use DateTimeImmutable;
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
|
||||||
@@ -17,4 +18,21 @@ final class BonusRepository extends ServiceEntityRepository
|
|||||||
{
|
{
|
||||||
parent::__construct($registry, Bonus::class);
|
parent::__construct($registry, Bonus::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Bonus[]
|
||||||
|
*/
|
||||||
|
public function findByMonth(DateTimeImmutable $from, DateTimeImmutable $to): array
|
||||||
|
{
|
||||||
|
return $this->createQueryBuilder('b')
|
||||||
|
->andWhere('b.month >= :from')
|
||||||
|
->andWhere('b.month <= :to')
|
||||||
|
->setParameter('from', $from)
|
||||||
|
->setParameter('to', $to)
|
||||||
|
->innerJoin('b.employee', 'e')
|
||||||
|
->addSelect('e')
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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')
|
||||||
|
|||||||
@@ -43,4 +43,21 @@ final class EmployeeRttPaymentRepository extends ServiceEntityRepository
|
|||||||
->getResult()
|
->getResult()
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return EmployeeRttPayment[]
|
||||||
|
*/
|
||||||
|
public function findByYearAndMonth(int $year, int $month): array
|
||||||
|
{
|
||||||
|
return $this->createQueryBuilder('p')
|
||||||
|
->andWhere('p.year = :year')
|
||||||
|
->andWhere('p.month = :month')
|
||||||
|
->setParameter('year', $year)
|
||||||
|
->setParameter('month', $month)
|
||||||
|
->innerJoin('p.employee', 'e')
|
||||||
|
->addSelect('e')
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace App\Repository;
|
namespace App\Repository;
|
||||||
|
|
||||||
use App\Entity\MileageAllowance;
|
use App\Entity\MileageAllowance;
|
||||||
|
use DateTimeImmutable;
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
|
||||||
@@ -17,4 +18,21 @@ final class MileageAllowanceRepository extends ServiceEntityRepository
|
|||||||
{
|
{
|
||||||
parent::__construct($registry, MileageAllowance::class);
|
parent::__construct($registry, MileageAllowance::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return MileageAllowance[]
|
||||||
|
*/
|
||||||
|
public function findByMonth(DateTimeImmutable $from, DateTimeImmutable $to): array
|
||||||
|
{
|
||||||
|
return $this->createQueryBuilder('m')
|
||||||
|
->andWhere('m.month >= :from')
|
||||||
|
->andWhere('m.month <= :to')
|
||||||
|
->setParameter('from', $from)
|
||||||
|
->setParameter('to', $to)
|
||||||
|
->innerJoin('m.employee', 'e')
|
||||||
|
->addSelect('e')
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -191,6 +191,57 @@ final class WorkHourRepository extends ServiceEntityRepository implements WorkHo
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count weekend and public holiday worked days for forfait bonus leave (PRESENCE mode only).
|
||||||
|
* Morning + afternoon = 1.0 day, one only = 0.5 day.
|
||||||
|
*
|
||||||
|
* @param list<string> $publicHolidayDates Y-m-d formatted weekday public holiday dates
|
||||||
|
*/
|
||||||
|
public function countWeekendAndHolidayWorkedDays(Employee $employee, DateTimeImmutable $from, DateTimeImmutable $to, array $publicHolidayDates = []): float
|
||||||
|
{
|
||||||
|
$targetDates = [];
|
||||||
|
|
||||||
|
// Collect weekend dates in range
|
||||||
|
for ($cursor = $from; $cursor <= $to; $cursor = $cursor->modify('+1 day')) {
|
||||||
|
if ((int) $cursor->format('N') >= 6) {
|
||||||
|
$targetDates[] = $cursor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add weekday public holidays
|
||||||
|
foreach ($publicHolidayDates as $date) {
|
||||||
|
$targetDates[] = new DateTimeImmutable($date);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ([] === $targetDates) {
|
||||||
|
return 0.0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$dateStrings = array_map(static fn (DateTimeImmutable $d): string => $d->format('Y-m-d'), $targetDates);
|
||||||
|
|
||||||
|
/** @var list<WorkHour> $rows */
|
||||||
|
$rows = $this->createQueryBuilder('w')
|
||||||
|
->andWhere('w.employee = :employee')
|
||||||
|
->andWhere('w.workDate IN (:dates)')
|
||||||
|
->andWhere('w.isPresentMorning = true OR w.isPresentAfternoon = true')
|
||||||
|
->setParameter('employee', $employee)
|
||||||
|
->setParameter('dates', $dateStrings)
|
||||||
|
->getQuery()
|
||||||
|
->getResult()
|
||||||
|
;
|
||||||
|
|
||||||
|
$total = 0.0;
|
||||||
|
foreach ($rows as $row) {
|
||||||
|
if ($row->isPresentMorning() && $row->isPresentAfternoon()) {
|
||||||
|
$total += 1.0;
|
||||||
|
} else {
|
||||||
|
$total += 0.5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $total;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the set of Y-m-d dates where the employee has worked hours on the given dates.
|
* Return the set of Y-m-d dates where the employee has worked hours on the given dates.
|
||||||
*
|
*
|
||||||
@@ -228,6 +279,55 @@ final class WorkHourRepository extends ServiceEntityRepository implements WorkHo
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function isWeekFullyValidated(Employee $employee, DateTimeImmutable $from, DateTimeImmutable $to): bool
|
||||||
|
{
|
||||||
|
// Count weekdays (Mon-Fri) in range
|
||||||
|
$expectedWeekdays = 0;
|
||||||
|
for ($d = $from; $d <= $to; $d = $d->modify('+1 day')) {
|
||||||
|
if ((int) $d->format('N') <= 5) {
|
||||||
|
++$expectedWeekdays;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (0 === $expectedWeekdays) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Every weekday must have a work_hour row
|
||||||
|
$totalCount = (int) $this->createQueryBuilder('w')
|
||||||
|
->select('COUNT(w.id)')
|
||||||
|
->andWhere('w.employee = :employee')
|
||||||
|
->andWhere('w.workDate >= :from')
|
||||||
|
->andWhere('w.workDate <= :to')
|
||||||
|
->setParameter('employee', $employee)
|
||||||
|
->setParameter('from', $from)
|
||||||
|
->setParameter('to', $to)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleScalarResult()
|
||||||
|
;
|
||||||
|
|
||||||
|
if ($totalCount < $expectedWeekdays) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// All rows must be validated
|
||||||
|
$nonValidatedCount = (int) $this->createQueryBuilder('w')
|
||||||
|
->select('COUNT(w.id)')
|
||||||
|
->andWhere('w.employee = :employee')
|
||||||
|
->andWhere('w.workDate >= :from')
|
||||||
|
->andWhere('w.workDate <= :to')
|
||||||
|
->andWhere('w.isValid = :isValid')
|
||||||
|
->setParameter('employee', $employee)
|
||||||
|
->setParameter('from', $from)
|
||||||
|
->setParameter('to', $to)
|
||||||
|
->setParameter('isValid', false)
|
||||||
|
->getQuery()
|
||||||
|
->getSingleScalarResult()
|
||||||
|
;
|
||||||
|
|
||||||
|
return 0 === $nonValidatedCount;
|
||||||
|
}
|
||||||
|
|
||||||
public function hasPendingSiteValidationForSiteAndDate(int $siteId, DateTimeInterface $date): bool
|
public function hasPendingSiteValidationForSiteAndDate(int $siteId, DateTimeInterface $date): bool
|
||||||
{
|
{
|
||||||
$workDate = DateTimeImmutable::createFromInterface($date);
|
$workDate = DateTimeImmutable::createFromInterface($date);
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ final readonly class EmployeeContractChangeRequest
|
|||||||
public ?DateTimeImmutable $contractEndDate,
|
public ?DateTimeImmutable $contractEndDate,
|
||||||
public ?bool $contractPaidLeaveSettled,
|
public ?bool $contractPaidLeaveSettled,
|
||||||
public ?string $contractComment,
|
public ?string $contractComment,
|
||||||
|
public ?bool $isDriver = null,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
public function hasPeriodChangeRequest(): bool
|
public function hasPeriodChangeRequest(): bool
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ final class EmployeeContractChangeRequestFactory
|
|||||||
contractEndDate: $this->parseOptionalYmd($employee->getContractEndDate(), 'contractEndDate'),
|
contractEndDate: $this->parseOptionalYmd($employee->getContractEndDate(), 'contractEndDate'),
|
||||||
contractPaidLeaveSettled: $employee->getContractPaidLeaveSettled(),
|
contractPaidLeaveSettled: $employee->getContractPaidLeaveSettled(),
|
||||||
contractComment: $employee->getContractComment(),
|
contractComment: $employee->getContractComment(),
|
||||||
|
isDriver: $employee->getIsDriverInput(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ final class EmployeeContractPeriodBuilder
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
|
bool $isDriver = false,
|
||||||
): EmployeeContractPeriod {
|
): EmployeeContractPeriod {
|
||||||
return new EmployeeContractPeriod()
|
return new EmployeeContractPeriod()
|
||||||
->setEmployee($employee)
|
->setEmployee($employee)
|
||||||
@@ -25,6 +26,7 @@ final class EmployeeContractPeriodBuilder
|
|||||||
->setStartDate($startDate)
|
->setStartDate($startDate)
|
||||||
->setEndDate($endDate)
|
->setEndDate($endDate)
|
||||||
->setContractNature($nature)
|
->setContractNature($nature)
|
||||||
|
->setIsDriver($isDriver)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ final readonly class EmployeeContractPeriodManager implements EmployeeContractPe
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
|
bool $isDriver = false,
|
||||||
): void {
|
): void {
|
||||||
$this->periodValidator->assertPeriodDates($startDate, $endDate, $nature);
|
$this->periodValidator->assertPeriodDates($startDate, $endDate, $nature);
|
||||||
|
|
||||||
@@ -36,7 +37,7 @@ final readonly class EmployeeContractPeriodManager implements EmployeeContractPe
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->persistNewPeriod($employee, $contract, $startDate, $endDate, $nature);
|
$this->persistNewPeriod($employee, $contract, $startDate, $endDate, $nature, $isDriver);
|
||||||
$this->entityManager->flush();
|
$this->entityManager->flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,7 +70,8 @@ final readonly class EmployeeContractPeriodManager implements EmployeeContractPe
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
?EmployeeContractPeriod $todayPeriod
|
?EmployeeContractPeriod $todayPeriod,
|
||||||
|
bool $isDriver = false,
|
||||||
): void {
|
): void {
|
||||||
$this->periodValidator->assertPeriodDates($startDate, $endDate, $nature);
|
$this->periodValidator->assertPeriodDates($startDate, $endDate, $nature);
|
||||||
|
|
||||||
@@ -81,7 +83,7 @@ final readonly class EmployeeContractPeriodManager implements EmployeeContractPe
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->persistNewPeriod($employee, $contract, $startDate, $endDate, $nature);
|
$this->persistNewPeriod($employee, $contract, $startDate, $endDate, $nature, $isDriver);
|
||||||
$this->entityManager->flush();
|
$this->entityManager->flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -91,8 +93,9 @@ final readonly class EmployeeContractPeriodManager implements EmployeeContractPe
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
|
bool $isDriver = false,
|
||||||
): void {
|
): void {
|
||||||
$period = $this->periodBuilder->build($employee, $contract, $startDate, $endDate, $nature);
|
$period = $this->periodBuilder->build($employee, $contract, $startDate, $endDate, $nature, $isDriver);
|
||||||
$this->entityManager->persist($period);
|
$this->entityManager->persist($period);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ interface EmployeeContractPeriodManagerInterface
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
|
bool $isDriver = false,
|
||||||
): void;
|
): void;
|
||||||
|
|
||||||
public function closeCurrentPeriod(
|
public function closeCurrentPeriod(
|
||||||
@@ -33,6 +34,7 @@ interface EmployeeContractPeriodManagerInterface
|
|||||||
DateTimeImmutable $startDate,
|
DateTimeImmutable $startDate,
|
||||||
?DateTimeImmutable $endDate,
|
?DateTimeImmutable $endDate,
|
||||||
ContractNature $nature,
|
ContractNature $nature,
|
||||||
?EmployeeContractPeriod $todayPeriod
|
?EmployeeContractPeriod $todayPeriod,
|
||||||
|
bool $isDriver = false,
|
||||||
): void;
|
): void;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,6 +23,60 @@ readonly class EmployeeContractResolver
|
|||||||
return $period?->getContract();
|
return $period?->getContract();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function resolveIsDriverForEmployeeAndDate(Employee $employee, DateTimeImmutable $date): bool
|
||||||
|
{
|
||||||
|
$period = $this->periodRepository->findOneCoveringDate($employee, $date);
|
||||||
|
|
||||||
|
return $period?->getIsDriver() ?? false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Employee> $employees
|
||||||
|
* @param list<string> $days
|
||||||
|
*
|
||||||
|
* @return array<int, array<string, bool>>
|
||||||
|
*/
|
||||||
|
public function resolveIsDriverForEmployeesAndDays(array $employees, array $days): array
|
||||||
|
{
|
||||||
|
$resolved = [];
|
||||||
|
if ([] === $employees || [] === $days) {
|
||||||
|
return $resolved;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($employees as $employee) {
|
||||||
|
$employeeId = $employee->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($days as $day) {
|
||||||
|
$resolved[$employeeId][$day] = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$from = new DateTimeImmutable(min($days));
|
||||||
|
$to = new DateTimeImmutable(max($days));
|
||||||
|
$periods = $this->periodRepository->findByEmployeesAndDateRange($employees, $from, $to);
|
||||||
|
foreach ($periods as $period) {
|
||||||
|
$employeeId = $period->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$start = $period->getStartDate()->format('Y-m-d');
|
||||||
|
$end = $period->getEndDate()?->format('Y-m-d') ?? '9999-12-31';
|
||||||
|
$isDriver = $period->getIsDriver();
|
||||||
|
foreach ($days as $day) {
|
||||||
|
if ($day < $start || $day > $end) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$resolved[$employeeId][$day] = $isDriver;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resolved;
|
||||||
|
}
|
||||||
|
|
||||||
public function resolveNatureForEmployeeAndDate(Employee $employee, DateTimeImmutable $date): ContractNature
|
public function resolveNatureForEmployeeAndDate(Employee $employee, DateTimeImmutable $date): ContractNature
|
||||||
{
|
{
|
||||||
$period = $this->periodRepository->findOneCoveringDate($employee, $date);
|
$period = $this->periodRepository->findOneCoveringDate($employee, $date);
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ final readonly class LeaveBalanceComputationService
|
|||||||
private const float STANDARD_SATURDAY_ACCRUAL_PER_MONTH = self::STANDARD_ANNUAL_SATURDAYS / 12.0;
|
private const float STANDARD_SATURDAY_ACCRUAL_PER_MONTH = self::STANDARD_ANNUAL_SATURDAYS / 12.0;
|
||||||
private const float FOUR_HOUR_ANNUAL_DAYS = 10.0;
|
private const float FOUR_HOUR_ANNUAL_DAYS = 10.0;
|
||||||
private const float FOUR_HOUR_ACCRUAL_PER_MONTH = 0.83;
|
private const float FOUR_HOUR_ACCRUAL_PER_MONTH = 0.83;
|
||||||
|
private const float LONG_MALADIE_MONTHLY_ACCRUAL = 2.0;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private AbsenceRepository $absenceRepository,
|
private AbsenceRepository $absenceRepository,
|
||||||
@@ -31,6 +32,7 @@ final readonly class LeaveBalanceComputationService
|
|||||||
private EmployeeLeaveBalanceRepository $leaveBalanceRepository,
|
private EmployeeLeaveBalanceRepository $leaveBalanceRepository,
|
||||||
private PublicHolidayServiceInterface $publicHolidayService,
|
private PublicHolidayServiceInterface $publicHolidayService,
|
||||||
private SuspensionDaysCalculator $suspensionDaysCalculator,
|
private SuspensionDaysCalculator $suspensionDaysCalculator,
|
||||||
|
private LongMaladieService $longMaladieService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -83,19 +85,34 @@ final readonly class LeaveBalanceComputationService
|
|||||||
$suspensions = $this->suspensionDaysCalculator->applyFirstMonthGrace(
|
$suspensions = $this->suspensionDaysCalculator->applyFirstMonthGrace(
|
||||||
$this->resolveSuspensionsForEmployeePeriod($employee, $from, $to)
|
$this->resolveSuspensionsForEmployeePeriod($employee, $from, $to)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
$longMaladiePeriods = [];
|
||||||
|
$longMaladieReductionFactor = 1.0;
|
||||||
|
if (4 !== $employee->getContract()?->getWeeklyHours()) {
|
||||||
|
$longMaladiePeriods = $this->longMaladieService->findReducedRatePeriods($employee, $effectiveFrom, $to);
|
||||||
|
if ([] !== $longMaladiePeriods) {
|
||||||
|
$totalNormalAccrual = $this->resolveDaysAccrualPerMonth($employee) + $this->resolveSaturdayAccrualPerMonth($employee);
|
||||||
|
$longMaladieReductionFactor = self::LONG_MALADIE_MONTHLY_ACCRUAL / $totalNormalAccrual;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$generatedDays = $this->computeAccruedDays(
|
$generatedDays = $this->computeAccruedDays(
|
||||||
$this->resolveAnnualDays($employee),
|
$this->resolveAnnualDays($employee),
|
||||||
$this->resolveDaysAccrualPerMonth($employee),
|
$this->resolveDaysAccrualPerMonth($employee),
|
||||||
$effectiveFrom,
|
$effectiveFrom,
|
||||||
$to,
|
$to,
|
||||||
$suspensions
|
$suspensions,
|
||||||
|
$longMaladiePeriods,
|
||||||
|
$longMaladieReductionFactor
|
||||||
);
|
);
|
||||||
$generatedSaturdays = $this->computeAccruedDays(
|
$generatedSaturdays = $this->computeAccruedDays(
|
||||||
$this->resolveAnnualSaturdays($employee),
|
$this->resolveAnnualSaturdays($employee),
|
||||||
$this->resolveSaturdayAccrualPerMonth($employee),
|
$this->resolveSaturdayAccrualPerMonth($employee),
|
||||||
$effectiveFrom,
|
$effectiveFrom,
|
||||||
$to,
|
$to,
|
||||||
$suspensions
|
$suspensions,
|
||||||
|
$longMaladiePeriods,
|
||||||
|
$longMaladieReductionFactor
|
||||||
);
|
);
|
||||||
|
|
||||||
$absences = $this->absenceRepository->findByEmployeeAndOverlappingDateRange($employee, $effectiveFrom, $to);
|
$absences = $this->absenceRepository->findByEmployeeAndOverlappingDateRange($employee, $effectiveFrom, $to);
|
||||||
@@ -267,21 +284,29 @@ final readonly class LeaveBalanceComputationService
|
|||||||
: self::STANDARD_SATURDAY_ACCRUAL_PER_MONTH;
|
: self::STANDARD_SATURDAY_ACCRUAL_PER_MONTH;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<ContractSuspension> $suspensions
|
||||||
|
* @param list<array{start: DateTimeImmutable, end: DateTimeImmutable}> $longMaladiePeriods
|
||||||
|
*/
|
||||||
private function computeAccruedDays(
|
private function computeAccruedDays(
|
||||||
float $annualCap,
|
float $annualCap,
|
||||||
float $accrualPerMonth,
|
float $accrualPerMonth,
|
||||||
DateTimeImmutable $periodStart,
|
DateTimeImmutable $periodStart,
|
||||||
DateTimeImmutable $periodEnd,
|
DateTimeImmutable $periodEnd,
|
||||||
array $suspensions = []
|
array $suspensions = [],
|
||||||
|
array $longMaladiePeriods = [],
|
||||||
|
float $longMaladieReductionFactor = 1.0
|
||||||
): float {
|
): float {
|
||||||
if ($accrualPerMonth <= 0.0 || $periodEnd < $periodStart) {
|
if ($accrualPerMonth <= 0.0 || $periodEnd < $periodStart) {
|
||||||
return 0.0;
|
return 0.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$periodStart = $this->normalizeDate($periodStart);
|
$periodStart = $this->normalizeDate($periodStart);
|
||||||
$periodEnd = $this->normalizeDate($periodEnd);
|
$periodEnd = $this->normalizeDate($periodEnd);
|
||||||
$coveredMonths = 0.0;
|
$publicHolidays = [] !== $suspensions ? $this->buildPublicHolidayMap($periodStart, $periodEnd) : [];
|
||||||
$cursor = $periodStart->modify('first day of this month')->setTime(0, 0);
|
$normalMonths = 0.0;
|
||||||
|
$reducedMonths = 0.0;
|
||||||
|
$cursor = $periodStart->modify('first day of this month')->setTime(0, 0);
|
||||||
while ($cursor <= $periodEnd) {
|
while ($cursor <= $periodEnd) {
|
||||||
$monthStart = $cursor > $periodStart ? $cursor : $periodStart;
|
$monthStart = $cursor > $periodStart ? $cursor : $periodStart;
|
||||||
$monthEnd = $cursor->modify('last day of this month')->setTime(0, 0);
|
$monthEnd = $cursor->modify('last day of this month')->setTime(0, 0);
|
||||||
@@ -289,18 +314,39 @@ final readonly class LeaveBalanceComputationService
|
|||||||
$monthEnd = $periodEnd;
|
$monthEnd = $periodEnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
$coveredDays = ((int) $monthEnd->diff($monthStart)->format('%a')) + 1;
|
|
||||||
if ([] !== $suspensions) {
|
if ([] !== $suspensions) {
|
||||||
$suspendedDays = $this->suspensionDaysCalculator->countSuspendedDaysInMonth($monthStart, $monthEnd, $suspensions);
|
$suspendedDays = $this->suspensionDaysCalculator->countSuspendedDaysInMonth($monthStart, $monthEnd, $suspensions);
|
||||||
$coveredDays = max(0, $coveredDays - $suspendedDays);
|
if ($suspendedDays > 0) {
|
||||||
|
$businessDays = $this->countBusinessDaysInRange($monthStart, $monthEnd, $publicHolidays);
|
||||||
|
$suspendedBusinessDays = $this->suspensionDaysCalculator->countSuspendedBusinessDays($monthStart, $monthEnd, $suspensions, $publicHolidays);
|
||||||
|
$normalMonths += max(0, $businessDays - $suspendedBusinessDays) / 22.0;
|
||||||
|
$cursor = $cursor->modify('first day of next month');
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$coveredDays = ((int) $monthEnd->diff($monthStart)->format('%a')) + 1;
|
||||||
$daysInMonth = (int) $cursor->format('t');
|
$daysInMonth = (int) $cursor->format('t');
|
||||||
$coveredMonths += $coveredDays / $daysInMonth;
|
|
||||||
|
if ([] !== $longMaladiePeriods) {
|
||||||
|
$reducedDays = $this->longMaladieService->countReducedDaysInMonth($monthStart, $monthEnd, $longMaladiePeriods);
|
||||||
|
if ($reducedDays > 0) {
|
||||||
|
$normalDays = max(0, $coveredDays - $reducedDays);
|
||||||
|
$normalMonths += $normalDays / $daysInMonth;
|
||||||
|
$reducedMonths += min($coveredDays, $reducedDays) / $daysInMonth;
|
||||||
|
$cursor = $cursor->modify('first day of next month');
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$normalMonths += $coveredDays / $daysInMonth;
|
||||||
|
|
||||||
$cursor = $cursor->modify('first day of next month');
|
$cursor = $cursor->modify('first day of next month');
|
||||||
}
|
}
|
||||||
|
|
||||||
return min($annualCap, $coveredMonths * $accrualPerMonth);
|
return min($annualCap, ($normalMonths + $reducedMonths * $longMaladieReductionFactor) * $accrualPerMonth);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function parseYmdDate(string $value): ?DateTimeImmutable
|
private function parseYmdDate(string $value): ?DateTimeImmutable
|
||||||
@@ -317,8 +363,15 @@ final readonly class LeaveBalanceComputationService
|
|||||||
|
|
||||||
private function countBusinessDays(DateTimeImmutable $from, DateTimeImmutable $to): int
|
private function countBusinessDays(DateTimeImmutable $from, DateTimeImmutable $to): int
|
||||||
{
|
{
|
||||||
$publicHolidays = $this->buildPublicHolidayMap($from, $to);
|
return $this->countBusinessDaysInRange($from, $to, $this->buildPublicHolidayMap($from, $to));
|
||||||
$count = 0;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, string> $publicHolidays pre-built map
|
||||||
|
*/
|
||||||
|
private function countBusinessDaysInRange(DateTimeImmutable $from, DateTimeImmutable $to, array $publicHolidays): int
|
||||||
|
{
|
||||||
|
$count = 0;
|
||||||
for ($cursor = $from; $cursor <= $to; $cursor = $cursor->modify('+1 day')) {
|
for ($cursor = $from; $cursor <= $to; $cursor = $cursor->modify('+1 day')) {
|
||||||
$weekDay = (int) $cursor->format('N');
|
$weekDay = (int) $cursor->format('N');
|
||||||
$dayKey = $cursor->format('Y-m-d');
|
$dayKey = $cursor->format('Y-m-d');
|
||||||
|
|||||||
116
src/Service/Leave/LongMaladieService.php
Normal file
116
src/Service/Leave/LongMaladieService.php
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\Service\Leave;
|
||||||
|
|
||||||
|
use App\Entity\Employee;
|
||||||
|
use App\Repository\AbsenceRepository;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
|
||||||
|
use function count;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detects continuous MALADIE (sick leave) periods and computes
|
||||||
|
* the date ranges where reduced accrual applies (after the first month grace).
|
||||||
|
*/
|
||||||
|
final readonly class LongMaladieService
|
||||||
|
{
|
||||||
|
private const int MAX_GAP_DAYS = 3;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private AbsenceRepository $absenceRepository,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns date ranges where the reduced maladie accrual rate applies.
|
||||||
|
* For continuous maladie periods > 1 month, the first month is excluded (grace period).
|
||||||
|
*
|
||||||
|
* @return list<array{start: DateTimeImmutable, end: DateTimeImmutable}>
|
||||||
|
*/
|
||||||
|
public function findReducedRatePeriods(
|
||||||
|
Employee $employee,
|
||||||
|
DateTimeImmutable $from,
|
||||||
|
DateTimeImmutable $to
|
||||||
|
): array {
|
||||||
|
// Look back 13 months to catch maladie that started before the exercise period
|
||||||
|
$extendedFrom = $from->modify('-13 months');
|
||||||
|
$dates = $this->absenceRepository->findMaladieDatesByEmployee($employee, $extendedFrom, $to);
|
||||||
|
if ([] === $dates) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$periods = $this->consolidateIntoPeriods($dates);
|
||||||
|
|
||||||
|
return $this->applyFirstMonthGrace($periods);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count calendar days in [monthStart, monthEnd] that fall within reduced maladie periods.
|
||||||
|
*
|
||||||
|
* @param list<array{start: DateTimeImmutable, end: DateTimeImmutable}> $reducedPeriods
|
||||||
|
*/
|
||||||
|
public function countReducedDaysInMonth(
|
||||||
|
DateTimeImmutable $monthStart,
|
||||||
|
DateTimeImmutable $monthEnd,
|
||||||
|
array $reducedPeriods
|
||||||
|
): int {
|
||||||
|
$total = 0;
|
||||||
|
foreach ($reducedPeriods as $period) {
|
||||||
|
$overlapStart = $period['start'] > $monthStart ? $period['start'] : $monthStart;
|
||||||
|
$overlapEnd = $period['end'] < $monthEnd ? $period['end'] : $monthEnd;
|
||||||
|
|
||||||
|
if ($overlapStart > $overlapEnd) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$total += ((int) $overlapEnd->diff($overlapStart)->format('%a')) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $total;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<DateTimeImmutable> $dates sorted chronologically
|
||||||
|
*
|
||||||
|
* @return list<array{start: DateTimeImmutable, end: DateTimeImmutable}>
|
||||||
|
*/
|
||||||
|
private function consolidateIntoPeriods(array $dates): array
|
||||||
|
{
|
||||||
|
$periods = [];
|
||||||
|
$start = $dates[0];
|
||||||
|
$prev = $start;
|
||||||
|
|
||||||
|
for ($i = 1, $count = count($dates); $i < $count; ++$i) {
|
||||||
|
$current = $dates[$i];
|
||||||
|
$gap = (int) $prev->diff($current)->format('%a');
|
||||||
|
if ($gap > self::MAX_GAP_DAYS) {
|
||||||
|
$periods[] = ['start' => $start, 'end' => $prev];
|
||||||
|
$start = $current;
|
||||||
|
}
|
||||||
|
$prev = $current;
|
||||||
|
}
|
||||||
|
$periods[] = ['start' => $start, 'end' => $prev];
|
||||||
|
|
||||||
|
return $periods;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<array{start: DateTimeImmutable, end: DateTimeImmutable}> $periods
|
||||||
|
*
|
||||||
|
* @return list<array{start: DateTimeImmutable, end: DateTimeImmutable}>
|
||||||
|
*/
|
||||||
|
private function applyFirstMonthGrace(array $periods): array
|
||||||
|
{
|
||||||
|
$result = [];
|
||||||
|
foreach ($periods as $period) {
|
||||||
|
$gracedStart = $period['start']->modify('+1 month');
|
||||||
|
if ($gracedStart > $period['end']) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$result[] = ['start' => $gracedStart, 'end' => $period['end']];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,8 @@ namespace App\Service;
|
|||||||
|
|
||||||
use Exception;
|
use Exception;
|
||||||
use RuntimeException;
|
use RuntimeException;
|
||||||
|
use Symfony\Contracts\Cache\CacheInterface;
|
||||||
|
use Symfony\Contracts\Cache\ItemInterface;
|
||||||
use Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface;
|
use Symfony\Contracts\HttpClient\Exception\ClientExceptionInterface;
|
||||||
use Symfony\Contracts\HttpClient\Exception\RedirectionExceptionInterface;
|
use Symfony\Contracts\HttpClient\Exception\RedirectionExceptionInterface;
|
||||||
use Symfony\Contracts\HttpClient\Exception\ServerExceptionInterface;
|
use Symfony\Contracts\HttpClient\Exception\ServerExceptionInterface;
|
||||||
@@ -17,7 +19,8 @@ final readonly class PublicHolidayService implements PublicHolidayServiceInterfa
|
|||||||
{
|
{
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private HttpClientInterface $client,
|
private HttpClientInterface $client,
|
||||||
private string $holidayUrl
|
private string $holidayUrl,
|
||||||
|
private CacheInterface $cache,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -30,24 +33,29 @@ final readonly class PublicHolidayService implements PublicHolidayServiceInterfa
|
|||||||
public function getHolidaysDay(string $zone): array
|
public function getHolidaysDay(string $zone): array
|
||||||
{
|
{
|
||||||
$zone = strtolower(trim($zone));
|
$zone = strtolower(trim($zone));
|
||||||
$url = $this->holidayUrl."{$zone}.json";
|
$key = "public_holidays_{$zone}_all";
|
||||||
|
|
||||||
try {
|
return $this->cache->get($key, function (ItemInterface $item) use ($zone): array {
|
||||||
$response = $this->client->request(
|
$item->expiresAfter(30 * 86400);
|
||||||
'GET',
|
$url = $this->holidayUrl."{$zone}.json";
|
||||||
$url
|
|
||||||
);
|
|
||||||
} catch (TransportExceptionInterface) {
|
|
||||||
throw new RuntimeException('Unable to reach public holidays API.');
|
|
||||||
} catch (ClientExceptionInterface) {
|
|
||||||
throw new RuntimeException('Invalid zone provided for public holidays.');
|
|
||||||
} catch (ServerExceptionInterface) {
|
|
||||||
throw new RuntimeException('Public holidays API is temporarily unavailable.');
|
|
||||||
} catch (Throwable) {
|
|
||||||
throw new RuntimeException('Unexpected error while fetching public holidays.');
|
|
||||||
}
|
|
||||||
|
|
||||||
return json_decode($response->getContent(), true);
|
try {
|
||||||
|
$response = $this->client->request(
|
||||||
|
'GET',
|
||||||
|
$url
|
||||||
|
);
|
||||||
|
} catch (TransportExceptionInterface) {
|
||||||
|
throw new RuntimeException('Unable to reach public holidays API.');
|
||||||
|
} catch (ClientExceptionInterface) {
|
||||||
|
throw new RuntimeException('Invalid zone provided for public holidays.');
|
||||||
|
} catch (ServerExceptionInterface) {
|
||||||
|
throw new RuntimeException('Public holidays API is temporarily unavailable.');
|
||||||
|
} catch (Throwable) {
|
||||||
|
throw new RuntimeException('Unexpected error while fetching public holidays.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return json_decode($response->getContent(), true);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -60,20 +68,25 @@ final readonly class PublicHolidayService implements PublicHolidayServiceInterfa
|
|||||||
{
|
{
|
||||||
$zone = strtolower(trim($zone));
|
$zone = strtolower(trim($zone));
|
||||||
$years = trim($years);
|
$years = trim($years);
|
||||||
$url = $this->holidayUrl."{$zone}/{$years}.json";
|
$key = "public_holidays_{$zone}_{$years}";
|
||||||
|
|
||||||
try {
|
return $this->cache->get($key, function (ItemInterface $item) use ($zone, $years): array {
|
||||||
$response = $this->client->request('GET', $url);
|
$item->expiresAfter(30 * 86400);
|
||||||
} catch (TransportExceptionInterface) {
|
$url = $this->holidayUrl."{$zone}/{$years}.json";
|
||||||
throw new RuntimeException('Unable to reach public holidays API.');
|
|
||||||
} catch (ClientExceptionInterface) {
|
|
||||||
throw new RuntimeException('Invalid zone or year provided for public holidays.');
|
|
||||||
} catch (ServerExceptionInterface) {
|
|
||||||
throw new RuntimeException('Public holidays API is temporarily unavailable.');
|
|
||||||
} catch (Throwable) {
|
|
||||||
throw new RuntimeException('Unexpected error while fetching public holidays.');
|
|
||||||
}
|
|
||||||
|
|
||||||
return json_decode($response->getContent(), true);
|
try {
|
||||||
|
$response = $this->client->request('GET', $url);
|
||||||
|
} catch (TransportExceptionInterface) {
|
||||||
|
throw new RuntimeException('Unable to reach public holidays API.');
|
||||||
|
} catch (ClientExceptionInterface) {
|
||||||
|
throw new RuntimeException('Invalid zone or year provided for public holidays.');
|
||||||
|
} catch (ServerExceptionInterface) {
|
||||||
|
throw new RuntimeException('Public holidays API is temporarily unavailable.');
|
||||||
|
} catch (Throwable) {
|
||||||
|
throw new RuntimeException('Unexpected error while fetching public holidays.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return json_decode($response->getContent(), true);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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');
|
||||||
@@ -195,20 +206,36 @@ final readonly class RttRecoveryComputationService
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$weekAnchorNature = $naturesByDate[$employeeId][$weekDays[0]] ?? ContractNature::CDI;
|
$weekAnchorNature = $naturesByDate[$employeeId][$weekDays[0]] ?? ContractNature::CDI;
|
||||||
$weekAnchorContract = $employeeContractsByDate[$weekDays[0]] ?? null;
|
$weekAnchorContract = $employeeContractsByDate[$weekDays[0]] ?? null;
|
||||||
$isWeekPresenceTracking = TrackingMode::PRESENCE->value === $weekAnchorContract?->getTrackingMode();
|
$isWeekPresenceTracking = TrackingMode::PRESENCE->value === $weekAnchorContract?->getTrackingMode();
|
||||||
$disableOvertimeBonuses = $this->hasDisabledOvertimeBonuses($weekAnchorContract, $weekAnchorNature);
|
$disableOvertimeBonuses = $this->hasDisabledOvertimeBonuses($weekAnchorContract, $weekAnchorNature);
|
||||||
$overtimeReferenceMinutes = $this->computeWeeklyOvertimeReferenceMinutes($weekDays, $employeeContractsByDate);
|
$weekContractType = ContractType::resolve(
|
||||||
|
$weekAnchorContract?->getName(),
|
||||||
|
$weekAnchorContract?->getTrackingMode(),
|
||||||
|
$weekAnchorContract?->getWeeklyHours()
|
||||||
|
);
|
||||||
|
$isCustomContract = ContractType::CUSTOM === $weekContractType;
|
||||||
|
$overtimeReferenceMinutes = $isCustomContract
|
||||||
|
? $this->computeWeeklyCustomReferenceMinutes($weekDays, $employeeContractsByDate)
|
||||||
|
: $this->computeWeeklyOvertimeReferenceMinutes($weekDays, $employeeContractsByDate);
|
||||||
$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 || $isCustomContract) ? 0 : max(0, min($weeklyTotalMinutes, 43 * 60) - $overtime25StartMinutes);
|
||||||
$bonus25 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : (int) round($base25 * 0.25);
|
$bonus25 = ($isWeekPresenceTracking || $disableOvertimeBonuses || $isCustomContract) ? 0 : (int) round($base25 * 0.25);
|
||||||
$base50 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : max(0, $weeklyTotalMinutes - 43 * 60);
|
$base50 = ($isWeekPresenceTracking || $disableOvertimeBonuses || $isCustomContract) ? 0 : max(0, $weeklyTotalMinutes - 43 * 60);
|
||||||
$bonus50 = ($isWeekPresenceTracking || $disableOvertimeBonuses) ? 0 : (int) round($base50 * 0.5);
|
$bonus50 = ($isWeekPresenceTracking || $disableOvertimeBonuses || $isCustomContract) ? 0 : (int) round($base50 * 0.5);
|
||||||
|
|
||||||
|
if ($isWeekPresenceTracking || $disableOvertimeBonuses) {
|
||||||
|
$totalMinutes = 0;
|
||||||
|
} elseif ($isCustomContract) {
|
||||||
|
$totalMinutes = max(0, $weeklyOvertimeTotalMinutes);
|
||||||
|
} else {
|
||||||
|
$totalMinutes = $weeklyOvertimeTotalMinutes + $bonus25 + $bonus50;
|
||||||
|
}
|
||||||
|
|
||||||
$results[$weekKey] = new WeekRecoveryDetail(
|
$results[$weekKey] = new WeekRecoveryDetail(
|
||||||
overtimeMinutes: $weeklyOvertimeTotalMinutes,
|
overtimeMinutes: $weeklyOvertimeTotalMinutes,
|
||||||
@@ -216,9 +243,7 @@ final readonly class RttRecoveryComputationService
|
|||||||
bonus25Minutes: $bonus25,
|
bonus25Minutes: $bonus25,
|
||||||
base50Minutes: $base50,
|
base50Minutes: $base50,
|
||||||
bonus50Minutes: $bonus50,
|
bonus50Minutes: $bonus50,
|
||||||
totalMinutes: ($isWeekPresenceTracking || $disableOvertimeBonuses)
|
totalMinutes: $totalMinutes,
|
||||||
? 0
|
|
||||||
: $weeklyOvertimeTotalMinutes + $bonus25 + $bonus50,
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -227,6 +252,20 @@ final readonly class RttRecoveryComputationService
|
|||||||
|
|
||||||
private function computeMetrics(WorkHour $workHour): WorkMetrics
|
private function computeMetrics(WorkHour $workHour): WorkMetrics
|
||||||
{
|
{
|
||||||
|
$driverDay = $workHour->getDayHoursMinutes() ?? 0;
|
||||||
|
$driverNight = $workHour->getNightHoursMinutes() ?? 0;
|
||||||
|
$driverWorkshop = $workHour->getWorkshopHoursMinutes() ?? 0;
|
||||||
|
|
||||||
|
if ($driverDay > 0 || $driverNight > 0 || $driverWorkshop > 0) {
|
||||||
|
$totalMinutes = $driverDay + $driverNight + $driverWorkshop;
|
||||||
|
|
||||||
|
return new WorkMetrics(
|
||||||
|
dayMinutes: $driverDay + $driverWorkshop,
|
||||||
|
nightMinutes: $driverNight,
|
||||||
|
totalMinutes: $totalMinutes,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
$ranges = [
|
$ranges = [
|
||||||
[$workHour->getMorningFrom(), $workHour->getMorningTo()],
|
[$workHour->getMorningFrom(), $workHour->getMorningTo()],
|
||||||
[$workHour->getAfternoonFrom(), $workHour->getAfternoonTo()],
|
[$workHour->getAfternoonFrom(), $workHour->getAfternoonTo()],
|
||||||
@@ -315,6 +354,23 @@ final readonly class RttRecoveryComputationService
|
|||||||
return max(0, $end - $start);
|
return max(0, $end - $start);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<string> $days
|
||||||
|
* @param array<string, ?Contract> $contractsByDate
|
||||||
|
*/
|
||||||
|
private function computeWeeklyCustomReferenceMinutes(array $days, array $contractsByDate): int
|
||||||
|
{
|
||||||
|
$total = 0;
|
||||||
|
foreach ($days as $date) {
|
||||||
|
$isoDay = (int) new DateTimeImmutable($date)->format('N');
|
||||||
|
$contract = $contractsByDate[$date] ?? null;
|
||||||
|
$hours = $contract?->getWeeklyHours();
|
||||||
|
$total += $this->resolveDailyReferenceMinutes($hours, $isoDay);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $total;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param list<string> $days
|
* @param list<string> $days
|
||||||
* @param array<string, ?Contract> $contractsByDate
|
* @param array<string, ?Contract> $contractsByDate
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ use App\Repository\EmployeeRepository;
|
|||||||
use App\Repository\WorkHourRepository;
|
use App\Repository\WorkHourRepository;
|
||||||
use App\Security\EmployeeScopeService;
|
use App\Security\EmployeeScopeService;
|
||||||
use App\Service\Leave\LeaveBalanceComputationService;
|
use App\Service\Leave\LeaveBalanceComputationService;
|
||||||
|
use App\Service\Leave\LongMaladieService;
|
||||||
use App\Service\Leave\SuspensionDaysCalculator;
|
use App\Service\Leave\SuspensionDaysCalculator;
|
||||||
use App\Service\PublicHolidayServiceInterface;
|
use App\Service\PublicHolidayServiceInterface;
|
||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
@@ -42,6 +43,7 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
private const float CDI_NON_FORFAIT_4H_ACQUIRED_SATURDAYS = 0.0;
|
private const float CDI_NON_FORFAIT_4H_ACQUIRED_SATURDAYS = 0.0;
|
||||||
private const float CDI_NON_FORFAIT_4H_ACCRUAL_PER_MONTH = 0.83;
|
private const float CDI_NON_FORFAIT_4H_ACCRUAL_PER_MONTH = 0.83;
|
||||||
private const float CDI_NON_FORFAIT_4H_SATURDAY_ACCRUAL_PER_MONTH = 0.0;
|
private const float CDI_NON_FORFAIT_4H_SATURDAY_ACCRUAL_PER_MONTH = 0.0;
|
||||||
|
private const float LONG_MALADIE_MONTHLY_ACCRUAL = 2.0;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private Security $security,
|
private Security $security,
|
||||||
@@ -52,6 +54,7 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
private EmployeeContractPeriodRepository $periodRepository,
|
private EmployeeContractPeriodRepository $periodRepository,
|
||||||
private EmployeeLeaveBalanceRepository $leaveBalanceRepository,
|
private EmployeeLeaveBalanceRepository $leaveBalanceRepository,
|
||||||
private LeaveBalanceComputationService $leaveBalanceComputationService,
|
private LeaveBalanceComputationService $leaveBalanceComputationService,
|
||||||
|
private LongMaladieService $longMaladieService,
|
||||||
private PublicHolidayServiceInterface $publicHolidayService,
|
private PublicHolidayServiceInterface $publicHolidayService,
|
||||||
private SuspensionDaysCalculator $suspensionDaysCalculator,
|
private SuspensionDaysCalculator $suspensionDaysCalculator,
|
||||||
private WorkHourRepository $workHourRepository,
|
private WorkHourRepository $workHourRepository,
|
||||||
@@ -126,9 +129,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;
|
||||||
}
|
}
|
||||||
@@ -187,13 +190,29 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
$suspensions = $this->suspensionDaysCalculator->applyFirstMonthGrace(
|
$suspensions = $this->suspensionDaysCalculator->applyFirstMonthGrace(
|
||||||
$this->resolveSuspensionsForPeriod($employee, $effectiveFrom, $to)
|
$this->resolveSuspensionsForPeriod($employee, $effectiveFrom, $to)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
$longMaladiePeriods = [];
|
||||||
|
$longMaladieReductionFactor = 1.0;
|
||||||
|
if (LeaveRuleCode::CDI_CDD_NON_FORFAIT->value === $leavePolicy['ruleCode']
|
||||||
|
&& 4 !== $employee->getContract()?->getWeeklyHours()
|
||||||
|
&& null !== $accrualCalculationEnd
|
||||||
|
) {
|
||||||
|
$longMaladiePeriods = $this->longMaladieService->findReducedRatePeriods($employee, $effectiveFrom, $accrualCalculationEnd);
|
||||||
|
if ([] !== $longMaladiePeriods) {
|
||||||
|
$totalNormalAccrual = $leavePolicy['accrualPerMonth'] + $leavePolicy['saturdayAccrualPerMonth'];
|
||||||
|
$longMaladieReductionFactor = self::LONG_MALADIE_MONTHLY_ACCRUAL / $totalNormalAccrual;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$generatedDays = $leavePolicy['accrualPerMonth'] > 0.0
|
$generatedDays = $leavePolicy['accrualPerMonth'] > 0.0
|
||||||
? $this->computeAccruedDaysFromStart(
|
? $this->computeAccruedDaysFromStart(
|
||||||
$leavePolicy['acquiredDays'],
|
$leavePolicy['acquiredDays'],
|
||||||
$leavePolicy['accrualPerMonth'],
|
$leavePolicy['accrualPerMonth'],
|
||||||
$effectiveFrom,
|
$effectiveFrom,
|
||||||
$accrualCalculationEnd,
|
$accrualCalculationEnd,
|
||||||
$suspensions
|
$suspensions,
|
||||||
|
$longMaladiePeriods,
|
||||||
|
$longMaladieReductionFactor
|
||||||
)
|
)
|
||||||
: 0.0;
|
: 0.0;
|
||||||
$generatedSaturdays = $leavePolicy['saturdayAccrualPerMonth'] > 0.0
|
$generatedSaturdays = $leavePolicy['saturdayAccrualPerMonth'] > 0.0
|
||||||
@@ -202,7 +221,9 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
$leavePolicy['saturdayAccrualPerMonth'],
|
$leavePolicy['saturdayAccrualPerMonth'],
|
||||||
$effectiveFrom,
|
$effectiveFrom,
|
||||||
$accrualCalculationEnd,
|
$accrualCalculationEnd,
|
||||||
$suspensions
|
$suspensions,
|
||||||
|
$longMaladiePeriods,
|
||||||
|
$longMaladieReductionFactor
|
||||||
)
|
)
|
||||||
: 0.0;
|
: 0.0;
|
||||||
$absences = $this->absenceRepository->findByEmployeeAndOverlappingDateRange($employee, $from, $to);
|
$absences = $this->absenceRepository->findByEmployeeAndOverlappingDateRange($employee, $from, $to);
|
||||||
@@ -286,6 +307,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,
|
||||||
@@ -365,12 +396,18 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
return $year;
|
return $year;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<ContractSuspension> $suspensions
|
||||||
|
* @param list<array{start: DateTimeImmutable, end: DateTimeImmutable}> $longMaladiePeriods
|
||||||
|
*/
|
||||||
private function computeAccruedDaysFromStart(
|
private function computeAccruedDaysFromStart(
|
||||||
float $acquiredDays,
|
float $acquiredDays,
|
||||||
float $accrualPerMonth,
|
float $accrualPerMonth,
|
||||||
DateTimeImmutable $periodStart,
|
DateTimeImmutable $periodStart,
|
||||||
?DateTimeImmutable $periodEnd,
|
?DateTimeImmutable $periodEnd,
|
||||||
array $suspensions = []
|
array $suspensions = [],
|
||||||
|
array $longMaladiePeriods = [],
|
||||||
|
float $longMaladieReductionFactor = 1.0
|
||||||
): float {
|
): float {
|
||||||
if ($accrualPerMonth <= 0.0) {
|
if ($accrualPerMonth <= 0.0) {
|
||||||
return $acquiredDays;
|
return $acquiredDays;
|
||||||
@@ -380,10 +417,12 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
return 0.0;
|
return 0.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$periodStart = $this->normalizeDate($periodStart);
|
$periodStart = $this->normalizeDate($periodStart);
|
||||||
$periodEnd = $this->normalizeDate($periodEnd);
|
$periodEnd = $this->normalizeDate($periodEnd);
|
||||||
$coveredMonths = 0.0;
|
$publicHolidays = [] !== $suspensions ? $this->buildPublicHolidayMap($periodStart, $periodEnd) : [];
|
||||||
$cursor = $periodStart->modify('first day of this month')->setTime(0, 0);
|
$normalMonths = 0.0;
|
||||||
|
$reducedMonths = 0.0;
|
||||||
|
$cursor = $periodStart->modify('first day of this month')->setTime(0, 0);
|
||||||
while ($cursor <= $periodEnd) {
|
while ($cursor <= $periodEnd) {
|
||||||
$monthStart = $cursor > $periodStart ? $cursor : $periodStart;
|
$monthStart = $cursor > $periodStart ? $cursor : $periodStart;
|
||||||
$monthEnd = $cursor->modify('last day of this month')->setTime(0, 0);
|
$monthEnd = $cursor->modify('last day of this month')->setTime(0, 0);
|
||||||
@@ -391,18 +430,39 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
$monthEnd = $periodEnd;
|
$monthEnd = $periodEnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
$coveredDays = ((int) $monthEnd->diff($monthStart)->format('%a')) + 1;
|
|
||||||
if ([] !== $suspensions) {
|
if ([] !== $suspensions) {
|
||||||
$suspendedDays = $this->suspensionDaysCalculator->countSuspendedDaysInMonth($monthStart, $monthEnd, $suspensions);
|
$suspendedDays = $this->suspensionDaysCalculator->countSuspendedDaysInMonth($monthStart, $monthEnd, $suspensions);
|
||||||
$coveredDays = max(0, $coveredDays - $suspendedDays);
|
if ($suspendedDays > 0) {
|
||||||
|
$businessDays = $this->countBusinessDays($monthStart, $monthEnd, $publicHolidays);
|
||||||
|
$suspendedBusinessDays = $this->suspensionDaysCalculator->countSuspendedBusinessDays($monthStart, $monthEnd, $suspensions, $publicHolidays);
|
||||||
|
$normalMonths += max(0, $businessDays - $suspendedBusinessDays) / 22.0;
|
||||||
|
$cursor = $cursor->modify('first day of next month');
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$coveredDays = ((int) $monthEnd->diff($monthStart)->format('%a')) + 1;
|
||||||
$daysInMonth = (int) $cursor->format('t');
|
$daysInMonth = (int) $cursor->format('t');
|
||||||
$coveredMonths += $coveredDays / $daysInMonth;
|
|
||||||
|
if ([] !== $longMaladiePeriods) {
|
||||||
|
$reducedDays = $this->longMaladieService->countReducedDaysInMonth($monthStart, $monthEnd, $longMaladiePeriods);
|
||||||
|
if ($reducedDays > 0) {
|
||||||
|
$normalDays = max(0, $coveredDays - $reducedDays);
|
||||||
|
$normalMonths += $normalDays / $daysInMonth;
|
||||||
|
$reducedMonths += min($coveredDays, $reducedDays) / $daysInMonth;
|
||||||
|
$cursor = $cursor->modify('first day of next month');
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$normalMonths += $coveredDays / $daysInMonth;
|
||||||
|
|
||||||
$cursor = $cursor->modify('first day of next month');
|
$cursor = $cursor->modify('first day of next month');
|
||||||
}
|
}
|
||||||
|
|
||||||
return min($acquiredDays, $coveredMonths * $accrualPerMonth);
|
return min($acquiredDays, ($normalMonths + $reducedMonths * $longMaladieReductionFactor) * $accrualPerMonth);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function resolveAccrualCalculationEndDate(
|
private function resolveAccrualCalculationEndDate(
|
||||||
@@ -475,10 +535,21 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
$type = $employee->getContract()?->getType();
|
$type = $employee->getContract()?->getType();
|
||||||
if (ContractType::FORFAIT === $type) {
|
if (ContractType::FORFAIT === $type) {
|
||||||
$businessDaysInPeriod = $this->countBusinessDays($from, $to);
|
$businessDaysInPeriod = $this->countBusinessDays($from, $to);
|
||||||
|
$publicHolidays = $this->buildPublicHolidayMap($from, $to);
|
||||||
|
$weekdayHolidays = array_filter(
|
||||||
|
array_keys($publicHolidays),
|
||||||
|
static fn (string $date): bool => (int) new DateTimeImmutable($date)->format('N') <= 5
|
||||||
|
);
|
||||||
|
$bonusDays = $this->workHourRepository->countWeekendAndHolidayWorkedDays(
|
||||||
|
$employee,
|
||||||
|
$from,
|
||||||
|
$to,
|
||||||
|
array_values($weekdayHolidays)
|
||||||
|
);
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'ruleCode' => LeaveRuleCode::FORFAIT_218->value,
|
'ruleCode' => LeaveRuleCode::FORFAIT_218->value,
|
||||||
'acquiredDays' => (float) max(0, $businessDaysInPeriod - self::FORFAIT_TARGET_WORKED_DAYS),
|
'acquiredDays' => (float) max(0, $businessDaysInPeriod - self::FORFAIT_TARGET_WORKED_DAYS) + $bonusDays,
|
||||||
'acquiredSaturdays' => 0.0,
|
'acquiredSaturdays' => 0.0,
|
||||||
'accrualPerMonth' => 0.0,
|
'accrualPerMonth' => 0.0,
|
||||||
'saturdayAccrualPerMonth' => 0.0,
|
'saturdayAccrualPerMonth' => 0.0,
|
||||||
@@ -516,10 +587,13 @@ final readonly class EmployeeLeaveSummaryProvider implements ProviderInterface
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
private function countBusinessDays(DateTimeImmutable $from, DateTimeImmutable $to): int
|
/**
|
||||||
|
* @param null|array<string, string> $publicHolidays pre-built map (built if null)
|
||||||
|
*/
|
||||||
|
private function countBusinessDays(DateTimeImmutable $from, DateTimeImmutable $to, ?array $publicHolidays = null): int
|
||||||
{
|
{
|
||||||
$publicHolidays = $this->buildPublicHolidayMap($from, $to);
|
$publicHolidays ??= $this->buildPublicHolidayMap($from, $to);
|
||||||
$count = 0;
|
$count = 0;
|
||||||
for ($cursor = $from; $cursor <= $to; $cursor = $cursor->modify('+1 day')) {
|
for ($cursor = $from; $cursor <= $to; $cursor = $cursor->modify('+1 day')) {
|
||||||
$weekDay = (int) $cursor->format('N');
|
$weekDay = (int) $cursor->format('N');
|
||||||
$dayKey = $cursor->format('Y-m-d');
|
$dayKey = $cursor->format('Y-m-d');
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ use App\Entity\User;
|
|||||||
use App\Repository\EmployeeRepository;
|
use App\Repository\EmployeeRepository;
|
||||||
use App\Repository\EmployeeRttBalanceRepository;
|
use App\Repository\EmployeeRttBalanceRepository;
|
||||||
use App\Repository\EmployeeRttPaymentRepository;
|
use App\Repository\EmployeeRttPaymentRepository;
|
||||||
|
use App\Repository\WorkHourRepository;
|
||||||
use App\Security\EmployeeScopeService;
|
use App\Security\EmployeeScopeService;
|
||||||
use App\Service\Rtt\RttRecoveryComputationService;
|
use App\Service\Rtt\RttRecoveryComputationService;
|
||||||
use DateTimeImmutable;
|
use DateTimeImmutable;
|
||||||
@@ -26,6 +27,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 +37,11 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
private EmployeeRttBalanceRepository $rttBalanceRepository,
|
private EmployeeRttBalanceRepository $rttBalanceRepository,
|
||||||
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
||||||
private RttRecoveryComputationService $rttRecoveryService,
|
private RttRecoveryComputationService $rttRecoveryService,
|
||||||
) {}
|
private WorkHourRepository $workHourRepository,
|
||||||
|
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 +79,22 @@ 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');
|
||||||
|
|
||||||
|
// Include the current week if all existing days are admin-validated
|
||||||
|
if (7 !== $isoDay) {
|
||||||
|
$currentWeekStart = $today->modify('monday this week');
|
||||||
|
$currentWeekEnd = $currentWeekStart->modify('+6 days');
|
||||||
|
$checkEnd = $this->resolveWeekEndForEmployee($employee, $currentWeekStart, $currentWeekEnd, $today);
|
||||||
|
if ($this->workHourRepository->isWeekFullyValidated($employee, $currentWeekStart, $checkEnd)) {
|
||||||
|
$limitDate = $currentWeekEnd;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$currentByWeekStart = $this->rttRecoveryService->computeRecoveryByWeek($employee, $weekRanges, $periodFrom, $periodTo, $limitDate);
|
$currentByWeekStart = $this->rttRecoveryService->computeRecoveryByWeek($employee, $weekRanges, $periodFrom, $periodTo, $limitDate);
|
||||||
@@ -90,7 +110,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 +138,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 = [];
|
||||||
|
|
||||||
@@ -189,4 +248,25 @@ final readonly class EmployeeRttSummaryProvider implements ProviderInterface
|
|||||||
|
|
||||||
return $month >= 6 ? $year + 1 : $year;
|
return $month >= 6 ? $year + 1 : $year;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the employee's contract ends within the current week, cap the check range to that end date.
|
||||||
|
*/
|
||||||
|
private function resolveWeekEndForEmployee(Employee $employee, DateTimeImmutable $weekStart, DateTimeImmutable $weekEnd, DateTimeImmutable $today): DateTimeImmutable
|
||||||
|
{
|
||||||
|
foreach ($employee->getContractPeriods() as $period) {
|
||||||
|
if ($period->getStartDate() > $today) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$endDate = $period->getEndDate();
|
||||||
|
if (null === $endDate) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($endDate >= $weekStart && $endDate <= $weekEnd) {
|
||||||
|
return $endDate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $weekEnd;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -65,7 +65,8 @@ final readonly class EmployeeWriteProcessor implements ProcessorInterface
|
|||||||
contract: $currentContract,
|
contract: $currentContract,
|
||||||
startDate: $startDate,
|
startDate: $startDate,
|
||||||
endDate: $changeRequest->contractEndDate,
|
endDate: $changeRequest->contractEndDate,
|
||||||
nature: $nature
|
nature: $nature,
|
||||||
|
isDriver: $changeRequest->isDriver ?? false,
|
||||||
);
|
);
|
||||||
|
|
||||||
$data->setEntryDate($startDate);
|
$data->setEntryDate($startDate);
|
||||||
@@ -108,7 +109,8 @@ final readonly class EmployeeWriteProcessor implements ProcessorInterface
|
|||||||
startDate: $startDate,
|
startDate: $startDate,
|
||||||
endDate: $changeRequest->contractEndDate,
|
endDate: $changeRequest->contractEndDate,
|
||||||
nature: $nature,
|
nature: $nature,
|
||||||
todayPeriod: $todayPeriod
|
todayPeriod: $todayPeriod,
|
||||||
|
isDriver: $changeRequest->isDriver ?? false,
|
||||||
);
|
);
|
||||||
|
|
||||||
return $result;
|
return $result;
|
||||||
|
|||||||
211
src/State/LeaveRecapPrintProvider.php
Normal file
211
src/State/LeaveRecapPrintProvider.php
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
<?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\Repository\WorkHourRepository;
|
||||||
|
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,
|
||||||
|
private WorkHourRepository $workHourRepository,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
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');
|
||||||
|
|
||||||
|
// Include the current week if all existing days are admin-validated
|
||||||
|
if (7 !== $isoDay) {
|
||||||
|
$currentWeekStart = $today->modify('monday this week');
|
||||||
|
$currentWeekEnd = $currentWeekStart->modify('+6 days');
|
||||||
|
$checkEnd = $this->resolveWeekEndForEmployee($employee, $currentWeekStart, $currentWeekEnd, $today);
|
||||||
|
if ($this->workHourRepository->isWeekFullyValidated($employee, $currentWeekStart, $checkEnd)) {
|
||||||
|
$limitDate = $currentWeekEnd;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 resolveWeekEndForEmployee(Employee $employee, DateTimeImmutable $weekStart, DateTimeImmutable $weekEnd, DateTimeImmutable $today): DateTimeImmutable
|
||||||
|
{
|
||||||
|
foreach ($employee->getContractPeriods() as $period) {
|
||||||
|
if ($period->getStartDate() > $today) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$endDate = $period->getEndDate();
|
||||||
|
if (null === $endDate) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($endDate >= $weekStart && $endDate <= $weekEnd) {
|
||||||
|
return $endDate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $weekEnd;
|
||||||
|
}
|
||||||
|
|
||||||
|
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";
|
||||||
|
}
|
||||||
|
}
|
||||||
53
src/State/MileageAllowanceAmountReceiptDownloadProvider.php
Normal file
53
src/State/MileageAllowanceAmountReceiptDownloadProvider.php
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\State;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\Operation;
|
||||||
|
use ApiPlatform\State\ProviderInterface;
|
||||||
|
use App\Entity\MileageAllowance;
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
|
use Symfony\Component\HttpFoundation\BinaryFileResponse;
|
||||||
|
use Symfony\Component\HttpFoundation\HeaderUtils;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||||||
|
|
||||||
|
final readonly class MileageAllowanceAmountReceiptDownloadProvider implements ProviderInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private EntityManagerInterface $entityManager,
|
||||||
|
#[Autowire('%kernel.project_dir%/var/uploads')]
|
||||||
|
private string $uploadDir,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function provide(Operation $operation, array $uriVariables = [], array $context = []): BinaryFileResponse
|
||||||
|
{
|
||||||
|
$mileageAllowance = $this->entityManager->find(MileageAllowance::class, $uriVariables['id']);
|
||||||
|
|
||||||
|
if (null === $mileageAllowance) {
|
||||||
|
throw new NotFoundHttpException('Mileage allowance not found.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$receiptPath = $mileageAllowance->getAmountReceiptPath();
|
||||||
|
|
||||||
|
if (null === $receiptPath) {
|
||||||
|
throw new NotFoundHttpException('No amount receipt found for this mileage allowance.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$absolutePath = sprintf('%s/%s', $this->uploadDir, $receiptPath);
|
||||||
|
|
||||||
|
if (!file_exists($absolutePath)) {
|
||||||
|
throw new NotFoundHttpException('Amount receipt file not found.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$response = new BinaryFileResponse($absolutePath);
|
||||||
|
$disposition = HeaderUtils::makeDisposition(
|
||||||
|
HeaderUtils::DISPOSITION_ATTACHMENT,
|
||||||
|
$mileageAllowance->getAmountReceiptName() ?? 'justificatif.pdf'
|
||||||
|
);
|
||||||
|
$response->headers->set('Content-Disposition', $disposition);
|
||||||
|
|
||||||
|
return $response;
|
||||||
|
}
|
||||||
|
}
|
||||||
66
src/State/MileageAllowanceAmountReceiptUploadProcessor.php
Normal file
66
src/State/MileageAllowanceAmountReceiptUploadProcessor.php
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\State;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\Operation;
|
||||||
|
use ApiPlatform\State\ProcessorInterface;
|
||||||
|
use App\Entity\MileageAllowance;
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\DependencyInjection\Attribute\Autowire;
|
||||||
|
use Symfony\Component\HttpFoundation\JsonResponse;
|
||||||
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
|
||||||
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
|
||||||
|
final readonly class MileageAllowanceAmountReceiptUploadProcessor implements ProcessorInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private EntityManagerInterface $entityManager,
|
||||||
|
private RequestStack $requestStack,
|
||||||
|
#[Autowire('%kernel.project_dir%/var/uploads')]
|
||||||
|
private string $uploadDir,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function process(mixed $data, Operation $operation, array $uriVariables = [], array $context = []): JsonResponse
|
||||||
|
{
|
||||||
|
if (!$data instanceof MileageAllowance) {
|
||||||
|
throw new BadRequestHttpException('Invalid entity.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$request = $this->requestStack->getCurrentRequest();
|
||||||
|
$file = $request?->files->get('file');
|
||||||
|
|
||||||
|
if (null === $file) {
|
||||||
|
throw new BadRequestHttpException('No file uploaded.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if ('application/pdf' !== $file->getMimeType()) {
|
||||||
|
throw new BadRequestHttpException('Only PDF files are accepted.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$month = $data->getMonth();
|
||||||
|
$year = $month?->format('Y') ?? date('Y');
|
||||||
|
$monthNumber = $month?->format('m') ?? date('m');
|
||||||
|
$relativePath = sprintf('mileage-receipts/%s/%s', $year, $monthNumber);
|
||||||
|
$absoluteDir = sprintf('%s/%s', $this->uploadDir, $relativePath);
|
||||||
|
|
||||||
|
if (!is_dir($absoluteDir)) {
|
||||||
|
mkdir($absoluteDir, 0o755, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
$filename = Uuid::v4()->toRfc4122().'.pdf';
|
||||||
|
$fullRelative = sprintf('%s/%s', $relativePath, $filename);
|
||||||
|
$originalName = $file->getClientOriginalName();
|
||||||
|
|
||||||
|
$file->move($absoluteDir, $filename);
|
||||||
|
|
||||||
|
$data->setAmountReceiptPath($fullRelative);
|
||||||
|
$data->setAmountReceiptName($originalName);
|
||||||
|
$this->entityManager->flush();
|
||||||
|
|
||||||
|
return new JsonResponse(['path' => $fullRelative, 'name' => $originalName], Response::HTTP_OK);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,6 +34,16 @@ final readonly class MileageAllowanceDeleteProcessor implements ProcessorInterfa
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$amountReceiptPath = $data->getAmountReceiptPath();
|
||||||
|
|
||||||
|
if (null !== $amountReceiptPath) {
|
||||||
|
$absolutePath = sprintf('%s/%s', $this->uploadDir, $amountReceiptPath);
|
||||||
|
|
||||||
|
if (file_exists($absolutePath)) {
|
||||||
|
unlink($absolutePath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$this->entityManager->remove($data);
|
$this->entityManager->remove($data);
|
||||||
$this->entityManager->flush();
|
$this->entityManager->flush();
|
||||||
|
|
||||||
|
|||||||
588
src/State/SalaryRecapPrintProvider.php
Normal file
588
src/State/SalaryRecapPrintProvider.php
Normal file
@@ -0,0 +1,588 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace App\State;
|
||||||
|
|
||||||
|
use ApiPlatform\Metadata\Operation;
|
||||||
|
use ApiPlatform\State\ProviderInterface;
|
||||||
|
use App\Entity\Absence;
|
||||||
|
use App\Entity\Employee;
|
||||||
|
use App\Entity\WorkHour;
|
||||||
|
use App\Enum\TrackingMode;
|
||||||
|
use App\Repository\AbsenceRepository;
|
||||||
|
use App\Repository\BonusRepository;
|
||||||
|
use App\Repository\EmployeeRepository;
|
||||||
|
use App\Repository\EmployeeRttPaymentRepository;
|
||||||
|
use App\Repository\MileageAllowanceRepository;
|
||||||
|
use App\Repository\WorkHourRepository;
|
||||||
|
use App\Service\Contracts\EmployeeContractResolver;
|
||||||
|
use DateInterval;
|
||||||
|
use DateTimeImmutable;
|
||||||
|
use Dompdf\Dompdf;
|
||||||
|
use Dompdf\Options;
|
||||||
|
use Symfony\Component\HttpFoundation\RequestStack;
|
||||||
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Twig\Environment;
|
||||||
|
|
||||||
|
class SalaryRecapPrintProvider implements ProviderInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private Environment $twig,
|
||||||
|
private readonly RequestStack $requestStack,
|
||||||
|
private EmployeeRepository $employeeRepository,
|
||||||
|
private WorkHourRepository $workHourRepository,
|
||||||
|
private AbsenceRepository $absenceRepository,
|
||||||
|
private EmployeeRttPaymentRepository $rttPaymentRepository,
|
||||||
|
private BonusRepository $bonusRepository,
|
||||||
|
private MileageAllowanceRepository $mileageAllowanceRepository,
|
||||||
|
private EmployeeContractResolver $contractResolver,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function provide(Operation $operation, array $uriVariables = [], array $context = []): Response
|
||||||
|
{
|
||||||
|
$request = $this->requestStack->getCurrentRequest();
|
||||||
|
if (!$request) {
|
||||||
|
return new Response('Missing request.', Response::HTTP_BAD_REQUEST);
|
||||||
|
}
|
||||||
|
|
||||||
|
$month = $request->query->get('month');
|
||||||
|
if (!$month || !preg_match('/^\d{4}-\d{2}$/', $month)) {
|
||||||
|
return new Response('Missing or invalid month query param (expected YYYY-MM).', Response::HTTP_BAD_REQUEST);
|
||||||
|
}
|
||||||
|
|
||||||
|
$from = DateTimeImmutable::createFromFormat('Y-m-d', $month.'-01');
|
||||||
|
$to = $from->modify('last day of this month');
|
||||||
|
|
||||||
|
$employees = $this->employeeRepository->findForPrintBySiteIds([]);
|
||||||
|
$workHours = $this->workHourRepository->findByDateRangeAndEmployees($from, $to, $employees);
|
||||||
|
$absences = $this->absenceRepository->findForPrint($from, $to, $employees);
|
||||||
|
|
||||||
|
$year = (int) $from->format('Y');
|
||||||
|
$monthNumber = (int) $from->format('n');
|
||||||
|
$rttPayments = $this->rttPaymentRepository->findByYearAndMonth($year, $monthNumber);
|
||||||
|
|
||||||
|
$bonuses = $this->bonusRepository->findByMonth($from, $to);
|
||||||
|
$mileages = $this->mileageAllowanceRepository->findByMonth($from, $to);
|
||||||
|
|
||||||
|
$days = $this->buildDays($from, $to);
|
||||||
|
$contractMap = $this->contractResolver->resolveForEmployeesAndDays($employees, $days);
|
||||||
|
$driverMap = $this->contractResolver->resolveIsDriverForEmployeesAndDays($employees, $days);
|
||||||
|
|
||||||
|
$workHourMap = $this->buildWorkHourMap($workHours);
|
||||||
|
$absenceMap = $this->buildAbsenceMap($absences);
|
||||||
|
$rttPaymentMap = $this->buildRttPaymentMap($rttPayments);
|
||||||
|
$bonusMap = $this->buildBonusMap($bonuses);
|
||||||
|
$mileageMap = $this->buildMileageMap($mileages);
|
||||||
|
|
||||||
|
$siteGroups = $this->aggregateBySite($employees, $days, $contractMap, $driverMap, $workHourMap, $absenceMap, $rttPaymentMap, $bonusMap, $mileageMap);
|
||||||
|
|
||||||
|
$options = new Options();
|
||||||
|
$options->set('isRemoteEnabled', true);
|
||||||
|
|
||||||
|
$dompdf = new Dompdf($options);
|
||||||
|
$html = $this->twig->render('salary-recap/print.html.twig', [
|
||||||
|
'from' => $from,
|
||||||
|
'to' => $to,
|
||||||
|
'siteGroups' => $siteGroups,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$dompdf->loadHtml($html);
|
||||||
|
$dompdf->setPaper('A4', 'landscape');
|
||||||
|
$dompdf->render();
|
||||||
|
|
||||||
|
$filename = sprintf(
|
||||||
|
'recap_salaire_%s.pdf',
|
||||||
|
$from->format('Y-m')
|
||||||
|
);
|
||||||
|
|
||||||
|
return new Response($dompdf->output(), Response::HTTP_OK, [
|
||||||
|
'Content-Type' => 'application/pdf',
|
||||||
|
'Content-Disposition' => 'inline; filename="'.$filename.'"',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function buildDays(DateTimeImmutable $from, DateTimeImmutable $to): array
|
||||||
|
{
|
||||||
|
$days = [];
|
||||||
|
$current = $from;
|
||||||
|
|
||||||
|
while ($current <= $to) {
|
||||||
|
$days[] = $current->format('Y-m-d');
|
||||||
|
$current = $current->add(new DateInterval('P1D'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $days;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, array<string, WorkHour>>
|
||||||
|
*/
|
||||||
|
private function buildWorkHourMap(array $workHours): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
foreach ($workHours as $wh) {
|
||||||
|
$employeeId = $wh->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$date = $wh->getWorkDate()->format('Y-m-d');
|
||||||
|
$map[$employeeId][$date] = $wh;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, list<Absence>>
|
||||||
|
*/
|
||||||
|
private function buildAbsenceMap(array $absences): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
foreach ($absences as $absence) {
|
||||||
|
$employeeId = $absence->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$map[$employeeId][] = $absence;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, int>
|
||||||
|
*/
|
||||||
|
private function buildRttPaymentMap(array $rttPayments): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
foreach ($rttPayments as $payment) {
|
||||||
|
$employeeId = $payment->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$map[$employeeId] = ($map[$employeeId] ?? 0) + $payment->getBase25Minutes() + $payment->getBase50Minutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, float>
|
||||||
|
*/
|
||||||
|
private function buildBonusMap(array $bonuses): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
foreach ($bonuses as $bonus) {
|
||||||
|
$employeeId = $bonus->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$map[$employeeId] = ($map[$employeeId] ?? 0.0) + $bonus->getAmount();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, float>
|
||||||
|
*/
|
||||||
|
private function buildMileageMap(array $mileages): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
foreach ($mileages as $mileage) {
|
||||||
|
$employeeId = $mileage->getEmployee()?->getId();
|
||||||
|
if (!$employeeId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$map[$employeeId] = ($map[$employeeId] ?? 0.0) + $mileage->getKilometers();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $map;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function aggregateBySite(
|
||||||
|
array $employees,
|
||||||
|
array $days,
|
||||||
|
array $contractMap,
|
||||||
|
array $driverMap,
|
||||||
|
array $workHourMap,
|
||||||
|
array $absenceMap,
|
||||||
|
array $rttPaymentMap,
|
||||||
|
array $bonusMap,
|
||||||
|
array $mileageMap,
|
||||||
|
): array {
|
||||||
|
$siteGroups = [];
|
||||||
|
|
||||||
|
foreach ($employees as $employee) {
|
||||||
|
$employeeId = $employee->getId();
|
||||||
|
$site = $employee->getSite();
|
||||||
|
$siteName = $site ? $site->getName() : 'Sans site';
|
||||||
|
$siteId = $site ? $site->getId() : 0;
|
||||||
|
|
||||||
|
$row = $this->buildEmployeeRow(
|
||||||
|
$employee,
|
||||||
|
$employeeId,
|
||||||
|
$days,
|
||||||
|
$contractMap[$employeeId] ?? [],
|
||||||
|
$driverMap[$employeeId] ?? [],
|
||||||
|
$workHourMap[$employeeId] ?? [],
|
||||||
|
$absenceMap[$employeeId] ?? [],
|
||||||
|
$rttPaymentMap[$employeeId] ?? 0,
|
||||||
|
$bonusMap[$employeeId] ?? 0.0,
|
||||||
|
$mileageMap[$employeeId] ?? 0.0,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isset($siteGroups[$siteId])) {
|
||||||
|
$siteGroups[$siteId] = [
|
||||||
|
'name' => $siteName,
|
||||||
|
'color' => $site?->getColor() ?? '#ffd7d7',
|
||||||
|
'employees' => [],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
$siteGroups[$siteId]['employees'][] = $row;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $siteGroups;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function buildEmployeeRow(
|
||||||
|
Employee $employee,
|
||||||
|
int $employeeId,
|
||||||
|
array $days,
|
||||||
|
array $contractsByDate,
|
||||||
|
array $driverByDate,
|
||||||
|
array $workHoursByDate,
|
||||||
|
array $absences,
|
||||||
|
int $rttPaidMinutes,
|
||||||
|
float $bonusAmount,
|
||||||
|
float $mileageKm,
|
||||||
|
): array {
|
||||||
|
$contractName = null;
|
||||||
|
$presenceDays = 0.0;
|
||||||
|
$nightMinutesTotal = 0;
|
||||||
|
$nightBasketCount = 0;
|
||||||
|
$sundayMinutesTotal = 0;
|
||||||
|
$isDriverAnyDay = false;
|
||||||
|
$driverBreakfast = 0;
|
||||||
|
$driverMeals = 0;
|
||||||
|
$driverOvernight = 0;
|
||||||
|
$driverSaturdays = 0;
|
||||||
|
$isForfait = false;
|
||||||
|
|
||||||
|
foreach ($days as $date) {
|
||||||
|
$contract = $contractsByDate[$date] ?? null;
|
||||||
|
$isDriver = $driverByDate[$date] ?? false;
|
||||||
|
$wh = $workHoursByDate[$date] ?? null;
|
||||||
|
|
||||||
|
if ($contract && null === $contractName) {
|
||||||
|
$contractName = $contract->getName();
|
||||||
|
$isForfait = TrackingMode::PRESENCE === $contract->getTrackingModeEnum();
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($isDriver) {
|
||||||
|
$isDriverAnyDay = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$wh) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$dayOfWeek = (int) new DateTimeImmutable($date)->format('N');
|
||||||
|
|
||||||
|
if ($isDriver) {
|
||||||
|
$nightMinutesTotal += $wh->getNightHoursMinutes() ?? 0;
|
||||||
|
$dayMin = $wh->getDayHoursMinutes() ?? 0;
|
||||||
|
$nightMin = $wh->getNightHoursMinutes() ?? 0;
|
||||||
|
if (($nightMin > $dayMin && $nightMin > 0) || $nightMin >= 240) {
|
||||||
|
++$nightBasketCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($wh->getHasBreakfast()) {
|
||||||
|
++$driverBreakfast;
|
||||||
|
}
|
||||||
|
if ($wh->getHasLunch() || $wh->getHasDinner()) {
|
||||||
|
++$driverMeals;
|
||||||
|
}
|
||||||
|
if ($wh->getHasOvernight()) {
|
||||||
|
++$driverOvernight;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (6 === $dayOfWeek && ($dayMin > 0 || $nightMin > 0 || ($wh->getWorkshopHoursMinutes() ?? 0) > 0)) {
|
||||||
|
++$driverSaturdays;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (7 === $dayOfWeek) {
|
||||||
|
$sundayMinutesTotal += $dayMin + $nightMin + ($wh->getWorkshopHoursMinutes() ?? 0);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$metrics = $this->computeNightMinutes($wh);
|
||||||
|
$nightMinutesTotal += $metrics['nightMinutes'];
|
||||||
|
if (($metrics['nightMinutes'] > $metrics['dayMinutes'] && $metrics['nightMinutes'] > 0) || $metrics['nightMinutes'] >= 240) {
|
||||||
|
++$nightBasketCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (7 === $dayOfWeek) {
|
||||||
|
$sundayMinutesTotal += $metrics['dayMinutes'] + $metrics['nightMinutes'];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Samedi : les minutes après minuit débordent sur le dimanche
|
||||||
|
if (6 === $dayOfWeek) {
|
||||||
|
$sundayMinutesTotal += $this->computeOverflowAfterMidnight($wh);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($isForfait) {
|
||||||
|
if ($wh->getIsPresentMorning()) {
|
||||||
|
$presenceDays += 0.5;
|
||||||
|
}
|
||||||
|
if ($wh->getIsPresentAfternoon()) {
|
||||||
|
$presenceDays += 0.5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$conges = $this->countAbsencesByCode($absences, ['C']);
|
||||||
|
$maladie = $this->countAbsencesByCode($absences, ['M', 'AT']);
|
||||||
|
|
||||||
|
$nightHours = round($nightMinutesTotal / 60, 2);
|
||||||
|
$paidHours = round($rttPaidMinutes / 60, 2);
|
||||||
|
$sundayHours = round($sundayMinutesTotal / 60, 2);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'lastName' => mb_strimwidth($employee->getLastName() ?? '', 0, 15, '...'),
|
||||||
|
'firstName' => mb_strimwidth($employee->getFirstName() ?? '', 0, 15, '...'),
|
||||||
|
'contractName' => $contractName,
|
||||||
|
'presenceDays' => $presenceDays,
|
||||||
|
'mileageKm' => $mileageKm,
|
||||||
|
'nightHours' => $nightHours,
|
||||||
|
'nightBasketCount' => $nightBasketCount,
|
||||||
|
'paidHours' => $paidHours,
|
||||||
|
'sundayHours' => $sundayHours,
|
||||||
|
'bonusAmount' => $bonusAmount,
|
||||||
|
'congesCount' => $conges['count'],
|
||||||
|
'congesDates' => $conges['dates'],
|
||||||
|
'maladieCount' => $maladie['count'],
|
||||||
|
'maladieDates' => $maladie['dates'],
|
||||||
|
'isDriver' => $isDriverAnyDay,
|
||||||
|
'driverBreakfast' => $driverBreakfast,
|
||||||
|
'driverMeals' => $driverMeals,
|
||||||
|
'driverOvernight' => $driverOvernight,
|
||||||
|
'driverSaturdays' => $driverSaturdays,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{nightMinutes: int, dayMinutes: int}
|
||||||
|
*/
|
||||||
|
private function computeNightMinutes(WorkHour $workHour): array
|
||||||
|
{
|
||||||
|
$ranges = [
|
||||||
|
[$workHour->getMorningFrom(), $workHour->getMorningTo()],
|
||||||
|
[$workHour->getAfternoonFrom(), $workHour->getAfternoonTo()],
|
||||||
|
[$workHour->getEveningFrom(), $workHour->getEveningTo()],
|
||||||
|
];
|
||||||
|
|
||||||
|
$totalMinutes = 0;
|
||||||
|
$nightMinutes = 0;
|
||||||
|
|
||||||
|
foreach ($ranges as [$from, $to]) {
|
||||||
|
$totalMinutes += $this->intervalMinutes($from, $to);
|
||||||
|
$nightMinutes += $this->nightIntervalMinutes($from, $to);
|
||||||
|
}
|
||||||
|
|
||||||
|
$dayMinutes = max(0, $totalMinutes - $nightMinutes);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'nightMinutes' => $nightMinutes,
|
||||||
|
'dayMinutes' => $dayMinutes,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return null|array{int, int}
|
||||||
|
*/
|
||||||
|
private function resolveInterval(?string $from, ?string $to): ?array
|
||||||
|
{
|
||||||
|
$fromMinutes = $this->toMinutes($from);
|
||||||
|
$toMinutes = $this->toMinutes($to);
|
||||||
|
if (null === $fromMinutes || null === $toMinutes) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$end = $toMinutes <= $fromMinutes ? $toMinutes + 1440 : $toMinutes;
|
||||||
|
|
||||||
|
return [$fromMinutes, $end];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function toMinutes(?string $time): ?int
|
||||||
|
{
|
||||||
|
if (null === $time || '' === $time) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$hours, $minutes] = array_map('intval', explode(':', $time));
|
||||||
|
|
||||||
|
return ($hours * 60) + $minutes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function intervalMinutes(?string $from, ?string $to): int
|
||||||
|
{
|
||||||
|
$interval = $this->resolveInterval($from, $to);
|
||||||
|
if (null === $interval) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$start, $end] = $interval;
|
||||||
|
|
||||||
|
return max(0, $end - $start);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function nightIntervalMinutes(?string $from, ?string $to): int
|
||||||
|
{
|
||||||
|
$interval = $this->resolveInterval($from, $to);
|
||||||
|
if (null === $interval) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$start, $end] = $interval;
|
||||||
|
$windows = [[0, 360], [1260, 1440]];
|
||||||
|
$total = 0;
|
||||||
|
|
||||||
|
for ($dayOffset = 0; $dayOffset <= 1; ++$dayOffset) {
|
||||||
|
$shift = $dayOffset * 1440;
|
||||||
|
foreach ($windows as [$windowStart, $windowEnd]) {
|
||||||
|
$total += $this->overlap($start, $end, $windowStart + $shift, $windowEnd + $shift);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $total;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calcule les minutes qui débordent après minuit (> 1440) pour les créneaux d'un WorkHour.
|
||||||
|
* Ex: créneau soir 21:00-05:00 → interval [1260, 1740] → overflow = 1740-1440 = 300 min (5h).
|
||||||
|
*/
|
||||||
|
private function computeOverflowAfterMidnight(WorkHour $workHour): int
|
||||||
|
{
|
||||||
|
$ranges = [
|
||||||
|
[$workHour->getMorningFrom(), $workHour->getMorningTo()],
|
||||||
|
[$workHour->getAfternoonFrom(), $workHour->getAfternoonTo()],
|
||||||
|
[$workHour->getEveningFrom(), $workHour->getEveningTo()],
|
||||||
|
];
|
||||||
|
|
||||||
|
$overflow = 0;
|
||||||
|
|
||||||
|
foreach ($ranges as [$from, $to]) {
|
||||||
|
$interval = $this->resolveInterval($from, $to);
|
||||||
|
if (null === $interval) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$start, $end] = $interval;
|
||||||
|
|
||||||
|
// Si le créneau dépasse minuit (1440), la partie au-delà est sur le jour suivant
|
||||||
|
if ($end > 1440) {
|
||||||
|
$overflow += $end - max($start, 1440);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $overflow;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function overlap(int $startA, int $endA, int $startB, int $endB): int
|
||||||
|
{
|
||||||
|
$start = max($startA, $startB);
|
||||||
|
$end = min($endA, $endB);
|
||||||
|
|
||||||
|
return max(0, $end - $start);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Absence> $absences
|
||||||
|
* @param list<string> $codes
|
||||||
|
*
|
||||||
|
* @return array{count: float, dates: string}
|
||||||
|
*/
|
||||||
|
private function countAbsencesByCode(array $absences, array $codes): array
|
||||||
|
{
|
||||||
|
$count = 0.0;
|
||||||
|
$dayKeys = [];
|
||||||
|
|
||||||
|
foreach ($absences as $absence) {
|
||||||
|
$type = $absence->getType();
|
||||||
|
if (!$type || !in_array($type->getCode(), $codes, true)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$startHalf = $absence->getStartHalf();
|
||||||
|
$endHalf = $absence->getEndHalf();
|
||||||
|
|
||||||
|
if ($startHalf === $endHalf) {
|
||||||
|
$count += 0.5;
|
||||||
|
} else {
|
||||||
|
$count += 1.0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$dayKeys[] = $absence->getStartDate()->format('Y-m-d');
|
||||||
|
}
|
||||||
|
|
||||||
|
sort($dayKeys);
|
||||||
|
$dayKeys = array_unique($dayKeys);
|
||||||
|
|
||||||
|
$periods = $this->mergeDaysIntoPeriods($dayKeys);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'count' => $count,
|
||||||
|
'dates' => implode(', ', $periods),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<string> $sortedDates Y-m-d sorted
|
||||||
|
*
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function mergeDaysIntoPeriods(array $sortedDates): array
|
||||||
|
{
|
||||||
|
if ([] === $sortedDates) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$periods = [];
|
||||||
|
$rangeStart = $sortedDates[0];
|
||||||
|
$rangeEnd = $sortedDates[0];
|
||||||
|
|
||||||
|
for ($i = 1, $len = count($sortedDates); $i < $len; ++$i) {
|
||||||
|
$prev = new DateTimeImmutable($rangeEnd);
|
||||||
|
$current = new DateTimeImmutable($sortedDates[$i]);
|
||||||
|
|
||||||
|
if (1 === $current->diff($prev)->days) {
|
||||||
|
$rangeEnd = $sortedDates[$i];
|
||||||
|
} else {
|
||||||
|
$periods[] = $this->formatPeriod($rangeStart, $rangeEnd);
|
||||||
|
$rangeStart = $sortedDates[$i];
|
||||||
|
$rangeEnd = $sortedDates[$i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$periods[] = $this->formatPeriod($rangeStart, $rangeEnd);
|
||||||
|
|
||||||
|
return $periods;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function formatPeriod(string $start, string $end): string
|
||||||
|
{
|
||||||
|
$s = new DateTimeImmutable($start)->format('d/m');
|
||||||
|
|
||||||
|
if ($start === $end) {
|
||||||
|
return $s;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'Du '.$s.' au '.new DateTimeImmutable($end)->format('d/m');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -95,7 +95,8 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
$isPresenceTracking = TrackingMode::PRESENCE->value === $contract?->getTrackingMode();
|
$isPresenceTracking = TrackingMode::PRESENCE->value === $contract?->getTrackingMode();
|
||||||
$normalized = $this->normalizeEntry($entry, $employeeId, $isPresenceTracking);
|
$isDriver = $this->contractResolver->resolveIsDriverForEmployeeAndDate($employee, $workDate);
|
||||||
|
$normalized = $this->normalizeEntry($entry, $employeeId, $isPresenceTracking, $isDriver);
|
||||||
$existing = $existingByEmployeeId[$employeeId] ?? null;
|
$existing = $existingByEmployeeId[$employeeId] ?? null;
|
||||||
$isAdmin = in_array('ROLE_ADMIN', $user->getRoles(), true);
|
$isAdmin = in_array('ROLE_ADMIN', $user->getRoles(), true);
|
||||||
$isSelf = in_array('ROLE_SELF', $user->getRoles(), true);
|
$isSelf = in_array('ROLE_SELF', $user->getRoles(), true);
|
||||||
@@ -225,21 +226,55 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
* eveningFrom:?string,
|
* eveningFrom:?string,
|
||||||
* eveningTo:?string,
|
* eveningTo:?string,
|
||||||
* isPresentMorning:bool,
|
* isPresentMorning:bool,
|
||||||
* isPresentAfternoon:bool
|
* isPresentAfternoon:bool,
|
||||||
|
* dayHoursMinutes:?int,
|
||||||
|
* nightHoursMinutes:?int,
|
||||||
|
* workshopHoursMinutes:?int,
|
||||||
|
* hasBreakfast:bool,
|
||||||
|
* hasLunch:bool,
|
||||||
|
* hasDinner:bool,
|
||||||
|
* hasOvernight:bool
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
private function normalizeEntry(array $entry, int $employeeId, bool $isPresenceTracking): array
|
private function normalizeEntry(array $entry, int $employeeId, bool $isPresenceTracking, bool $isDriver): array
|
||||||
{
|
{
|
||||||
|
if ($isDriver) {
|
||||||
|
return [
|
||||||
|
'morningFrom' => null,
|
||||||
|
'morningTo' => null,
|
||||||
|
'afternoonFrom' => null,
|
||||||
|
'afternoonTo' => null,
|
||||||
|
'eveningFrom' => null,
|
||||||
|
'eveningTo' => null,
|
||||||
|
'isPresentMorning' => false,
|
||||||
|
'isPresentAfternoon' => false,
|
||||||
|
'dayHoursMinutes' => $this->normalizeMinutes($entry['dayHoursMinutes'] ?? null, $employeeId, 'dayHoursMinutes'),
|
||||||
|
'nightHoursMinutes' => $this->normalizeMinutes($entry['nightHoursMinutes'] ?? null, $employeeId, 'nightHoursMinutes'),
|
||||||
|
'workshopHoursMinutes' => $this->normalizeMinutes($entry['workshopHoursMinutes'] ?? null, $employeeId, 'workshopHoursMinutes'),
|
||||||
|
'hasBreakfast' => $this->normalizePresence($entry['hasBreakfast'] ?? false, $employeeId, 'hasBreakfast'),
|
||||||
|
'hasLunch' => $this->normalizePresence($entry['hasLunch'] ?? false, $employeeId, 'hasLunch'),
|
||||||
|
'hasDinner' => $this->normalizePresence($entry['hasDinner'] ?? false, $employeeId, 'hasDinner'),
|
||||||
|
'hasOvernight' => $this->normalizePresence($entry['hasOvernight'] ?? false, $employeeId, 'hasOvernight'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
if ($isPresenceTracking) {
|
if ($isPresenceTracking) {
|
||||||
return [
|
return [
|
||||||
'morningFrom' => null,
|
'morningFrom' => null,
|
||||||
'morningTo' => null,
|
'morningTo' => null,
|
||||||
'afternoonFrom' => null,
|
'afternoonFrom' => null,
|
||||||
'afternoonTo' => null,
|
'afternoonTo' => null,
|
||||||
'eveningFrom' => null,
|
'eveningFrom' => null,
|
||||||
'eveningTo' => null,
|
'eveningTo' => null,
|
||||||
'isPresentMorning' => $this->normalizePresence($entry['isPresentMorning'] ?? false, $employeeId, 'isPresentMorning'),
|
'isPresentMorning' => $this->normalizePresence($entry['isPresentMorning'] ?? false, $employeeId, 'isPresentMorning'),
|
||||||
'isPresentAfternoon' => $this->normalizePresence($entry['isPresentAfternoon'] ?? false, $employeeId, 'isPresentAfternoon'),
|
'isPresentAfternoon' => $this->normalizePresence($entry['isPresentAfternoon'] ?? false, $employeeId, 'isPresentAfternoon'),
|
||||||
|
'dayHoursMinutes' => null,
|
||||||
|
'nightHoursMinutes' => null,
|
||||||
|
'workshopHoursMinutes' => null,
|
||||||
|
'hasBreakfast' => false,
|
||||||
|
'hasLunch' => false,
|
||||||
|
'hasDinner' => false,
|
||||||
|
'hasOvernight' => false,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -252,8 +287,15 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
'eveningTo' => $this->normalizeTime($entry['eveningTo'] ?? null, $employeeId, 'eveningTo'),
|
'eveningTo' => $this->normalizeTime($entry['eveningTo'] ?? null, $employeeId, 'eveningTo'),
|
||||||
// On conserve aussi la présence si envoyée (cas forfait affiché côté UI),
|
// On conserve aussi la présence si envoyée (cas forfait affiché côté UI),
|
||||||
// même si le contrat résolu ce jour est en suivi horaire.
|
// même si le contrat résolu ce jour est en suivi horaire.
|
||||||
'isPresentMorning' => $this->normalizePresence($entry['isPresentMorning'] ?? false, $employeeId, 'isPresentMorning'),
|
'isPresentMorning' => $this->normalizePresence($entry['isPresentMorning'] ?? false, $employeeId, 'isPresentMorning'),
|
||||||
'isPresentAfternoon' => $this->normalizePresence($entry['isPresentAfternoon'] ?? false, $employeeId, 'isPresentAfternoon'),
|
'isPresentAfternoon' => $this->normalizePresence($entry['isPresentAfternoon'] ?? false, $employeeId, 'isPresentAfternoon'),
|
||||||
|
'dayHoursMinutes' => null,
|
||||||
|
'nightHoursMinutes' => null,
|
||||||
|
'workshopHoursMinutes' => null,
|
||||||
|
'hasBreakfast' => false,
|
||||||
|
'hasLunch' => false,
|
||||||
|
'hasDinner' => false,
|
||||||
|
'hasOvernight' => false,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,6 +325,32 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
return $time;
|
return $time;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function normalizeMinutes(mixed $value, int $employeeId, string $field): ?int
|
||||||
|
{
|
||||||
|
if (null === $value || '' === $value) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_int($value) && !is_float($value)) {
|
||||||
|
throw new UnprocessableEntityHttpException(sprintf(
|
||||||
|
'Employee %d: %s must be an integer (minutes).',
|
||||||
|
$employeeId,
|
||||||
|
$field
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
$minutes = (int) $value;
|
||||||
|
if ($minutes < 0) {
|
||||||
|
throw new UnprocessableEntityHttpException(sprintf(
|
||||||
|
'Employee %d: %s must be >= 0.',
|
||||||
|
$employeeId,
|
||||||
|
$field
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $minutes;
|
||||||
|
}
|
||||||
|
|
||||||
private function normalizePresence(mixed $value, int $employeeId, string $field): bool
|
private function normalizePresence(mixed $value, int $employeeId, string $field): bool
|
||||||
{
|
{
|
||||||
if (!is_bool($value)) {
|
if (!is_bool($value)) {
|
||||||
@@ -305,7 +373,14 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
* eveningFrom:?string,
|
* eveningFrom:?string,
|
||||||
* eveningTo:?string,
|
* eveningTo:?string,
|
||||||
* isPresentMorning:bool,
|
* isPresentMorning:bool,
|
||||||
* isPresentAfternoon:bool
|
* isPresentAfternoon:bool,
|
||||||
|
* dayHoursMinutes:?int,
|
||||||
|
* nightHoursMinutes:?int,
|
||||||
|
* workshopHoursMinutes:?int,
|
||||||
|
* hasBreakfast:bool,
|
||||||
|
* hasLunch:bool,
|
||||||
|
* hasDinner:bool,
|
||||||
|
* hasOvernight:bool
|
||||||
* } $entry
|
* } $entry
|
||||||
*/
|
*/
|
||||||
private function isEntryEmpty(array $entry): bool
|
private function isEntryEmpty(array $entry): bool
|
||||||
@@ -317,7 +392,14 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
&& null === $entry['eveningFrom']
|
&& null === $entry['eveningFrom']
|
||||||
&& null === $entry['eveningTo']
|
&& null === $entry['eveningTo']
|
||||||
&& false === $entry['isPresentMorning']
|
&& false === $entry['isPresentMorning']
|
||||||
&& false === $entry['isPresentAfternoon'];
|
&& false === $entry['isPresentAfternoon']
|
||||||
|
&& (null === $entry['dayHoursMinutes'] || 0 === $entry['dayHoursMinutes'])
|
||||||
|
&& (null === $entry['nightHoursMinutes'] || 0 === $entry['nightHoursMinutes'])
|
||||||
|
&& (null === $entry['workshopHoursMinutes'] || 0 === $entry['workshopHoursMinutes'])
|
||||||
|
&& false === $entry['hasBreakfast']
|
||||||
|
&& false === $entry['hasLunch']
|
||||||
|
&& false === $entry['hasDinner']
|
||||||
|
&& false === $entry['hasOvernight'];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -329,7 +411,14 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
* eveningFrom:?string,
|
* eveningFrom:?string,
|
||||||
* eveningTo:?string,
|
* eveningTo:?string,
|
||||||
* isPresentMorning:bool,
|
* isPresentMorning:bool,
|
||||||
* isPresentAfternoon:bool
|
* isPresentAfternoon:bool,
|
||||||
|
* dayHoursMinutes:?int,
|
||||||
|
* nightHoursMinutes:?int,
|
||||||
|
* workshopHoursMinutes:?int,
|
||||||
|
* hasBreakfast:bool,
|
||||||
|
* hasLunch:bool,
|
||||||
|
* hasDinner:bool,
|
||||||
|
* hasOvernight:bool
|
||||||
* } $entry
|
* } $entry
|
||||||
*/
|
*/
|
||||||
private function hydrateWorkHour(WorkHour $workHour, array $entry): void
|
private function hydrateWorkHour(WorkHour $workHour, array $entry): void
|
||||||
@@ -343,6 +432,13 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
->setEveningTo($entry['eveningTo'])
|
->setEveningTo($entry['eveningTo'])
|
||||||
->setIsPresentMorning($entry['isPresentMorning'])
|
->setIsPresentMorning($entry['isPresentMorning'])
|
||||||
->setIsPresentAfternoon($entry['isPresentAfternoon'])
|
->setIsPresentAfternoon($entry['isPresentAfternoon'])
|
||||||
|
->setDayHoursMinutes($entry['dayHoursMinutes'])
|
||||||
|
->setNightHoursMinutes($entry['nightHoursMinutes'])
|
||||||
|
->setWorkshopHoursMinutes($entry['workshopHoursMinutes'])
|
||||||
|
->setHasBreakfast($entry['hasBreakfast'])
|
||||||
|
->setHasLunch($entry['hasLunch'])
|
||||||
|
->setHasDinner($entry['hasDinner'])
|
||||||
|
->setHasOvernight($entry['hasOvernight'])
|
||||||
// Toute modification invalide la validation chef de site.
|
// Toute modification invalide la validation chef de site.
|
||||||
->setIsSiteValid(false)
|
->setIsSiteValid(false)
|
||||||
// Toute modification utilisateur repasse la ligne en attente de validation RH.
|
// Toute modification utilisateur repasse la ligne en attente de validation RH.
|
||||||
@@ -359,7 +455,14 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
* eveningFrom:?string,
|
* eveningFrom:?string,
|
||||||
* eveningTo:?string,
|
* eveningTo:?string,
|
||||||
* isPresentMorning:bool,
|
* isPresentMorning:bool,
|
||||||
* isPresentAfternoon:bool
|
* isPresentAfternoon:bool,
|
||||||
|
* dayHoursMinutes:?int,
|
||||||
|
* nightHoursMinutes:?int,
|
||||||
|
* workshopHoursMinutes:?int,
|
||||||
|
* hasBreakfast:bool,
|
||||||
|
* hasLunch:bool,
|
||||||
|
* hasDinner:bool,
|
||||||
|
* hasOvernight:bool
|
||||||
* } $entry
|
* } $entry
|
||||||
*/
|
*/
|
||||||
private function isSameAsExisting(WorkHour $workHour, array $entry): bool
|
private function isSameAsExisting(WorkHour $workHour, array $entry): bool
|
||||||
@@ -371,6 +474,13 @@ final readonly class WorkHourBulkUpsertProcessor implements ProcessorInterface
|
|||||||
&& $workHour->getEveningFrom() === $entry['eveningFrom']
|
&& $workHour->getEveningFrom() === $entry['eveningFrom']
|
||||||
&& $workHour->getEveningTo() === $entry['eveningTo']
|
&& $workHour->getEveningTo() === $entry['eveningTo']
|
||||||
&& $workHour->getIsPresentMorning() === $entry['isPresentMorning']
|
&& $workHour->getIsPresentMorning() === $entry['isPresentMorning']
|
||||||
&& $workHour->getIsPresentAfternoon() === $entry['isPresentAfternoon'];
|
&& $workHour->getIsPresentAfternoon() === $entry['isPresentAfternoon']
|
||||||
|
&& $workHour->getDayHoursMinutes() === $entry['dayHoursMinutes']
|
||||||
|
&& $workHour->getNightHoursMinutes() === $entry['nightHoursMinutes']
|
||||||
|
&& $workHour->getWorkshopHoursMinutes() === $entry['workshopHoursMinutes']
|
||||||
|
&& $workHour->getHasBreakfast() === $entry['hasBreakfast']
|
||||||
|
&& $workHour->getHasLunch() === $entry['hasLunch']
|
||||||
|
&& $workHour->getHasDinner() === $entry['hasDinner']
|
||||||
|
&& $workHour->getHasOvernight() === $entry['hasOvernight'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,9 +52,11 @@ final readonly class WorkHourDayContextProvider implements ProviderInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
// On initialise toutes les lignes, même sans absence ce jour-là.
|
// On initialise toutes les lignes, même sans absence ce jour-là.
|
||||||
|
$contract = $this->contractResolver->resolveForEmployeeAndDate($employee, $workDate);
|
||||||
$rowsByEmployeeId[$employeeId] = new DayContextRow(
|
$rowsByEmployeeId[$employeeId] = new DayContextRow(
|
||||||
employeeId: $employeeId,
|
employeeId: $employeeId,
|
||||||
hasContractAtDate: null !== $this->contractResolver->resolveForEmployeeAndDate($employee, $workDate)
|
hasContractAtDate: null !== $contract,
|
||||||
|
isDriverContract: $this->contractResolver->resolveIsDriverForEmployeeAndDate($employee, $workDate),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -116,6 +116,7 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
{
|
{
|
||||||
$contractsByEmployeeDate = $this->contractResolver->resolveForEmployeesAndDays($employees, $days);
|
$contractsByEmployeeDate = $this->contractResolver->resolveForEmployeesAndDays($employees, $days);
|
||||||
$contractNaturesByEmployeeDate = $this->contractResolver->resolveNaturesForEmployeesAndDays($employees, $days);
|
$contractNaturesByEmployeeDate = $this->contractResolver->resolveNaturesForEmployeesAndDays($employees, $days);
|
||||||
|
$isDriverByEmployeeDate = $this->contractResolver->resolveIsDriverForEmployeesAndDays($employees, $days);
|
||||||
$metricsByEmployeeDate = [];
|
$metricsByEmployeeDate = [];
|
||||||
foreach ($workHours as $workHour) {
|
foreach ($workHours as $workHour) {
|
||||||
$employeeId = $workHour->getEmployee()?->getId();
|
$employeeId = $workHour->getEmployee()?->getId();
|
||||||
@@ -126,9 +127,16 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
// Pré-calcul des métriques par salarié/date pour simplifier l'agrégation finale.
|
// Pré-calcul des métriques par salarié/date pour simplifier l'agrégation finale.
|
||||||
$dateKey = $workHour->getWorkDate()->format('Y-m-d');
|
$dateKey = $workHour->getWorkDate()->format('Y-m-d');
|
||||||
$metricsByEmployeeDate[$employeeId][$dateKey] = [
|
$metricsByEmployeeDate[$employeeId][$dateKey] = [
|
||||||
'metrics' => $this->computeMetrics($workHour),
|
'metrics' => $this->computeMetrics($workHour),
|
||||||
'isPresentMorning' => $workHour->getIsPresentMorning(),
|
'isPresentMorning' => $workHour->getIsPresentMorning(),
|
||||||
'isPresentAfternoon' => $workHour->getIsPresentAfternoon(),
|
'isPresentAfternoon' => $workHour->getIsPresentAfternoon(),
|
||||||
|
'dayHoursMinutes' => $workHour->getDayHoursMinutes(),
|
||||||
|
'nightHoursMinutes' => $workHour->getNightHoursMinutes(),
|
||||||
|
'workshopHoursMinutes' => $workHour->getWorkshopHoursMinutes(),
|
||||||
|
'hasBreakfast' => $workHour->getHasBreakfast(),
|
||||||
|
'hasLunch' => $workHour->getHasLunch(),
|
||||||
|
'hasDinner' => $workHour->getHasDinner(),
|
||||||
|
'hasOvernight' => $workHour->getHasOvernight(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -156,7 +164,7 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
[$absentMorning, $absentAfternoon] = $this->absenceSegmentsResolver->resolveForDate($absence, $date);
|
[$absentMorning, $absentAfternoon] = $this->absenceSegmentsResolver->resolveForDate($absence, $date);
|
||||||
if ($absentMorning || $absentAfternoon) {
|
if ($absentMorning || $absentAfternoon) {
|
||||||
$absenceByEmployeeDate[$employeeId][$date] = true;
|
$absenceByEmployeeDate[$employeeId][$date] = true;
|
||||||
$absentMorningByEmployeeDate[$employeeId][$date] = ($absentMorningByEmployeeDate[$employeeId][$date] ?? false) || $absentMorning;
|
$absentMorningByEmployeeDate[$employeeId][$date] = ($absentMorningByEmployeeDate[$employeeId][$date] ?? false) || $absentMorning;
|
||||||
$absentAfternoonByEmployeeDate[$employeeId][$date] = ($absentAfternoonByEmployeeDate[$employeeId][$date] ?? false) || $absentAfternoon;
|
$absentAfternoonByEmployeeDate[$employeeId][$date] = ($absentAfternoonByEmployeeDate[$employeeId][$date] ?? false) || $absentAfternoon;
|
||||||
if (!isset($absenceLabelByEmployeeDate[$employeeId][$date])) {
|
if (!isset($absenceLabelByEmployeeDate[$employeeId][$date])) {
|
||||||
$absenceLabelByEmployeeDate[$employeeId][$date] = $absence->getType()?->getLabel();
|
$absenceLabelByEmployeeDate[$employeeId][$date] = $absence->getType()?->getLabel();
|
||||||
@@ -179,33 +187,82 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$weeklyDayMinutes = 0;
|
$weeklyDayMinutes = 0;
|
||||||
$weeklyNightMinutes = 0;
|
$weeklyNightMinutes = 0;
|
||||||
$weeklyTotalMinutes = 0;
|
$weeklyWorkshopMinutes = 0;
|
||||||
$weeklyPresenceCount = 0.0;
|
$weeklyTotalMinutes = 0;
|
||||||
$daily = [];
|
$weeklyPresenceCount = 0.0;
|
||||||
|
$weeklyNightBasketCount = 0;
|
||||||
|
$weeklyBreakfastCount = 0;
|
||||||
|
$weeklyLunchCount = 0;
|
||||||
|
$weeklyDinnerCount = 0;
|
||||||
|
$weeklyOvernightCount = 0;
|
||||||
|
$daily = [];
|
||||||
// Les contrats au suivi "présence" ne manipulent pas les heures, mais des demi-journées.
|
// Les contrats au suivi "présence" ne manipulent pas les heures, mais des demi-journées.
|
||||||
$weekAnchorContract = $contractsByEmployeeDate[$employeeId][$anchorDateYmd]
|
$weekAnchorContract = $contractsByEmployeeDate[$employeeId][$anchorDateYmd]
|
||||||
?? $contractsByEmployeeDate[$employeeId][$days[0]]
|
?? $contractsByEmployeeDate[$employeeId][$days[0]]
|
||||||
?? null;
|
?? null;
|
||||||
|
$isDriver = $isDriverByEmployeeDate[$employeeId][$anchorDateYmd]
|
||||||
|
?? $isDriverByEmployeeDate[$employeeId][$days[0]]
|
||||||
|
?? false;
|
||||||
$weekAnchorContractNature = $contractNaturesByEmployeeDate[$employeeId][$anchorDateYmd]
|
$weekAnchorContractNature = $contractNaturesByEmployeeDate[$employeeId][$anchorDateYmd]
|
||||||
?? $contractNaturesByEmployeeDate[$employeeId][$days[0]]
|
?? $contractNaturesByEmployeeDate[$employeeId][$days[0]]
|
||||||
?? ContractNature::CDI;
|
?? ContractNature::CDI;
|
||||||
$employeeContractsByDate = [];
|
$employeeContractsByDate = [];
|
||||||
|
$hasContractForWeek = false;
|
||||||
foreach ($days as $date) {
|
foreach ($days as $date) {
|
||||||
$employeeContractsByDate[$date] = $contractsByEmployeeDate[$employeeId][$date] ?? null;
|
$employeeContractsByDate[$date] = $contractsByEmployeeDate[$employeeId][$date] ?? null;
|
||||||
|
if (null !== $employeeContractsByDate[$date]) {
|
||||||
|
$hasContractForWeek = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($days as $date) {
|
foreach ($days as $date) {
|
||||||
$entry = $metricsByEmployeeDate[$employeeId][$date] ?? null;
|
$entry = $metricsByEmployeeDate[$employeeId][$date] ?? null;
|
||||||
$metrics = $entry['metrics'] ?? new WorkMetrics();
|
|
||||||
$creditedMinutes = $creditedByEmployeeDate[$employeeId][$date] ?? 0;
|
$creditedMinutes = $creditedByEmployeeDate[$employeeId][$date] ?? 0;
|
||||||
$contractAtDate = $employeeContractsByDate[$date] ?? null;
|
$contractAtDate = $employeeContractsByDate[$date] ?? null;
|
||||||
$isPresenceTracking = TrackingMode::PRESENCE->value === $contractAtDate?->getTrackingMode();
|
$isPresenceTracking = TrackingMode::PRESENCE->value === $contractAtDate?->getTrackingMode();
|
||||||
// Les absences "comptées comme travaillées" alimentent le total du jour.
|
$isDateDriver = $isDriverByEmployeeDate[$employeeId][$date] ?? false;
|
||||||
$metrics->addCreditedMinutes($creditedMinutes);
|
|
||||||
|
$hasBreakfast = false;
|
||||||
|
$hasLunch = false;
|
||||||
|
$hasDinner = false;
|
||||||
|
$hasOvernight = false;
|
||||||
|
|
||||||
|
if ($isDateDriver) {
|
||||||
|
$dayMinutes = ($entry['dayHoursMinutes'] ?? 0);
|
||||||
|
$nightMinutes = ($entry['nightHoursMinutes'] ?? 0);
|
||||||
|
$workshopMinutes = ($entry['workshopHoursMinutes'] ?? 0);
|
||||||
|
$totalMinutes = $dayMinutes + $nightMinutes + $workshopMinutes + $creditedMinutes;
|
||||||
|
$dayMinutes += $creditedMinutes;
|
||||||
|
$hasBreakfast = $entry['hasBreakfast'] ?? false;
|
||||||
|
$hasLunch = $entry['hasLunch'] ?? false;
|
||||||
|
$hasDinner = $entry['hasDinner'] ?? false;
|
||||||
|
$hasOvernight = $entry['hasOvernight'] ?? false;
|
||||||
|
if ($hasBreakfast) {
|
||||||
|
++$weeklyBreakfastCount;
|
||||||
|
}
|
||||||
|
if ($hasLunch) {
|
||||||
|
++$weeklyLunchCount;
|
||||||
|
}
|
||||||
|
if ($hasDinner) {
|
||||||
|
++$weeklyDinnerCount;
|
||||||
|
}
|
||||||
|
if ($hasOvernight) {
|
||||||
|
++$weeklyOvernightCount;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$metrics = $entry['metrics'] ?? new WorkMetrics();
|
||||||
|
// Les absences "comptées comme travaillées" alimentent le total du jour.
|
||||||
|
$metrics->addCreditedMinutes($creditedMinutes);
|
||||||
|
$dayMinutes = $metrics->dayMinutes;
|
||||||
|
$nightMinutes = $metrics->nightMinutes;
|
||||||
|
$workshopMinutes = 0;
|
||||||
|
$totalMinutes = $metrics->totalMinutes;
|
||||||
|
}
|
||||||
|
|
||||||
$present = null;
|
$present = null;
|
||||||
if ($isPresenceTracking) {
|
if ($isPresenceTracking && !$isDateDriver) {
|
||||||
$absentMorning = $absentMorningByEmployeeDate[$employeeId][$date] ?? false;
|
$absentMorning = $absentMorningByEmployeeDate[$employeeId][$date] ?? false;
|
||||||
$absentAfternoon = $absentAfternoonByEmployeeDate[$employeeId][$date] ?? false;
|
$absentAfternoon = $absentAfternoonByEmployeeDate[$employeeId][$date] ?? false;
|
||||||
$morning = (($entry['isPresentMorning'] ?? false) && !$absentMorning) ? 0.5 : 0.0;
|
$morning = (($entry['isPresentMorning'] ?? false) && !$absentMorning) ? 0.5 : 0.0;
|
||||||
@@ -214,30 +271,42 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
$present = min(1.0, $morning + $afternoon + $creditedPresence);
|
$present = min(1.0, $morning + $afternoon + $creditedPresence);
|
||||||
}
|
}
|
||||||
|
|
||||||
$weeklyDayMinutes += $metrics->dayMinutes;
|
$hasNightBasket = ($nightMinutes > $dayMinutes && $nightMinutes > 0) || $nightMinutes >= 240;
|
||||||
$weeklyNightMinutes += $metrics->nightMinutes;
|
if ($hasNightBasket) {
|
||||||
$weeklyTotalMinutes += $metrics->totalMinutes;
|
++$weeklyNightBasketCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
$weeklyDayMinutes += $dayMinutes;
|
||||||
|
$weeklyNightMinutes += $nightMinutes;
|
||||||
|
$weeklyWorkshopMinutes += $workshopMinutes;
|
||||||
|
$weeklyTotalMinutes += $totalMinutes;
|
||||||
if (null !== $present) {
|
if (null !== $present) {
|
||||||
$weeklyPresenceCount += $present;
|
$weeklyPresenceCount += $present;
|
||||||
}
|
}
|
||||||
|
|
||||||
$daily[] = new WeeklyDaySummary(
|
$daily[] = new WeeklyDaySummary(
|
||||||
date: $date,
|
date: $date,
|
||||||
dayMinutes: $metrics->dayMinutes,
|
dayMinutes: $dayMinutes,
|
||||||
nightMinutes: $metrics->nightMinutes,
|
nightMinutes: $nightMinutes,
|
||||||
totalMinutes: $metrics->totalMinutes,
|
workshopMinutes: $workshopMinutes,
|
||||||
|
totalMinutes: $totalMinutes,
|
||||||
present: $present,
|
present: $present,
|
||||||
hasAbsence: $absenceByEmployeeDate[$employeeId][$date] ?? false,
|
hasAbsence: $absenceByEmployeeDate[$employeeId][$date] ?? false,
|
||||||
absenceLabel: $absenceLabelByEmployeeDate[$employeeId][$date] ?? null,
|
absenceLabel: $absenceLabelByEmployeeDate[$employeeId][$date] ?? null,
|
||||||
absenceColor: $absenceColorByEmployeeDate[$employeeId][$date] ?? null,
|
absenceColor: $absenceColorByEmployeeDate[$employeeId][$date] ?? null,
|
||||||
|
hasNightBasket: $hasNightBasket,
|
||||||
|
hasBreakfast: $hasBreakfast,
|
||||||
|
hasLunch: $hasLunch,
|
||||||
|
hasDinner: $hasDinner,
|
||||||
|
hasOvernight: $hasOvernight,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
$isWeekPresenceTracking = TrackingMode::PRESENCE->value === $weekAnchorContract?->getTrackingMode();
|
$isWeekPresenceTracking = TrackingMode::PRESENCE->value === $weekAnchorContract?->getTrackingMode();
|
||||||
$disableOvertimeBonuses = $this->hasDisabledOvertimeBonuses($weekAnchorContract, $weekAnchorContractNature);
|
$disableOvertimeBonuses = $isDriver || $this->hasDisabledOvertimeBonuses($weekAnchorContract, $weekAnchorContractNature);
|
||||||
$overtimeReferenceMinutes = $this->computeWeeklyOvertimeReferenceMinutes($days, $employeeContractsByDate);
|
$overtimeReferenceMinutes = $this->computeWeeklyOvertimeReferenceMinutes($days, $employeeContractsByDate);
|
||||||
$overtime25StartMinutes = $this->computeWeeklyOvertime25StartMinutes($days, $employeeContractsByDate);
|
$overtime25StartMinutes = $this->computeWeeklyOvertime25StartMinutes($days, $employeeContractsByDate);
|
||||||
$weeklyOvertimeTotalMinutes = $isWeekPresenceTracking
|
$weeklyOvertimeTotalMinutes = ($isWeekPresenceTracking || $isDriver)
|
||||||
? 0
|
? 0
|
||||||
: max(0, $weeklyTotalMinutes - $overtimeReferenceMinutes);
|
: max(0, $weeklyTotalMinutes - $overtimeReferenceMinutes);
|
||||||
$weeklyOvertime25Minutes = ($isWeekPresenceTracking || $disableOvertimeBonuses)
|
$weeklyOvertime25Minutes = ($isWeekPresenceTracking || $disableOvertimeBonuses)
|
||||||
@@ -261,12 +330,20 @@ final readonly class WorkHourWeeklySummaryProvider implements ProviderInterface
|
|||||||
daily: $daily,
|
daily: $daily,
|
||||||
weeklyDayMinutes: $weeklyDayMinutes,
|
weeklyDayMinutes: $weeklyDayMinutes,
|
||||||
weeklyNightMinutes: $weeklyNightMinutes,
|
weeklyNightMinutes: $weeklyNightMinutes,
|
||||||
|
weeklyWorkshopMinutes: $weeklyWorkshopMinutes,
|
||||||
weeklyTotalMinutes: $weeklyTotalMinutes,
|
weeklyTotalMinutes: $weeklyTotalMinutes,
|
||||||
weeklyPresenceCount: $weeklyPresenceCount,
|
weeklyPresenceCount: $weeklyPresenceCount,
|
||||||
weeklyOvertimeTotalMinutes: $weeklyOvertimeTotalMinutes,
|
weeklyOvertimeTotalMinutes: $weeklyOvertimeTotalMinutes,
|
||||||
weeklyOvertime25Minutes: $weeklyOvertime25Minutes,
|
weeklyOvertime25Minutes: $weeklyOvertime25Minutes,
|
||||||
weeklyOvertime50Minutes: $weeklyOvertime50Minutes,
|
weeklyOvertime50Minutes: $weeklyOvertime50Minutes,
|
||||||
weeklyRecoveryMinutes: $weeklyRecoveryMinutes
|
weeklyRecoveryMinutes: $weeklyRecoveryMinutes,
|
||||||
|
weeklyNightBasketCount: $weeklyNightBasketCount,
|
||||||
|
isDriver: $isDriver,
|
||||||
|
weeklyBreakfastCount: $weeklyBreakfastCount,
|
||||||
|
weeklyLunchCount: $weeklyLunchCount,
|
||||||
|
weeklyDinnerCount: $weeklyDinnerCount,
|
||||||
|
weeklyOvernightCount: $weeklyOvernightCount,
|
||||||
|
hasContractForWeek: $hasContractForWeek,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
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>
|
||||||
163
templates/salary-recap/print.html.twig
Normal file
163
templates/salary-recap/print.html.twig
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="fr">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>Récapitulatif Salaire</title>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
@page { size: A4 landscape; 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
.month-box {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
border: 2px solid #000;
|
||||||
|
padding: 4px 12px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.recap {
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
table-layout: auto;
|
||||||
|
border: 4px solid #0a0a0a;
|
||||||
|
}
|
||||||
|
|
||||||
|
th, td {
|
||||||
|
border: 2px solid #0a0a0a;
|
||||||
|
padding: 3px 3px;
|
||||||
|
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.dates {
|
||||||
|
text-align: left;
|
||||||
|
white-space: normal;
|
||||||
|
word-break: break-word;
|
||||||
|
font-size: 10px;
|
||||||
|
}
|
||||||
|
td.obs { }
|
||||||
|
|
||||||
|
tbody td { font-size: 10px; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
{% set months = {
|
||||||
|
1:'Janvier', 2:'Février', 3:'Mars', 4:'Avril', 5:'Mai', 6:'Juin',
|
||||||
|
7:'Juillet', 8:'Août', 9:'Septembre', 10:'Octobre', 11:'Novembre', 12:'Décembre'
|
||||||
|
} %}
|
||||||
|
|
||||||
|
<div class="title-bar">
|
||||||
|
<h1>RECAPITULATIF SALAIRE DU {{ from|date('d/m/Y') }} au {{ to|date('d/m/Y') }}</h1>
|
||||||
|
<div class="month-box">{{ months[from|date('n')|number_format] }} {{ from|date('Y') }}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table class="recap">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th rowspan="2" style="width: 24mm; text-align: left;">Nom</th>
|
||||||
|
<th rowspan="2" style="width: 12mm;">Base</th>
|
||||||
|
<th rowspan="2" style="width: 12mm;">Jour de<br>présence<br>Cadre</th>
|
||||||
|
<th rowspan="2" style="width: 9mm;">Frais<br>Kms</th>
|
||||||
|
<th rowspan="2" style="width: 9mm;">Heures<br>de<br>nuit</th>
|
||||||
|
<th rowspan="2" style="width: 9mm;">Panier<br>de<br>nuit</th>
|
||||||
|
<th rowspan="2" style="width: 12mm;">Heures<br>payés</th>
|
||||||
|
<th rowspan="2" style="width: 9mm;">Heures<br>dim.</th>
|
||||||
|
<th rowspan="2" style="width: 9mm;">Prime</th>
|
||||||
|
<th colspan="2">Congés</th>
|
||||||
|
<th colspan="2">Maladie</th>
|
||||||
|
<th colspan="4">CHAUFFEUR</th>
|
||||||
|
<th rowspan="2" style="width: 26mm;">Observations</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th style="width: 10mm;">Nbre</th>
|
||||||
|
<th style="width: 26mm;">Date</th>
|
||||||
|
<th style="width: 10mm;">Nbre</th>
|
||||||
|
<th style="width: 26mm;">Date</th>
|
||||||
|
<th style="width: 8mm;">PDJ</th>
|
||||||
|
<th style="width: 10mm;">REPAS</th>
|
||||||
|
<th style="width: 12mm;">NUITEE</th>
|
||||||
|
<th style="width: 12mm;">samedi</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="18">
|
||||||
|
{{ group.name }}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% for row in group.employees %}
|
||||||
|
<tr>
|
||||||
|
<td class="name">{{ row.lastName }}<br>{{ row.firstName }}</td>
|
||||||
|
<td class="base">{{ row.contractName ?? '' }}</td>
|
||||||
|
<td class="num">{{ row.presenceDays > 0 ? row.presenceDays : '' }}</td>
|
||||||
|
<td class="num">{{ row.mileageKm > 0 ? row.mileageKm : '' }}</td>
|
||||||
|
<td class="num">{{ row.nightHours > 0 ? row.nightHours : '' }}</td>
|
||||||
|
<td class="num">{{ row.nightBasketCount > 0 ? row.nightBasketCount : '' }}</td>
|
||||||
|
<td class="num">{{ row.paidHours > 0 ? row.paidHours : '' }}</td>
|
||||||
|
<td class="num">{{ row.sundayHours > 0 ? row.sundayHours : '' }}</td>
|
||||||
|
<td class="num">{{ row.bonusAmount > 0 ? row.bonusAmount : '' }}</td>
|
||||||
|
<td class="num">{{ row.congesCount > 0 ? row.congesCount : '' }}</td>
|
||||||
|
<td class="dates">{{ row.congesDates }}</td>
|
||||||
|
<td class="num">{{ row.maladieCount > 0 ? row.maladieCount : '' }}</td>
|
||||||
|
<td class="dates">{{ row.maladieDates }}</td>
|
||||||
|
<td class="num">{{ row.isDriver and row.driverBreakfast > 0 ? row.driverBreakfast : '' }}</td>
|
||||||
|
<td class="num">{{ row.isDriver and row.driverMeals > 0 ? row.driverMeals : '' }}</td>
|
||||||
|
<td class="num">{{ row.isDriver and row.driverOvernight > 0 ? row.driverOvernight : '' }}</td>
|
||||||
|
<td class="num">{{ row.isDriver and row.driverSaturdays > 0 ? row.driverSaturdays : '' }}</td>
|
||||||
|
<td class="obs"></td>
|
||||||
|
</tr>
|
||||||
|
{% else %}
|
||||||
|
<tr>
|
||||||
|
<td colspan="18">Aucun employé.</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
Reference in New Issue
Block a user