fix(logistique) : bon de pesée — cartouche tiers + filtrage des listes contrepartie par site (ERP-208)
- PDF : cartouche bordé en haut à droite avec le type (Client/Fournisseur/Autre) et le nom du tiers (getCounterpartyName + getCounterpartyTypeLabel). - Écran ticket : listes Client/Fournisseur filtrées sur le site courant (param siteId[]) et rechargées au changement de site ; reset du tiers sélectionné s'il sort du périmètre du nouveau site.
This commit is contained in:
+44
@@ -0,0 +1,44 @@
|
||||
import { describe, it, expect, vi, beforeEach } from 'vitest'
|
||||
import { useWeighingTicketReferentials } from '../useWeighingTicketReferentials'
|
||||
|
||||
const mockApiGet = vi.hoisted(() => vi.fn())
|
||||
vi.stubGlobal('useApi', () => ({ get: mockApiGet }))
|
||||
|
||||
/**
|
||||
* Tests des référentiels Client/Fournisseur de l'écran ticket de pesée (M5).
|
||||
* Contrat couvert (ERP-208) : `load(siteId)` filtre les deux endpoints par site
|
||||
* courant via `siteId[]` ; sans site → listes complètes (param absent).
|
||||
*/
|
||||
describe('useWeighingTicketReferentials', () => {
|
||||
beforeEach(() => {
|
||||
mockApiGet.mockReset()
|
||||
mockApiGet.mockResolvedValue({ member: [] })
|
||||
})
|
||||
|
||||
it('passe siteId[] aux deux endpoints quand un site courant est fourni', async () => {
|
||||
const { load } = useWeighingTicketReferentials()
|
||||
await load(7)
|
||||
|
||||
const clientsCall = mockApiGet.mock.calls.find(c => c[0] === '/clients')
|
||||
const suppliersCall = mockApiGet.mock.calls.find(c => c[0] === '/suppliers')
|
||||
expect(clientsCall?.[1]).toMatchObject({ pagination: 'false', 'siteId[]': [7] })
|
||||
expect(suppliersCall?.[1]).toMatchObject({ pagination: 'false', 'siteId[]': [7] })
|
||||
})
|
||||
|
||||
it('ne passe pas siteId[] quand aucun site (liste complète)', async () => {
|
||||
const { load } = useWeighingTicketReferentials()
|
||||
await load(null)
|
||||
|
||||
const clientsCall = mockApiGet.mock.calls.find(c => c[0] === '/clients')
|
||||
expect(clientsCall?.[1]).not.toHaveProperty('siteId[]')
|
||||
expect(clientsCall?.[1]).toMatchObject({ pagination: 'false' })
|
||||
})
|
||||
|
||||
it('mappe les membres Hydra en options { value: @id, label: companyName }', async () => {
|
||||
mockApiGet.mockResolvedValue({ member: [{ '@id': '/api/clients/3', companyName: 'ACME' }] })
|
||||
const { load, clients } = useWeighingTicketReferentials()
|
||||
await load(7)
|
||||
|
||||
expect(clients.value).toEqual([{ value: '/api/clients/3', label: 'ACME' }])
|
||||
})
|
||||
})
|
||||
@@ -32,11 +32,19 @@ export function useWeighingTicketReferentials() {
|
||||
const clients = ref<RefOption[]>([])
|
||||
const suppliers = ref<RefOption[]>([])
|
||||
|
||||
/** Récupère une collection complète (pagination désactivée) en Hydra. */
|
||||
async function fetchAll(url: string): Promise<PartyMember[]> {
|
||||
/**
|
||||
* Récupère une collection complète (pagination désactivée) en Hydra. Filtre par
|
||||
* site courant si `siteId` est fourni (ERP-208) : un tiers est rattaché à un site
|
||||
* via les sites de ses adresses — param `siteId[]` déjà géré par les providers M1/M2.
|
||||
*/
|
||||
async function fetchAll(url: string, siteId?: number | null): Promise<PartyMember[]> {
|
||||
const query: Record<string, unknown> = { pagination: 'false' }
|
||||
if (siteId !== null && siteId !== undefined) {
|
||||
query['siteId[]'] = [siteId]
|
||||
}
|
||||
const res = await api.get<{ member?: PartyMember[] }>(
|
||||
url,
|
||||
{ pagination: 'false' },
|
||||
query,
|
||||
{ headers: LD_JSON_HEADERS, toast: false },
|
||||
)
|
||||
return res.member ?? []
|
||||
@@ -45,14 +53,15 @@ export function useWeighingTicketReferentials() {
|
||||
/**
|
||||
* Charge en parallèle clients + fournisseurs (résilient : un référentiel en
|
||||
* échec — ex. 403 selon le rôle — laisse simplement son select vide sans
|
||||
* faire échouer l'autre).
|
||||
* faire échouer l'autre). `siteId` (site courant) filtre les listes par site
|
||||
* (ERP-208) ; absent → listes complètes.
|
||||
*/
|
||||
async function load(): Promise<void> {
|
||||
async function load(siteId?: number | null): Promise<void> {
|
||||
await Promise.allSettled([
|
||||
fetchAll('/clients').then((list) => {
|
||||
fetchAll('/clients', siteId).then((list) => {
|
||||
clients.value = list.map(c => ({ value: c['@id'], label: c.companyName }))
|
||||
}),
|
||||
fetchAll('/suppliers').then((list) => {
|
||||
fetchAll('/suppliers', siteId).then((list) => {
|
||||
suppliers.value = list.map(s => ({ value: s['@id'], label: s.companyName }))
|
||||
}),
|
||||
])
|
||||
|
||||
Reference in New Issue
Block a user