c76c447aa2
Auto Tag Develop / tag (push) Successful in 8s
Empilée sur ERP-144 (#106). ## Périmètre ERP-145 Écrans **Consultation** (lecture seule) et **Modification** (édition par onglet), peuplés depuis la **seule** réponse `GET /api/providers/{id}` (embed contacts/adresses/ribs + refs comptables — pas de N+1). ### Consultation — `pages/providers/[id]/index.vue` (`/providers/{id}`) - Ouverture par défaut sur **Contacts** ; tous champs readonly ; onglets **Contacts · Adresse · Rapports · Échanges · Comptabilité** (navigation libre). Rapports/Échanges = placeholders « À venir ». - Flèche retour → répertoire. Bouton **Modifier** (si `manage` OU `accounting.manage`). Bouton **Archiver** (Admin seul, `archive`) → modal → PATCH `{isArchived:true}` ; **Restaurer** si archivé. - Comptabilité visible seulement si `accounting.view` ; banque/RIB affichés selon le type de règlement (VIREMENT/LCR). ### Modification — `pages/providers/[id]/edit.vue` (`/providers/{id}/edit`) - Pré-rempli ; **bloc principal éditable** (Nom/Catégories/Sites, PATCH `provider:write:main` via `updateMain`) ; onglets Contact/Adresse/Comptabilité en **navigation libre**, PATCH partiel par onglet (réutilise `useProviderForm` en `editMode`). - Onglets sans permission `manage` / `accounting.manage` restent **readonly** (pas de bouton Valider / suppression). Accès réservé à `manage` OU `accounting.manage`. ### Composables / helpers - **`useProvider(id)`** : charge le détail (ld+json) + archive/restore (PATCH isArchived seul, puis rechargement). - **`useProviderForm`** étendu : `updateMain()` (PATCH principal en édition) + `editMode` (completeTab ne verrouille/avance plus). - **`providerDetail.ts`** : mapping embed → brouillons + options role-indépendantes (libellés depuis l'embed) + règles d'actions (Modifier/Archiver/Restaurer). ## Conformité - `useApi()` only ; `Malio*` only ; `usePermissions()` pour boutons/onglets ; aucun texte FR en dur ; pas d'import inter-module (règle ABSOLUE n°1). ## Vérifications - Vitest : 470/470 (16 nouveaux : mapping détail, actions par permission, updateMain + editMode). - ESLint : OK · `nuxi typecheck` : 0 erreur sur les fichiers source du ticket. - Golden path navigateur : **Consultation** (ACME) — bloc principal readonly + libellés catégories/sites résolus depuis l'embed, 5 onglets, Modifier+Archiver visibles (admin), Comptabilité readonly. **Modification** — bloc principal éditable pré-rempli (Site « 86 17 »), 3 onglets navigation libre, onglet Contact pré-rempli. Reviewed-on: #107 Co-authored-by: tristan <tristan@yuno.malio.fr> Co-committed-by: tristan <tristan@yuno.malio.fr>
71 lines
2.5 KiB
TypeScript
71 lines
2.5 KiB
TypeScript
import { ref } from 'vue'
|
|
import type { ProviderDetail } from '~/modules/technique/utils/forms/providerDetail'
|
|
|
|
/**
|
|
* Chargement et actions d'archivage d'un prestataire unique (ecrans Consultation /
|
|
* Modification, ERP-145). Miroir de `useSupplier` (M2). Lit le detail embarque via
|
|
* `GET /api/providers/{id}` (contacts / adresses + leurs sous-collections / ribs
|
|
* sous `provider:item:read` / `provider:read:accounting`) — une SEULE requete
|
|
* peuple les deux ecrans (embed borne, pas de N+1).
|
|
*
|
|
* L'en-tete `Accept: application/ld+json` est impose pour obtenir le payload Hydra
|
|
* complet (avec les `@id` des relations embarquees, indispensables au pre-remplissage).
|
|
*
|
|
* Etat 100 % local a l'instance (refs). Les erreurs d'archivage / restauration
|
|
* (notamment le 409 d'homonyme actif a la restauration) sont PROPAGEES a l'appelant,
|
|
* qui decide du toast a afficher.
|
|
*/
|
|
export function useProvider(id: number | string) {
|
|
const api = useApi()
|
|
|
|
const provider = ref<ProviderDetail | null>(null)
|
|
const loading = ref(false)
|
|
const error = ref(false)
|
|
|
|
/** Recupere le detail complet (embed contacts/adresses/ribs + comptabilite). */
|
|
function fetchDetail(): Promise<ProviderDetail> {
|
|
return api.get<ProviderDetail>(
|
|
`/providers/${id}`,
|
|
{},
|
|
{ headers: { Accept: 'application/ld+json' }, toast: false },
|
|
)
|
|
}
|
|
|
|
/** Charge le detail du prestataire. En cas d'echec : `error = true`, `provider = null`. */
|
|
async function load(): Promise<void> {
|
|
loading.value = true
|
|
error.value = false
|
|
try {
|
|
provider.value = await fetchDetail()
|
|
}
|
|
catch {
|
|
error.value = true
|
|
provider.value = null
|
|
}
|
|
finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Bascule l'archivage (PATCH `isArchived` SEUL — groupe provider:write:archive ;
|
|
* tout autre champ => 422), puis RECHARGE le detail complet : la reponse du PATCH
|
|
* ne porte que `provider:read` (ni l'embed des sous-collections ni les libelles
|
|
* comptables), un simple merge laisserait l'affichage incoherent. Toute erreur est
|
|
* propagee a l'appelant AVANT le rechargement.
|
|
*/
|
|
async function setArchived(isArchived: boolean): Promise<void> {
|
|
await api.patch(`/providers/${id}`, { isArchived }, { toast: false })
|
|
provider.value = await fetchDetail()
|
|
}
|
|
|
|
return {
|
|
provider,
|
|
loading,
|
|
error,
|
|
load,
|
|
archive: () => setArchived(true),
|
|
restore: () => setArchived(false),
|
|
}
|
|
}
|