bbd8a38c95
Auto Tag Develop / tag (push) Successful in 9s
Améliorations frontend de la partie **Répertoire** (Client / Prospect / Prestataire). Onglet **Rapport** retravaillé en fin de parcours ; le reste de la logique métier inchangé. ## Navigation & liste - Onglet actif conservé au retour liste ↔ fiche (flèche app **et** navigateur) via `history.state` (hors URL) — util `historyTab.ts` - Colonne « Action » (entête alignée) + feedback hover sur les boutons d'action - Conversion prospect → client : modal de confirmation - Boutons « Ajouter » : label court + taille Malio standard ; barres d'outils à hauteur homogène (plus de saut entre onglets) ## Fiches (Info / Contact / Adresse) - Style **plat** sans box-shadow (comme Starseed) - Champs email/téléphone : `MalioInputEmail` / `MalioInputPhone` - Grilles en **4 colonnes** (Info + blocs) - Boutons « Nouveau contact/adresse » en secondary ; « Enregistrer » en taille Malio ; marge form↔bouton homogène - Bouton retour **ghost** (`mdi:arrow-left-bold`) - **Adresse** : flux CP → ville → rue (rue conditionnée au CP+ville, cascade de reset) ; titre du bloc = libellé saisi - Suppression d'un bloc Contact/Adresse : **modal** de confirmation (centralisée dans `useDirectoryDetail`) - Modals (suppression, conversion) basées sur `MalioModal` (design Starseed) avec nom en gras ## Onglet Rapport - Bouton d'ajout en taille Malio (« Ajouter ») - Suppression compte-rendu : `ConfirmModal` partagée (remplace l'ancienne modal maison) - Suppression d'un document joint : ajout d'une modal de confirmation - Upload via `MalioInputUpload` ; bouton supprimer document aligné (`mdi:delete-outline` ghost) ## Divers - `fix(auth)` : cookie JWT renommé `BEARER_LESSTIME` (collision localhost avec d'autres apps Symfony) - `fix(infra)` : target makefile `fix-uploads-perm` (volume `uploads_data` root → upload local OK) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Reviewed-on: #27
36 lines
1.4 KiB
TypeScript
36 lines
1.4 KiB
TypeScript
/**
|
|
* Onglet actif transmis d'une page à l'autre via l'état d'historique
|
|
* (`history.state`), SANS le mettre dans l'URL. Sert à préserver l'onglet courant
|
|
* du Répertoire (Clients / Prospects / Prestataires) lors de l'aller-retour
|
|
* liste ↔ fiche, dans les deux sens (flèche de l'app ET flèche du navigateur).
|
|
*
|
|
* On reste fidèle à la règle « état d'UI local, pas dans l'URL » : l'onglet
|
|
* voyage dans l'entrée d'historique de la navigation, l'URL ne change pas.
|
|
*/
|
|
|
|
/**
|
|
* Lit la clé d'onglet posée dans `history.state.tab` si elle fait partie des
|
|
* onglets valides. Retourne `null` sinon : navigation directe / deep link,
|
|
* rechargement de page, ou onglet inexistant.
|
|
*/
|
|
export function readHistoryTab(validKeys: string[]): string | null {
|
|
if (typeof window === 'undefined') {
|
|
return null
|
|
}
|
|
const tab = (window.history.state as Record<string, unknown> | null)?.tab
|
|
return typeof tab === 'string' && validKeys.includes(tab) ? tab : null
|
|
}
|
|
|
|
/**
|
|
* Estampille l'entrée d'historique COURANTE avec l'onglet actif, sans créer de
|
|
* nouvelle entrée ni changer l'URL. À appeler juste avant de naviguer vers une
|
|
* fiche : au retour via la flèche du navigateur (popstate), cette entrée
|
|
* « liste » est restaurée avec son onglet.
|
|
*/
|
|
export function stampHistoryTab(tab: string): void {
|
|
if (typeof window === 'undefined') {
|
|
return
|
|
}
|
|
window.history.replaceState({ ...window.history.state, tab }, '')
|
|
}
|