// ============================================================================
// viewer.js v3 — vue claire du planning techniciens
// ============================================================================
// Différences clés avec v2 :
// 1. Fetch direct EasyVista (plus besoin de capturer la page manuellement)
// 2. Parsing XML (planning_xhr.php?div=calendar_block) au lieu de HTML
// 3. Fetch des fiches individuelles pour détecter les statuts Clôturé/Résolu
// 4. Cache persistant 7 jours par date (chrome.storage.local)
// 5. Navigation ◀ / date picker / ▶
// 6. Refresh auto 12h / 15h
//
// Les fetches se font dans le service worker (background.js) pour éviter
// les problèmes de CORS : viewer.js envoie des messages, background fait les
// requêtes et renvoie les données.
// ============================================================================
// ============================================================================
// Configuration
// ============================================================================
// Équipe : ID EasyVista → nom affiché
const TEAM = {
"76272": "Ciuppa, Mathieu",
"83725": "De Almeida Martins, Solange",
"66635": "Makonda, Yannick",
"92235": "Mamouni, Anas",
"90070": "Paisana, David",
"40944": "Pillonel, Olivier",
"72485": "Rosset, Pascal",
"86874": "Rouiller, Quentin"
};
// Absences récurrentes (id tech → [jour JS, 0=dim..6=sam])
const RECURRING_ABSENCES = {
"40944": [5] // Pillonel absent tous les vendredis
};
// Statuts EasyVista qui déclenchent l'affichage "clos"
const CLOSED_STATUS = ["Clôturé", "Cloture", "Clôture"];
const RESOLVED_STATUS = ["Résolu", "Resolu"];
// Statuts qui indiquent qu'une intervention a été supprimée/annulée
// → si présente dans le cache mais disparue du planning : on retire
const CANCELLED_STATUS = ["Annulé", "Annule", "Supprimé", "Supprime"];
// Clés de stockage
const LS_THEME = "planning_theme";
const CACHE_PREFIX = "planning_cache_"; // + YYYY-MM-DD
const CACHE_DAYS = 7;
// Concurrence des fetches en parallèle.
// En v3.1.1 : xhr2 (bulles) et fetches fiches tournent SIMULTANÉMENT pour
// que les refs arrivent plus vite. Chacun a sa propre concurrency, et le
// total reste raisonnable pour le serveur EasyVista.
// - xhr2 : petits (~400 o) et rapides → 10 workers suffisent
// - fiches : gros (~250 Ko) → 15 workers pour vraiment accélérer
// Total max simultané : 25 requêtes, ce qui reste confortable.
// Si le serveur renvoie des erreurs ou XML vides → baisser les deux.
const FETCH_CONCURRENCY_BULLES = 10;
const FETCH_CONCURRENCY_FICHES = 15;
// ============================================================================
// Mapping de catégorie → titre court + couleur
// ============================================================================
const CATEGORY_TO_TITLE = [
// Arrivées / nouvelles installations → Installation (bleu)
[/Arriv[ée]e\s+ou\s+mutation/i, "Installation", "installation"],
[/Accessoire\s+pour\s+PC/i, "Installation", "installation"],
[/Nouveau\s+Poste\s+Windows/i, "Installation", "installation"],
[/Nouveau\s+Poste\s+macOS/i, "Installation", "installation"],
// Récupération / départ (vert)
[/D[ée]part\s+d[\u2018\u2019']un\s+utilisateur/i, "Récupération", "recup"],
[/Reprise\s+du\s+mat[ée]riel/i, "Récupération", "recup"],
// Remplacement (orange)
[/Remplacement\s+de\s+mat[ée]riel/i, "Remplacement", "remplacement"],
];
/**
* Détecte si le texte de l'action commence par "Roll Out".
*/
function isRollOut(iv) {
const texts = [
iv.bulleDescription,
iv.actionText,
iv.infobulle && iv.infobulle.aFaire,
iv.label
];
for (const t of texts) {
if (!t) continue;
if (/^\s*[«"']?\s*roll[\s\-]*out/i.test(String(t))) return true;
if (/(?:^|\bA faire\s*:\s*)roll[\s\-]*out/i.test(String(t))) return true;
}
return false;
}
/**
* Détecte si le texte de l'action mentionne une récupération de matériel.
* Accepté : "RÉCUPÉRATION DE MATÉRIEL" ou "Récupération" au début de l'action,
* ou dans "A faire : Récupération ...".
*/
function isRecupAction(iv) {
const texts = [
iv.bulleDescription,
iv.actionText,
iv.infobulle && iv.infobulle.aFaire,
iv.label
];
for (const t of texts) {
if (!t) continue;
const s = String(t);
if (/^\s*r[ée]cup[ée]ration/i.test(s)) return true;
if (/\bA\s+faire\s*:\s*r[ée]cup[ée]ration/i.test(s)) return true;
}
return false;
}
/**
* Dérive un titre court et une clé de couleur à partir d'une intervention.
* Priorité :
* 1. Si la ref commence par I260 → "Incident" (violet)
* 2. Si l'action commence par "Roll Out" → "Roll Out" (brun)
* 3. Si l'action mentionne récupération → "Récupération" (vert)
* 4. Sinon, mapping par catégorie (fiche)
* 5. Sinon, "Autres" (gris)
*/
function deriveShortTitle(iv) {
if (iv.type === "AL-Reservation") return "Réservation";
if (iv.ref && /^I2\d/.test(iv.ref)) return "Incident";
if (isRollOut(iv)) return "Roll Out";
if (isRecupAction(iv)) return "Récupération";
const cat = iv.categoryLine || "";
if (!cat) return "Autres";
for (const [regex, title] of CATEGORY_TO_TITLE) {
if (regex.test(cat)) return title;
}
return "Autres";
}
function deriveColorKey(iv) {
if (iv.type === "AL-Reservation") return "reservation";
if (iv.ref && /^I2\d/.test(iv.ref)) return "incident";
if (isRollOut(iv)) return "rollout";
if (isRecupAction(iv)) return "recup";
const cat = iv.categoryLine || "";
if (!cat) return "autre";
for (const [regex, , colorKey] of CATEGORY_TO_TITLE) {
if (regex.test(cat)) return colorKey;
}
return "autre";
}
// ============================================================================
// État global
// ============================================================================
let state = {
session: null, // { phpsessid, origin, tabId }
currentDate: null, // "YYYY-MM-DD" affiché
currentData: null, // résultat parsé (techs, stats, ...)
loading: false
};
// ─── Annulation coopérative d'un refresh manuel (v3.1) ──────────────────────
// Chaque refresh reçoit un "jeton" (entier incrémenté). Les workers lisent
// isRefreshAborted() avant chaque fetch : si le jeton a changé ou si
// l'utilisateur a cliqué sur "Arrêter", ils s'arrêtent proprement.
//
// v3.2 : on ajoute une "abortPromise" par refresh. loadForDate race cette
// promesse avec son Promise.all, donc dès qu'on clique Arrêter, loadForDate
// sort immédiatement (masque le bouton, fait un toast), même si les fetches
// en cours continuent silencieusement. Le changement de token les rend
// inoffensifs (ils ne peuvent plus écrire le cache ni updater le DOM).
let currentRefreshToken = 0;
let abortedToken = -1;
let abortResolvers = new Map(); // token → resolve fn of the abort promise
function startNewRefresh() {
currentRefreshToken++;
return currentRefreshToken;
}
function makeAbortPromise(myToken) {
return new Promise(resolve => {
abortResolvers.set(myToken, resolve);
});
}
function abortCurrentRefresh() {
abortedToken = currentRefreshToken;
// Réveiller tous les loadForDate en attente (normalement un seul)
for (const [token, resolve] of abortResolvers) {
if (token <= currentRefreshToken) {
resolve("aborted");
abortResolvers.delete(token);
}
}
}
function isRefreshAborted() {
return abortedToken === currentRefreshToken;
}
function cleanupAbortResolver(myToken) {
abortResolvers.delete(myToken);
}
// ============================================================================
// Boot
// ============================================================================
document.addEventListener("DOMContentLoaded", init);
async function init() {
initTheme();
bindTopbar();
// Initialiser la date = aujourd'hui
state.currentDate = todayISO();
document.getElementById("date-picker").value = state.currentDate;
// Écouter les messages d'auto-refresh du service worker
chrome.runtime.onMessage.addListener((msg) => {
if (msg && msg.type === "autoRefresh") {
console.log("Auto-refresh 12h/15h déclenché");
refreshPlanning({ keepStatuses: true });
}
});
// Charger la session puis le planning
await refreshSessionAndLoad();
}
async function refreshSessionAndLoad() {
const resp = await sendMessage({ type: "getSession" });
if (!resp.ok || !resp.session) {
showSessionNeeded();
return;
}
state.session = resp.session;
hideSessionNeeded();
await loadForDate(state.currentDate);
}
// ============================================================================
// Thème clair/sombre
// ============================================================================
function initTheme() {
const saved = localStorage.getItem(LS_THEME);
const theme = (saved === "light" || saved === "dark") ? saved : detectDefaultTheme();
applyTheme(theme);
}
function detectDefaultTheme() {
if (window.matchMedia && window.matchMedia("(prefers-color-scheme: dark)").matches) {
return "dark";
}
return "light";
}
function applyTheme(theme) {
document.documentElement.setAttribute("data-theme", theme);
const icon = document.getElementById("theme-icon");
if (icon) icon.textContent = theme === "dark" ? "☀️" : "🌙";
}
function toggleTheme() {
const current = document.documentElement.getAttribute("data-theme") || "light";
const next = current === "dark" ? "light" : "dark";
applyTheme(next);
localStorage.setItem(LS_THEME, next);
}
// ============================================================================
// Topbar handlers
// ============================================================================
function bindTopbar() {
document.getElementById("theme-toggle").addEventListener("click", toggleTheme);
document.getElementById("refresh-btn").addEventListener("click", () => refreshPlanning());
document.getElementById("abort-btn").addEventListener("click", () => {
// Feedback visuel instantané : masquer le bouton tout de suite, sans
// attendre que loadForDate finisse sa race.
showAbortButton(false);
abortCurrentRefresh();
showAbortToast();
});
document.getElementById("clear-cache-btn").addEventListener("click", onClearCache);
document.getElementById("nav-prev").addEventListener("click", () => navigateDate(-1));
document.getElementById("nav-next").addEventListener("click", () => navigateDate(+1));
document.getElementById("nav-today").addEventListener("click", () => loadForDate(todayISO()));
document.getElementById("date-picker").addEventListener("change", (e) => {
if (e.target.value) loadForDate(e.target.value);
});
document.getElementById("open-ev-btn").addEventListener("click", openEasyVista);
}
async function openEasyVista() {
// Ouvrir sur le domaine externe (accessible depuis l'extérieur).
// Le domaine interne (itsma.etat-de-vaud.ch) n'est accessible que depuis le réseau VD.
// Une fois connecté, l'extension détectera automatiquement le PHPSESSID quel que
// soit le domaine où tu es connecté.
await chrome.tabs.create({ url: "https://itsma.vd.ch/" });
}
// Navigation ±1 jour en sautant les week-ends
function navigateDate(direction) {
const d = isoToDate(state.currentDate);
d.setDate(d.getDate() + direction);
// Sauter les week-ends
while (d.getDay() === 0 || d.getDay() === 6) {
d.setDate(d.getDate() + direction);
}
loadForDate(dateToISO(d));
}
async function onClearCache() {
if (!confirm(`Vider le cache du ${formatDateDM(state.currentDate)} ?`)) return;
await chrome.storage.local.remove(CACHE_PREFIX + state.currentDate);
await loadForDate(state.currentDate, { forceRefetch: true });
}
// ============================================================================
// Date helpers
// ============================================================================
function todayISO() {
const d = new Date();
return dateToISO(d);
}
function dateToISO(d) {
const yyyy = d.getFullYear();
const mm = String(d.getMonth() + 1).padStart(2, "0");
const dd = String(d.getDate()).padStart(2, "0");
return `${yyyy}-${mm}-${dd}`;
}
function isoToDate(iso) {
const [y, m, d] = iso.split("-").map(n => parseInt(n, 10));
return new Date(y, m - 1, d);
}
function isoToDDMMYYYY(iso) {
const [y, m, d] = iso.split("-");
return `${d}/${m}/${y}`;
}
function formatDateDM(iso) {
const [, m, d] = iso.split("-");
return `${d}/${m}`;
}
function isoToUnixDate(iso) {
// Renvoie le timestamp Unix à midi local du jour (pour que le serveur comprenne bien le jour demandé)
const d = isoToDate(iso);
d.setHours(12, 0, 0, 0);
return Math.floor(d.getTime() / 1000);
}
// ============================================================================
// Messages → background
// ============================================================================
function sendMessage(msg) {
return new Promise((resolve, reject) => {
chrome.runtime.sendMessage(msg, (response) => {
if (chrome.runtime.lastError) {
reject(new Error(chrome.runtime.lastError.message));
return;
}
resolve(response || {});
});
});
}
// ============================================================================
// Cache (chrome.storage.local)
// ============================================================================
async function readCache(isoDate) {
const key = CACHE_PREFIX + isoDate;
const obj = await chrome.storage.local.get(key);
return obj[key] || null;
}
async function writeCache(isoDate, data) {
const key = CACHE_PREFIX + isoDate;
await chrome.storage.local.set({ [key]: { ...data, savedAt: Date.now() } });
}
// ============================================================================
// Flux principal : charger une date
// ============================================================================
async function loadForDate(isoDate, opts = {}) {
state.currentDate = isoDate;
document.getElementById("date-picker").value = isoDate;
if (!state.session) {
showSessionNeeded();
return;
}
// (v3.1.1) Tout chargement = un nouveau jeton d'annulation. Le bouton
// "Arrêter" apparaît pour TOUT refresh (clic manuel, navigation date,
// ouverture vue claire), pas juste refreshPlanning(). Le bouton disparaît
// quand le chargement est vraiment fini (finally).
const myToken = startNewRefresh();
showAbortButton(true);
const t0 = performance.now();
console.log(`[load] début pour ${isoDate} (token=${myToken})`);
try {
// 1. Afficher immédiatement depuis le cache si disponible
const cached = await readCache(isoDate);
if (cached && !opts.forceRefetch) {
renderFromData({
techs: cached.techs,
targetDate: isoDate,
captureTime: cached.savedAt || null,
source: "cache"
});
// Si cache présent ET pas de refresh explicite demandé, on s'arrête là.
if (!opts.doStatusRefresh) {
return;
}
} else {
showLoading();
}
if (isRefreshAborted()) return;
// 2. Fetch frais du planning (XML calendar_block — rapide, ~40 ko)
const tXml = performance.now();
const fresh = await fetchPlanningForDate(isoDate);
console.log(`[load] XML planning récupéré en ${Math.round(performance.now() - tXml)} ms`);
if (!fresh) return;
if (isRefreshAborted()) return;
// 3. Fusionner cache + frais
const merged = mergeCacheAndFresh(cached, fresh);
// 4. Afficher immédiatement avec ce qu'on a
renderFromData({
techs: merged.techs,
targetDate: isoDate,
captureTime: Date.now(),
source: "fresh"
});
console.log(`[load] 1er rendu (sans refs) à ${Math.round(performance.now() - t0)} ms`);
// 5. PARALLÈLE : xhr2 (lieu/contact) + fetches fiches (ref/statut)
// Avant v3.1.1 : séquentiel, on devait attendre les 34 xhr2 avant de
// lancer les 34 fiches. Résultat : première ref arrivait après ~1s.
// Maintenant : les deux démarrent en même temps, chacun met à jour
// la ligne correspondante via le rendu incrémental.
const bulleNeeded = [];
for (const tech of merged.techs) {
for (const iv of tech.interventions) {
if (iv.type !== "AL-Intervention") continue;
if (iv.infobulle && iv.bulleContact) continue;
bulleNeeded.push(iv);
}
}
// On refetche les fiches si :
// - au moins une intervention n'a jamais été fetchée (pas de ficheTarget), OU
// - au moins une intervention n'a pas encore l'actionDescription complète de la fiche
// (cas du cache chargé depuis une version antérieure à v3.2)
const needFetch = merged.techs.some(tech =>
tech.interventions.some(iv =>
iv.type === "AL-Intervention" &&
(!iv.ficheTarget || !iv.actionDescriptionFetched)
)
);
const promises = [];
if (bulleNeeded.length > 0 && !isRefreshAborted()) {
const tBulles = performance.now();
console.log(`[load] fetch xhr2 pour ${bulleNeeded.length} interventions…`);
promises.push(
fetchBullesForInterventions(bulleNeeded).then(() => {
console.log(`[load] xhr2 finis en ${Math.round(performance.now() - tBulles)} ms`);
if (!isRefreshAborted()) {
renderFromData({
techs: merged.techs,
targetDate: isoDate,
captureTime: Date.now(),
source: "fresh+bulles"
});
}
})
);
}
if ((opts.doStatusRefresh || needFetch) && !isRefreshAborted()) {
const tFiches = performance.now();
const nFiches = merged.techs.flatMap(t=>t.interventions).filter(i=>i.type==="AL-Intervention").length;
console.log(`[load] début fetch des ${nFiches} fiches…`);
promises.push(
refreshStatuses(merged.techs, isoDate).then(() => {
console.log(`[load] fiches finies en ${Math.round(performance.now() - tFiches)} ms`);
})
);
}
// Race du Promise.all avec le signal d'annulation : dès que l'user clique
// Arrêter, loadForDate sort immédiatement (masque le bouton, fait un toast)
// sans attendre que les 15 workers en cours finissent leurs fetches.
// Les fetches continuent en arrière-plan mais le token a changé donc ils
// ne peuvent plus écrire le cache ni rafraîchir le DOM.
const abortPromise = makeAbortPromise(myToken);
const allDone = Promise.all(promises).then(() => "done");
const raceResult = await Promise.race([allDone, abortPromise]);
// 6. Sauvegarder dans le cache (une seule fois, après que tout soit enrichi)
// Uniquement si on est allé au bout (pas d'annulation).
if (raceResult === "done" && !isRefreshAborted()) {
await writeCache(isoDate, { techs: merged.techs });
}
if (raceResult === "done" && !isRefreshAborted()) {
showRefreshDone();
console.log(`[load] TOTAL: ${Math.round(performance.now() - t0)} ms`);
// Retry silencieux en arrière-plan pour les interventions dont le texte
// d'action n'a pas pu être récupéré (timeline partielle au 1er coup).
// Lancé SANS await : l'user peut continuer à utiliser l'extension.
// La fonction respecte le token : si l'user change de jour, elle s'arrête.
runBackgroundTimelineRetry(merged.techs, isoDate, myToken).catch(() => {});
} else {
console.log(`[load] annulé par l'utilisateur à ${Math.round(performance.now() - t0)} ms`);
showAbortToast();
}
} finally {
// Masquer le bouton "Arrêter" uniquement si c'est NOTRE chargement qui
// se termine (pas un chargement postérieur que l'utilisateur aurait lancé
// entre-temps en naviguant ailleurs).
if (currentRefreshToken === myToken) {
showAbortButton(false);
}
cleanupAbortResolver(myToken);
}
}
async function refreshPlanning(opts = {}) {
if (!state.session) {
await refreshSessionAndLoad();
return;
}
// Refresh manuel : force le refetch des fiches. Le bouton "Arrêter" est
// géré par loadForDate lui-même.
await loadForDate(state.currentDate, { ...opts, doStatusRefresh: true });
}
// ============================================================================
// Fetch du planning (via background)
// ============================================================================
async function fetchPlanningForDate(isoDate) {
setRefreshing(true);
try {
const unixDate = isoToUnixDate(isoDate);
const resp = await sendMessage({
type: "fetchPlanning",
session: state.session,
unixDate: unixDate
});
if (!resp.ok) {
if (resp.error === "no_session" || resp.error === "session_expired") {
state.session = null;
showSessionNeeded();
} else {
showError("Erreur de fetch : " + (resp.error || "inconnue"));
}
return null;
}
// Safeguard (v3.1) : le serveur EasyVista répond parfois 200 avec un
// corps vide — typiquement quand la session vient d'être invalidée, ou
// quand il soupçonne du scraping (trop de requêtes parallèles). Dans
// les deux cas, on traite ça comme une session expirée : inutile de
// parser (ça ferait "Document is empty") ni de retry en boucle.
if (!resp.xml || resp.xml.length < 20) {
console.warn("[viewer] XML planning vide — session probablement invalide");
state.session = null;
showSessionNeeded();
return null;
}
// Parser le HTML complet du planning (contient TOUT : ref, catégorie,
// contact, lieu, description, formLinks, request_id + checksum)
const techs = parsePlanningXml(resp.xml, isoDate);
return { techs };
} catch (err) {
showError("Erreur inattendue : " + (err.message || err));
return null;
} finally {
setRefreshing(false);
}
}
// ============================================================================
// Parsing du XML du planning
// ============================================================================
/**
* Parse le XML retourné par planning_xhr.php?div=calendar_block.
* Contient les interventions (actions) par technicien, avec :
* - action_id, done_by_id, action_label (parfois juste "AL-Intervention"),
* - start_time / end_time, start_date / end_date,
* - formLink (eventName=formEvent&target=ACTIONID&checksum=...) pour ouvrir l'action,
* - request_id (ID de la fiche SD_REQUEST, utilisé pour ouvrir la fiche).
*/
function parsePlanningXml(xml, isoDate) {
const doc = new DOMParser().parseFromString(xml, "text/xml");
const parserError = doc.querySelector("parsererror");
if (parserError) {
console.warn("Parser error:", parserError.textContent);
}
const actionNodes = doc.querySelectorAll("action");
const byTechId = new Map();
for (const id of Object.keys(TEAM)) {
byTechId.set(id, { id, name: TEAM[id], interventions: [] });
}
for (const node of actionNodes) {
const iv = actionNodeToIntervention(node);
if (!iv) continue;
if (!byTechId.has(iv.techId)) continue;
if (!actionCoversDate(iv, isoDate)) continue;
byTechId.get(iv.techId).interventions.push(iv);
}
for (const tech of byTechId.values()) {
tech.interventions.sort((a, b) =>
(a.startTime || "").localeCompare(b.startTime || "")
);
}
return [...byTechId.values()];
}
function actionNodeToIntervention(node) {
const get = name => node.getAttribute(name) || "";
const actionId = get("action_id");
if (!actionId) return null;
const actionType = get("action_type");
const techId = get("done_by_id");
const label = get("action_label");
const cssClass = get("Css_Class");
const startDate = get("start_date");
const endDate = get("end_date");
const startTime = get("start_time");
const endTime = get("end_time");
const currentDate = get("current_date");
const formLink = get("formLink");
const deadline = get("max_resolution_date") || get("max_intervention_date");
const requestId = get("request_id");
// Extraire la ref S260/I260 du label si présente
const refMatch = label.match(/\b([SI]2\d{5}_\d{5})\b/);
const ref = refMatch ? refMatch[1] : null;
// Détection du type "Réservation" : un coordinateur a bloqué un créneau.
// Dans le XML, action_type = "AL-Absence" pour ce genre de créneau, mais
// action_label contient le vrai pattern :
// action_label = "Xxxxx / Créé par : Nom, Prénom"
// Ex: "Ecrans / Créé par : Nom20, Prénom20"
// "Rollout / Créé par : Nom24, Prénom24"
// "Congés / Créé par : ..." → pas une réservation, c'est une absence
// "Maladie / Créé par : ..." → idem
// "Pompier / Créé par : ..." → idem
let effectiveType = actionType;
let reservationLabel = null;
let reservationCreator = null;
const reservationMatch = label.match(/^([^/]+?)\s*\/\s*Créé par\s*:\s*(.+)$/i);
if (reservationMatch) {
const label1 = reservationMatch[1].trim();
const creator = reservationMatch[2].trim();
// Les "absences" connues (Congés/Maladie/Pompier) restent des absences
if (/^(cong[ée]s|maladie|pompier)$/i.test(label1)) {
effectiveType = "AL-Absence";
} else {
// Tout autre label (Ecrans, Rollout, ...) → Réservation
effectiveType = "AL-Reservation";
reservationLabel = label1;
reservationCreator = creator;
}
}
return {
actionId: actionId,
requestId: requestId,
techId: techId,
label: label,
type: effectiveType, // "AL-Intervention" | "AL-Absence" | "AL-Reservation"
originalType: actionType, // type brut (pour debug)
reservationLabel: reservationLabel, // "Ecrans", "Rollout", etc.
reservationCreator: reservationCreator, // "Nom, Prénom" du coordinateur
cssClass: cssClass,
isPompier: /pompier/i.test(label) || /pompier/i.test(actionType),
ref: ref,
startDate: startDate,
endDate: endDate,
startTime: startTime,
endTime: endTime,
currentDate: currentDate,
formLink: formLink,
deadline: deadline,
bulleContact: null,
bulleLieu: null,
bulleDescription: null,
infobulle: null,
status: null,
categoryLine: null,
commentaireTech: null,
ficheTarget: null,
ficheChecksum: null,
ficheFetched: false,
ficheFetchError: null,
ghost: false
};
}
/**
* Parse le body de planning_xhr_2.php?id=ACTIONID (ou similaire).
* Format observé :
* @@DESCRIPTION_S@@...texte complet de l'action...@@DESCRIPTION_E@@
* @@LABEL_S@@AL-Intervention@@LABEL_E@@
* @@LAST_S@@Nom, Prénom@@LAST_E@@
* @@PLANNED_TIME_S@@@@PLANNED_TIME_E@@
* @@PLANNED_CHANGE_S@@@@PLANNED_CHANGE_E@@
*/
function parseXhr2Body(body) {
if (!body || typeof body !== "string") return null;
const out = { description: null, label: null, last: null };
const rxD = /@@DESCRIPTION_S@@([\s\S]*?)@@DESCRIPTION_E@@/;
const rxL = /@@LABEL_S@@([\s\S]*?)@@LABEL_E@@/;
const rxLa = /@@LAST_S@@([\s\S]*?)@@LAST_E@@/;
const md = body.match(rxD);
const ml = body.match(rxL);
const mla = body.match(rxLa);
if (md) out.description = md[1].trim();
if (ml) out.label = ml[1].trim();
if (mla) out.last = mla[1].trim();
return out;
}
/**
* Fetch planning_xhr_2.php pour chaque intervention en parallèle (12 workers)
* et renseigne bulleContact / bulleLieu / bulleDescription / infobulle.
*/
async function fetchBullesForInterventions(interventions) {
if (!interventions || interventions.length === 0) return { ok: 0, fail: 0 };
setRefreshing(true);
let idx = 0;
let ok = 0, fail = 0;
async function worker() {
while (idx < interventions.length) {
if (isRefreshAborted()) return;
const i = idx++;
const iv = interventions[i];
try {
const resp = await sendMessage({ type: "fetchXhr2", actionId: iv.actionId });
if (!resp || !resp.ok) { fail++; continue; }
const parsed = parseXhr2Body(resp.body);
if (!parsed) { fail++; continue; }
if (parsed.description) {
iv.bulleDescription = parsed.description;
const infob = parseActionText(parsed.description);
if (infob) {
iv.infobulle = infob;
if (infob.contact) iv.bulleContact = infob.contact;
if (infob.lieu) iv.bulleLieu = infob.lieu;
}
}
if (parsed.label) iv.label = parsed.label;
iv.xhr2Fetched = true;
ok++;
} catch (err) {
fail++;
console.warn("[xhr2] erreur iv", iv.actionId, err);
}
}
}
const workers = [];
const nWorkers = Math.min(FETCH_CONCURRENCY_BULLES, interventions.length);
for (let w = 0; w < nWorkers; w++) workers.push(worker());
await Promise.all(workers);
console.log(`[xhr2] ${ok} OK, ${fail} échecs sur ${interventions.length}`);
setRefreshing(false);
return { ok, fail };
}
function actionCoversDate(iv, isoDate) {
if (!iv.startDate || !iv.endDate) return true; // manque info → on garde
const target = isoToDDMMYYYY(isoDate);
return ddmmyyyyLE(iv.startDate, target) && ddmmyyyyLE(target, iv.endDate);
}
function ddmmyyyyLE(a, b) {
// Compare deux dates JJ/MM/AAAA
const toNum = s => {
const m = s.match(/^(\d{2})\/(\d{2})\/(\d{4})$/);
return m ? parseInt(m[3] + m[2] + m[1], 10) : 0;
};
return toNum(a) <= toNum(b);
}
// ============================================================================
// Fusion cache ↔ fresh
// ============================================================================
function mergeCacheAndFresh(cached, fresh) {
// fresh.techs : liste des techs avec interventions d'aujourd'hui (depuis EasyVista)
// cached.techs : dernière liste sauvegardée pour ce jour (avec statuts)
//
// Règles :
// - Chaque intervention fresh APPORTE : actionId, type, startTime, endTime, formLink...
// - Le cache APPORTE : ref, categoryLine, status, infobulle (contact/lieu/...),
// commentaireTech, actionText, ficheFetched
// - Pour les CHAMPS ENRICHIS : cache wins (sauf si fresh en a de meilleurs)
// - Une intervention en cache mais plus en fresh → marquée "ghost"
if (!cached || !cached.techs) {
return { techs: fresh.techs };
}
// Indexer le cache par actionId
const cachedByAction = new Map();
for (const tech of cached.techs) {
for (const iv of tech.interventions || []) {
cachedByAction.set(iv.actionId, iv);
}
}
const resultTechs = fresh.techs.map(t => ({ ...t, interventions: [] }));
const freshActionIds = new Set();
for (const tech of fresh.techs) {
const outTech = resultTechs.find(t => t.id === tech.id);
for (const iv of tech.interventions) {
freshActionIds.add(iv.actionId);
const cachedIv = cachedByAction.get(iv.actionId);
if (cachedIv) {
// On part du cache (qui a les champs enrichis), puis on remplace
// les champs "live" depuis le fresh (horaires, type, formLink).
const merged = {
...cachedIv,
// Champs live venant du fresh (le planning peut avoir bougé)
techId: iv.techId || cachedIv.techId,
type: iv.type || cachedIv.type,
label: iv.label || cachedIv.label,
cssClass: iv.cssClass || cachedIv.cssClass,
isPompier: iv.isPompier,
startDate: iv.startDate || cachedIv.startDate,
endDate: iv.endDate || cachedIv.endDate,
startTime: iv.startTime || cachedIv.startTime,
endTime: iv.endTime || cachedIv.endTime,
currentDate: iv.currentDate || cachedIv.currentDate,
formLink: iv.formLink || cachedIv.formLink,
deadline: iv.deadline || cachedIv.deadline,
requestId: iv.requestId || cachedIv.requestId,
// Ref : on privilégie celle qu'on a (fresh ou cached)
ref: cachedIv.ref || iv.ref,
// Bulle (HTML planning) : fresh est plus à jour
bulleContact: iv.bulleContact || cachedIv.bulleContact,
bulleLieu: iv.bulleLieu || cachedIv.bulleLieu,
bulleDescription: iv.bulleDescription || cachedIv.bulleDescription,
// ghost : on retire (cette intervention est bien là dans le fresh)
ghost: false
};
outTech.interventions.push(merged);
} else {
outTech.interventions.push(iv);
}
}
}
// Ajouter les interventions qui sont en cache mais plus en fresh
for (const tech of cached.techs) {
const outTech = resultTechs.find(t => t.id === tech.id);
if (!outTech) continue;
for (const iv of tech.interventions || []) {
if (!freshActionIds.has(iv.actionId)) {
const ghost = { ...iv, ghost: true };
outTech.interventions.push(ghost);
}
}
// Retrier
outTech.interventions.sort((a, b) =>
(a.startTime || "").localeCompare(b.startTime || "")
);
}
return { techs: resultTechs };
}
// ============================================================================
// Fetch des fiches individuelles (pour obtenir le statut et les détails)
// ============================================================================
async function refreshStatuses(techs, isoDate) {
// Construire la liste des interventions à fetcher, dans l'ordre de priorité :
// 1. Interventions du (des) pompier(s) en premier
// 2. Puis les autres techs par ordre alphabétique du nom de famille
// 3. (Les absents n'ont pas d'interventions à fetcher)
const sortedTechs = [...techs].sort((a, b) => compareTechs(a, b, isoDate));
const toFetch = [];
for (const tech of sortedTechs) {
for (const iv of tech.interventions) {
if (iv.type !== "AL-Intervention") continue;
if (!iv.formLink) continue;
// On skip si :
// - Déjà clos / résolu ET ficheTarget déjà connu (statut + requestId OK)
// ET actionDescription déjà remplacée depuis la fiche
// - Sinon on garde (pour avoir statut frais OU ficheTarget pour clic
// OU le texte complet de l'action)
const statusClosed = isClosedStatus(iv.status) || isResolvedStatus(iv.status);
if (statusClosed && iv.ficheTarget && iv.actionDescriptionFetched) continue;
toFetch.push(iv);
}
}
if (toFetch.length === 0) return;
setRefreshing(true);
try {
// Fetcher avec concurrence = FETCH_CONCURRENCY_FICHES (15)
// Chaque worker vérifie isRefreshAborted() AVANT de prendre la prochaine
// intervention : si l'utilisateur a cliqué "Arrêter", les workers
// s'arrêtent proprement dans ~100ms.
let idx = 0;
async function worker() {
while (idx < toFetch.length) {
if (isRefreshAborted()) return;
const i = idx++;
await fetchAndUpdateIntervention(toFetch[i]);
}
}
const workers = [];
const nWorkers = Math.min(FETCH_CONCURRENCY_FICHES, toFetch.length);
for (let w = 0; w < nWorkers; w++) workers.push(worker());
await Promise.all(workers);
// Si annulé : on laisse les refs déjà arrivées s'afficher (le rendu
// incrémental les a mises dans le DOM), on skip juste le re-render
// final et le nettoyage ghosts/cache.
if (isRefreshAborted()) {
return;
}
// Résoudre le sort des ghosts
for (const tech of techs) {
tech.interventions = tech.interventions.filter(iv => {
if (!iv.ghost) return true;
if (CANCELLED_STATUS.includes(iv.status)) return false;
return true;
});
}
// Sauvegarder le résultat enrichi dans le cache
await writeCache(isoDate, { techs });
// Re-rendre pour afficher les mises à jour (un seul rendu à la fin)
renderFromData({
techs,
targetDate: isoDate,
captureTime: Date.now(),
source: "fresh+statuses"
});
} finally {
setRefreshing(false);
}
}
async function fetchAndUpdateIntervention(iv) {
try {
// Bail-out coopératif : si l'utilisateur a cliqué sur "Arrêter",
// on ne fetch pas cette intervention.
if (isRefreshAborted()) {
iv.ficheFetched = true;
iv.ficheFetchError = "aborted";
return;
}
// Fetch de la fiche (HTML) pour récupérer statut + commentaire tech +
// extraire target/checksum qui servent à :
// - l'API timeline (texte validé de l'action, si xhr2 n'avait pas été assez)
// - construire une URL d'ouverture qui marche (clic sur intervention)
//
// Retry léger : une erreur réseau ponctuelle (timeout, 5xx) ne doit pas
// perdre la ligne. 1 seul retry après 400ms. Session expirée n'est PAS
// retryée (ça ne passera pas mieux la 2e fois).
let ficheResp = await sendMessage({ type: "fetchFiche", formLink: iv.formLink });
if (!ficheResp.ok && ficheResp.error !== "session_expired" && !isRefreshAborted()) {
await new Promise(r => setTimeout(r, 400));
if (!isRefreshAborted()) {
ficheResp = await sendMessage({ type: "fetchFiche", formLink: iv.formLink });
}
}
if (!ficheResp.ok) {
iv.ficheFetched = true;
iv.ficheFetchError = ficheResp.error || "fetch_failed";
if (ficheResp.error === "session_expired") {
state.session = null;
}
return;
}
const fiche = parseFicheHtml(ficheResp.html);
iv.status = fiche.status;
iv.categoryLine = fiche.categoryLine || iv.categoryLine;
if (fiche.rfc && !iv.ref) {
iv.ref = fiche.rfc;
}
iv.commentaireTech = fiche.commentaireTech;
// ─── Remplacement du texte d'action + contact/lieu depuis la fiche ─────────
// Le texte de la bulle (planning_xhr_2.php) est parfois tronqué/incomplet.
// La fiche contient le texte complet dans AM_ACTION.DESCRIPTION.
// SÉCURITÉ : on ne remplace QUE si l'Intervenant de la fiche correspond au
// tech de la ligne du planning (car une même fiche peut avoir plusieurs
// actions assignées à différents techs, et on fetche la MÊME fiche pour tous).
const expectedTechName = iv.techId ? TEAM[iv.techId] : null;
const matchOk = fiche.intervenant && expectedTechName &&
namesMatch(fiche.intervenant, expectedTechName);
if (fiche.actionDescription && matchOk) {
// Remplace le texte d'action (affiché dans la popup)
iv.bulleDescription = fiche.actionDescription;
iv.actionDescriptionFetched = true; // flag : déjà remplacé depuis la fiche
// Reparse contact/lieu depuis le nouveau texte : la carte affiche
// bulleContact/bulleLieu, donc il faut les mettre à jour aussi.
const infob = parseActionText(fiche.actionDescription);
if (infob) {
iv.infobulle = infob;
if (infob.contact) iv.bulleContact = infob.contact;
if (infob.lieu) iv.bulleLieu = infob.lieu;
}
}
// Si ça ne matche pas : on garde bulleDescription/Contact/Lieu tels quels (sécurité)
// Extraire le checksum CORRECT pour ouvrir la fiche :
// - Le target de la FICHE = iv.requestId (vient du XML)
// - Il faut trouver le checksum qui est accolé à ce target dans le HTML
// La regex principale cherche "target=REQUEST_ID&checksum=XXX" mais peut
// échouer si ce pattern n'apparaît pas dans le HTML (selon les sections
// hydratées par Angular). On a plusieurs fallbacks robustes.
if (iv.requestId) {
let checksumFound = false;
// Tentative 1 : target=ID&checksum=... (pattern le plus courant dans les liens)
const rx1 = new RegExp(`target=${iv.requestId}&(?:amp;)?checksum=([a-f0-9]{40})`);
const m1 = ficheResp.html.match(rx1);
if (m1) {
iv.ficheTarget = iv.requestId;
iv.ficheChecksum = m1[1];
checksumFound = true;
} else {
// Tentative 2 : dans le JSON formData : "id":"REQUEST_ID"..."checksum":"..."
// ou l'inverse : "checksum":"..."..."id":"REQUEST_ID"
const rx2a = new RegExp(`"id"\\s*:\\s*"${iv.requestId}"[\\s\\S]{0,200}?"checksum"\\s*:\\s*"([a-f0-9]{40})"`);
const m2a = ficheResp.html.match(rx2a);
if (m2a) {
iv.ficheTarget = iv.requestId;
iv.ficheChecksum = m2a[1];
checksumFound = true;
} else {
const rx2b = new RegExp(`"checksum"\\s*:\\s*"([a-f0-9]{40})"[\\s\\S]{0,200}?"id"\\s*:\\s*"${iv.requestId}"`);
const m2b = ficheResp.html.match(rx2b);
if (m2b) {
iv.ficheTarget = iv.requestId;
iv.ficheChecksum = m2b[1];
checksumFound = true;
}
}
}
// Tentative 3 (ultime) : le checksum global du form principal.
if (!checksumFound) {
const rx3 = /"form"\s*:\s*\{[^}]*?"checksum"\s*:\s*"([a-f0-9]{40})"[\s\S]{0,2000}?"id"\s*:\s*"(\d+)"/;
const m3 = ficheResp.html.match(rx3);
if (m3 && m3[2] === String(iv.requestId)) {
iv.ficheTarget = iv.requestId;
iv.ficheChecksum = m3[1];
}
}
}
iv.ficheFetched = true;
// ─── RENDU INCRÉMENTAL (v3.1) ─────────────────────────────────────────
// La ref (RFC_NUMBER) et le statut sont déjà connus : on met à jour la
// ligne correspondante DANS LE DOM immédiatement, sans attendre que les
// autres workers aient fini. Pas de re-rendu global.
updateInterventionRow(iv);
// Pour l'API timeline, on utilise le MÊME target + checksum (celui de la fiche)
const timelineTarget = iv.ficheTarget;
const timelineChecksum = iv.ficheChecksum;
// Étape timeline API : on veut le texte COMPLET de l'action.
// planning_xhr_2.php tronque souvent à ~300 chars, mais l'API timeline
// retourne le texte intégral. On la fetch à chaque fois que possible.
//
// PROBLÈME OBSERVÉ : EasyVista retourne parfois une timeline "partielle"
// au 1er appel (ex: 8 Ko au lieu de 44 Ko), sans le texte de l'action
// courante. Le serveur a besoin de "construire" le contexte après le fetch
// de la fiche. Dans ce cas on MARQUE l'intervention pour un retry silencieux
// en arrière-plan (fait plus tard par runBackgroundTimelineRetry).
const needsTimelineValidation = !iv.actionText;
if (needsTimelineValidation && timelineTarget && timelineChecksum) {
if (isRefreshAborted()) return;
const tlResp = await sendMessage({
type: "fetchTimeline",
target: timelineTarget,
checksum: timelineChecksum
});
if (tlResp && tlResp.ok) {
const actionDetails = parseTimelineJson(tlResp.body, iv.actionId);
if (actionDetails && actionDetails.text) {
applyActionTextToIv(iv, actionDetails);
} else {
// Timeline partielle : marquer pour retry silencieux en arrière-plan
iv.actionTextPending = true;
}
} else {
iv.actionTextPending = true;
}
}
} catch (err) {
iv.ficheFetched = true;
iv.ficheFetchError = String(err);
console.warn("fetchAndUpdate error:", err);
}
}
/**
* Applique les détails d'action (texte timeline) à une intervention :
* - met à jour bulleDescription (texte affiché dans la popup)
* - reparse contact/lieu pour mettre à jour la carte
* - rafraîchit la ligne dans le DOM
* Utilisé à la fois par le flow principal et par le retry silencieux.
*/
function applyActionTextToIv(iv, actionDetails) {
iv.actionText = actionDetails.text;
iv.actionDone = actionDetails.doneById;
iv.bulleDescription = actionDetails.text;
iv.actionDescriptionFetched = true;
iv.actionTextPending = false;
const infob = parseActionText(actionDetails.text);
if (infob) {
iv.infobulle = infob;
if (infob.contact) iv.bulleContact = infob.contact;
if (infob.lieu) iv.bulleLieu = infob.lieu;
}
// Rafraîchir la ligne dans le DOM (lieu/contact mis à jour en live)
updateInterventionRow(iv);
}
/**
* Retry silencieux en arrière-plan : liste les interventions dont le texte
* d'action n'a pas pu être récupéré (timeline partielle au 1er coup), et
* refait un fetch timeline pour chacune, avec un petit délai entre les appels
* pour ne pas surcharger le serveur.
*
* Cette fonction est lancée sans await — elle tourne en tâche de fond pendant
* que l'utilisateur navigue. Elle respecte le jeton de refresh : si l'user
* change de jour, le jeton change et le retry s'arrête silencieusement.
*
* Aucun spinner ni indication visuelle : l'user ne voit rien, sauf que les
* popups se mettent à jour quand le texte arrive.
*/
async function runBackgroundTimelineRetry(techs, isoDate, myToken) {
// Collecter les interventions qui ont besoin d'un retry
const pending = [];
for (const tech of techs) {
for (const iv of tech.interventions) {
if (iv.actionTextPending && iv.ficheTarget && iv.ficheChecksum) {
pending.push(iv);
}
}
}
if (pending.length === 0) return;
// Attendre un peu avant de démarrer (laisser le serveur "respirer")
await new Promise(r => setTimeout(r, 1500));
// Si l'user a changé de jour entre-temps, abandonner
if (currentRefreshToken !== myToken) return;
for (const iv of pending) {
// Si l'user a navigué ailleurs OU cliqué arrêter : on sort sans bruit
if (currentRefreshToken !== myToken) return;
if (isRefreshAborted()) return;
try {
const tlResp = await sendMessage({
type: "fetchTimeline",
target: iv.ficheTarget,
checksum: iv.ficheChecksum
});
if (tlResp && tlResp.ok) {
const actionDetails = parseTimelineJson(tlResp.body, iv.actionId);
if (actionDetails && actionDetails.text) {
applyActionTextToIv(iv, actionDetails);
}
}
} catch {
// Silence : c'est du retry en arrière-plan, on ne dérange pas l'user
}
// Petit délai entre chaque retry pour ménager le serveur
await new Promise(r => setTimeout(r, 400));
}
// Sauvegarder le cache avec les nouvelles infos (si on est toujours
// sur la même date et même token)
if (currentRefreshToken === myToken && !isRefreshAborted()) {
try {
await writeCache(isoDate, { techs });
} catch {}
}
}
function isClosedStatus(s) {
return !!s && CLOSED_STATUS.some(x => s.includes(x));
}
function isResolvedStatus(s) {
return !!s && RESOLVED_STATUS.some(x => s.includes(x));
}
function isCancelledStatus(s) {
return !!s && CANCELLED_STATUS.some(x => s.includes(x));
}
// ============================================================================
// Parsing d'une fiche individuelle (HTML)
// ============================================================================
function parseFicheHtml(html) {
const out = {
status: null,
rfc: null,
categoryLine: null,
commentaireTech: null,
intervenant: null, // Nom du tech assigné à l'action (format "Nom, Prénom")
actionDescription: null // Texte complet "Date:... Lieu:... Contact:..." (propre, sans HTML)
};
// STATUS_FR (valeur parfois encodée en \u00XX)
let m = html.match(/"dbFieldName"\s*:\s*"STATUS_FR"[^}]*?"value"\s*:\s*"([^"]{2,30})"/);
if (m) out.status = decodeJsonString(m[1]);
// RFC_NUMBER
m = html.match(/"dbFieldName"\s*:\s*"RFC_NUMBER"[^}]*?"value"\s*:\s*"([^"]{5,30})"/);
if (m) out.rfc = m[1];
// TITLE_FR contient la catégorie complète
m = html.match(/"dbFieldName"\s*:\s*"TITLE_FR"[^}]*?"value"\s*:\s*"([^"]{5,300})"/);
if (m) out.categoryLine = decodeJsonString(m[1]);
// Commentaire tech à la fin de DESCRIPTION : "
techN: ..."
m = html.match(/"dbFieldName"\s*:\s*"DESCRIPTION"[^}]*?"value"\s*:\s*"((?:[^"\\]|\\.)+)"/);
if (m) {
const desc = decodeJsonString(m[1]);
const ctm = desc.match(/
\s*
\s*([a-z][a-z0-9]{2,14})\s*:\s*([^<]{3,500})/i);
if (ctm) {
out.commentaireTech = ctm[1] + ": " + ctm[2].trim();
}
}
// ─── Intervenant assigné (AM_EMPLOYEE.LAST_NAME dans la section "Action") ───
// HTML Angular rendu :