forked from FroSteel/Planification
Compare commits
13 Commits
v2026.5.42
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 06c0195130 | |||
| c65e943dac | |||
| d23824359f | |||
| 2d242d26ec | |||
| 54b8f826df | |||
| 8ecf2d3df4 | |||
| 9d8d8102d7 | |||
| 48b00a8585 | |||
| 6bb97addd6 | |||
| 05275a3be5 | |||
| d7b680fb3f | |||
| 3c7e7c0c25 | |||
| b3677d661a |
-56
@@ -1,56 +0,0 @@
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
desktop.ini
|
||||
|
||||
# Editors
|
||||
.vscode/
|
||||
.idea/
|
||||
*.swp
|
||||
*.swo
|
||||
*~
|
||||
|
||||
# Backups
|
||||
*.bak
|
||||
*.bak-*
|
||||
*.orig
|
||||
*.old
|
||||
|
||||
# Build artifacts (les ZIP/XPI livrés ne sont pas dans le repo, ils sont buildés à la demande)
|
||||
dist/
|
||||
*.zip
|
||||
*.xpi
|
||||
*.crx
|
||||
|
||||
# Node (si jamais utilisé pour build)
|
||||
node_modules/
|
||||
package-lock.json
|
||||
npm-debug.log*
|
||||
|
||||
# Logs
|
||||
*.log
|
||||
rebuild.log
|
||||
|
||||
# Dossiers de travail temporaires
|
||||
extracted/
|
||||
temp/
|
||||
tmp/
|
||||
|
||||
# Tests
|
||||
test-output/
|
||||
|
||||
# Archives historiques locales (jamais sur Gitea)
|
||||
_archives/
|
||||
Old.zip
|
||||
Old/
|
||||
|
||||
# Mémoire / config Claude Code (ne jamais commit, contient potentiellement
|
||||
# des tokens, des notes user, etc.)
|
||||
.claude/
|
||||
CLAUDE.local.md
|
||||
|
||||
# Variables d'environnement / secrets
|
||||
.env
|
||||
.env.*
|
||||
*.token
|
||||
secrets.json
|
||||
+216
-11
@@ -1,14 +1,218 @@
|
||||
# CHANGELOG — Extension Planification EasyVista Canton de Vaud
|
||||
|
||||
> Ce changelog documente l'évolution de l'extension Chrome/Firefox "Planification"
|
||||
> développée par Quentin Rouiller pour les techniciens DGNSI (Canton de Vaud).
|
||||
> développée par Quentin Rouiller pour les coordinateurs DGNSI (Canton de Vaud).
|
||||
>
|
||||
> Les versions documentées ci-dessous sont celles dont les détails sont connus.
|
||||
> Pour les versions plus anciennes, Claude Code se basera sur l'analyse du code
|
||||
> source pour déterminer un message de commit pertinent.
|
||||
> Pour les versions plus anciennes, l'analyse du code source permet de
|
||||
> reconstituer un message de version pertinent.
|
||||
|
||||
---
|
||||
|
||||
## v2026.5.45 — Dock latéral drag&drop, fix verdicts ghost, multi-onglets EZV résolu
|
||||
|
||||
> Refonte de l'expérience drag&drop avec dock latéral pour parquer des interventions entre les jours, fix critique du parser de fiches qui marquait à tort des interventions terminées comme annulées, résolution des 6 issues ouvertes (multi-onglets EZV, absences récurrentes, popups épinglés, pompier absent), et nombreux ajustements d'ergonomie.
|
||||
|
||||
### Issues résolues
|
||||
|
||||
- **#3** — Coches « Absences récurrentes » : merge propre avec l'état stocké au lieu d'écrasement, les coches sont retenues lors d'un changement de groupe ou d'une réouverture des paramètres.
|
||||
- **#4** — Perte de session EZV multi-onglets : permission optionnelle `cookies` + listener `cookieChanged` côté background, toggle dans Paramètres → Diagnostics. La session reste valable même après reconnexion et fermeture d'un onglet EZV.
|
||||
- **#5** — Bouton de copie de référence dans une popup épinglée : handler `copy-ref` ajouté.
|
||||
- **#6** — Popup épinglée au premier plan : clic sur une popup recalcule le z-index pour la passer au-dessus des autres.
|
||||
- **#7** — Notification « +2 min » fantôme : reset des flags d'alerte slide au retour de la prolongation.
|
||||
- **#8** — Compteur pompier : un pompier absent toute la journée est exclu du compteur.
|
||||
|
||||
### Dock latéral drag&drop
|
||||
|
||||
Le dock à droite permet de mettre des interventions de côté pendant qu'on navigue entre les jours, puis de les redéposer plus tard.
|
||||
|
||||
- Apparition graduelle pendant un drag : peep-min (sans contenu), peep (avec cartes), expanded (au survol ou au bord droit).
|
||||
- **Délai 500 ms** pour expand/collapse pour éviter le flicker quand le curseur effleure le bord du dock.
|
||||
- Card du dock : référence + durée prévue (`1h`, `1h30`, `45min`) avec barre verticale 4 px sur la gauche dans la couleur de la catégorie. Fond transparent.
|
||||
- Bouton de retrait `×` : **appui long 2 s** avec animation `conic-gradient`. Un clic simple ne fait rien.
|
||||
- Drag depuis le dock : retrait différé à l'activation effective du drag (5 px). Ghost flottant sans heure à gauche. Drop sur tech → modal de confirmation. Drop hors zone / Échap / sur le même slot d'origine → restauration dans le dock.
|
||||
- Plus de scrollbar horizontale en bas du dock, plus de clic-through dans la zone autour du bouton « Tout annuler ».
|
||||
|
||||
### Verdicts ghost — fix critique
|
||||
|
||||
Une intervention terminée par le tech mais dont la fiche était passée en statut « Redirigé » / « Finalisation » / « Exécution » était à tort marquée comme « annulée » et retirée du planning, parce que le parser de fiche cherchait les dates d'action aux indices `[8]`/`[9]` du tableau `rows` alors que le layout EV récent les place en `[6]`/`[7]` (et la description en `[9]` au lieu de `[11]`).
|
||||
|
||||
- **Détection robuste** : scan des valeurs pour le pattern `DD/MM/YYYY HH:MM:SS`, garde des 2 dernières occurrences comme (création, fin). Survit aux variations de layout EV.
|
||||
- **Décorrélation logs / KEEP-forcé** : nouvelle case dédiée « Garder les disparitions » dans Paramètres → Diagnostics, indépendante des « Logs verbeux ». Mode prod par défaut → verdict `REMOVE` appliqué (statut Annulé/Supprimé, ou statut clos sans commentaire du tech, ou action sans commentaire `login:`).
|
||||
|
||||
### Drag&drop bloqué pour les interventions non-déplaçables
|
||||
|
||||
`_canRescheduleIv` refuse maintenant explicitement le drag pour :
|
||||
|
||||
- iv en verdict `terminated-pending` (gris « fait »).
|
||||
- iv en verdict `terminated-clos` (vert ✓✓).
|
||||
- iv dont le statut EV est dans `CLOSED_STATUS` ou `RESOLVED_STATUS`.
|
||||
- iv en cours d'analyse de disparition (`_disappearChecking`).
|
||||
|
||||
### Popups dépinglés
|
||||
|
||||
- Auto-fermeture du popup dépinglé quand la souris quitte sa zone et celle des cards liées de la même iv (300 ms de grâce). N'interfère pas avec un drag de planning en cours.
|
||||
|
||||
### Tooltip / contact
|
||||
|
||||
- Le contact n'inclut plus les labels « fiche » (`Étage`, `Bureau`, `Service`, `Matériel`, `Problème`, `TFS`, `Date`, `Heure`, `Lieu`, `Bénéficiaire`, `Nom utilisateur`) qui se collent parfois à la valeur du contact à cause de séparateurs perdus dans la source EV. Le tooltip insère un saut de ligne avant chaque label collé pour la lisibilité.
|
||||
|
||||
|
||||
## v2026.5.44 — Refonte topbar, personnalisation Apparence, onboarding équipe, refresh séquentiel
|
||||
|
||||
> Refonte visuelle de la topbar (vue classique + horizontale), nouveau panneau
|
||||
> de personnalisation (couleur de la barre du haut + police de l'application
|
||||
> sur toute la page), nouvelle expérience d'onboarding quand aucun technicien
|
||||
> n'est sélectionné, refonte du système de verdicts ghost (✓✓ clos / ✓ Fait /
|
||||
> ✓ Suspendu), refresh strictement séquentiel avec arrêt instantané, et
|
||||
> plusieurs corrections.
|
||||
|
||||
### Refresh / cache / verdicts ghost
|
||||
|
||||
- Rafraîchissement **séquentiel** (1 fiche à la fois) au lieu de 5 workers
|
||||
parallèles → arrêt instantané via le bouton « ✕ Arrêter » (AbortController),
|
||||
plus de races DOM, ordre d'affichage cohérent (pompier d'abord, puis alpha,
|
||||
puis matin → après-midi).
|
||||
- Re-fetch du checksum frais via `basicAutoComplete` + `redirectHeader`
|
||||
(plus de fiche périmée entre sessions).
|
||||
- Cache merge robuste (fallback `cachedByRef` quand `actionId` change) et
|
||||
cache écrit toutes les 5 fiches pendant le refresh (incrémental).
|
||||
- **Système de verdicts ghost unifié** : ✓✓ vert (clos / résolu officiel),
|
||||
✓ gris « Fait » (terminated-pending), ✓ jaune « Suspendu »
|
||||
(terminated-suspended), retrait silencieux pour cancelled / cancelled-
|
||||
reservation / cancelled-absence.
|
||||
- Statuts EV (clos / résolu / annulé / suspendu) éditables depuis Paramètres
|
||||
→ EasyVista avec matching insensible à la casse, accents et conjugaisons.
|
||||
- Mise à jour live du tooltip et du popup épinglé après un verdict (plus
|
||||
besoin de fermer/réouvrir).
|
||||
- Clic immédiat sur la carte dès que le verdict tombe (avant la fin du
|
||||
refresh complet).
|
||||
- Boutons « Actualiser » (rapide, ne re-télécharge pas les fiches déjà
|
||||
connues) vs « Tout recharger » (force tout sauf les ✓✓ déjà clos).
|
||||
- **Mode diagnostic optionnel** (Paramètres → Diagnostics) : aucune
|
||||
intervention disparue n'est retirée silencieusement, tout est tracé sous
|
||||
le préfixe `[disparition]` dans la console F12 pour debug. En PROD
|
||||
(par défaut), les iv `cancelled` sont bien retirées comme avant.
|
||||
|
||||
### Topbar — vue classique
|
||||
|
||||
- Sélecteur de date du planning **ancré au centre absolu** : il ne se décale
|
||||
plus quand le bouton « ✕ Arrêter » apparaît à droite pendant un
|
||||
rafraîchissement.
|
||||
- Bouton **« Aujourd'hui »** affiché en toutes lettres (au lieu de « Auj. »).
|
||||
- Horloge contextuelle (date du jour + heure) réduite et discrète, à côté
|
||||
du bouton Aujourd'hui dans un cadre encadré.
|
||||
- Date du planning agrandie et neutre (couleur stable, plus de bascule
|
||||
selon la date sélectionnée).
|
||||
|
||||
### Personnalisation — Paramètres → Apparence
|
||||
|
||||
- **Couleur de la barre du haut** : 12 presets cliquables (Défaut, Blanc,
|
||||
Gris clair, Anthracite, Bleu DGNSI, Marine, Vert sapin, Brique, Violet,
|
||||
Rouge, Bleu pastel, Vert pastel) + picker custom + champ hex `#rrggbb`
|
||||
+ bouton « Réinitialiser ».
|
||||
- La couleur s'applique uniquement à la topbar (et à la sidebar quand on
|
||||
est en vue horizontale).
|
||||
- Le texte de la topbar (titre, horloge, date, capture-info, badges,
|
||||
boutons) s'adapte automatiquement (clair/foncé) selon la **luminance**
|
||||
de la couleur choisie pour rester toujours lisible.
|
||||
- **Police de l'application** : 28 choix organisés en familles
|
||||
(sans-serif : Arial, Helvetica, Verdana, Tahoma, Trebuchet, Calibri,
|
||||
Segoe UI, Gill Sans, Futura, Optima ; serif : Times New Roman, Georgia,
|
||||
Cambria, Garamond, Palatino, Bookman ; monospace : Courier New, Consolas,
|
||||
Lucida Console, JetBrains Mono ; display : Comic Sans MS, Impact,
|
||||
Brush Script, Copperplate ; condensée : Arial Narrow). La police choisie
|
||||
s'applique à **toute la page** (topbar, cards, popups, tooltips, panel
|
||||
admin) et chaque option du select s'affiche dans sa propre police pour
|
||||
prévisualiser le rendu, avec un aperçu live à droite.
|
||||
- Export / import du cache et de `admin_config` depuis Paramètres →
|
||||
Diagnostics.
|
||||
|
||||
### Vue horizontale
|
||||
|
||||
- Bloc « Aujourd'hui + horloge » empilé verticalement dans la sidebar, dans
|
||||
le même cadre encadré que la vue classique.
|
||||
- Date sélectionnée mise en avant (taille augmentée, en gras), date du
|
||||
jour et heure réduites à la même petite taille pour rester discrètes.
|
||||
- **Barre verticale verte** ajoutée à droite des mini-cards quand le
|
||||
ticket est officiellement clôturé / résolu (✓✓), avec léger décalage du
|
||||
✓✓ pour ne pas chevaucher la barre.
|
||||
- Quand l'utilisateur a choisi une couleur de topbar, la sidebar prend
|
||||
aussi la couleur : titre, horloge, capture-info, stats, today-block,
|
||||
date sélectionnée, boutons, theme-toggle et séparateurs adoptent une
|
||||
teinte translucide cohérente (via `color-mix`) qui contraste correctement
|
||||
sur n'importe quel fond.
|
||||
|
||||
### Statistiques globales
|
||||
|
||||
- Nouveau compteur **« X faits / Y clos »** entre `(matin · après-midi)`
|
||||
et `tech. dispo`. Inclut tous les tickets terminés (clos/résolus officiels
|
||||
+ verdicts ghost « Fait » / « Suspendu »).
|
||||
- En vue classique, séparateur `//` après `clos` (au lieu de `·`).
|
||||
- En vue horizontale (sidebar), une **barre horizontale 1px** sépare le
|
||||
bloc interventions/faits/clos du bloc tech. dispo + pompiers / absents.
|
||||
|
||||
### Onboarding équipe (1ʳᵉ install ou config vide)
|
||||
|
||||
- L'erreur générique « Aucun technicien sélectionné » est remplacée par une
|
||||
**carte d'onboarding centrée** comprenant :
|
||||
- icône (👥) cerclée en couleur accent du thème ;
|
||||
- titre « Aucune équipe configurée » ;
|
||||
- description claire ;
|
||||
- bouton primary **« Ouvrir paramètres »** qui ouvre directement le panel
|
||||
admin sur la section Équipe.
|
||||
- Carte centrée verticalement et horizontalement dans la zone disponible,
|
||||
identique en vue classique et horizontale.
|
||||
|
||||
### Bugfix
|
||||
|
||||
- **Issue #1 (Pompier + Absence)** : si un tech est à la fois pompier ET
|
||||
absent, les deux badges s'affichent désormais avec un séparateur `/` au
|
||||
lieu de masquer l'absence derrière le badge pompier.
|
||||
- **Absences récurrentes** : quand on changeait de groupe puis revenait au
|
||||
groupe initial, les jours d'absence cochés pour les techniciens
|
||||
disparaissaient visuellement (la donnée elle-même restait en storage).
|
||||
Correction : restauration depuis `cfg.recurringAbsences` à chaque
|
||||
re-render.
|
||||
- **Barre de progression / bannière session expirée** : suivent désormais
|
||||
la hauteur dynamique de la topbar (variable CSS `--topbar-height` mesurée
|
||||
par un `ResizeObserver`). Plus de chevauchement quand on scrolle.
|
||||
- **STATUS_FR regex** : limite augmentée de 30 à 200 caractères (battait
|
||||
sur « Suspendu : Attente info bénéficiaire/demandeur »).
|
||||
- **Description action** : décodage `" → "`, `<br> → \n`, HTML
|
||||
strippé. Préfixe « login: » retiré du commentaire technicien dans le
|
||||
tooltip / popup.
|
||||
- **Tooltip référence** : flèche « ↗ » retirée du lien cliquable.
|
||||
|
||||
---
|
||||
|
||||
## v2026.5.43 — Fix Firefox : positionnement menu dock + stabilité popup pin/unpin
|
||||
|
||||
### Menu hover sur pastille du dock (popup réduit)
|
||||
- Bug Firefox uniquement : quand un popup épinglé était réduit dans la
|
||||
taskbar du bas, le menu qui apparaît au survol de la pastille
|
||||
(Agrandir / Fermer) se positionnait trop haut, pas juste au-dessus de
|
||||
la pastille.
|
||||
- Cause : `getBoundingClientRect()` était appelé immédiatement après
|
||||
`appendChild`, avant que Firefox n'ait calculé la mise en page.
|
||||
Combiné avec un `transform: translateY(4px)` dans l'animation
|
||||
`pill-hover-menu-appear`, Firefox lisait des dimensions décalées.
|
||||
- Fix : positionnement hors écran initial, force-layout via
|
||||
`void offsetHeight`, mesure des dimensions, puis pose finale. CSS de
|
||||
l'animation simplifiée en opacité-only (plus de transform).
|
||||
|
||||
### Stabilité popup au pin/unpin
|
||||
- Bug : la popup épinglée bougeait visuellement et changeait légèrement
|
||||
de taille quand on la dé-épinglait avec le bouton 📌 (puis l'inverse).
|
||||
- Cause : `.pinned-popup` avait `padding-top: 28px` (place pour la
|
||||
dragbar) et `border: 2px`, alors que `.soft-unpinned` avait
|
||||
`padding-top: 12px` et `border: 1px`. Le contenu se décalait de 16px
|
||||
vers le haut et la popup devenait 1px plus fine de chaque côté.
|
||||
- Fix : `.soft-unpinned` conserve désormais `padding-top: 28px` et
|
||||
`border: 2px` comme `.pinned-popup`. Bordure passe juste en
|
||||
`--border-strong` (gris discret) plutôt que `--accent` (bleu) pour
|
||||
signaler visuellement le mode "détaché". Position et taille stables.
|
||||
|
||||
## v2026.5.42 — Nettoyage de commentaires + exemples génériques
|
||||
|
||||
- Passage en revue des commentaires de `src/viewer.js` : les exemples qui
|
||||
@@ -66,7 +270,7 @@
|
||||
au viewer via `err.kind`.
|
||||
- Toutes les anciennes constantes hardcodées (`EV_ORIGINS`,
|
||||
`DEFAULT_SUPPORT_IDS` interne à `detectTeamFromEV`,
|
||||
`isPillonelAbsentFriday`) ont été remplacées ou retirées.
|
||||
`isXXXAbsentFriday`) ont été remplacées ou retirées.
|
||||
|
||||
### Conflits absence/réservation × intervention
|
||||
- Nouveau code visuel : si une intervention est planifiée pendant
|
||||
@@ -259,12 +463,12 @@
|
||||
- Stats rapides .tech-row-stats ajoutés au header (nb interv, Xm · Ya)
|
||||
|
||||
## v2026.5.31 — Sarcelle pour absence récurrente (REJETÉ par utilisateur)
|
||||
- Couleur Pillonel vendredi : sarcelle foncée #0f766e / soft #ccfbf1
|
||||
- Couleur absence récurrente (jour fixe) : sarcelle foncée #0f766e / soft #ccfbf1
|
||||
- Variables --c-recurring, --c-recurring-soft
|
||||
- Layout 4 colonnes forcées + scroll interne cartes (REJETÉ : "scroll en continu")
|
||||
|
||||
## v2026.5.30 — Absence récurrente cyan + mode compact 24"
|
||||
- Absence récurrente Pillonel vendredi en cyan
|
||||
- Absences récurrentes (configurées par tech) en cyan
|
||||
- Mode compact @media (max-width: 1920px) avec grid-template-columns: repeat(4, 1fr)
|
||||
|
||||
## v2026.5.29 — Contraste++ + footer
|
||||
@@ -313,8 +517,9 @@
|
||||
|
||||
## Versions antérieures (v5.x et v4.x)
|
||||
|
||||
> Ces versions sont à analyser par Claude Code à partir des fichiers source.
|
||||
> Indices clés à chercher dans le viewer.js :
|
||||
> Ces versions ne sont pas documentées en détail. Pour les analyser à partir
|
||||
> des fichiers source historiques (consultables via les tags git), voici les
|
||||
> indices clés à chercher dans `viewer.js` :
|
||||
>
|
||||
> - Présence de `pinTooltip` → version >= v4.x
|
||||
> - Présence de `_softUnpinPopup` → version >= v4.3.3
|
||||
@@ -342,14 +547,14 @@
|
||||
- v4.3.3 : _softUnpinPopup (désépinglage mou)
|
||||
|
||||
### v3.x et antérieures — Versions de base
|
||||
- À analyser par Claude Code
|
||||
- Code historique consultable via les tags git correspondants.
|
||||
|
||||
---
|
||||
|
||||
## Notes techniques persistantes (toutes versions)
|
||||
|
||||
- 8 techs hardcodés : "76272,83725,66635,92235,90070,40944,72485,86874"
|
||||
- Pillonel Olivier (ID 40944) absent tous les vendredis (hardcodé)
|
||||
- 8 techs hardcodés à l'origine (depuis v2026.5.41 : retirés, alimentés par admin_config)
|
||||
- Absences récurrentes (un tech absent un jour fixe par semaine) hardcodées à l'origine, depuis v2026.5.41 configurables via Paramètres → Équipe
|
||||
- Group ID EasyVista : 191
|
||||
- Domaines cibles : itsma.etat-de-vaud.ch (interne), itsma.vd.ch (externe)
|
||||
- SSO : Canton ForgeRock OpenAM
|
||||
|
||||
@@ -1,192 +0,0 @@
|
||||
# CLAUDE.md — Workflow de développement Planification
|
||||
|
||||
> **À lire avant toute modification.** Ce fichier décrit le processus complet
|
||||
> que Claude doit suivre quand Quentin demande une modification de l'extension.
|
||||
|
||||
---
|
||||
|
||||
## Stack du projet
|
||||
|
||||
- **Type** : extension navigateur Manifest V3 (Chrome / Edge / Firefox 140+)
|
||||
- **Fonction** : viewer du planning des techniciens DGNSI dans EasyVista
|
||||
- **Cible utilisateurs** : techniciens DGNSI (Canton de Vaud)
|
||||
- **Auteur** : Quentin Rouiller (email dans la mémoire Claude `user_role.md`)
|
||||
- **Repo Gitea** : https://gitea.netaplaid.ch/FroSteel/Planification
|
||||
- **Config runtime** : `chrome.storage.local["admin_config"]` (persiste entre updates)
|
||||
|
||||
## Structure du repo
|
||||
|
||||
```
|
||||
src/ # Sources de l'extension (chargées par le navigateur)
|
||||
├── manifest.json # Manifest V3 — version YYYY.M.PATCH
|
||||
├── background.js # Service worker
|
||||
├── viewer.{html,js,css}
|
||||
└── icons/
|
||||
|
||||
Autres/ # Méta + build
|
||||
├── build.sh # Génère dist/{chromium,firefox}/, .zip, .xpi, met à jour firefox-updates.json
|
||||
├── CHANGELOG.md # Synchronisé avec le CHANGELOG.md à la racine
|
||||
├── README.md # Synchronisé avec le README.md à la racine
|
||||
└── LICENSE
|
||||
|
||||
Builds/ # Artefacts distribués (Chromium/, Firefox/, .zip, .xpi)
|
||||
dist/ # Sortie de build (gitignoré)
|
||||
firefox-updates.json # Manifest d'auto-update Firefox (servi via update_url)
|
||||
CLAUDE.md # Ce fichier
|
||||
CHANGELOG.md # Source de vérité du changelog (lu par AMO + GitHub-style)
|
||||
README.md # Source de vérité du README
|
||||
```
|
||||
|
||||
> **NB** : le repo Gitea utilise un **layout flat à la racine** pour le code
|
||||
> source historique (`build.sh`, `README.md`, `CHANGELOG.md`, `LICENSE`,
|
||||
> `firefox-updates.json` à la racine, et `src/` pour le code). En local,
|
||||
> le dossier `Autres/` contient une copie de ces fichiers — tu peux éditer
|
||||
> l'un ou l'autre, mais quand tu pousses sur Gitea c'est la racine qui doit
|
||||
> être mise à jour.
|
||||
|
||||
---
|
||||
|
||||
## Workflow standard d'une demande de modification
|
||||
|
||||
Quand Quentin demande une nouvelle fonctionnalité ou un bugfix :
|
||||
|
||||
### Phase 1 — Code + build local (toi seul, pas encore de push)
|
||||
|
||||
1. **Comprendre la demande**, poser des questions si nécessaire avant d'écrire du code.
|
||||
2. **Coder les modifications** dans `src/` (jamais directement dans `dist/` ou `Builds/`).
|
||||
3. **Bumper la version** : incrémenter le 3e chiffre dans `src/manifest.json`
|
||||
(ex: `2026.5.41` → `2026.5.42`). Bump majeur (2e chiffre) seulement pour
|
||||
les refontes ; année (1er chiffre) au passage à 2027.
|
||||
4. **Mettre à jour le CHANGELOG** (`Autres/CHANGELOG.md` ET la copie racine
|
||||
`CHANGELOG.md`) en ajoutant une nouvelle entrée en haut.
|
||||
5. **Mettre à jour le README** (`Autres/README.md` ET racine `README.md`)
|
||||
si la nouvelle version touche aux fonctionnalités principales.
|
||||
6. **Builder** : `./Autres/build.sh` — ça produit `dist/chromium/`,
|
||||
`dist/firefox/`, le `.zip` et le `.xpi` avec la nouvelle version.
|
||||
7. **Annoncer à Quentin** : "v2026.5.X buildée, recharge l'extension dans
|
||||
ton navigateur et teste". Décrire brièvement ce qui a changé visuellement.
|
||||
8. **Attendre son retour**. Tant qu'il n'a pas dit "OK", ne pas pousser sur
|
||||
Gitea. Si correction demandée, retourner à l'étape 2.
|
||||
|
||||
### Phase 2 — Push sur Gitea (uniquement après validation explicite)
|
||||
|
||||
Quand Quentin dit "OK push" / "valide" / équivalent :
|
||||
|
||||
1. **Préparer un clone Gitea à jour** dans `/tmp/planif-push/` (clone si pas
|
||||
présent, sinon `git fetch origin && git reset --hard origin/main`).
|
||||
2. **Synchroniser** :
|
||||
- `rsync -a --delete /Users/quentin/Documents/Planning/src/ /tmp/planif-push/src/`
|
||||
- Copier les versions racine de `CHANGELOG.md`, `README.md`, `LICENSE`,
|
||||
`build.sh` (les versions racine sur Gitea, pas celles de `Autres/`)
|
||||
3. **Régénérer `firefox-updates.json`** à la racine du repo : ajouter
|
||||
l'entrée de la nouvelle version en haut de la liste `updates` (les
|
||||
anciennes entrées restent — Firefox prend la version la plus haute
|
||||
parmi celles listées). Le `update_link` pointe vers la release Gitea :
|
||||
`https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/vYYYY.M.PATCH/planification-vYYYY.M.PATCH-firefox.xpi`.
|
||||
Le `update_hash` est calculé après signature AMO (cf. Phase 3).
|
||||
|
||||
Le repo Gitea est **public**, donc l'URL fixe `update_url` =
|
||||
`https://gitea.netaplaid.ch/FroSteel/Planification/raw/branch/main/firefox-updates.json`
|
||||
est accessible sans auth → Firefox peut le fetcher directement.
|
||||
`build.sh` maintient automatiquement ce JSON à chaque build (ajoute /
|
||||
met à jour l'entrée de la version courante).
|
||||
4. **Commit + push** :
|
||||
```bash
|
||||
cd /tmp/planif-push
|
||||
git add -A
|
||||
git commit -m "vYYYY.M.PATCH — <description courte>"
|
||||
git push origin main
|
||||
git tag -a vYYYY.M.PATCH -m "vYYYY.M.PATCH"
|
||||
git push origin vYYYY.M.PATCH
|
||||
```
|
||||
5. **Créer la release Gitea** via l'API (POST
|
||||
`/repos/FroSteel/Planification/releases`) avec :
|
||||
- `tag_name`: `vYYYY.M.PATCH`
|
||||
- `name`: `vYYYY.M.PATCH`
|
||||
- `body`: extrait du CHANGELOG (la section de cette version)
|
||||
6. **Uploader les binaires** comme assets de la release :
|
||||
- `dist/planification-vYYYY.M.PATCH-chromium.zip`
|
||||
- `dist/planification-vYYYY.M.PATCH-firefox.xpi` (NON signé pour le moment)
|
||||
7. **Mettre à jour le wiki Gitea** :
|
||||
- Page **Versions** : ajouter une entrée détaillée en haut (dérivée du CHANGELOG)
|
||||
- Page **Utilisation** : si le changement modifie l'UX (ajout d'un bouton,
|
||||
d'une section admin, d'un comportement) → documenter
|
||||
- Page **Architecture** : si nouvelles fonctions clés / nouvelle config
|
||||
persistée → documenter
|
||||
|
||||
### Phase 3 — Signature Firefox (manuel, fait par Quentin)
|
||||
|
||||
C'est la seule étape que Claude ne peut pas automatiser :
|
||||
|
||||
1. Quentin va sur https://addons.mozilla.org/developers/
|
||||
2. Submit New Version → uploade le `.xpi` non signé de la release Gitea
|
||||
3. Choisit **"On your own"** (Unlisted, self-distributed)
|
||||
4. Mozilla signe → Quentin télécharge le `.xpi` signé
|
||||
|
||||
Quentin revient ensuite avec le `.xpi` signé et demande "remplace par le signé".
|
||||
À ce moment Claude fait :
|
||||
1. Remplacer l'asset `.xpi` de la release Gitea (delete + upload)
|
||||
2. Calculer le `sha256` du `.xpi` signé
|
||||
3. Mettre à jour `firefox-updates.json` : ajouter `"update_hash": "sha256:<hash>"`
|
||||
4. Commit + push le JSON mis à jour
|
||||
|
||||
À partir de ce moment, l'auto-update Firefox fonctionne pour cette version.
|
||||
|
||||
---
|
||||
|
||||
## Token Gitea
|
||||
|
||||
⚠️ **Le token API Gitea ne doit JAMAIS apparaître dans ce fichier ni dans le
|
||||
repo Gitea**. Il est stocké uniquement dans la mémoire Claude locale
|
||||
(`~/.claude/projects/-Users-quentin-Documents-Planning/memory/gitea_token.md`,
|
||||
hors repo). Si Claude perd la mémoire (nouvelle session non héritée),
|
||||
demander à Quentin de redonner le token.
|
||||
|
||||
Header API à utiliser : `Authorization: token <TOKEN>` + `User-Agent: curl/8.4.0`
|
||||
(le User-Agent évite que Cloudflare bloque les requêtes Python urllib).
|
||||
|
||||
---
|
||||
|
||||
## Règles importantes
|
||||
|
||||
- **Ne jamais hardcoder** dans `src/` : groupe EV, équipe, domaines, absences
|
||||
récurrentes. Tout passe par `admin_config`. Les seuls hardcodes acceptables
|
||||
sont les **filets de sécurité** (DEFAULT_GROUP_ID, DEFAULT_EV_ORIGINS pour
|
||||
le 1er install). Cf. v2026.5.41 pour la migration complète.
|
||||
- **Ne jamais pousser sur Gitea sans validation explicite** de Quentin.
|
||||
- **Toujours bumper la version** avant un push qui modifie le code.
|
||||
- **Toujours mettre à jour le CHANGELOG** avant un push.
|
||||
- **Tags non touchés** sur Gitea : `v1.0.0`-`v3.3.0`, `v4.1.x`-`v4.3.0`,
|
||||
`v5.0.0`, `v2026.5.27`-`v2026.5.32` (ceux-là pointent vers du code
|
||||
reconstitué historique, ne jamais les bouger).
|
||||
- **Force-push uniquement si Quentin le demande explicitement.**
|
||||
- **L'email de l'auteur** ne doit apparaître nulle part dans `src/` ni dans
|
||||
les fichiers Markdown du repo (CLAUDE.md, README.md inclus). Il est stocké
|
||||
uniquement en mémoire Claude (`user_role.md` / `gitea_token.md`) et exposé
|
||||
obfusqué (entités HTML) sur la page wiki Contact.
|
||||
|
||||
---
|
||||
|
||||
## Pages wiki Gitea
|
||||
|
||||
| Page | Contenu | Quand mettre à jour |
|
||||
|---|---|---|
|
||||
| **Home** | Pitch, contexte, démarrage rapide | Rarement |
|
||||
| **Utilisation** | Guide complet pour l'utilisateur | À chaque changement UX |
|
||||
| **Versions** | Historique détaillé des versions | À chaque release |
|
||||
| **Architecture** | Doc technique (fonctions, config, structure) | À chaque ajout d'helper / changement structurel |
|
||||
| **Contact** | Email obfusqué + lien Issue Gitea | Rarement |
|
||||
|
||||
URL de base wiki : `https://gitea.netaplaid.ch/FroSteel/Planification/wiki/<NOM>`
|
||||
Endpoint API : `/api/v1/repos/FroSteel/Planification/wiki/page/<NOM>` (PATCH avec
|
||||
`content_base64`).
|
||||
|
||||
---
|
||||
|
||||
## Pour résumer ton rôle, Claude
|
||||
|
||||
Quentin demande une modif → tu codes → tu builds → il teste → il valide →
|
||||
tu push tout (Gitea + wiki + firefox-updates.json). Plus tard il revient avec
|
||||
le `.xpi` signé d'AMO → tu mets à jour la release et le `update_hash` du JSON.
|
||||
|
||||
Si tu hésites sur quoi faire à un moment, **demande**. Ne suppose pas.
|
||||
@@ -1,16 +1,39 @@
|
||||
# Planification — Extension EasyVista Canton de Vaud
|
||||
|
||||
Extension Chrome / Firefox pour visualiser de manière claire et rapide le planning des techniciens DGNSI (Canton de Vaud) dans EasyVista.
|
||||
Extension Chrome / Firefox pour visualiser de manière claire et rapide le planning EasyVista de l'équipe technicienne DGNSI (Canton de Vaud).
|
||||
|
||||
> 📖 **Documentation utilisateur complète** : [wiki](https://gitea.netaplaid.ch/FroSteel/Planification/wiki) ([Home](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Home) · [Utilisation](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Utilisation) · [Versions](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Versions) · [Architecture](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Architecture) · [Dépannage](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/D%C3%A9pannage) · [Contact](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Contact))
|
||||
|
||||
## Aperçu rapide
|
||||
|
||||
- **Auteur** : Quentin Rouiller (QRO)
|
||||
- **Cible** : techniciens DGNSI (Canton de Vaud), EasyVista (`itsma.etat-de-vaud.ch` / `itsma.vd.ch`)
|
||||
- **Auteur** : Quentin Rouiller (QRO), Technicien DGNSI — Canton de Vaud
|
||||
- **Public cible** : coordinateurs DGNSI qui pilotent dans EasyVista (`itsma.etat-de-vaud.ch` / `itsma.vd.ch`) le planning de l'équipe technicienne
|
||||
- **Démarrage projet** : jeudi 16 avril 2026
|
||||
- **Version actuelle** : `v2026.5.42`
|
||||
- **Contact** : voir [page wiki Contact](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Contact)
|
||||
- **Manifest** : V3 (Chrome/Edge/Firefox)
|
||||
- **Format** : `.zip` (Chromium) + `.xpi` signé (Firefox)
|
||||
- **Version actuelle** : [`v2026.5.45`](https://gitea.netaplaid.ch/FroSteel/Planification/releases/tag/v2026.5.45) (latest)
|
||||
- **Contact** : voir [page wiki Contact](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Contact) ou [ouvrir une issue](https://gitea.netaplaid.ch/FroSteel/Planification/issues/new)
|
||||
- **Manifest** : V3 (Chrome/Edge/Firefox 140+)
|
||||
- **Format** : `.zip` (Chromium) + `.xpi` signé Mozilla (Firefox)
|
||||
- **Distribution** : auto-update natif Firefox via `firefox-updates.json`
|
||||
|
||||
## Installation rapide
|
||||
|
||||
### Firefox 🦊 (recommandé — auto-update)
|
||||
|
||||
1. Télécharger le `.xpi` signé depuis la **[release courante](https://gitea.netaplaid.ch/FroSteel/Planification/releases/latest)**.
|
||||
2. Drag-and-drop dans `about:addons` de Firefox.
|
||||
3. Cliquer "Ajouter".
|
||||
|
||||
À partir de là, l'extension se met à jour **automatiquement** à chaque nouvelle version (vérification toutes les ~24 h via `firefox-updates.json`).
|
||||
|
||||
### Chrome / Edge / Brave 🌐 (manuel)
|
||||
|
||||
1. Télécharger le `.zip` depuis la **[release courante](https://gitea.netaplaid.ch/FroSteel/Planification/releases/latest)**.
|
||||
2. Décompresser dans un dossier permanent.
|
||||
3. `chrome://extensions/` (ou `edge://extensions/`) → activer **Mode développeur** → "Charger l'extension non empaquetée" → sélectionner le dossier décompressé.
|
||||
|
||||
Les mises à jour sont **manuelles** : à chaque nouvelle release, retélécharger le `.zip`, écraser le dossier, puis cliquer ⟳ (Recharger) sur la carte de l'extension.
|
||||
|
||||
➡ Pour le détail complet (stockage, désinstallation, comparatif), voir [wiki Utilisation → Installation et navigateurs](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Utilisation#installation-et-navigateurs).
|
||||
|
||||
## Fonctionnalités principales
|
||||
|
||||
@@ -40,7 +63,7 @@ Extension Chrome / Firefox pour visualiser de manière claire et rapide le plann
|
||||
- **Congé / Congés** : cyan `#06b6d4` (suffixe `s` adaptatif)
|
||||
- **Pompier** : rouge `#b03030`
|
||||
- Badge + barre gauche colorée + dégradé fond
|
||||
- Absence récurrente Pillonel vendredi : cyan (depuis v2026.5.30)
|
||||
- Absences récurrentes (configurées par tech) : cyan (depuis v2026.5.30)
|
||||
|
||||
### User et session
|
||||
- Badge user avec photo/initiales en topbar
|
||||
@@ -55,6 +78,9 @@ Extension Chrome / Firefox pour visualiser de manière claire et rapide le plann
|
||||
- Sélecteur de groupe EasyVista (SI-CSS, SI-EXT, …) en tête de l'onglet Équipe (depuis v2026.5.40) — détection automatique via le `<select id="plan_group_id">` de la page Planning EV, robuste aux ajouts/renommages côté EV
|
||||
- Édition manuelle des domaines EasyVista interne / externe (depuis v2026.5.40)
|
||||
- Tri des techniciens : actifs d'abord, puis exclus, alphabétique dans chaque groupe (depuis v2026.5.40)
|
||||
- **Personnalisation Apparence (depuis v2026.5.44)** : couleur de la topbar (12 presets + custom + hex) avec contraste auto-calculé sur le texte ; police de l'application (28 choix : Arial, Helvetica, Verdana, Tahoma, Trebuchet, Calibri, Segoe UI, Times New Roman, Georgia, Cambria, Garamond, Palatino, Courier, Consolas, Comic Sans, Impact, …) appliquée à toute la page
|
||||
- **Onboarding équipe (depuis v2026.5.44)** : carte centrée propre quand aucun tech n'est sélectionné, avec bouton « Ouvrir paramètres » qui dépose directement sur la section Équipe
|
||||
- **Statuts EV configurables (depuis v2026.5.44)** : clos / résolu / annulé / suspendu éditables depuis Paramètres → EasyVista, matching insensible à la casse / accents / conjugaisons
|
||||
|
||||
## Versionning — historique et conventions
|
||||
|
||||
@@ -64,7 +90,7 @@ L'extension a connu **3 systèmes de versionning successifs** :
|
||||
|---|---|---|
|
||||
| 16-17 avril 2026 | Versions de base | `1.0.0`, `2.0.0`, `3.0.0` |
|
||||
| 18-20 avril 2026 | SemVer classique | `4.1.3`, `4.2.8`, `5.0.12` |
|
||||
| 21 avril 2026 → maintenant | **`ANNÉE.MAJEURE.PATCH`** | `2026.5.16` → `2026.5.40` |
|
||||
| 21 avril 2026 → maintenant | **`ANNÉE.MAJEURE.PATCH`** | `2026.5.16` → `2026.5.45` |
|
||||
|
||||
### Format actuel : `ANNÉE.MAJEURE.PATCH`
|
||||
|
||||
@@ -87,7 +113,36 @@ Le numéro de **majeure** n'est **pas** un mois et **pas** un chiffre lié au ca
|
||||
|
||||
## Versions notables
|
||||
|
||||
### `v2026.5.42` (latest, 27 avril 2026) — Nettoyage de commentaires + exemples génériques
|
||||
### `v2026.5.45` (latest, 8 mai 2026) — Dock latéral drag&drop, fix verdicts ghost, multi-onglets EZV résolu
|
||||
- **Dock latéral drag&drop** : nouvelle zone à droite pour parquer des interventions et les redéposer plus tard, avec apparition graduelle (peep-min/peep/expanded), délai 500 ms pour expand/collapse anti-flicker, card compacte (réf + durée + barre catégorie 4 px à gauche), bouton de retrait par appui long 2 s.
|
||||
- **Verdicts ghost** : fix critique du parser de fiche (les dates d’action ne sont plus cherchées aux indices fixes \`[8]\`/\`[9]\` mais détectées par scan regex pattern \`DD/MM/YYYY HH:MM:SS\`). Décorrélation des « Logs verbeux » et « Garder les disparitions » dans Paramètres → Diagnostics.
|
||||
- **Drag&drop bloqué** sur les interventions « Fait » (gris), « Clos » (vert ✓✓), statut Clôturé/Résolu/Terminé, et pendant l’analyse de disparition d’un ghost.
|
||||
- **Issue #3** : coches « Absences récurrentes » retenues (merge propre au lieu d’écrasement) lors d’un changement de groupe.
|
||||
- **Issue #4** : multi-onglets EZV — permission optionnelle \`cookies\` + listener \`cookieChanged\`, la session reste valable après reconnexion + fermeture d’un onglet EZV.
|
||||
- **Issue #5** : copie de référence dans une popup épinglée fonctionne.
|
||||
- **Issue #6** : popup épinglée passe au premier plan au clic.
|
||||
- **Issue #7** : plus de notification « +2 min » fantôme après prolongation.
|
||||
- **Issue #8** : pompier absent toute la journée n’est plus compté.
|
||||
- **Auto-fermeture** des popups dépinglés quand la souris quitte leur zone et celle des cards liées.
|
||||
- **Tooltip / contact nettoyé** : les labels « fiche » (Étage/Bureau/Service/etc.) collés à la valeur du contact sont retirés.
|
||||
|
||||
### `v2026.5.44` (1 mai 2026) — Refonte topbar, personnalisation Apparence, onboarding équipe, fix #1
|
||||
- **Topbar refondue (vue classique)** : sélecteur de date du planning ancré au centre absolu (ne se décale plus quand le bouton « Arrêter » apparaît), bouton « Aujourd'hui » en toutes lettres, horloge contextuelle dans un cadre encadré.
|
||||
- **Personnalisation Apparence** : couleur de la barre du haut (12 presets + picker custom + champ hex), contraste de texte calculé automatiquement par luminance ; police de l'application avec 28 choix (Arial, Helvetica, Verdana, Tahoma, Trebuchet, Calibri, Segoe UI, Times New Roman, Georgia, Cambria, Garamond, Palatino, Courier, Consolas, Comic Sans, Impact, …) appliquée à toute la page.
|
||||
- **Vue horizontale** : bloc Aujourd'hui + horloge dans le même cadre que la classique, barre verte verticale à droite des mini-cards clos/résolu, sidebar adopte la couleur de topbar custom de manière cohérente.
|
||||
- **Stats globales** : nouveau compteur « X faits / Y clos » ; vue classique avec « // » après clos, vue horizontale avec barre horizontale 1px de séparation.
|
||||
- **Onboarding équipe** : carte centrée propre (icône, titre, description, bouton « Ouvrir paramètres ») au lieu du bandeau d'erreur ; le bouton ouvre directement la section Équipe.
|
||||
- **Refresh / cache / verdicts ghost** : rafraîchissement séquentiel avec arrêt instantané (AbortController), checksum frais via basicAutoComplete + redirectHeader, cache merge robuste, verdicts unifiés (✓✓ clos, ✓ Fait, ✓ Suspendu jaune, retrait silencieux pour cancelled), mode diagnostic optionnel.
|
||||
- **Issue #1 (Pompier + Absence)** : les deux badges s'affichent désormais avec un séparateur `/` au lieu de masquer l'absence.
|
||||
- **Bugfix divers** : absences récurrentes restaurées au switch de groupe, barre de progression / bannière session expirée suivent la hauteur dynamique de la topbar (`--topbar-height` via `ResizeObserver`), description action décodée proprement, flèche `↗` retirée des références cliquables.
|
||||
|
||||
### `v2026.5.43` (27 avril 2026) — Fix Firefox : menu dock + stabilité popup pin/unpin
|
||||
- Firefox : le menu hover sur les pastilles du dock (popup réduit) se
|
||||
positionne désormais correctement au-dessus de la pastille.
|
||||
- Pin/unpin : la popup épinglée ne bouge plus et garde la même taille
|
||||
quand on la dé-épingle / re-épingle.
|
||||
|
||||
### `v2026.5.42` — Nettoyage de commentaires + exemples génériques
|
||||
- Uniformisation des exemples utilisés dans les commentaires de `viewer.js`
|
||||
(parsing contacts/lieux/références/codes-barres) en placeholders abstraits.
|
||||
Comportement runtime strictement inchangé.
|
||||
@@ -99,7 +154,7 @@ Le numéro de **majeure** n'est **pas** un mois et **pas** un chiffre lié au ca
|
||||
- **Heures de la journée** : bouton ✓ Appliquer explicite (au lieu de save direct), toast de confirmation, refetch automatique du planning. Synchronisation effective avec les requêtes EV (`day_start_hour` / `day_end_hour` / `begin_hour` / `end_hour`) — avant, l'affichage changeait mais les requêtes restaient sur 8h-19h hardcodés.
|
||||
- **Thème unifié** : le toggle 🌙 de la topbar et le sélecteur Apparence du panel admin écrivent dans la même clé (`cfg.theme`). Le mode "Automatique" est résolu en JS via `prefers-color-scheme` (le CSS n'avait pas de bloc `@media`, ce qui faisait retomber sur le clair même quand l'OS était en sombre). Listener `matchMedia` pour bascule live en mode auto.
|
||||
- **Conflit absence/réservation × intervention** : si une intervention est planifiée pendant qu'un tech a une absence (toute la journée ou demi-journée) ou une réservation au même créneau, sa carte est peinte en **rouge plein** (intervention conflictuelle). Logique : full-day → toutes en rouge ; partiel → seules celles en chevauchement.
|
||||
- **Pillonel & Cie** : suppression de la fonction hardcodée `isPillonelAbsentFriday()`. L'absence récurrente est désormais générique : `RECURRING_ABSENCES[tech.id]` lit `cfg.recurringAbsences` et le label "Absent le X" est calculé dynamiquement depuis le jour de la semaine.
|
||||
- **Absences récurrentes génériques** : suppression de la fonction hardcodée `isXXXAbsentFriday()`. L'absence récurrente est désormais générique : `RECURRING_ABSENCES[tech.id]` lit `cfg.recurringAbsences` et le label "Absent le X" est calculé dynamiquement depuis le jour de la semaine.
|
||||
- **Notifications au-dessus du flou** : z-index `.toast-stack` relevé à 11000 (le panel admin est à 10000) pour que les toasts de feedback restent visibles quand l'admin est ouvert.
|
||||
- **Vue horizontale** : popups au survol/clic limités aux candidats `dessous`/`dessus` (la sidebar à gauche et la timeline pleine largeur rendent gauche/droite peu praticables).
|
||||
- **Tri équipe** : inclus d'abord, puis exclus, alphabétique dans chaque sous-groupe (ne saute plus quand on coche/décoche).
|
||||
@@ -109,7 +164,6 @@ Le numéro de **majeure** n'est **pas** un mois et **pas** un chiffre lié au ca
|
||||
|
||||
### `v2026.5.40` — Sélection groupe EV + édition domaines + tri équipe + vue horizontale enrichie
|
||||
- **Onglet Équipe** : sélecteur de groupe EasyVista (SI-CSS, SI-EXT, …) en tête de section, détecté automatiquement à l'ouverture du panel via le `<select id="plan_group_id">` de la page Planning EV. Robuste aux ajouts/renommages côté EV.
|
||||
- **Onglet Équipe** : sélecteur de groupe EasyVista (SI-CSS, SI-EXT, …) en tête de section, détecté automatiquement à l'ouverture du panel via le `<select id="plan_group_id">` de la page Planning EV. Robuste aux ajouts/renommages côté EV.
|
||||
- ID groupe affiché en italique (ex: `ID groupe : 191`).
|
||||
- Quand on change de groupe, la liste d'équipe se rafraîchit automatiquement avec les membres du nouveau groupe.
|
||||
- Plus de bouton "Détecter" : tout est auto à l'ouverture.
|
||||
@@ -122,110 +176,29 @@ Le numéro de **majeure** n'est **pas** un mois et **pas** un chiffre lié au ca
|
||||
### `v2026.5.39` — Séparation Matin / Après-midi + Apparence
|
||||
- Pills "MATIN" / "APRÈS-MIDI" entre les interventions de chaque tech
|
||||
- Section **Apparence** dans les paramètres : thème, taille du texte, durée du cache, heures de la journée
|
||||
- Section **À propos** (version, auteur, licence)
|
||||
|
||||
### `v2026.5.38` — Attribution auteur + nettoyage + observabilité
|
||||
- Module `LOG` unifié + handlers globaux d'erreur
|
||||
- Toggle "Logs verbeux (debug)" dans le panel admin
|
||||
- En-têtes copyright dans tous les fichiers source
|
||||
|
||||
### `v2026.5.37` — Refonte vue horizontale (sidebar complète)
|
||||
- Topbar entièrement déplacée en sidebar verticale
|
||||
|
||||
### `v2026.5.36` — Sidebar verticale en vue horizontale
|
||||
- Wrapper flex-row `#horizontal-wrapper` [sidebar 200px] + [main]
|
||||
|
||||
### `v2026.5.32` — Vue horizontale togglable
|
||||
- Bouton ⊞ "Vue" dans popup user-badge
|
||||
|
||||
### `v2026.5.27` — Classification absences
|
||||
- Maladie indigo, Congé cyan, Pompier rouge
|
||||
|
||||
### `v4.2.3` — Grande popup timeline persistante
|
||||
- Clic segment timeline = popup persistante
|
||||
|
||||
### `v4.1.3` — Tooltips épinglables
|
||||
- Introduction de `pinTooltip`
|
||||
|
||||
### `v1.0.0` (16 avril 2026) — Initiale
|
||||
- Premier viewer EasyVista pour le canton
|
||||
|
||||
Voir [CHANGELOG.md](CHANGELOG.md) pour l'historique complet (40+ versions taggées).
|
||||
➡ Pour l'historique complet (40+ versions depuis le 16 avril 2026), voir le **[CHANGELOG.md](CHANGELOG.md)** ou la **[page wiki Versions](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Versions)**.
|
||||
|
||||
## Architecture technique
|
||||
|
||||
```
|
||||
Planning/
|
||||
├── src/ # Sources de l'extension (chargées par le navigateur)
|
||||
│ ├── manifest.json # Manifest V3 (Chrome) + gecko_settings (Firefox)
|
||||
│ ├── background.js # Service worker : fetch planning XML, gestion session, fetch fiches
|
||||
│ ├── viewer.html # Interface principale
|
||||
│ ├── viewer.js # Logique (~9 500 lignes)
|
||||
│ ├── viewer.css # Styles + thèmes clair/sombre
|
||||
│ └── icons/ # icon16, icon48, icon128
|
||||
├── Autres/ # Méta : build script + docs (depuis v2026.5.40)
|
||||
│ ├── build.sh # Génère dist/chromium/, dist/firefox/, .zip, .xpi
|
||||
│ ├── CHANGELOG.md
|
||||
│ ├── LICENSE
|
||||
│ └── README.md
|
||||
├── Builds/ # Artefacts distribués aux techniciens
|
||||
│ ├── Chromium/
|
||||
│ ├── Firefox/
|
||||
│ ├── planification-vYYYY.M.PATCH-chromium.zip
|
||||
│ └── planification-vYYYY.M.PATCH-firefox.xpi
|
||||
└── dist/ # Sortie de build (gitignoré)
|
||||
Planification/ # Layout du repo Gitea (public)
|
||||
├── src/ # Sources de l'extension (chargées par le navigateur)
|
||||
│ ├── manifest.json # Manifest V3 (Chrome) + browser_specific_settings (Firefox)
|
||||
│ ├── background.js # Service worker (~1 600 lignes)
|
||||
│ ├── viewer.html # Interface principale
|
||||
│ ├── viewer.js # Logique UI (~10 700 lignes)
|
||||
│ ├── viewer.css # Styles + thèmes clair/sombre (~4 800 lignes)
|
||||
│ └── icons/ # icon16, icon48, icon128
|
||||
├── build.sh # Génère dist/chromium/, dist/firefox/, .zip, .xpi, met à jour firefox-updates.json
|
||||
├── firefox-updates.json # Manifest auto-update Firefox (servi via update_url)
|
||||
├── README.md
|
||||
├── CHANGELOG.md
|
||||
├── LICENSE # MIT
|
||||
└── .gitignore
|
||||
```
|
||||
|
||||
### `viewer.js` — fonctions clés
|
||||
|
||||
| Fonction | Introduite | Rôle |
|
||||
|---|---|---|
|
||||
| `loadForDate` | v1.0.0 | Fetch + parse planning pour une date donnée |
|
||||
| `buildTooltipHTML` | v1.0.0 | Construction HTML du tooltip d'intervention |
|
||||
| `pinTooltip` | v4.1.3 | Épingler un tooltip (le rendre permanent) |
|
||||
| `bindTimelinePopover` | v4.2.3 | Lier popover timeline aux segments |
|
||||
| `showTimelinePopover` | v4.2.3 | Afficher popover persistante |
|
||||
| `openPersistentTimelinePopup` | v4.2.3 | Grande popup détaillée |
|
||||
| `setTooltipViewportPosition` | v4.2.4 | Détection auto fixed/abs |
|
||||
| `_softUnpinPopup` | v4.3.3 | Désépinglage mou (popup reste visible) |
|
||||
| `initAppClock` | v5.0.0 | Horloge HH:MM topbar |
|
||||
| `initSessionTimer` | v5.0.0 | Compteur session EV (tick 1s) |
|
||||
| `_applyViewMode` | v2026.5.32 | Toggle vue classique/horizontale |
|
||||
| `_maybeRetryFetchUser` | v2026.5.34 | Relance opportuniste fetch user |
|
||||
| `positionTooltipAnchored` | v2026.5.34 | Positionnement unifié (4 candidats) |
|
||||
| `renderAdminSectionTeam` | v5.0.0 | Onglet admin Équipe (sélecteur groupe EV depuis v2026.5.40) |
|
||||
| `renderAdminSectionEV` | v5.0.0 | Onglet admin EasyVista (édition domaines depuis v2026.5.40) |
|
||||
|
||||
### `background.js` — fonctions clés
|
||||
|
||||
| Fonction | Rôle |
|
||||
|---|---|
|
||||
| `findEasyVistaSession` | Trouve l'onglet EV ouvert + extrait PHPSESSID |
|
||||
| `fetchPlanningXml` | Fetch XML calendar_block du planning |
|
||||
| `fetchFicheHtml` | Fetch HTML d'une fiche EV (avec retry SSO) |
|
||||
| `fetchCurrentUser` | Identifie l'user EV connecté |
|
||||
| `detectGroupsFromEV` (v2026.5.40) | Parse le `<select id="plan_group_id">` → liste des groupes EV |
|
||||
| `detectTeamFromEV` | Liste les membres d'un groupe (paramétrable depuis v2026.5.40) |
|
||||
| `evFetch` | Wrapper fetch avec headers EV (Referer, X-Requested-With) |
|
||||
|
||||
### Constantes / valeurs hardcodées (toutes versions)
|
||||
|
||||
- Group ID EV par défaut : `191` (SI-CSS) — surchargeable via le sélecteur depuis v2026.5.40
|
||||
- Pillonel Olivier (ID 40944) : absent tous les vendredis (récurrent)
|
||||
- GUIDs forms EV :
|
||||
- Demande : `S={C99ECD05-3D48-4C62-ABF0-66292053AED6}`
|
||||
- Incident : `I={07ED9C68-6172-48EA-8A58-90912B0A283E}`
|
||||
- SSO : Canton ForgeRock OpenAM
|
||||
- Storage keys : `admin_config`, `view_mode`
|
||||
- Domaines : `itsma.etat-de-vaud.ch` (interne), `itsma.vd.ch` (externe)
|
||||
|
||||
## Installation
|
||||
|
||||
### Firefox
|
||||
Télécharger le `.xpi` depuis `Builds/` ou le serveur de mises à jour interne, puis drag-and-drop dans `about:addons`.
|
||||
|
||||
### Chrome / Edge
|
||||
Mode développeur : décompresser `Builds/planification-vYYYY.M.PATCH-chromium.zip` (ou utiliser directement `dist/chromium/`) et charger en tant qu'extension non empaquetée.
|
||||
➡ Pour le détail des composants, fonctions clés et flux de données, voir la **[page wiki Architecture](https://gitea.netaplaid.ch/FroSteel/Planification/wiki/Architecture)**.
|
||||
|
||||
## Développement
|
||||
|
||||
@@ -234,23 +207,21 @@ git clone https://gitea.netaplaid.ch/FroSteel/Planification.git
|
||||
cd Planification
|
||||
|
||||
# Modifier les sources dans src/
|
||||
# Bumper la version dans src/manifest.json + ajouter une entrée dans Autres/CHANGELOG.md
|
||||
# Builder :
|
||||
./Autres/build.sh
|
||||
# → produit dist/chromium/, dist/firefox/, dist/*.zip, dist/*.xpi
|
||||
|
||||
# Copier dans Builds/ pour distribution :
|
||||
cp -r dist/chromium Builds/Chromium
|
||||
cp -r dist/firefox Builds/Firefox
|
||||
cp dist/*.zip dist/*.xpi Builds/
|
||||
# Bumper src/manifest.json + entrée CHANGELOG.md
|
||||
./build.sh
|
||||
# → dist/chromium/, dist/firefox/, dist/*.zip, dist/*.xpi
|
||||
# → firefox-updates.json mis à jour (sha256 .xpi NON SIGNÉ)
|
||||
|
||||
git add -A
|
||||
git commit -m "vYYYY.M.PATCH — description"
|
||||
git tag vYYYY.M.PATCH
|
||||
git push origin main
|
||||
git push --tags
|
||||
git push origin main vYYYY.M.PATCH
|
||||
```
|
||||
|
||||
Pour Firefox : signer le `.xpi` sur AMO en mode "On your own" (Unlisted),
|
||||
remplacer l'asset `.xpi` de la release Gitea, puis mettre à jour le sha256
|
||||
de cette version dans `firefox-updates.json`.
|
||||
|
||||
## Licence
|
||||
|
||||
[MIT License](LICENSE) — Copyright (c) 2026 Quentin Rouiller
|
||||
|
||||
@@ -27,7 +27,7 @@ import json
|
||||
with open('src/manifest.json', 'r') as f: m = json.load(f)
|
||||
m['browser_specific_settings'] = {
|
||||
'gecko': {
|
||||
'id': 'planification@netaplaid.ch',
|
||||
'id': 'planification-dgnsi@netaplaid.ch',
|
||||
'strict_min_version': '140.0',
|
||||
'update_url': 'https://gitea.netaplaid.ch/FroSteel/Planification/raw/branch/main/firefox-updates.json',
|
||||
'data_collection_permissions': {'required': ['none']}
|
||||
@@ -64,7 +64,7 @@ xpi = f"dist/planification-v${VERSION}-firefox.xpi"
|
||||
with open(xpi, 'rb') as f: sha = hashlib.sha256(f.read()).hexdigest()
|
||||
|
||||
JSON_PATH = "firefox-updates.json"
|
||||
ADDON_ID = "planification@netaplaid.ch"
|
||||
ADDON_ID = "planification-dgnsi@netaplaid.ch"
|
||||
update_link = f"https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/v${VERSION}/planification-v${VERSION}-firefox.xpi"
|
||||
|
||||
if os.path.exists(JSON_PATH):
|
||||
|
||||
+17
-2
@@ -1,11 +1,26 @@
|
||||
{
|
||||
"addons": {
|
||||
"planification@netaplaid.ch": {
|
||||
"planification-dgnsi@netaplaid.ch": {
|
||||
"updates": [
|
||||
{
|
||||
"version": "2026.5.45",
|
||||
"update_link": "https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/v2026.5.45/planification-v2026.5.45-firefox.xpi",
|
||||
"update_hash": "sha256:42ac47aeda23912e80051fc941ba45a0dec74ec4a6c509a25270c27b73dddead"
|
||||
},
|
||||
{
|
||||
"version": "2026.5.44",
|
||||
"update_link": "https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/v2026.5.44/planification-v2026.5.44-firefox.xpi",
|
||||
"update_hash": "sha256:e56e87d59c465e5df828b18d74376f561bf34e81e21bf4d70989a709e89217e0"
|
||||
},
|
||||
{
|
||||
"version": "2026.5.43",
|
||||
"update_link": "https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/v2026.5.43/planification-v2026.5.43-firefox.xpi",
|
||||
"update_hash": "sha256:7052200fab3c9266d5b809398a00dac768679ab2e96e4e147e4bb86c4ab648e5"
|
||||
},
|
||||
{
|
||||
"version": "2026.5.42",
|
||||
"update_link": "https://gitea.netaplaid.ch/FroSteel/Planification/releases/download/v2026.5.42/planification-v2026.5.42-firefox.xpi",
|
||||
"update_hash": "sha256:3d8cf762bf0921f9da473a9a5e31368fee21f0b9fd71f9f9432d256127de8674"
|
||||
"update_hash": "sha256:a5291a1eab6768d430dfc1cf032f93aeb788083da620ae0568b9dee68f14734d"
|
||||
},
|
||||
{
|
||||
"version": "2026.5.41",
|
||||
|
||||
+424
-16
@@ -188,6 +188,48 @@ async function getDayBounds() {
|
||||
// Clic sur l'icône → ouvrir le viewer
|
||||
// ============================================================================
|
||||
|
||||
// ============================================================================
|
||||
// badge "!" clignotant sur l'icône de l'extension quand la session
|
||||
// EasyVista va expirer. Permet à l'utilisateur de voir l'alerte même si
|
||||
// l'onglet planning n'est pas au premier plan.
|
||||
// ============================================================================
|
||||
|
||||
let _expirationBlinkInterval = null;
|
||||
let _expirationBlinkTimeout = null;
|
||||
|
||||
function _setExpirationBadge(active, durationMs) {
|
||||
// Stop l'état précédent
|
||||
if (_expirationBlinkInterval) {
|
||||
clearInterval(_expirationBlinkInterval);
|
||||
_expirationBlinkInterval = null;
|
||||
}
|
||||
if (_expirationBlinkTimeout) {
|
||||
clearTimeout(_expirationBlinkTimeout);
|
||||
_expirationBlinkTimeout = null;
|
||||
}
|
||||
try { chrome.action.setBadgeText({ text: "" }); } catch (e) {}
|
||||
|
||||
if (!active) return;
|
||||
|
||||
// Couleur de fond rouge pour le "!"
|
||||
try { chrome.action.setBadgeBackgroundColor({ color: "#d6443d" }); } catch (e) {}
|
||||
let on = true;
|
||||
const tick = () => {
|
||||
try {
|
||||
chrome.action.setBadgeText({ text: on ? "!" : "" });
|
||||
} catch (e) {}
|
||||
on = !on;
|
||||
};
|
||||
tick();
|
||||
_expirationBlinkInterval = setInterval(tick, 700);
|
||||
|
||||
if (durationMs && durationMs > 0) {
|
||||
_expirationBlinkTimeout = setTimeout(() => {
|
||||
_setExpirationBadge(false);
|
||||
}, durationMs);
|
||||
}
|
||||
}
|
||||
|
||||
chrome.action.onClicked.addListener(async () => {
|
||||
const viewerUrl = chrome.runtime.getURL("viewer.html");
|
||||
// Si le viewer est déjà ouvert, on focus cet onglet plutôt que d'en ouvrir un autre
|
||||
@@ -207,24 +249,146 @@ chrome.action.onClicked.addListener(async () => {
|
||||
/**
|
||||
* Trouve l'onglet EasyVista ouvert et récupère phpsessid + origin.
|
||||
*
|
||||
* v2026.5.45 (issue #4) : si la permission optionnelle "cookies" est
|
||||
* accordée, on lit le PHPSESSID directement depuis le cookie HttpOnly du
|
||||
* domaine EZV — toujours à jour, immune au PHPSESSID périmé qu'on trouve
|
||||
* dans l'URL des onglets historiques après un relog. Sinon, fallback sur
|
||||
* l'ancienne logique URL (compat sans la permission).
|
||||
*
|
||||
* @author Quentin Rouiller
|
||||
*/
|
||||
async function findEasyVistaSession() {
|
||||
// v2026.5.41 : les origines EV viennent de admin_config (éditables dans
|
||||
// Paramètres → EasyVista), avec fallback sur DEFAULT_EV_ORIGINS.
|
||||
async function _hasCookiesPermission() {
|
||||
return new Promise(resolve => {
|
||||
try {
|
||||
chrome.permissions.contains({ permissions: ["cookies"] }, granted => {
|
||||
resolve(!!granted);
|
||||
});
|
||||
} catch (e) {
|
||||
resolve(false);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
async function _readPhpsessidFromCookie(origin) {
|
||||
if (!chrome.cookies) return null;
|
||||
return new Promise(resolve => {
|
||||
try {
|
||||
chrome.cookies.get({ url: origin, name: "PHPSESSID" }, c => {
|
||||
if (!c || !c.value) { resolve(null); return; }
|
||||
resolve({
|
||||
value: c.value,
|
||||
expirationDate: c.expirationDate || null
|
||||
});
|
||||
});
|
||||
} catch (e) {
|
||||
resolve(null);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// ─── findEasyVistaSession ───────────────────────────────────────────────────
|
||||
//
|
||||
// Stratégie unifiée : pour chaque origine EV configurée, on cherche le
|
||||
// PHPSESSID dans cet ordre :
|
||||
// 1. cookie HttpOnly (autoritatif, donne aussi expirationDate)
|
||||
// 2. fallback URL d'un onglet ouvert (?PHPSESSID=…)
|
||||
// On retourne la première origine qui a un onglet ouvert ET un PHPSESSID.
|
||||
//
|
||||
// Anti-rafale : la fonction est appelée par ~16 endroits dans le service
|
||||
// worker, parfois en parallèle. On cache le résultat 200 ms pour éviter
|
||||
// le bombardement de chrome.cookies.get + chrome.tabs.query, et de logs
|
||||
// dupliqués. Le cache est invalidé immédiatement quand un événement
|
||||
// cookies.onChanged tombe (relog, expiration).
|
||||
//
|
||||
// Logs : une seule ligne `LOG.warn` par CHANGEMENT d'état (transition
|
||||
// cookie↔url, nouveau PHPSESSID, ou perte de session). Pas de log
|
||||
// répétitif par appel — utiliser `LOG.info` pour les détails internes.
|
||||
|
||||
let _sessionCache = null;
|
||||
let _sessionCacheTs = 0;
|
||||
// Cache désactivé temporairement (TTL=0) — soupçon que la durée de
|
||||
// fenêtre stale puisse causer une réponse "page de login" (~8 Ko) lors
|
||||
// d'un fetch fiche. À remettre à 200 après confirmation que ce n'était
|
||||
// pas le souci.
|
||||
const _SESSION_CACHE_TTL_MS = 0;
|
||||
let _lastSessionSignature = null;
|
||||
|
||||
function _invalidateSessionCache() {
|
||||
_sessionCache = null;
|
||||
_sessionCacheTs = 0;
|
||||
}
|
||||
|
||||
async function _findEasyVistaSessionRaw() {
|
||||
const origins = await getEvOrigins();
|
||||
const hasCookies = await _hasCookiesPermission();
|
||||
|
||||
for (const origin of origins) {
|
||||
const tabs = await chrome.tabs.query({ url: origin + "/*" });
|
||||
for (const tab of tabs) {
|
||||
const m = (tab.url || "").match(/[?&]PHPSESSID=([a-zA-Z0-9]+)/);
|
||||
if (m) {
|
||||
return { phpsessid: m[1], origin: origin, tabId: tab.id };
|
||||
if (!tabs.length) continue;
|
||||
|
||||
let phpsessid = null;
|
||||
let expirationDate = null;
|
||||
let source = null;
|
||||
|
||||
if (hasCookies) {
|
||||
const c = await _readPhpsessidFromCookie(origin);
|
||||
if (c && c.value) {
|
||||
phpsessid = c.value;
|
||||
expirationDate = c.expirationDate;
|
||||
source = "cookie";
|
||||
}
|
||||
}
|
||||
|
||||
if (!phpsessid) {
|
||||
for (const tab of tabs) {
|
||||
const m = (tab.url || "").match(/[?&]PHPSESSID=([a-zA-Z0-9]+)/);
|
||||
if (m) {
|
||||
phpsessid = m[1];
|
||||
source = "url";
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (phpsessid) {
|
||||
return { phpsessid, origin, tabId: tabs[0].id, source, expirationDate };
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
async function findEasyVistaSession() {
|
||||
const now = Date.now();
|
||||
if (_sessionCache !== null && now - _sessionCacheTs < _SESSION_CACHE_TTL_MS) {
|
||||
return _sessionCache;
|
||||
}
|
||||
if (_sessionCache === null && _sessionCacheTs > 0 && now - _sessionCacheTs < _SESSION_CACHE_TTL_MS) {
|
||||
// Cache négatif (pas de session trouvée à l'instant) — même TTL.
|
||||
return null;
|
||||
}
|
||||
|
||||
const result = await _findEasyVistaSessionRaw();
|
||||
_sessionCache = result;
|
||||
_sessionCacheTs = now;
|
||||
|
||||
// Log uniquement si l'état change (source, origine, ou PHPSESSID).
|
||||
const sig = result
|
||||
? `${result.source}|${result.origin}|${result.phpsessid.slice(0, 8)}`
|
||||
: "none";
|
||||
if (sig !== _lastSessionSignature) {
|
||||
if (result) {
|
||||
LOG.warn("session",
|
||||
`🔑 PHPSESSID via ${result.source.toUpperCase()} (${result.phpsessid.slice(0, 8)}…) sur ${result.origin}`,
|
||||
{ source: result.source, origin: result.origin,
|
||||
expirationDate: result.expirationDate || null });
|
||||
} else {
|
||||
LOG.warn("session", "❌ Aucun PHPSESSID disponible (pas d'onglet EZV ouvert ou cookie absent)");
|
||||
}
|
||||
_lastSessionSignature = sig;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// Fetch helpers (s'exécutent dans le contexte du service worker,
|
||||
// les cookies du domaine sont automatiquement inclus via credentials: include)
|
||||
@@ -299,14 +463,39 @@ async function fetchPlanningXml(origin, phpsessid, unixDate) {
|
||||
* @param {string} origin - origine EasyVista (pour construire le Referer)
|
||||
* @param {object} [opts] - options fetch (method, body, headers supplémentaires)
|
||||
*/
|
||||
// registre global des AbortController des fetchs EV en vol. Permet
|
||||
// au foreground (viewer.js) d'envoyer un message "abortAllFetches" pour
|
||||
// tuer instantanément les requêtes en cours quand l'user clique "Arrêter".
|
||||
const _evFetchControllers = new Set();
|
||||
function _abortAllEvFetches() {
|
||||
for (const c of _evFetchControllers) {
|
||||
try { c.abort(); } catch (e) { /* ignore */ }
|
||||
}
|
||||
_evFetchControllers.clear();
|
||||
}
|
||||
|
||||
async function evFetch(url, origin, opts = {}) {
|
||||
const defaultHeaders = {
|
||||
"Referer": `${origin}/index.php?eventName=HelpDesk_PlanningItem`,
|
||||
"X-Requested-With": "XMLHttpRequest"
|
||||
};
|
||||
const headers = Object.assign({}, defaultHeaders, opts.headers || {});
|
||||
const fetchOpts = Object.assign({ credentials: "include" }, opts, { headers });
|
||||
return await fetch(url, fetchOpts);
|
||||
// on ne remplace pas un signal explicitement passé par l'appelant.
|
||||
let controller = null;
|
||||
if (!opts.signal) {
|
||||
controller = new AbortController();
|
||||
_evFetchControllers.add(controller);
|
||||
}
|
||||
const fetchOpts = Object.assign(
|
||||
{ credentials: "include" },
|
||||
opts,
|
||||
{ headers, signal: opts.signal || (controller && controller.signal) }
|
||||
);
|
||||
try {
|
||||
return await fetch(url, fetchOpts);
|
||||
} finally {
|
||||
if (controller) _evFetchControllers.delete(controller);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -376,10 +565,10 @@ async function fetchFicheHtml(origin, phpsessid, formLink) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Sinon : on retourne ce qu'on a
|
||||
return html;
|
||||
// on signale au foreground si la dernière réponse est tronquée pour
|
||||
// qu'il puisse afficher un ⚠ et probe la session.
|
||||
return { html, truncated: html.length < MIN_VALID_SIZE, size: html.length };
|
||||
}
|
||||
// Ne devrait pas arriver (la boucle fait return avant)
|
||||
throw new Error("fetchFicheHtml: max retries reached");
|
||||
}
|
||||
|
||||
@@ -1175,7 +1364,7 @@ async function detectTeamFromEV(origin, phpsessid, groupIdArg, supportIdsArg) {
|
||||
}
|
||||
console.log("[bg] parsing pattern 1 (checkbox) :", results.length, "résultats");
|
||||
|
||||
// Pattern 2 : fallback <option value="76272">Nom...</option>
|
||||
// Pattern 2 : fallback <option value="NNNNN">Nom...</option>
|
||||
if (results.length === 0) {
|
||||
const rxOption = /<option[^>]*value=["'](\d{4,7})["'][^>]*>([^<]+)<\/option>/gi;
|
||||
let mO;
|
||||
@@ -1225,12 +1414,27 @@ async function detectTeamFromEV(origin, phpsessid, groupIdArg, supportIdsArg) {
|
||||
chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
(async () => {
|
||||
try {
|
||||
// abort de toutes les requêtes EV en vol (clic sur "Arrêter").
|
||||
if (msg.type === "abortAllFetches") {
|
||||
_abortAllEvFetches();
|
||||
sendResponse({ ok: true });
|
||||
return;
|
||||
}
|
||||
|
||||
if (msg.type === "getSession") {
|
||||
const session = await findEasyVistaSession();
|
||||
sendResponse({ ok: true, session });
|
||||
return;
|
||||
}
|
||||
|
||||
// badge "!" clignotant sur l'icône de l'extension.
|
||||
// { active: true, durationMs?: number }
|
||||
if (msg.type === "setExpirationBadge") {
|
||||
_setExpirationBadge(!!msg.active, msg.durationMs || 0);
|
||||
sendResponse({ ok: true });
|
||||
return;
|
||||
}
|
||||
|
||||
if (msg.type === "fetchPlanning") {
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) {
|
||||
@@ -1282,12 +1486,14 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
const html = await fetchFicheHtml(session.origin, session.phpsessid, msg.formLink);
|
||||
// fetchFicheHtml renvoie maintenant { html, truncated, size }.
|
||||
const result = await fetchFicheHtml(session.origin, session.phpsessid, msg.formLink);
|
||||
const html = result.html;
|
||||
if (looksLikeLoginPage(html)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true, html, session });
|
||||
sendResponse({ ok: true, html, session, truncated: !!result.truncated, size: result.size });
|
||||
} catch (err) {
|
||||
sendResponse({
|
||||
ok: false,
|
||||
@@ -1299,6 +1505,116 @@ chrome.runtime.onMessage.addListener((msg, sender, sendResponse) => {
|
||||
return;
|
||||
}
|
||||
|
||||
// probe rapide de session — fetch un endpoint léger pour vérifier
|
||||
// que PHPSESSID est toujours valide. Renvoie ok=false/error=session_expired
|
||||
// si la session est morte.
|
||||
// (feature reschedule) : déplace une intervention vers un autre
|
||||
// tech / nouvelle date / nouvel horaire de début. Durée préservée
|
||||
// automatiquement par EZV (l'API ne change que start_date+hour+minute).
|
||||
// Args : actionId, employeeId, date (DD/MM/YYYY), hour (0-23), minute (0-59).
|
||||
if (msg.type === "rescheduleAction") {
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) { sendResponse({ ok: false, error: "no_session" }); return; }
|
||||
try {
|
||||
const url = `${session.origin}/planning_updator_xhr.php` +
|
||||
`?PHPSESSID=${encodeURIComponent(session.phpsessid)}` +
|
||||
`&function_name=Planning_schedule_action_Employee` +
|
||||
`&action_id=${encodeURIComponent(msg.actionId)}` +
|
||||
`&employee_id=${encodeURIComponent(msg.employeeId)}` +
|
||||
`&date=${encodeURIComponent(msg.date)}` +
|
||||
`&hour=${encodeURIComponent(msg.hour)}` +
|
||||
`&minute=${encodeURIComponent(msg.minute)}` +
|
||||
`&multi_day_mode_act=0`;
|
||||
LOG.warn("reschedule",
|
||||
`📅 reschedule actionId=${msg.actionId} → tech=${msg.employeeId} ${msg.date} ${msg.hour}:${String(msg.minute).padStart(2,"0")}`);
|
||||
const r = await evFetch(url, session.origin);
|
||||
if (!r.ok) {
|
||||
sendResponse({ ok: false, error: classifyHttpStatus(r.status), httpStatus: r.status });
|
||||
return;
|
||||
}
|
||||
const txt = await r.text();
|
||||
if (looksLikeLoginPage(txt)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true, response: txt });
|
||||
} catch (err) {
|
||||
LOG.warn("reschedule", "rescheduleAction err", { err: err && err.message });
|
||||
sendResponse({ ok: false, error: err.kind || "fetch_failed", detail: err.message });
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// (feature reschedule) : modifie la durée (heure début/fin) d'une
|
||||
// action sans changer de tech. POST application/x-www-form-urlencoded
|
||||
// sur planning_updator_xhr.php avec function_name=fc_save_inspector.
|
||||
// Args : actionId, suffix (ex "act_<id>_nb_0_date_<DDMMYYYY>"),
|
||||
// startDate, endDate (DD/MM/YYYY), startTime, endTime (HH:MM).
|
||||
if (msg.type === "updateActionTimes") {
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) { sendResponse({ ok: false, error: "no_session" }); return; }
|
||||
try {
|
||||
const params = new URLSearchParams();
|
||||
params.set(`start_date_${msg.suffix}`, msg.startDate);
|
||||
params.set(`start_time_${msg.suffix}`, msg.startTime);
|
||||
params.set(`end_date_${msg.suffix}`, msg.endDate);
|
||||
params.set(`end_time_${msg.suffix}`, msg.endTime);
|
||||
params.set("action_id", msg.actionId);
|
||||
params.set("suffix_act", msg.suffix);
|
||||
params.set(`act_absence_${msg.suffix}`, "");
|
||||
params.set("function_name", "fc_save_inspector");
|
||||
const body = params.toString();
|
||||
const url = `${session.origin}/planning_updator_xhr.php` +
|
||||
`?PHPSESSID=${encodeURIComponent(session.phpsessid)}`;
|
||||
LOG.warn("reschedule",
|
||||
`⏱ updateActionTimes actionId=${msg.actionId} ${msg.startDate} ${msg.startTime}→${msg.endTime}`);
|
||||
const r = await evFetch(url, session.origin, {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
||||
body
|
||||
});
|
||||
if (!r.ok) {
|
||||
sendResponse({ ok: false, error: classifyHttpStatus(r.status), httpStatus: r.status });
|
||||
return;
|
||||
}
|
||||
const txt = await r.text();
|
||||
if (looksLikeLoginPage(txt)) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true, response: txt });
|
||||
} catch (err) {
|
||||
LOG.warn("reschedule", "updateActionTimes err", { err: err && err.message });
|
||||
sendResponse({ ok: false, error: err.kind || "fetch_failed", detail: err.message });
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (msg.type === "checkSession") {
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) {
|
||||
sendResponse({ ok: false, error: "no_session" });
|
||||
return;
|
||||
}
|
||||
try {
|
||||
const url = `${session.origin}/index.php?eventName=HelpDesk_PlanningItem&PHPSESSID=${encodeURIComponent(session.phpsessid)}`;
|
||||
const r = await evFetch(url, session.origin);
|
||||
if (!r.ok) {
|
||||
sendResponse({ ok: false, error: classifyHttpStatus(r.status), httpStatus: r.status });
|
||||
return;
|
||||
}
|
||||
const txt = await r.text();
|
||||
if (looksLikeLoginPage(txt) || txt.length < 5000) {
|
||||
sendResponse({ ok: false, error: "session_expired" });
|
||||
return;
|
||||
}
|
||||
sendResponse({ ok: true });
|
||||
} catch (err) {
|
||||
sendResponse({ ok: false, error: "fetch_failed", detail: err.message || String(err) });
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (msg.type === "fetchTimelineApi") {
|
||||
const session = await findEasyVistaSession();
|
||||
if (!session) {
|
||||
@@ -1593,10 +1909,33 @@ async function _getCacheDays() {
|
||||
}
|
||||
|
||||
// Au démarrage, nettoyer les anciennes alarmes et les anciens caches
|
||||
chrome.runtime.onInstalled.addListener(async () => {
|
||||
// v2026.5.45 (issue #3) : on stocke aussi un marqueur d'install/update
|
||||
// avec previousVersion. Au prochain ouverture du viewer, on affiche un
|
||||
// toast préventif si la version précédente était < 2026.5.44 (où le bug
|
||||
// structurel d'écrasement des absences récurrentes existait encore).
|
||||
chrome.runtime.onInstalled.addListener(async (details) => {
|
||||
clearLegacyRefreshAlarms();
|
||||
const days = await _getCacheDays();
|
||||
cleanupOldCaches(days).catch(err => LOG.warn("cleanup", "échec onInstalled", { err: err && err.message }));
|
||||
try {
|
||||
const reason = details && details.reason;
|
||||
const previousVersion = (details && details.previousVersion) || null;
|
||||
if (reason === "install" || reason === "update") {
|
||||
const currentVersion = (chrome.runtime.getManifest && chrome.runtime.getManifest().version) || null;
|
||||
await chrome.storage.local.set({
|
||||
_lastInstallEvent: {
|
||||
reason,
|
||||
previousVersion,
|
||||
currentVersion,
|
||||
ts: Date.now(),
|
||||
notified: false
|
||||
}
|
||||
});
|
||||
LOG.info("install", "marqueur posé", { reason, previousVersion, currentVersion });
|
||||
}
|
||||
} catch (e) {
|
||||
LOG.warn("install", "écriture _lastInstallEvent échouée", { err: e && e.message });
|
||||
}
|
||||
});
|
||||
|
||||
chrome.runtime.onStartup.addListener(async () => {
|
||||
@@ -1604,3 +1943,72 @@ chrome.runtime.onStartup.addListener(async () => {
|
||||
const days = await _getCacheDays();
|
||||
cleanupOldCaches(days).catch(err => LOG.warn("cleanup", "échec onStartup", { err: err && err.message }));
|
||||
});
|
||||
|
||||
// v2026.5.45 (issue #4 — bonus) : listener sur les changements de cookie
|
||||
// EZV. Quand la permission `cookies` est accordée, on s'abonne pour détecter
|
||||
// EN TEMPS RÉEL les changements de PHPSESSID (relog dans un autre onglet,
|
||||
// expiration côté serveur, révocation…). On envoie un message broadcast aux
|
||||
// viewers ouverts pour qu'ils recalibrent leur compteur de session.
|
||||
function _attachCookiesListener() {
|
||||
if (!chrome.cookies || !chrome.cookies.onChanged) return;
|
||||
if (_attachCookiesListener._attached) return;
|
||||
_attachCookiesListener._attached = true;
|
||||
LOG.warn("cookies", "🍪 listener PHPSESSID activé (permission cookies accordée)");
|
||||
chrome.cookies.onChanged.addListener((info) => {
|
||||
try {
|
||||
if (!info || !info.cookie) return;
|
||||
const c = info.cookie;
|
||||
if (c.name !== "PHPSESSID") return;
|
||||
const dom = (c.domain || "").replace(/^\./, "");
|
||||
if (!/itsma\.(etat-de-vaud|vd)\.ch$/.test(dom)) return;
|
||||
const phpsessid8 = c.value ? c.value.slice(0, 8) + "…" : null;
|
||||
// événements importants → LOG.warn (toujours visibles).
|
||||
// Détail (cause, expirationDate, …) → LOG.info (debug only).
|
||||
if (info.removed) {
|
||||
LOG.warn("cookies", `🚫 PHPSESSID supprimé sur ${dom} (cause=${info.cause}) → session morte`);
|
||||
} else {
|
||||
LOG.warn("cookies", `🍪 PHPSESSID ${phpsessid8} sur ${dom} (cause=${info.cause})`);
|
||||
}
|
||||
// Invalide immédiatement le cache findEasyVistaSession() pour que
|
||||
// le prochain appel reflète le nouveau cookie sans attendre la TTL.
|
||||
if (typeof _invalidateSessionCache === "function") _invalidateSessionCache();
|
||||
LOG.info("cookies", " détail onChanged",
|
||||
{ domain: dom, removed: !!info.removed, cause: info.cause,
|
||||
phpsessid8, expirationDate: c.expirationDate || null,
|
||||
exp: c.expirationDate ? new Date(c.expirationDate * 1000).toISOString() : null });
|
||||
try {
|
||||
chrome.runtime.sendMessage({
|
||||
type: "cookieChanged",
|
||||
domain: dom,
|
||||
removed: !!info.removed,
|
||||
cause: info.cause,
|
||||
phpsessid: c.value || null,
|
||||
expirationDate: c.expirationDate || null
|
||||
}).catch(() => { /* viewers fermés → pas grave */ });
|
||||
} catch (e) { /* idem */ }
|
||||
} catch (e) {
|
||||
LOG.warn("cookies", "listener err", { err: e && e.message });
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Au boot du service worker, on log l'état de la permission (toujours visible)
|
||||
// et on attache le listener si déjà accordée.
|
||||
(async () => {
|
||||
try {
|
||||
const has = await _hasCookiesPermission();
|
||||
LOG.warn("cookies", `🔧 permission cookies au boot : ${has ? "ACCORDÉE → lecture cookie active" : "non accordée → fallback URL"}`);
|
||||
if (has) _attachCookiesListener();
|
||||
} catch (e) { /* silent */ }
|
||||
})();
|
||||
chrome.permissions.onAdded && chrome.permissions.onAdded.addListener((p) => {
|
||||
if (p && p.permissions && p.permissions.includes("cookies")) {
|
||||
LOG.warn("cookies", "✅ permission cookies ACCORDÉE par l'utilisateur");
|
||||
_attachCookiesListener();
|
||||
}
|
||||
});
|
||||
chrome.permissions.onRemoved && chrome.permissions.onRemoved.addListener((p) => {
|
||||
if (p && p.permissions && p.permissions.includes("cookies")) {
|
||||
LOG.warn("cookies", "🛑 permission cookies RETIRÉE → fallback URL réactivé");
|
||||
}
|
||||
});
|
||||
|
||||
+4
-1
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"manifest_version": 3,
|
||||
"name": "Planification",
|
||||
"version": "2026.5.42",
|
||||
"version": "2026.5.45",
|
||||
"description": "Vue claire et rapide du planning des techniciens EasyVista. Développé par Quentin Rouiller — DGNSI, Canton de Vaud.",
|
||||
"permissions": [
|
||||
"activeTab",
|
||||
@@ -10,6 +10,9 @@
|
||||
"tabs",
|
||||
"alarms"
|
||||
],
|
||||
"optional_permissions": [
|
||||
"cookies"
|
||||
],
|
||||
"host_permissions": [
|
||||
"https://itsma.etat-de-vaud.ch/*",
|
||||
"https://itsma.vd.ch/*"
|
||||
|
||||
+1502
-206
File diff suppressed because it is too large
Load Diff
+13
-11
@@ -28,26 +28,28 @@
|
||||
type="button" aria-label="Utilisateur connecté"
|
||||
title="Utilisateur — cliquer pour accéder aux paramètres">?</button>
|
||||
<h1 id="app-title">Planification</h1>
|
||||
<!-- : bloc "Aujourd'hui + horloge" encadré, suivi DIRECTEMENT
|
||||
du statut d'actualisation (MAJ + ✓), puis le sélecteur de date
|
||||
du planning. -->
|
||||
<div id="today-block" class="today-block">
|
||||
<button id="nav-today" class="btn btn-today" title="Revenir au jour courant">Aujourd'hui</button>
|
||||
<div id="app-clock" class="app-clock" title="Date et heure actuelles">
|
||||
<div id="app-clock-date" class="app-clock-date"></div>
|
||||
<div id="app-clock-time" class="app-clock-time"></div>
|
||||
</div>
|
||||
</div>
|
||||
<span id="capture-info" class="capture-info"></span>
|
||||
<span id="refresh-check" class="refresh-check hidden" title="Mise à jour terminée">✓</span>
|
||||
<div class="date-nav">
|
||||
<button id="nav-prev" class="btn btn-nav" title="Jour précédent" aria-label="Jour précédent">◀</button>
|
||||
<!-- v2026.5.17 : input date custom qui affiche "Vendredi 24.04.2026" -->
|
||||
<div class="date-custom-wrapper">
|
||||
<div id="date-custom" class="date-custom" role="button" tabindex="0" title="Choisir une date">
|
||||
<div id="date-custom" class="date-custom" role="button" tabindex="0" title="Choisir une date du planning">
|
||||
<span id="date-custom-label"></span>
|
||||
<span class="date-custom-icon">📅</span>
|
||||
</div>
|
||||
<input type="date" id="date-picker" class="date-input-hidden">
|
||||
</div>
|
||||
<button id="nav-next" class="btn btn-nav" title="Jour suivant" aria-label="Jour suivant">▶</button>
|
||||
<button id="nav-today" class="btn btn-today" title="Aujourd'hui">Auj.</button>
|
||||
</div>
|
||||
<span id="capture-info" class="capture-info"></span>
|
||||
<span id="refresh-check" class="refresh-check hidden" title="Mise à jour terminée">✓</span>
|
||||
</div>
|
||||
<!-- v2026.5.16 : date complète du jour au-dessus de l'heure dans la topbar -->
|
||||
<div id="app-clock" class="app-clock" title="Date et heure actuelles">
|
||||
<div id="app-clock-date" class="app-clock-date"></div>
|
||||
<div id="app-clock-time" class="app-clock-time"></div>
|
||||
</div>
|
||||
<!-- v5.0.9 : compteur de session EasyVista (visible < 5 min restantes) -->
|
||||
<div id="app-session" class="app-session hidden"></div>
|
||||
|
||||
+5486
-461
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user