a340d8139a
Auto Tag Develop / tag (push) Successful in 8s
## Contexte ERP-148 — mise à jour @malio/layer-ui et amélioration des champs date (onglet Information, Client & Fournisseur). ## Changements - **MalioDate v1.7.10** : le composant expose désormais son état de validité (`@update:valid`) et la saisie brute invalide (`@update:rawValue`). - **Validation back-autoritaire du format** : `foundedAt` n'accepte plus que l'ISO strict `Y-m-d` (`#[Context]` DateTimeNormalizer) + `collectDenormalizationErrors` sur `Client` et `Supplier`. Toute saisie non-ISO renvoie un **422 porté sur le champ**. - Corrige un cas piège : `12/25/2026` (invalide en JJ/MM/AAAA côté front) était auparavant accepté par PHP en M/J/AAAA → 25 décembre. Désormais rejeté. - **Front** : la saisie invalide est transmise au back ; le message technique de type-error est surchargé par une clé i18n via le **code de violation** (`resolveViolationMessage` / `VIOLATION_MESSAGE_I18N`), affiché inline par `useFormErrors`. - Réorganisation des utils de formulaire sous `utils/forms/`. ## Tests - Back : `ClientFoundedAtFormatTest` / `SupplierFoundedAtFormatTest` (dont le cas piège `12/25/2026`). - Front : résolveur i18n (`api.test.ts`, `useFormErrors.test.ts`) + payloads (`clientEdit`/`supplierEdit` specs). - Suite Commercial verte ; vérifié bout-en-bout en navigateur (PATCH → 422, erreur inline, submit bloqué). ## Note Échecs JWT aléatoires connus du hook pre-commit (401/500 sur tests d'auth sans rapport) ; tous verts en isolation. Reviewed-on: #92 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 { ClientDetail } from '~/modules/commercial/utils/forms/clientConsultation'
|
|
|
|
/**
|
|
* Chargement et actions d'archivage d'un client unique (ecran « Consultation
|
|
* client », 1.11). Lit le detail embarque via `GET /api/clients/{id}` (contacts /
|
|
* adresses / ribs sous `client:item:read` / `client:read:accounting`) et expose
|
|
* les bascules d'archivage (PATCH `isArchived` SEUL — tout autre champ => 422).
|
|
*
|
|
* L'en-tete `Accept: application/ld+json` est impose pour obtenir le payload
|
|
* Hydra complet (sans lui, API Platform 4 renvoie une representation reduite).
|
|
*
|
|
* Etat 100 % local a l'instance (refs) — aucune persistance URL. Les erreurs
|
|
* d'archivage/restauration (notamment le 409 RG-1.23 : homonyme actif a la
|
|
* restauration) sont PROPAGEES a l'appelant, qui decide du toast a afficher.
|
|
*/
|
|
export function useClient(id: number | string) {
|
|
const api = useApi()
|
|
|
|
const client = ref<ClientDetail | null>(null)
|
|
const loading = ref(false)
|
|
const error = ref(false)
|
|
|
|
/** Recupere le detail complet (embed contacts/adresses/ribs + comptabilite). */
|
|
function fetchDetail(): Promise<ClientDetail> {
|
|
return api.get<ClientDetail>(
|
|
`/clients/${id}`,
|
|
{},
|
|
{ headers: { Accept: 'application/ld+json' }, toast: false },
|
|
)
|
|
}
|
|
|
|
/** Charge le detail du client. En cas d'echec : `error = true`, `client = null`. */
|
|
async function load(): Promise<void> {
|
|
loading.value = true
|
|
error.value = false
|
|
try {
|
|
client.value = await fetchDetail()
|
|
}
|
|
catch {
|
|
error.value = true
|
|
client.value = null
|
|
}
|
|
finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Bascule l'archivage (PATCH `isArchived` SEUL — tout autre champ => 422),
|
|
* puis RECHARGE le detail complet : la reponse du PATCH ne porte que le groupe
|
|
* `client:read` (ni l'embed contacts/adresses/ribs ni les libelles des
|
|
* referentiels comptables), un simple merge laisserait l'affichage incoherent.
|
|
* Toute erreur (notamment le 409 d'homonyme actif a la restauration, RG-1.23)
|
|
* est propagee a l'appelant AVANT le rechargement.
|
|
*/
|
|
async function setArchived(isArchived: boolean): Promise<void> {
|
|
await api.patch(`/clients/${id}`, { isArchived }, { toast: false })
|
|
client.value = await fetchDetail()
|
|
}
|
|
|
|
return {
|
|
client,
|
|
loading,
|
|
error,
|
|
load,
|
|
archive: () => setArchived(true),
|
|
restore: () => setArchived(false),
|
|
}
|
|
}
|