Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 251c939ba0 | |||
| b6fcd3c186 | |||
| e664731cb8 | |||
| 244d62dc71 | |||
| 29bd6abcfe | |||
| 90ed4a213f | |||
| 9f772a84ed | |||
| 1131420960 | |||
| 2a818a0c77 | |||
| 59230bbc7e | |||
| 49a5dc5252 | |||
| 9ff3e83c03 | |||
| b55050b2ad | |||
| 1d66e5dd31 | |||
| c0c39705c7 | |||
| acd531f69e | |||
| 7d7b2fb720 | |||
| 7ca5c5f4c5 | |||
| f3a18ace1d | |||
| d9023a0ddc |
@@ -12,7 +12,14 @@
|
|||||||
"Bash(mv buttonIcon.story.vue button/)",
|
"Bash(mv buttonIcon.story.vue button/)",
|
||||||
"Bash(mv inputText.story.vue inputAmount.story.vue inputNumber.story.vue inputPassword.story.vue inputTextArea.story.vue inputUpload.story.vue input/)",
|
"Bash(mv inputText.story.vue inputAmount.story.vue inputNumber.story.vue inputPassword.story.vue inputTextArea.story.vue inputUpload.story.vue input/)",
|
||||||
"Bash(mv InputSelect.story.vue selectCheckbox.story.vue select/)",
|
"Bash(mv InputSelect.story.vue selectCheckbox.story.vue select/)",
|
||||||
"Bash(mv inputCheckbox.story.vue checkbox/)"
|
"Bash(mv inputCheckbox.story.vue checkbox/)",
|
||||||
|
"Bash(npx eslint *)",
|
||||||
|
"Bash(echo \"LINT EXIT: $?\")",
|
||||||
|
"Bash(git commit *)",
|
||||||
|
"mcp__chrome__navigate_page",
|
||||||
|
"mcp__chrome__take_snapshot",
|
||||||
|
"mcp__chrome__click",
|
||||||
|
"mcp__chrome__evaluate_script"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -108,9 +108,9 @@ npm run lint # Pas d'erreurs
|
|||||||
|
|
||||||
### 5. Créer la page playground
|
### 5. Créer la page playground
|
||||||
|
|
||||||
**Fichier :** `.playground/pages/composant/<nomComposant>.vue` (camelCase)
|
**Fichier :** `.playground/pages/composant/<categorie>/<nomComposant>.vue` (camelCase, dans le sous-dossier de catégorie)
|
||||||
|
|
||||||
La page est auto-détectée par `index.vue` via `import.meta.glob`. Inclure des variantes représentatives dans une grille :
|
La page devient automatiquement une route Nuxt (`/composant/<categorie>/<nomComposant>`) et hérite du layout `default` (qui affiche la `MalioSidebar`). **Ajouter ensuite le lien dans la nav centralisée** `.playground/playground.nav.ts` : insérer un `{label, to}` dans la section appropriée (ou créer une nouvelle section), où `to` = `/composant/<categorie>/<nomComposant>`. Inclure des variantes représentatives dans une grille :
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
@@ -216,7 +216,7 @@ Cette section est alimentée au fur et à mesure des retours utilisateur et des
|
|||||||
|--------|----------|
|
|--------|----------|
|
||||||
| Stub IconifyIcon ne fonctionne pas dans les tests | Utiliser `findComponent(IconifyIcon)` avec l'import réel pour tester les props |
|
| Stub IconifyIcon ne fonctionne pas dans les tests | Utiliser `findComponent(IconifyIcon)` avec l'import réel pour tester les props |
|
||||||
| Oubli de `inheritAttrs: false` | Toujours dans `defineOptions` — sinon les attrs se dupliquent |
|
| Oubli de `inheritAttrs: false` | Toujours dans `defineOptions` — sinon les attrs se dupliquent |
|
||||||
| Page playground non détectée | Vérifier le nom du fichier en camelCase dans `.playground/pages/composant/` |
|
| Composant absent de la sidebar du playground | Ajouter son entrée `{label, to}` dans `.playground/playground.nav.ts` (la page n'est plus auto-découverte) |
|
||||||
| Padding input pas ajusté avec icône | Ajouter `!pr-10` (ou équivalent) quand une icône est présente à droite |
|
| Padding input pas ajusté avec icône | Ajouter `!pr-10` (ou équivalent) quand une icône est présente à droite |
|
||||||
| Story sans initial state | Toujours initialiser les `ref` avec des valeurs pour que les variantes soient visibles dès le chargement |
|
| Story sans initial state | Toujours initialiser les `ref` avec des valeurs pour que les variantes soient visibles dès le chargement |
|
||||||
| CHANGELOG oublié | Toujours ajouter la ligne dans `### Added` avant de commit |
|
| CHANGELOG oublié | Toujours ajouter la ligne dans `### Added` avant de commit |
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
<template>
|
||||||
|
<div class="flex h-screen">
|
||||||
|
<MalioSidebar :sections="navSections">
|
||||||
|
<template #logo>
|
||||||
|
<NuxtLink to="/">
|
||||||
|
<img src="/LOGO_MALIO.png" alt="Malio">
|
||||||
|
</NuxtLink>
|
||||||
|
</template>
|
||||||
|
<template #logo-collapsed>
|
||||||
|
<NuxtLink to="/">
|
||||||
|
<img src="/LOGO_MALIO_COLLAPSED.png" alt="Malio">
|
||||||
|
</NuxtLink>
|
||||||
|
</template>
|
||||||
|
</MalioSidebar>
|
||||||
|
|
||||||
|
<main class="flex-1 overflow-y-auto p-6">
|
||||||
|
<slot />
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {navSections} from '../playground.nav'
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Multiple (filtres) — défaut</h2>
|
||||||
|
<MalioAccordion v-model="multiple">
|
||||||
|
<MalioAccordionItem title="Prix" value="prix">
|
||||||
|
<p>Slider de prix ici…</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Catégorie" value="cat">
|
||||||
|
<p>Liste de checkboxes ici…</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Marque" value="marque">
|
||||||
|
<p>Recherche + liste ici…</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
<p class="mt-2 text-sm text-gray-500">Ouverts : {{ multiple }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Single (FAQ)</h2>
|
||||||
|
<MalioAccordion v-model="single" mode="single">
|
||||||
|
<MalioAccordionItem title="Question 1" value="q1">
|
||||||
|
<p>Réponse 1</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Question 2" value="q2">
|
||||||
|
<p>Réponse 2</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
<p class="mt-2 text-sm text-gray-500">Ouvert : {{ single }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Non contrôlé + defaultOpen</h2>
|
||||||
|
<MalioAccordion>
|
||||||
|
<MalioAccordionItem title="Section A" value="a" :default-open="true">
|
||||||
|
<p>Ouverte au montage</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Section B" value="b">
|
||||||
|
<p>Fermée au montage</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Section désactivée</h2>
|
||||||
|
<MalioAccordion>
|
||||||
|
<MalioAccordionItem title="Active" value="ok">
|
||||||
|
<p>Contenu accessible</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Désactivée" value="ko" :disabled="true">
|
||||||
|
<p>Inaccessible</p>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
const multiple = ref<string[]>(['prix'])
|
||||||
|
const single = ref('q1')
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,132 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">MalioDate</h1>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[480px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Large (480px)</h2>
|
||||||
|
<MalioDate
|
||||||
|
v-model="value"
|
||||||
|
label="Date de naissance"
|
||||||
|
hint="Clique pour ouvrir le calendrier"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur (ISO) : <code>{{ value ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDate
|
||||||
|
v-model="editableValue"
|
||||||
|
label="Date (saisie clavier)"
|
||||||
|
editable
|
||||||
|
hint="Tape JJ/MM/AAAA ou utilise le calendrier"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur éditable (ISO) : <code>{{ editableValue ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded bg-m-primary px-3 py-1.5 text-white"
|
||||||
|
@click="value = '2026-12-25'"
|
||||||
|
>
|
||||||
|
Forcer le 25/12/2026
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded border px-3 py-1.5"
|
||||||
|
@click="value = null"
|
||||||
|
>
|
||||||
|
Réinitialiser
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">ERP (396px)</h2>
|
||||||
|
<MalioDate
|
||||||
|
v-model="erpValue"
|
||||||
|
label="Date du rendez-vous"
|
||||||
|
hint="Largeur cible ERP"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur (ISO) : <code>{{ erpValue ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDate
|
||||||
|
v-model="bounded"
|
||||||
|
label="Date bornée"
|
||||||
|
:min="todayIso"
|
||||||
|
:max="maxIso"
|
||||||
|
hint="Entre aujourd'hui et +30 jours"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioDate
|
||||||
|
label="Date de naissance (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioDate
|
||||||
|
v-model="readonlyFilledDate"
|
||||||
|
label="Date de naissance (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">markedDates + @month-change</h2>
|
||||||
|
<MalioDate
|
||||||
|
v-model="markedValue"
|
||||||
|
label="Calendrier avec statuts par jour"
|
||||||
|
hint="Jours verts = validés, rouges = à corriger"
|
||||||
|
:marked-dates="markedDates"
|
||||||
|
@month-change="onMonthChange"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Mois affiché : <code>{{ shownMonth }}</code></p>
|
||||||
|
<p class="mt-1 text-m-success">● success : {{ successDays.join(', ') }}</p>
|
||||||
|
<p class="text-m-danger">● danger : {{ dangerDays.join(', ') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref} from 'vue'
|
||||||
|
|
||||||
|
const pad = (n: number) => String(n).padStart(2, '0')
|
||||||
|
const toIso = (d: Date) => `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}`
|
||||||
|
const now = new Date()
|
||||||
|
const todayIso = toIso(now)
|
||||||
|
const maxIso = toIso(new Date(now.getTime() + 30 * 86400000))
|
||||||
|
|
||||||
|
const readonlyFilledDate = ref<string | null>('2026-06-15')
|
||||||
|
const value = ref<string | null>(null)
|
||||||
|
const erpValue = ref<string | null>(null)
|
||||||
|
const bounded = ref<string | null>(null)
|
||||||
|
const editableValue = ref<string | null>(null)
|
||||||
|
|
||||||
|
// Démo markedDates : quelques jours du mois courant marqués success / danger.
|
||||||
|
const ym = `${now.getFullYear()}-${pad(now.getMonth() + 1)}`
|
||||||
|
const successDays = [`${ym}-05`, `${ym}-06`, `${ym}-12`]
|
||||||
|
const dangerDays = [`${ym}-09`, `${ym}-20`]
|
||||||
|
const markedDates = computed<Record<string, 'success' | 'danger'>>(() => ({
|
||||||
|
...Object.fromEntries(successDays.map(d => [d, 'success' as const])),
|
||||||
|
...Object.fromEntries(dangerDays.map(d => [d, 'danger' as const])),
|
||||||
|
}))
|
||||||
|
const markedValue = ref<string | null>(null)
|
||||||
|
const monthsLong = ['janvier', 'février', 'mars', 'avril', 'mai', 'juin',
|
||||||
|
'juillet', 'août', 'septembre', 'octobre', 'novembre', 'décembre']
|
||||||
|
const shownMonth = ref('—')
|
||||||
|
const onMonthChange = ({month, year}: {month: number, year: number}) => {
|
||||||
|
shownMonth.value = `${monthsLong[month]} ${year} (month=${month})`
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">MalioDateRange</h1>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[480px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Large (480px)</h2>
|
||||||
|
<MalioDateRange
|
||||||
|
v-model="value"
|
||||||
|
label="Période"
|
||||||
|
hint="Clique deux fois pour définir une plage"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Début : <code>{{ value?.start ?? 'null' }}</code></p>
|
||||||
|
<p>Fin : <code>{{ value?.end ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded bg-m-primary px-3 py-1.5 text-white"
|
||||||
|
@click="value = {start: '2026-12-20', end: '2026-12-31'}"
|
||||||
|
>
|
||||||
|
Forcer 20→31/12/2026
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded border px-3 py-1.5"
|
||||||
|
@click="value = null"
|
||||||
|
>
|
||||||
|
Réinitialiser
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">ERP (396px)</h2>
|
||||||
|
<MalioDateRange
|
||||||
|
v-model="erpValue"
|
||||||
|
label="Période"
|
||||||
|
hint="Largeur cible ERP"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Début : <code>{{ erpValue?.start ?? 'null' }}</code></p>
|
||||||
|
<p>Fin : <code>{{ erpValue?.end ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDateRange
|
||||||
|
v-model="bounded"
|
||||||
|
label="Plage bornée"
|
||||||
|
:min="todayIso"
|
||||||
|
:max="maxIso"
|
||||||
|
hint="Entre aujourd'hui et +30 jours"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
type RangeValue = {start: string; end: string}
|
||||||
|
|
||||||
|
const pad = (n: number) => String(n).padStart(2, '0')
|
||||||
|
const toIso = (d: Date) => `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}`
|
||||||
|
const now = new Date()
|
||||||
|
const todayIso = toIso(now)
|
||||||
|
const maxIso = toIso(new Date(now.getTime() + 30 * 86400000))
|
||||||
|
|
||||||
|
const value = ref<RangeValue | null>(null)
|
||||||
|
const erpValue = ref<RangeValue | null>(null)
|
||||||
|
const bounded = ref<RangeValue | null>(null)
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">MalioDateWeek</h1>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[480px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Large (480px)</h2>
|
||||||
|
<MalioDateWeek
|
||||||
|
v-model="value"
|
||||||
|
label="Semaine"
|
||||||
|
hint="Clique un jour ou un n° de semaine"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur : <code>{{ value ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded bg-m-primary px-3 py-1.5 text-white"
|
||||||
|
@click="value = '2026-W52'"
|
||||||
|
>
|
||||||
|
Forcer 2026-W52
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded border px-3 py-1.5"
|
||||||
|
@click="value = null"
|
||||||
|
>
|
||||||
|
Réinitialiser
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">ERP (396px)</h2>
|
||||||
|
<MalioDateWeek
|
||||||
|
v-model="erpValue"
|
||||||
|
label="Semaine"
|
||||||
|
hint="Largeur cible ERP"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur : <code>{{ erpValue ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDateWeek
|
||||||
|
v-model="bounded"
|
||||||
|
label="Semaine bornée"
|
||||||
|
:min="todayIso"
|
||||||
|
:max="maxIso"
|
||||||
|
hint="Entre aujourd'hui et +60 jours"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
const pad = (n: number) => String(n).padStart(2, '0')
|
||||||
|
const toIso = (d: Date) => `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}`
|
||||||
|
const now = new Date()
|
||||||
|
const todayIso = toIso(now)
|
||||||
|
const maxIso = toIso(new Date(now.getTime() + 60 * 86400000))
|
||||||
|
|
||||||
|
const value = ref<string | null>(null)
|
||||||
|
const erpValue = ref<string | null>(null)
|
||||||
|
const bounded = ref<string | null>(null)
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,84 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">MalioDateTime</h1>
|
||||||
|
|
||||||
|
<div class="flex flex-wrap items-start gap-10">
|
||||||
|
<div class="w-[480px] space-y-3">
|
||||||
|
<h2 class="font-semibold">Large (480px)</h2>
|
||||||
|
<MalioDateTime
|
||||||
|
v-model="value"
|
||||||
|
label="Date et heure du rendez-vous"
|
||||||
|
hint="Choisis un jour puis une heure"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur (ISO naïf) : <code>{{ value ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDateTime
|
||||||
|
v-model="editableValue"
|
||||||
|
label="Date et heure (saisie clavier)"
|
||||||
|
editable
|
||||||
|
hint="Tape JJ/MM/AAAA HH:MM ou utilise le calendrier"
|
||||||
|
@update:valid="editableValid = $event"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur éditable (ISO naïf) : <code>{{ editableValue ?? 'null' }}</code></p>
|
||||||
|
<p>
|
||||||
|
Saisie valide :
|
||||||
|
<code :class="editableValid ? 'text-m-success' : 'text-m-danger'">{{ editableValid }}</code>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded bg-m-primary px-3 py-1.5 text-white"
|
||||||
|
@click="value = '2026-12-25T09:30:00'"
|
||||||
|
>
|
||||||
|
Forcer le 25/12/2026 09:30
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="rounded border px-3 py-1.5"
|
||||||
|
@click="value = null"
|
||||||
|
>
|
||||||
|
Réinitialiser
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="w-[396px] space-y-3">
|
||||||
|
<h2 class="font-semibold">ERP (396px)</h2>
|
||||||
|
<MalioDateTime
|
||||||
|
v-model="erpValue"
|
||||||
|
label="Date et heure du rendez-vous"
|
||||||
|
hint="Largeur cible ERP"
|
||||||
|
/>
|
||||||
|
<div class="rounded border p-3 text-sm">
|
||||||
|
<p>Valeur (ISO naïf) : <code>{{ erpValue ?? 'null' }}</code></p>
|
||||||
|
</div>
|
||||||
|
<MalioDateTime
|
||||||
|
v-model="bounded"
|
||||||
|
label="Créneau borné"
|
||||||
|
:min="todayIso"
|
||||||
|
:max="maxIso"
|
||||||
|
hint="Entre aujourd'hui et +30 jours"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
const pad = (n: number) => String(n).padStart(2, '0')
|
||||||
|
const toIso = (d: Date) => `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T00:00:00`
|
||||||
|
const now = new Date()
|
||||||
|
const todayIso = toIso(now)
|
||||||
|
const maxIso = toIso(new Date(now.getTime() + 30 * 86400000))
|
||||||
|
|
||||||
|
const value = ref<string | null>(null)
|
||||||
|
const erpValue = ref<string | null>(null)
|
||||||
|
const bounded = ref<string | null>('2026-05-20T14:30:00')
|
||||||
|
const editableValue = ref<string | null>(null)
|
||||||
|
const editableValid = ref(true)
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,294 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">Champs désactivés (disabled)</h1>
|
||||||
|
<p class="text-sm text-m-muted">
|
||||||
|
Tous les champs de formulaire dans leur état <code>disabled</code>, vides puis remplis.
|
||||||
|
Règles : texte + label grisés, <code>cursor-not-allowed</code>, et <strong>aucune affordance
|
||||||
|
interactive</strong> (pas de bouton « + », pas de croix « x », pas de chevron, pas d'œil ;
|
||||||
|
tags et valeurs grisés).
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2 xl:grid-cols-3">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputText</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputText
|
||||||
|
label="Référence (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputText
|
||||||
|
model-value="Commande #A-2048"
|
||||||
|
label="Référence (rempli)"
|
||||||
|
icon-name="mdi:lock-outline"
|
||||||
|
icon-size="20"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputEmail (addable → pas de « + »)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email (vide)"
|
||||||
|
:addable="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="contact@malio.fr"
|
||||||
|
label="Adresse email (rempli)"
|
||||||
|
:addable="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputPhone (addable → pas de « + »)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone (vide)"
|
||||||
|
:addable="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="+33 6 12 34 56 78"
|
||||||
|
label="Téléphone (rempli)"
|
||||||
|
:addable="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputPassword (pas d'œil)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputPassword
|
||||||
|
label="Mot de passe (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputPassword
|
||||||
|
model-value="motdepasse123"
|
||||||
|
label="Mot de passe (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputAmount</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputAmount
|
||||||
|
label="Montant (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputAmount
|
||||||
|
model-value="1250.00"
|
||||||
|
label="Montant (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputNumber</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputNumber
|
||||||
|
label="Quantité (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputNumber
|
||||||
|
model-value="42"
|
||||||
|
label="Quantité (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputTextArea</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputTextArea
|
||||||
|
label="Description (vide)"
|
||||||
|
:size="3"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputTextArea
|
||||||
|
model-value="Ce texte est désactivé et ne peut pas être modifié."
|
||||||
|
label="Description (rempli)"
|
||||||
|
:size="3"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputUpload (pas de croix)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputUpload
|
||||||
|
label="Fichier (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputUpload
|
||||||
|
model-value="document.pdf"
|
||||||
|
label="Fichier (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputAutocomplete (pas de chevron)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
label="Pays (vide)"
|
||||||
|
:options="countryOptions"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="de"
|
||||||
|
label="Pays (rempli)"
|
||||||
|
:options="countryOptions"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioSelect (pas de chevron)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioSelect
|
||||||
|
label="Catégorie (vide)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
empty-option-label="Aucune sélection"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
:model-value="'a'"
|
||||||
|
label="Catégorie (rempli)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
empty-option-label="Aucune sélection"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioSelectCheckbox — version tags (grisés)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
label="Catégories (vide)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
:display-tag="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
:model-value="['a', 'b', 'c']"
|
||||||
|
label="Catégories (rempli)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
:display-tag="true"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDate (pas de croix)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDate
|
||||||
|
label="Date de naissance (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioDate
|
||||||
|
model-value="2026-06-15"
|
||||||
|
label="Date de naissance (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateTime</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateTime
|
||||||
|
label="Date et heure (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioDateTime
|
||||||
|
model-value="2026-12-25T09:30:00"
|
||||||
|
label="Date et heure (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateRange</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateRange
|
||||||
|
label="Période (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioDateRange
|
||||||
|
:model-value="rangeValue"
|
||||||
|
label="Période (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateWeek</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateWeek
|
||||||
|
label="Semaine (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioDateWeek
|
||||||
|
model-value="2026-W52"
|
||||||
|
label="Semaine (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioTimePicker (pas de croix)</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioTimePicker
|
||||||
|
label="Heure (vide)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
<MalioTimePicker
|
||||||
|
model-value="14:30"
|
||||||
|
label="Heure (rempli)"
|
||||||
|
:disabled="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
type Option = {label: string; value: string | number}
|
||||||
|
|
||||||
|
const countryOptions: Option[] = [
|
||||||
|
{label: 'France', value: 'fr'},
|
||||||
|
{label: 'Belgique', value: 'be'},
|
||||||
|
{label: 'Canada', value: 'ca'},
|
||||||
|
{label: 'Suisse', value: 'ch'},
|
||||||
|
{label: 'Luxembourg', value: 'lu'},
|
||||||
|
{label: 'Allemagne', value: 'de'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const categoryOptions: Option[] = [
|
||||||
|
{label: 'Catégorie A', value: 'a'},
|
||||||
|
{label: 'Catégorie B', value: 'b'},
|
||||||
|
{label: 'Catégorie C', value: 'c'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const rangeValue = ref<{start: string; end: string}>({start: '2026-12-20', end: '2026-12-31'})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,276 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6 p-4">
|
||||||
|
<h1 class="text-2xl font-bold">Champs en lecture seule (readonly)</h1>
|
||||||
|
<p class="text-sm text-m-muted">
|
||||||
|
Tous les champs de formulaire dans leur état <code>readonly</code>, vides puis remplis.
|
||||||
|
Règles : bordure noire même vide, label et icône gris quand vide → noir quand rempli,
|
||||||
|
pas de focus bleu ni de grossissement.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2 xl:grid-cols-3">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputText</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputText
|
||||||
|
label="Référence (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputText
|
||||||
|
model-value="Commande #A-2048"
|
||||||
|
label="Référence (rempli)"
|
||||||
|
icon-name="mdi:lock-outline"
|
||||||
|
icon-size="20"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputEmail</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="contact@malio.fr"
|
||||||
|
label="Adresse email (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputAmount</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputAmount
|
||||||
|
label="Montant (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputAmount
|
||||||
|
model-value="1250.00"
|
||||||
|
label="Montant (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputAutocomplete</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
label="Pays (vide)"
|
||||||
|
:options="countryOptions"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="de"
|
||||||
|
label="Pays (rempli)"
|
||||||
|
icon-name="mdi:lock-outline"
|
||||||
|
icon-position="left"
|
||||||
|
:options="countryOptions"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputPassword</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputPassword
|
||||||
|
label="Mot de passe (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputPassword
|
||||||
|
model-value="motdepasse123"
|
||||||
|
label="Mot de passe (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputTextArea</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputTextArea
|
||||||
|
label="Description (vide)"
|
||||||
|
:size="3"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputTextArea
|
||||||
|
model-value="Ce texte est en lecture seule et ne peut pas être modifié."
|
||||||
|
label="Description (rempli)"
|
||||||
|
:size="3"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputPhone</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="+33 6 12 34 56 78"
|
||||||
|
label="Téléphone (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioInputUpload</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioInputUpload
|
||||||
|
label="Fichier (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioInputUpload
|
||||||
|
model-value="document.pdf"
|
||||||
|
label="Fichier (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioSelect</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioSelect
|
||||||
|
label="Catégorie (readonly vide)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
:model-value="'a'"
|
||||||
|
label="Catégorie (readonly rempli)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioSelectCheckbox</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
label="Catégories (readonly vide)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
:display-tag="true"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
:model-value="['a']"
|
||||||
|
label="Catégories (readonly rempli)"
|
||||||
|
:options="categoryOptions"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:display-tag="true"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDate</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDate
|
||||||
|
label="Date de naissance (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioDate
|
||||||
|
model-value="2026-06-15"
|
||||||
|
label="Date de naissance (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateTime</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateTime
|
||||||
|
label="Date et heure (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioDateTime
|
||||||
|
model-value="2026-12-25T09:30:00"
|
||||||
|
label="Date et heure (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateRange</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateRange
|
||||||
|
label="Période (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioDateRange
|
||||||
|
:model-value="rangeValue"
|
||||||
|
label="Période (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioDateWeek</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioDateWeek
|
||||||
|
label="Semaine (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioDateWeek
|
||||||
|
model-value="2026-W52"
|
||||||
|
label="Semaine (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">MalioTimePicker</h2>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<MalioTimePicker
|
||||||
|
label="Heure (vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
<MalioTimePicker
|
||||||
|
model-value="14:30"
|
||||||
|
label="Heure (rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
type Option = {label: string; value: string | number}
|
||||||
|
|
||||||
|
const countryOptions: Option[] = [
|
||||||
|
{label: 'France', value: 'fr'},
|
||||||
|
{label: 'Belgique', value: 'be'},
|
||||||
|
{label: 'Canada', value: 'ca'},
|
||||||
|
{label: 'Suisse', value: 'ch'},
|
||||||
|
{label: 'Luxembourg', value: 'lu'},
|
||||||
|
{label: 'Allemagne', value: 'de'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const categoryOptions: Option[] = [
|
||||||
|
{label: 'Catégorie A', value: 'a'},
|
||||||
|
{label: 'Catégorie B', value: 'b'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const rangeValue = ref<{start: string; end: string}>({start: '2026-12-20', end: '2026-12-31'})
|
||||||
|
</script>
|
||||||
@@ -1,48 +1,83 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
|
|
||||||
const drawerDefault = ref(false)
|
const drawerRight = ref(false)
|
||||||
const drawerNoClose = ref(false)
|
const drawerLeft = ref(false)
|
||||||
const drawerCustomWidth = ref(false)
|
const drawerForm = ref(false)
|
||||||
const drawerWithForm = ref(false)
|
const drawerFixedFooter = ref(false)
|
||||||
|
const drawerNoDismiss = ref(false)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
<div class="rounded-lg border p-6">
|
<div class="rounded-lg border p-6">
|
||||||
<h2 class="mb-6 text-xl font-bold">Drawer simple</h2>
|
<h2 class="mb-6 text-xl font-bold">Drawer droite (défaut)</h2>
|
||||||
<MalioButton label="Ouvrir le drawer" @click="drawerDefault = true" />
|
<MalioButton label="Ouvrir à droite" @click="drawerRight = true" />
|
||||||
<MalioDrawer v-model="drawerDefault" title="Titre du drawer">
|
<MalioDrawer v-model="drawerRight">
|
||||||
<p class="text-m-text">Contenu du drawer. Lorem ipsum dolor sit amet, consectetur adipiscing elit.</p>
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Détails</h2>
|
||||||
|
</template>
|
||||||
|
<p class="text-m-text">Contenu du drawer. Échap, clic backdrop et croix le ferment.</p>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-6">
|
<div class="rounded-lg border p-6">
|
||||||
<h2 class="mb-6 text-xl font-bold">Sans bouton fermer</h2>
|
<h2 class="mb-6 text-xl font-bold">Drawer gauche</h2>
|
||||||
<MalioButton label="Ouvrir le drawer" variant="secondary" @click="drawerNoClose = true" />
|
<MalioButton label="Ouvrir à gauche" variant="secondary" @click="drawerLeft = true" />
|
||||||
<MalioDrawer v-model="drawerNoClose" title="Sans croix" :show-close="false">
|
<MalioDrawer v-model="drawerLeft" side="left">
|
||||||
<p class="text-m-text">Ce drawer n'a pas de bouton fermer. Cliquez sur le backdrop pour fermer.</p>
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Navigation</h2>
|
||||||
|
</template>
|
||||||
|
<p class="text-m-text">Ce drawer glisse depuis la gauche.</p>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-6">
|
<div class="rounded-lg border p-6">
|
||||||
<h2 class="mb-6 text-xl font-bold">Largeur personnalisée</h2>
|
<h2 class="mb-6 text-xl font-bold">Avec footer d'actions</h2>
|
||||||
<MalioButton label="Ouvrir le drawer large" variant="tertiary" @click="drawerCustomWidth = true" />
|
<MalioButton label="Ouvrir le formulaire" variant="tertiary" @click="drawerForm = true" />
|
||||||
<MalioDrawer v-model="drawerCustomWidth" title="Drawer large" drawer-class="max-w-2xl">
|
<MalioDrawer v-model="drawerForm" drawer-class="max-w-lg">
|
||||||
<p class="text-m-text">Ce drawer utilise une largeur personnalisée via drawerClass.</p>
|
<template #header>
|
||||||
</MalioDrawer>
|
<h2 class="text-[24px] font-bold text-black">Nouveau contact</h2>
|
||||||
</div>
|
</template>
|
||||||
|
<div class="flex flex-col gap-4 py-2">
|
||||||
<div class="rounded-lg border p-6">
|
|
||||||
<h2 class="mb-6 text-xl font-bold">Avec formulaire</h2>
|
|
||||||
<MalioButton label="Ouvrir le formulaire" variant="danger" @click="drawerWithForm = true" />
|
|
||||||
<MalioDrawer v-model="drawerWithForm" title="Formulaire">
|
|
||||||
<div class="flex flex-col gap-4">
|
|
||||||
<MalioInputText label="Nom" />
|
<MalioInputText label="Nom" />
|
||||||
<MalioInputText label="Prénom" />
|
<MalioInputText label="Prénom" />
|
||||||
<MalioInputText label="Email" />
|
<MalioInputText label="Email" />
|
||||||
<MalioButton label="Enregistrer" button-class="w-full" @click="drawerWithForm = false" />
|
|
||||||
</div>
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Annuler" variant="secondary" button-class="flex-1" @click="drawerForm = false" />
|
||||||
|
<MalioButton label="Enregistrer" button-class="flex-1" @click="drawerForm = false" />
|
||||||
|
</template>
|
||||||
|
</MalioDrawer>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Footer fixe avec contenu long</h2>
|
||||||
|
<MalioButton label="Ouvrir (contenu long)" variant="tertiary" @click="drawerFixedFooter = true" />
|
||||||
|
<MalioDrawer v-model="drawerFixedFooter">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Conditions</h2>
|
||||||
|
</template>
|
||||||
|
<!-- Pas de hack : le footer est hors zone scrollable, seul le body défile -->
|
||||||
|
<div class="flex flex-col gap-4">
|
||||||
|
<p v-for="n in 12" :key="n" class="text-m-text">
|
||||||
|
Paragraphe {{ n }} — contenu long pour forcer le scroll et montrer que seul le body défile, le footer restant fixé en bas.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Accepter" button-class="w-full" @click="drawerFixedFooter = false" />
|
||||||
|
</template>
|
||||||
|
</MalioDrawer>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Non dismissable (croix uniquement)</h2>
|
||||||
|
<MalioButton label="Ouvrir" variant="danger" @click="drawerNoDismiss = true" />
|
||||||
|
<MalioDrawer v-model="drawerNoDismiss" :dismissable="false" :close-on-escape="false">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Action requise</h2>
|
||||||
|
</template>
|
||||||
|
<p class="text-m-text">Ni le backdrop ni Échap ne ferment ce drawer. Utilisez la croix.</p>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,88 @@
|
|||||||
|
<template>
|
||||||
|
<div class="flex justify-center">
|
||||||
|
<div class="w-[1348px]">
|
||||||
|
<div class="flex items-center justify-between mt-[46px]">
|
||||||
|
<div class="flex items-center gap-3">
|
||||||
|
<MalioButtonIcon
|
||||||
|
icon="mdi:arrow-left-bold"
|
||||||
|
icon-size="24"
|
||||||
|
aria-label="Précédent"
|
||||||
|
variant="ghost"
|
||||||
|
/>
|
||||||
|
<h1 class="text-[32px] text-m-primary font-bold">Filtres</h1>
|
||||||
|
</div>
|
||||||
|
<MalioButton
|
||||||
|
label="Filtres"
|
||||||
|
variant="tertiary"
|
||||||
|
icon-name="mdi:tune"
|
||||||
|
icon-position="left"
|
||||||
|
button-class="w-[184px] px-2 py-2 justify-start text-black gap-4"
|
||||||
|
@click="drawerOpen = true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<MalioDrawer
|
||||||
|
v-model="drawerOpen"
|
||||||
|
side="right"
|
||||||
|
drawer-class="max-w-[450px]"
|
||||||
|
body-class="p-0"
|
||||||
|
footer-class="justify-between gap-4 py-7"
|
||||||
|
>
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold uppercase">Filtres</h2>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<MalioAccordion>
|
||||||
|
<MalioAccordionItem title="Type de camion" value="camion">
|
||||||
|
<div class="flex flex-col gap-6">
|
||||||
|
<MalioCheckbox v-model="semiBenne" label="Semi Benne" />
|
||||||
|
<MalioCheckbox v-model="benne" label="Benne" />
|
||||||
|
</div>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
|
||||||
|
<MalioAccordionItem title="Date à Date" value="date">
|
||||||
|
<div class="grid grid-cols-[auto_1fr] items-center gap-x-3 gap-y-4">
|
||||||
|
<span>Du</span>
|
||||||
|
<MalioDate v-model="dateDebut"/>
|
||||||
|
<span>Au</span>
|
||||||
|
<MalioDate v-model="dateFin"/>
|
||||||
|
</div>
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton
|
||||||
|
label="Réinitialiser"
|
||||||
|
variant="tertiary"
|
||||||
|
button-class="w-m-btn-action"
|
||||||
|
@click="resetFiltres"
|
||||||
|
/>
|
||||||
|
<MalioButton
|
||||||
|
label="Voir les résultats"
|
||||||
|
variant="primary"
|
||||||
|
button-class="w-[170px]"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</MalioDrawer>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
const drawerOpen = ref(false)
|
||||||
|
|
||||||
|
const semiBenne = ref(false)
|
||||||
|
const benne = ref(false)
|
||||||
|
|
||||||
|
const dateDebut = ref<string | null>(null)
|
||||||
|
const dateFin = ref<string | null>(null)
|
||||||
|
|
||||||
|
function resetFiltres() {
|
||||||
|
semiBenne.value = false
|
||||||
|
benne.value = false
|
||||||
|
dateDebut.value = null
|
||||||
|
dateFin.value = null
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,323 @@
|
|||||||
|
<template>
|
||||||
|
<div class="flex justify-center">
|
||||||
|
<div class="w-[1348px]">
|
||||||
|
<div class="flex gap-3 mt-[46px]">
|
||||||
|
<MalioButtonIcon
|
||||||
|
icon="mdi:arrow-left-bold"
|
||||||
|
icon-size="24"
|
||||||
|
aria-label="Précédent"
|
||||||
|
variant="ghost"
|
||||||
|
/>
|
||||||
|
<h1 class="text-[32px] text-m-primary font-bold">Ajouter un client</h1>
|
||||||
|
</div>
|
||||||
|
<div class="mt-[48px] grid grid-cols-3 gap-x-[80px] gap-y-5">
|
||||||
|
<MalioInputText
|
||||||
|
label="Nom du client (Entreprise)"
|
||||||
|
/>
|
||||||
|
<MalioInputText
|
||||||
|
label="Nom du contact principal"
|
||||||
|
/>
|
||||||
|
<MalioInputText
|
||||||
|
label="Prénom du contact principal"
|
||||||
|
/>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
v-model="multiselectValue"
|
||||||
|
error="test"
|
||||||
|
label="Catégorie"
|
||||||
|
:options="[
|
||||||
|
{label: 'Catégorie 1', value: 'Catégorie 1'},
|
||||||
|
{label: 'Catégorie 2', value: 'Catégorie 2'}
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
<MalioInputPhone
|
||||||
|
v-for="(_, index) in phones"
|
||||||
|
:key="index"
|
||||||
|
v-model="phones[index]"
|
||||||
|
label="Téléphone"
|
||||||
|
add-icon-name="mdi:plus"
|
||||||
|
:addable="phones.length === 1"
|
||||||
|
@add="addPhoneInput"
|
||||||
|
/>
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Email"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="distributeur"
|
||||||
|
value=""
|
||||||
|
label="Distributeur / Courtier"
|
||||||
|
:options="[
|
||||||
|
{label: 'Dépend du distributeur', value: 'Dépend du distributeur'},
|
||||||
|
{label: 'Distributeur', value: 'Distributeur'},
|
||||||
|
{label: 'Courtier', value: 'Courtier'},
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="nomCourtier"
|
||||||
|
value=""
|
||||||
|
label="Nom du courtier"
|
||||||
|
:options="[
|
||||||
|
{label: 'Nom 1', value: 'Nom 1'}
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="nomDistributeur"
|
||||||
|
value=""
|
||||||
|
label="Nom du distributeur"
|
||||||
|
:options="[
|
||||||
|
{label: 'Nom 1', value: 'Nom 1'}
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
<MalioCheckbox label="Prestation de triage" groupClass="self-center"/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mt-12 flex justify-center">
|
||||||
|
<MalioButton label="Valider" variant="primary"/>
|
||||||
|
</div>
|
||||||
|
<div class="mt-[60px]">
|
||||||
|
<MalioTabList :tabs="tabs" v-model="tabsValue">
|
||||||
|
<template #information>
|
||||||
|
<div class="grid grid-cols-3 gap-x-[80px] gap-y-5 mt-12 shadow-[0_4px_4px_0_rgba(0,0,0,0.25)] py-4 pl-[28px] pr-[60px]">
|
||||||
|
<MalioInputTextArea label="Descritpion" resize="none" groupClass="row-span-2" textInput="h-full"/>
|
||||||
|
<MalioInputText v-model="concurrent" label="Concurrent"/>
|
||||||
|
<MalioDate
|
||||||
|
v-model="dateCreation"
|
||||||
|
label="Date création"
|
||||||
|
/>
|
||||||
|
<MalioInputText label="Nombre de salariés" />
|
||||||
|
<MalioInputAmount label="CA"/>
|
||||||
|
<MalioInputText label="Dirigeant" />
|
||||||
|
<MalioInputText label="Résultat" />
|
||||||
|
</div>
|
||||||
|
<div class="mt-12 flex justify-center">
|
||||||
|
<MalioButton label="Valider" variant="primary"/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #adresses>
|
||||||
|
<div class="relative grid grid-cols-3 gap-x-[80px] gap-y-5 mt-12 bg-white shadow-[0_4px_4px_0_rgba(0,0,0,0.25)] py-4 pl-[28px] pr-[60px]">
|
||||||
|
<MalioButtonIcon
|
||||||
|
icon="mdi:delete-outline"
|
||||||
|
aria-label="Supprimer l'adresse"
|
||||||
|
variant="ghost"
|
||||||
|
button-class="absolute top-3 right-3"
|
||||||
|
@click="onDeleteAdresse"
|
||||||
|
/>
|
||||||
|
<MalioCheckbox label="Prospect" groupClass="self-center"/>
|
||||||
|
<MalioCheckbox label="Adresse de livraison" groupClass="self-center"/>
|
||||||
|
<MalioCheckbox label="Facturation" groupClass="self-center"/>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
v-model="multiselectValue"
|
||||||
|
label="Catégorie"
|
||||||
|
:options="[
|
||||||
|
{label: 'Catégorie 1', value: 'Catégorie 1'},
|
||||||
|
{label: 'Catégorie 2', value: 'Catégorie 2'}
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
<MalioSelect
|
||||||
|
label="Pays"
|
||||||
|
v-model="pays"
|
||||||
|
:options="[
|
||||||
|
{label: 'France', value: 'France'},
|
||||||
|
{label: 'Espagne', value: 'Espagne'}
|
||||||
|
]"/>
|
||||||
|
<MalioInputText v-model="codePostal" label="Code postal" />
|
||||||
|
<MalioSelect
|
||||||
|
v-model="ville"
|
||||||
|
label="Ville"
|
||||||
|
:options="villeOptions"
|
||||||
|
:no-options-text="villeNoOptionsText"
|
||||||
|
/>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="adresse"
|
||||||
|
label="Adresse"
|
||||||
|
:options="adresseOptions"
|
||||||
|
:loading="adresseLoading"
|
||||||
|
:min-search-length="2"
|
||||||
|
:no-results-text="adresseNoResultsText"
|
||||||
|
:min-search-text="adresseMinSearchText"
|
||||||
|
@search="onSearchAdresse"
|
||||||
|
/>
|
||||||
|
<MalioInputText label="Adresse complémentaire"/>
|
||||||
|
<div class="flex justify-between">
|
||||||
|
<MalioCheckbox
|
||||||
|
v-for="dep in departements"
|
||||||
|
:key="dep"
|
||||||
|
v-model="departementsSelected[dep]"
|
||||||
|
:label="dep"
|
||||||
|
group-class="w-auto self-center"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<MalioSelect label="Contact" :options="[]"/>
|
||||||
|
<MalioCheckbox label="Prestation de triage" groupClass="self-center"/>
|
||||||
|
</div>
|
||||||
|
<div class="mt-12 flex justify-center gap-6">
|
||||||
|
<MalioButton label="Nouvelle Adresse" variant="secondary" icon-name="mdi:add-bold" icon-position="left"/>
|
||||||
|
<MalioButton label="Valider" variant="primary"/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</MalioTabList>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref, computed, watch} from 'vue'
|
||||||
|
import MalioDate from "../../../../app/components/malio/date/Date.vue";
|
||||||
|
|
||||||
|
type Commune = {
|
||||||
|
nom: string
|
||||||
|
code: string
|
||||||
|
codesPostaux: string[]
|
||||||
|
}
|
||||||
|
|
||||||
|
type BanFeature = {
|
||||||
|
properties: {
|
||||||
|
label: string
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
housenumber?: string
|
||||||
|
street?: string
|
||||||
|
postcode: string
|
||||||
|
citycode: string
|
||||||
|
city: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const multiselectValue = ref<Array<string | number>>([])
|
||||||
|
const distributeur = ref<string>('')
|
||||||
|
const phones = ref<string[]>([''])
|
||||||
|
const nomDistributeur = ref<string>('')
|
||||||
|
const nomCourtier = ref<string>('')
|
||||||
|
|
||||||
|
function addPhoneInput() {
|
||||||
|
phones.value.push('')
|
||||||
|
}
|
||||||
|
|
||||||
|
function onDeleteAdresse() {
|
||||||
|
console.log('Supprimer cette adresse')
|
||||||
|
}
|
||||||
|
|
||||||
|
const departements = ['86', '17', '82']
|
||||||
|
const departementsSelected = ref<Record<string, boolean>>({86: false, 17: false, 82: false})
|
||||||
|
|
||||||
|
const pays = ref<string>('France')
|
||||||
|
const codePostal = ref<string>('')
|
||||||
|
const ville = ref<string | number | null>(null)
|
||||||
|
const villeOptions = ref<Array<{label: string; value: string}>>([])
|
||||||
|
const villeLoading = ref(false)
|
||||||
|
|
||||||
|
const villeNoOptionsText = computed(() => {
|
||||||
|
if (villeLoading.value) return 'Chargement…'
|
||||||
|
if (!/^\d{5}$/.test(codePostal.value)) return 'Saisir un code postal (5 chiffres)'
|
||||||
|
return 'Aucune ville pour ce code postal'
|
||||||
|
})
|
||||||
|
|
||||||
|
let villeFetchId = 0
|
||||||
|
watch(codePostal, async (cp) => {
|
||||||
|
ville.value = null
|
||||||
|
villeOptions.value = []
|
||||||
|
adresse.value = null
|
||||||
|
adresseOptions.value = []
|
||||||
|
if (!/^\d{5}$/.test(cp)) {
|
||||||
|
villeLoading.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const requestId = ++villeFetchId
|
||||||
|
villeLoading.value = true
|
||||||
|
try {
|
||||||
|
const response = await fetch(`https://geo.api.gouv.fr/communes?codePostal=${cp}`)
|
||||||
|
const data = await response.json() as Commune[]
|
||||||
|
if (requestId !== villeFetchId) return
|
||||||
|
villeOptions.value = data.map(c => ({label: c.nom, value: c.code}))
|
||||||
|
} catch (err) {
|
||||||
|
if (requestId !== villeFetchId) return
|
||||||
|
villeOptions.value = []
|
||||||
|
console.error('Erreur lors du chargement des villes', err)
|
||||||
|
} finally {
|
||||||
|
if (requestId === villeFetchId) villeLoading.value = false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const adresse = ref<string | number | null>(null)
|
||||||
|
const adresseOptions = ref<Array<{label: string; value: string}>>([])
|
||||||
|
const adresseLoading = ref(false)
|
||||||
|
|
||||||
|
const adresseMinSearchText = computed(() => {
|
||||||
|
if (!/^\d{5}$/.test(codePostal.value)) return 'Saisir d\'abord un code postal'
|
||||||
|
return 'Tapez au moins 3 caractères'
|
||||||
|
})
|
||||||
|
const adresseNoResultsText = computed(() => {
|
||||||
|
if (!/^\d{5}$/.test(codePostal.value)) return 'Saisir d\'abord un code postal'
|
||||||
|
return 'Aucune adresse trouvée'
|
||||||
|
})
|
||||||
|
|
||||||
|
let adresseFetchId = 0
|
||||||
|
const onSearchAdresse = async (query: string) => {
|
||||||
|
if (!/^\d{5}$/.test(codePostal.value) || query.length < 3) {
|
||||||
|
adresseOptions.value = []
|
||||||
|
adresseLoading.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const requestId = ++adresseFetchId
|
||||||
|
adresseLoading.value = true
|
||||||
|
try {
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
q: query,
|
||||||
|
postcode: codePostal.value,
|
||||||
|
type: 'housenumber',
|
||||||
|
})
|
||||||
|
const response = await fetch(`https://api-adresse.data.gouv.fr/search/?${params.toString()}`)
|
||||||
|
const data = await response.json() as {features: BanFeature[]}
|
||||||
|
if (requestId !== adresseFetchId) return
|
||||||
|
adresseOptions.value = data.features.map(f => ({
|
||||||
|
label: f.properties.name,
|
||||||
|
value: f.properties.name,
|
||||||
|
}))
|
||||||
|
} catch (err) {
|
||||||
|
if (requestId !== adresseFetchId) return
|
||||||
|
adresseOptions.value = []
|
||||||
|
console.error('Erreur lors du chargement des adresses', err)
|
||||||
|
} finally {
|
||||||
|
if (requestId === adresseFetchId) adresseLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const tabsValue = ref('information')
|
||||||
|
const concurrent = ref('')
|
||||||
|
const dateCreation = ref<string | null>(null)
|
||||||
|
|
||||||
|
const informationValid = computed(() => concurrent.value.trim().length > 0)
|
||||||
|
const adressesValid = computed(() => /^\d{5}$/.test(codePostal.value))
|
||||||
|
|
||||||
|
const tabs = computed(() => [
|
||||||
|
{
|
||||||
|
key: 'information',
|
||||||
|
label: 'Information',
|
||||||
|
icon: 'mdi:account-outline',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'contacts',
|
||||||
|
label: 'Contacts',
|
||||||
|
icon: 'mdi:account-box-plus-outline',
|
||||||
|
disabled: !informationValid.value,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'adresses',
|
||||||
|
label: 'Adresses',
|
||||||
|
icon: 'mdi:map-marker-outline',
|
||||||
|
disabled: !informationValid.value,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'transport',
|
||||||
|
label: 'Transport',
|
||||||
|
icon: 'mdi:truck-delivery-outline',
|
||||||
|
disabled: !informationValid.value || !adressesValid.value,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'comptabilité',
|
||||||
|
label: 'Comptabilité',
|
||||||
|
icon: 'mdi:bank-circle-outline',
|
||||||
|
disabled: !informationValid.value || !adressesValid.value,
|
||||||
|
},
|
||||||
|
])
|
||||||
|
</script>
|
||||||
@@ -14,6 +14,17 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Grand montant (séparateurs)</h2>
|
||||||
|
<MalioInputAmount
|
||||||
|
v-model="bigValue"
|
||||||
|
label="Budget"
|
||||||
|
/>
|
||||||
|
<div class="mt-2 rounded border p-3 text-sm">
|
||||||
|
<p>modelValue émis : <code>{{ bigValue || 'vide' }}</code></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
||||||
<MalioInputAmount
|
<MalioInputAmount
|
||||||
@@ -36,6 +47,23 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputAmount
|
||||||
|
label="Montant (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputAmount
|
||||||
|
v-model="readonlyFilledAmount"
|
||||||
|
label="Montant (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Erreur et succès</h2>
|
<h2 class="mb-4 text-xl font-bold">Erreur et succès</h2>
|
||||||
<div class="mt-4">
|
<div class="mt-4">
|
||||||
@@ -57,4 +85,8 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { ref } from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledAmount = ref('1250.00')
|
||||||
|
const bigValue = ref('1234567.89')
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -0,0 +1,202 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple (statique)</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="simpleValue"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
local-filter
|
||||||
|
/>
|
||||||
|
<p class="mt-2 text-sm text-m-muted">
|
||||||
|
Valeur sélectionnée : <code>{{ simpleValue ?? 'null' }}</code>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec icône à gauche</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="leftIconValue"
|
||||||
|
label="Recherche"
|
||||||
|
icon-name="mdi:magnify"
|
||||||
|
icon-position="left"
|
||||||
|
:options="staticOptions"
|
||||||
|
local-filter
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4 md:col-span-2">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Branché sur une API (simulé)</h2>
|
||||||
|
<p class="mb-3 text-sm text-m-muted">
|
||||||
|
Le parent écoute l'event <code>search</code> et alimente <code>options</code> + <code>loading</code>.
|
||||||
|
Tapez au moins 2 caractères.
|
||||||
|
</p>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="apiValue"
|
||||||
|
label="Client"
|
||||||
|
:options="apiOptions"
|
||||||
|
:loading="apiLoading"
|
||||||
|
:min-search-length="2"
|
||||||
|
icon-name="mdi:magnify"
|
||||||
|
icon-position="left"
|
||||||
|
@search="onSearchApi"
|
||||||
|
@select="onSelectApi"
|
||||||
|
/>
|
||||||
|
<p v-if="apiSelected" class="mt-2 text-sm text-m-muted">
|
||||||
|
Sélection : <code>{{ apiSelected.label }} (id={{ apiSelected.value }})</code>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec création (allowCreate)</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="createValue"
|
||||||
|
label="Catégorie"
|
||||||
|
:options="staticOptions"
|
||||||
|
allow-create
|
||||||
|
hint="Taper Entrée pour créer une nouvelle valeur"
|
||||||
|
@create="onCreate"
|
||||||
|
/>
|
||||||
|
<p v-if="createdItems.length > 0" class="mt-2 text-sm text-m-muted">
|
||||||
|
Créés : {{ createdItems.join(', ') }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="fr"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="fr"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
readonly
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
label="Pays (readonly vide)"
|
||||||
|
:options="staticOptions"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="readonlyFilledAutocomplete"
|
||||||
|
label="Pays (readonly rempli)"
|
||||||
|
:options="staticOptions"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="hintValue"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
hint="Sélectionne un pays dans la liste"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="fr"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
error="Sélection invalide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
model-value="fr"
|
||||||
|
label="Pays"
|
||||||
|
:options="staticOptions"
|
||||||
|
success="Sélection valide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Liste vide</h2>
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="emptyValue"
|
||||||
|
label="Recherche"
|
||||||
|
:options="[]"
|
||||||
|
no-results-text="Aucun élément disponible"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
type Option = {label: string; value: string | number}
|
||||||
|
|
||||||
|
const staticOptions: Option[] = [
|
||||||
|
{label: 'France', value: 'fr'},
|
||||||
|
{label: 'Belgique', value: 'be'},
|
||||||
|
{label: 'Canada', value: 'ca'},
|
||||||
|
{label: 'Suisse', value: 'ch'},
|
||||||
|
{label: 'Luxembourg', value: 'lu'},
|
||||||
|
{label: 'Allemagne', value: 'de'},
|
||||||
|
{label: 'Espagne', value: 'es'},
|
||||||
|
{label: 'Italie', value: 'it'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const readonlyFilledAutocomplete = ref<string | number | null>('de')
|
||||||
|
const simpleValue = ref<string | number | null>(null)
|
||||||
|
const leftIconValue = ref<string | number | null>(null)
|
||||||
|
const createValue = ref<string | number | null>(null)
|
||||||
|
const hintValue = ref<string | number | null>(null)
|
||||||
|
const emptyValue = ref<string | number | null>(null)
|
||||||
|
|
||||||
|
const createdItems = ref<string[]>([])
|
||||||
|
const onCreate = (value: string) => {
|
||||||
|
createdItems.value.push(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiValue = ref<string | number | null>(null)
|
||||||
|
const apiOptions = ref<Option[]>([])
|
||||||
|
const apiLoading = ref(false)
|
||||||
|
const apiSelected = ref<Option | null>(null)
|
||||||
|
|
||||||
|
const fakeClients: Option[] = [
|
||||||
|
{label: 'Yuno Malio', value: 1},
|
||||||
|
{label: 'Yuna Corp', value: 2},
|
||||||
|
{label: 'Yum Foods', value: 3},
|
||||||
|
{label: 'Yumi Studio', value: 4},
|
||||||
|
{label: 'Acme Inc.', value: 5},
|
||||||
|
{label: 'Globex Corp', value: 6},
|
||||||
|
{label: 'Initech', value: 7},
|
||||||
|
{label: 'Soylent Corp', value: 8},
|
||||||
|
]
|
||||||
|
|
||||||
|
const onSearchApi = async (query: string) => {
|
||||||
|
apiLoading.value = true
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 400))
|
||||||
|
apiOptions.value = fakeClients.filter(c =>
|
||||||
|
c.label.toLowerCase().includes(query.toLowerCase()),
|
||||||
|
)
|
||||||
|
apiLoading.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSelectApi = (option: Option | null) => {
|
||||||
|
apiSelected.value = option
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,159 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
|
<MalioInputEmail />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec label</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
v-model="emailValue"
|
||||||
|
label="Adresse email"
|
||||||
|
name="email"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Ajout dynamique (bouton +)</h2>
|
||||||
|
<div class="space-y-3">
|
||||||
|
<MalioInputEmail
|
||||||
|
v-for="(email, index) in emails"
|
||||||
|
:key="index"
|
||||||
|
v-model="emails[index]"
|
||||||
|
label="Adresse email"
|
||||||
|
addable
|
||||||
|
@add="emails.push('')"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Icône à gauche</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email"
|
||||||
|
icon-position="left"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Sans icône</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email"
|
||||||
|
:icon-name="''"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="contact@malio.fr"
|
||||||
|
disabled
|
||||||
|
label="Adresse email"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="readonly@malio.fr"
|
||||||
|
readonly
|
||||||
|
label="Adresse email"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
v-model="readonlyFilledEmail"
|
||||||
|
label="Adresse email (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
label="Adresse email"
|
||||||
|
hint="ex: prenom.nom@malio.fr"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="pas-un-email"
|
||||||
|
label="Adresse email"
|
||||||
|
error="Adresse email invalide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
model-value="contact@malio.fr"
|
||||||
|
label="Adresse email"
|
||||||
|
success="Adresse email valide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Validation dynamique</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
v-model="dynamicEmail"
|
||||||
|
label="Adresse email"
|
||||||
|
hint="Saisir une adresse au format prenom@domaine.tld"
|
||||||
|
:error="dynamicError"
|
||||||
|
:success="dynamicSuccess"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Email obligatoire</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
v-model="requiredEmail"
|
||||||
|
label="Email obligatoire"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Email normalisé (minuscules)</h2>
|
||||||
|
<MalioInputEmail
|
||||||
|
v-model="lowercaseEmail"
|
||||||
|
label="Email normalisé (minuscules)"
|
||||||
|
:lowercase="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, ref } from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledEmail = ref('contact@malio.fr')
|
||||||
|
const emailValue = ref('')
|
||||||
|
const emails = ref<string[]>([''])
|
||||||
|
const dynamicEmail = ref('')
|
||||||
|
const requiredEmail = ref('')
|
||||||
|
const lowercaseEmail = ref('')
|
||||||
|
|
||||||
|
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
|
||||||
|
const isDynamicValid = computed(() => emailRegex.test(dynamicEmail.value))
|
||||||
|
const dynamicError = computed(() => {
|
||||||
|
if (!dynamicEmail.value) return ''
|
||||||
|
return isDynamicValid.value ? '' : 'Adresse email invalide'
|
||||||
|
})
|
||||||
|
const dynamicSuccess = computed(() => {
|
||||||
|
if (!dynamicEmail.value) return ''
|
||||||
|
return isDynamicValid.value ? 'Adresse email valide' : ''
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -41,6 +41,23 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputPassword
|
||||||
|
label="Mot de passe (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputPassword
|
||||||
|
v-model="readonlyFilledPassword"
|
||||||
|
label="Mot de passe (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
||||||
<MalioInputPassword
|
<MalioInputPassword
|
||||||
@@ -83,6 +100,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, ref } from 'vue'
|
import { computed, ref } from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledPassword = ref('motdepasse123')
|
||||||
const passwordValue = ref('')
|
const passwordValue = ref('')
|
||||||
const dynamicPassword = ref('')
|
const dynamicPassword = ref('')
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,159 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
|
<MalioInputPhone />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec label</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
v-model="phoneValue"
|
||||||
|
label="Téléphone"
|
||||||
|
name="phone"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec bouton « ajouter »</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
v-model="phoneAddable"
|
||||||
|
label="Téléphone"
|
||||||
|
addable
|
||||||
|
@add="onAdd"
|
||||||
|
/>
|
||||||
|
<p v-if="addClicks > 0" class="mt-2 text-sm text-m-muted">
|
||||||
|
Bouton cliqué {{ addClicks }} fois
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Icône à droite (sans bouton +)</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone"
|
||||||
|
icon-position="right"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Sans icône</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone"
|
||||||
|
:icon-name="''"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec masque français</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
v-model="phoneFrench"
|
||||||
|
label="Téléphone (FR)"
|
||||||
|
mask="+33 # ## ## ## ##"
|
||||||
|
hint="Saisir uniquement les chiffres"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Désactivé (avec addable)</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="+33 6 12 34 56 78"
|
||||||
|
addable
|
||||||
|
disabled
|
||||||
|
label="Téléphone"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (avec addable)</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="+33 6 12 34 56 78"
|
||||||
|
addable
|
||||||
|
readonly
|
||||||
|
label="Téléphone"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
v-model="readonlyFilledPhone"
|
||||||
|
label="Téléphone (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
label="Téléphone"
|
||||||
|
hint="Format international recommandé"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="abc"
|
||||||
|
label="Téléphone"
|
||||||
|
error="Numéro de téléphone invalide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioInputPhone
|
||||||
|
model-value="+33 6 12 34 56 78"
|
||||||
|
label="Téléphone"
|
||||||
|
success="Numéro valide"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4 md:col-span-2">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Cas ERP — liste de téléphones (max 2)</h2>
|
||||||
|
<p class="mb-3 text-sm text-m-muted">
|
||||||
|
Le bouton + s'affiche sur le dernier champ tant que la liste contient moins de {{ MAX_PHONES }} numéros.
|
||||||
|
</p>
|
||||||
|
<div class="flex flex-col gap-4">
|
||||||
|
<MalioInputPhone
|
||||||
|
v-for="(phone, index) in phones"
|
||||||
|
:key="index"
|
||||||
|
v-model="phones[index]"
|
||||||
|
:label="`Téléphone ${index + 1}`"
|
||||||
|
:addable="index === phones.length - 1 && phones.length < MAX_PHONES"
|
||||||
|
@add="addPhone"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref } from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledPhone = ref('+33 6 12 34 56 78')
|
||||||
|
const phoneValue = ref('')
|
||||||
|
const phoneAddable = ref('')
|
||||||
|
const phoneFrench = ref('')
|
||||||
|
const addClicks = ref(0)
|
||||||
|
|
||||||
|
const onAdd = () => {
|
||||||
|
addClicks.value++
|
||||||
|
}
|
||||||
|
|
||||||
|
const MAX_PHONES = 2
|
||||||
|
const phones = ref<string[]>([''])
|
||||||
|
|
||||||
|
const addPhone = () => {
|
||||||
|
if (phones.value.length < MAX_PHONES) {
|
||||||
|
phones.value.push('')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -108,6 +108,33 @@
|
|||||||
icon-size="20"
|
icon-size="20"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputText
|
||||||
|
label="Référence (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputText
|
||||||
|
v-model="readonlyFilledValue"
|
||||||
|
label="Référence (readonly rempli)"
|
||||||
|
icon-name="mdi:lock-outline"
|
||||||
|
icon-size="20"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Champ obligatoire</h2>
|
||||||
|
<MalioInputText
|
||||||
|
label="Champ obligatoire"
|
||||||
|
:required="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Avec masque</h2>
|
<h2 class="mb-4 text-xl font-bold">Avec masque</h2>
|
||||||
<MalioInputText
|
<MalioInputText
|
||||||
@@ -154,6 +181,7 @@
|
|||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, ref } from 'vue'
|
import { computed, ref } from 'vue'
|
||||||
|
const readonlyFilledValue = ref('Commande #A-2048')
|
||||||
const nameValue = ref('')
|
const nameValue = ref('')
|
||||||
const searchValue = ref('')
|
const searchValue = ref('')
|
||||||
const codeValue = ref('')
|
const codeValue = ref('')
|
||||||
|
|||||||
@@ -61,6 +61,25 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputTextArea
|
||||||
|
label="Description (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
:size="3"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputTextArea
|
||||||
|
v-model="readonlyFilledTextArea"
|
||||||
|
label="Description (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
:size="3"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Resize avec limites</h2>
|
<h2 class="mb-4 text-xl font-bold">Resize avec limites</h2>
|
||||||
<MalioInputTextArea
|
<MalioInputTextArea
|
||||||
@@ -94,6 +113,7 @@
|
|||||||
import {ref} from 'vue'
|
import {ref} from 'vue'
|
||||||
import MalioInputTextArea from '../../../../app/components/malio/input/InputTextArea.vue'
|
import MalioInputTextArea from '../../../../app/components/malio/input/InputTextArea.vue'
|
||||||
|
|
||||||
|
const readonlyFilledTextArea = ref('Ce texte est en lecture seule et ne peut pas être modifié.')
|
||||||
const hintValue = ref('')
|
const hintValue = ref('')
|
||||||
const iconValue = ref('')
|
const iconValue = ref('')
|
||||||
const errorValue = ref('abc')
|
const errorValue = ref('abc')
|
||||||
|
|||||||
@@ -14,6 +14,17 @@
|
|||||||
<p class="mt-2 text-sm text-gray-500">Valeur : {{ uploadValue || '(aucun)' }}</p>
|
<p class="mt-2 text-sm text-gray-500">Valeur : {{ uploadValue || '(aucun)' }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Clearable (croix pour vider)</h2>
|
||||||
|
<MalioInputUpload
|
||||||
|
v-model="clearableUpload"
|
||||||
|
label="Téléverser un document"
|
||||||
|
clearable
|
||||||
|
@clear="onClearUpload"
|
||||||
|
/>
|
||||||
|
<p class="mt-2 text-sm text-gray-500">Valeur : {{ clearableUpload || '(aucun)' }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Avec accept (PDF)</h2>
|
<h2 class="mb-4 text-xl font-bold">Avec accept (PDF)</h2>
|
||||||
<MalioInputUpload
|
<MalioInputUpload
|
||||||
@@ -31,6 +42,23 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioInputUpload
|
||||||
|
label="Fichier (readonly vide)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioInputUpload
|
||||||
|
v-model="readonlyFilledUpload"
|
||||||
|
label="Fichier (readonly rempli)"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
<h2 class="mb-4 text-xl font-bold">Avec hint</h2>
|
||||||
<MalioInputUpload
|
<MalioInputUpload
|
||||||
@@ -74,8 +102,14 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, ref } from 'vue'
|
import { computed, ref } from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledUpload = ref('document.pdf')
|
||||||
const uploadValue = ref('')
|
const uploadValue = ref('')
|
||||||
const dynamicUpload = ref('')
|
const dynamicUpload = ref('')
|
||||||
|
const clearableUpload = ref('rapport-2026.pdf')
|
||||||
|
|
||||||
|
const onClearUpload = () => {
|
||||||
|
clearableUpload.value = ''
|
||||||
|
}
|
||||||
|
|
||||||
const dynamicError = computed(() => {
|
const dynamicError = computed(() => {
|
||||||
if (!dynamicUpload.value) return ''
|
if (!dynamicUpload.value) return ''
|
||||||
|
|||||||
@@ -0,0 +1,74 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { ref } from 'vue'
|
||||||
|
import MalioButton from "../../../../app/components/malio/button/Button.vue";
|
||||||
|
|
||||||
|
const modalBase = ref(false)
|
||||||
|
const modalForm = ref(false)
|
||||||
|
const modalLong = ref(false)
|
||||||
|
const modalNoDismiss = ref(false)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Modal simple</h2>
|
||||||
|
<MalioButton label="Ouvrir" @click="modalBase = true" />
|
||||||
|
<MalioModal v-model="modalBase" headerClass="py-7 px-[25px]" footerClass="flex justify-center pt-8">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Marquer comme vu ?</h2>
|
||||||
|
</template>
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Valider"/>
|
||||||
|
</template>
|
||||||
|
</MalioModal>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Avec footer d'actions</h2>
|
||||||
|
<MalioButton label="Ouvrir le formulaire" variant="tertiary" @click="modalForm = true" />
|
||||||
|
<MalioModal v-model="modalForm" modal-class="max-w-lg">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Nouveau contact</h2>
|
||||||
|
</template>
|
||||||
|
<div class="flex flex-col gap-4 py-2">
|
||||||
|
<MalioInputText label="Nom" />
|
||||||
|
<MalioInputText label="Prénom" />
|
||||||
|
<MalioInputText label="Email" />
|
||||||
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Annuler" variant="secondary" button-class="flex-1" @click="modalForm = false" />
|
||||||
|
<MalioButton label="Enregistrer" button-class="flex-1" @click="modalForm = false" />
|
||||||
|
</template>
|
||||||
|
</MalioModal>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Contenu long (body scrollable)</h2>
|
||||||
|
<MalioButton label="Ouvrir" variant="tertiary" @click="modalLong = true" />
|
||||||
|
<MalioModal v-model="modalLong">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Conditions</h2>
|
||||||
|
</template>
|
||||||
|
<div class="flex flex-col gap-4">
|
||||||
|
<p v-for="n in 20" :key="n" class="text-m-text">
|
||||||
|
Paragraphe {{ n }} — contenu long pour forcer le scroll interne ; le header et le footer restent fixes.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Accepter" button-class="w-full" @click="modalLong = false" />
|
||||||
|
</template>
|
||||||
|
</MalioModal>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-6">
|
||||||
|
<h2 class="mb-6 text-xl font-bold">Non dismissable (croix uniquement)</h2>
|
||||||
|
<MalioButton label="Ouvrir" variant="danger" @click="modalNoDismiss = true" />
|
||||||
|
<MalioModal v-model="modalNoDismiss" :dismissable="false" :close-on-escape="false">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold text-black">Action requise</h2>
|
||||||
|
</template>
|
||||||
|
<p class="text-m-text">Ni le backdrop ni Échap ne ferment cette modal. Utilisez la croix.</p>
|
||||||
|
</MalioModal>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@@ -82,6 +82,17 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Sélection obligatoire</h2>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="requiredValue"
|
||||||
|
:options="options"
|
||||||
|
label="Sélection obligatoire"
|
||||||
|
:required="true"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Peu d'elements (2)</h2>
|
<h2 class="mb-4 text-xl font-bold">Peu d'elements (2)</h2>
|
||||||
<MalioSelect
|
<MalioSelect
|
||||||
@@ -92,6 +103,28 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Lecture seule (vide)</h2>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="readonlyEmptyValue"
|
||||||
|
:options="options"
|
||||||
|
label="Pays"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Lecture seule (rempli)</h2>
|
||||||
|
<MalioSelect
|
||||||
|
v-model="readonlyFilledValue"
|
||||||
|
:options="options"
|
||||||
|
label="Pays"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4 md:col-span-2">
|
<div class="rounded-lg border p-4 md:col-span-2">
|
||||||
<h2 class="mb-4 text-xl font-bold">Liste longue</h2>
|
<h2 class="mb-4 text-xl font-bold">Liste longue</h2>
|
||||||
<MalioSelect
|
<MalioSelect
|
||||||
@@ -151,6 +184,7 @@ const longOptions = [
|
|||||||
{label: 'Republique tcheque', value: 'cz'},
|
{label: 'Republique tcheque', value: 'cz'},
|
||||||
]
|
]
|
||||||
|
|
||||||
|
const requiredValue = ref<string | number | null>(null)
|
||||||
const basicValue = ref<string | number | null>(null)
|
const basicValue = ref<string | number | null>(null)
|
||||||
const labelValue = ref<string | number | null>(null)
|
const labelValue = ref<string | number | null>(null)
|
||||||
const selectedValue = ref<string | number | null>('fr')
|
const selectedValue = ref<string | number | null>('fr')
|
||||||
@@ -162,4 +196,6 @@ const emptyValue = ref<string | number | null>(null)
|
|||||||
const shortListValue = ref<string | number | null>(null)
|
const shortListValue = ref<string | number | null>(null)
|
||||||
const longListValue = ref<string | number | null>(null)
|
const longListValue = ref<string | number | null>(null)
|
||||||
const bottomValue = ref<string | number | null>(null)
|
const bottomValue = ref<string | number | null>(null)
|
||||||
|
const readonlyEmptyValue = ref<string | number | null>(null)
|
||||||
|
const readonlyFilledValue = ref<string | number | null>('fr')
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
<MalioSelectCheckbox
|
<MalioSelectCheckbox
|
||||||
v-model="labelValue"
|
v-model="labelValue"
|
||||||
:options="options"
|
:options="options"
|
||||||
displayTag="true"
|
:display-tag="true"
|
||||||
empty-option-label=" "
|
empty-option-label=" "
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -22,7 +22,7 @@
|
|||||||
<MalioSelectCheckbox
|
<MalioSelectCheckbox
|
||||||
v-model="labelValue1"
|
v-model="labelValue1"
|
||||||
:options="options"
|
:options="options"
|
||||||
displayTag="true"
|
:display-tag="true"
|
||||||
label="Pays"
|
label="Pays"
|
||||||
empty-option-label=" "
|
empty-option-label=" "
|
||||||
/>
|
/>
|
||||||
@@ -123,6 +123,28 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Lecture seule (vide)</h2>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
v-model="readonlyEmptyValue"
|
||||||
|
:options="options"
|
||||||
|
label="Pays"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Lecture seule (rempli)</h2>
|
||||||
|
<MalioSelectCheckbox
|
||||||
|
v-model="readonlyFilledValue"
|
||||||
|
:options="options"
|
||||||
|
label="Pays"
|
||||||
|
empty-option-label="Aucune selection"
|
||||||
|
:readonly="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4 md:col-span-2">
|
<div class="rounded-lg border p-4 md:col-span-2">
|
||||||
<h2 class="mb-4 text-xl font-bold">Liste longue</h2>
|
<h2 class="mb-4 text-xl font-bold">Liste longue</h2>
|
||||||
<MalioSelectCheckbox
|
<MalioSelectCheckbox
|
||||||
@@ -145,6 +167,7 @@
|
|||||||
empty-option-label="Aucune selection"
|
empty-option-label="Aucune selection"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -190,4 +213,6 @@ const selectAllValue = ref<Array<string | number>>([])
|
|||||||
const selectAllCustomValue = ref<Array<string | number>>([])
|
const selectAllCustomValue = ref<Array<string | number>>([])
|
||||||
const longListValue = ref<Array<string | number>>([])
|
const longListValue = ref<Array<string | number>>([])
|
||||||
const bottomValue = ref<Array<string | number>>([])
|
const bottomValue = ref<Array<string | number>>([])
|
||||||
|
const readonlyEmptyValue = ref<Array<string | number>>([])
|
||||||
|
const readonlyFilledValue = ref<Array<string | number>>(['fr'])
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,5 +1,65 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="grid grid-cols-1 items-start gap-6">
|
<div class="grid grid-cols-1 items-start gap-6">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-1 text-xl font-bold">Bac à sable (tous les cas)</h2>
|
||||||
|
<p class="mb-4 text-sm text-m-muted">
|
||||||
|
Règle les paramètres puis <strong>redimensionne le cadre en pointillés</strong>
|
||||||
|
(poignée en bas à droite) pour voir le nombre d'onglets s'adapter et les flèches apparaître.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="mb-4 flex flex-wrap items-end gap-4 text-sm">
|
||||||
|
<label class="flex flex-col gap-1">Nb onglets : {{ sbCount }}
|
||||||
|
<input
|
||||||
|
v-model.number="sbCount"
|
||||||
|
type="range"
|
||||||
|
min="1"
|
||||||
|
max="15"
|
||||||
|
class="w-40"
|
||||||
|
>
|
||||||
|
</label>
|
||||||
|
<label class="flex flex-col gap-1">maxVisibleTabs (0 = auto)
|
||||||
|
<input
|
||||||
|
v-model.number="sbMax"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
max="15"
|
||||||
|
class="w-20 rounded border px-2 py-1"
|
||||||
|
>
|
||||||
|
</label>
|
||||||
|
<label class="flex items-center gap-2">
|
||||||
|
<input
|
||||||
|
v-model="sbIcons"
|
||||||
|
type="checkbox"
|
||||||
|
> Icônes
|
||||||
|
</label>
|
||||||
|
<label class="flex items-center gap-2">
|
||||||
|
<input
|
||||||
|
v-model="sbLong"
|
||||||
|
type="checkbox"
|
||||||
|
> Labels longs
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="resize-x overflow-hidden rounded border-2 border-dashed border-m-muted p-3"
|
||||||
|
style="width: 100%; min-width: 280px;"
|
||||||
|
>
|
||||||
|
<MalioTabList
|
||||||
|
v-model="sbValue"
|
||||||
|
:tabs="sbTabs"
|
||||||
|
:max-visible-tabs="sbMaxProp"
|
||||||
|
>
|
||||||
|
<template
|
||||||
|
v-for="t in sbTabs"
|
||||||
|
#[t.key]
|
||||||
|
:key="t.key"
|
||||||
|
>
|
||||||
|
<p class="p-4">Contenu : {{ t.label }}</p>
|
||||||
|
</template>
|
||||||
|
</MalioTabList>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="rounded-lg border p-4">
|
<div class="rounded-lg border p-4">
|
||||||
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
<MalioTabList v-model="simpleValue" :tabs="tabs">
|
<MalioTabList v-model="simpleValue" :tabs="tabs">
|
||||||
@@ -36,11 +96,69 @@
|
|||||||
<template #details><p class="p-4">Détails avancés</p></template>
|
<template #details><p class="p-4">Détails avancés</p></template>
|
||||||
</MalioTabList>
|
</MalioTabList>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-1 text-xl font-bold">Beaucoup d'onglets (fenêtré)</h2>
|
||||||
|
<p class="mb-4 text-sm text-m-muted">
|
||||||
|
7 onglets avec <code>:max-visible-tabs="5"</code> — flèches gauche/droite pour faire défiler
|
||||||
|
(1 par 1). L'onglet actif reste sélectionné même hors fenêtre.
|
||||||
|
</p>
|
||||||
|
<MalioTabList v-model="manyValue" :tabs="manyTabs" :max-visible-tabs="5">
|
||||||
|
<template #infos><p class="p-4">Contenu Informations</p></template>
|
||||||
|
<template #adresses><p class="p-4">Contenu Adresses</p></template>
|
||||||
|
<template #contacts><p class="p-4">Contenu Contacts</p></template>
|
||||||
|
<template #compta><p class="p-4">Contenu Comptabilité</p></template>
|
||||||
|
<template #documents><p class="p-4">Contenu Documents</p></template>
|
||||||
|
<template #historique><p class="p-4">Contenu Historique</p></template>
|
||||||
|
<template #parametres><p class="p-4">Contenu Paramètres</p></template>
|
||||||
|
</MalioTabList>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-1 text-xl font-bold">Peu d'onglets avec maxVisibleTabs</h2>
|
||||||
|
<p class="mb-4 text-sm text-m-muted">
|
||||||
|
3 onglets avec <code>:max-visible-tabs="5"</code> — le fenêtrage ne s'active pas
|
||||||
|
(onglets ≤ max), donc pas de flèches, affichage normal centré.
|
||||||
|
</p>
|
||||||
|
<MalioTabList v-model="fewValue" :tabs="fewTabs" :max-visible-tabs="5">
|
||||||
|
<template #general><p class="p-4">Contenu Général</p></template>
|
||||||
|
<template #adresses><p class="p-4">Contenu Adresses</p></template>
|
||||||
|
<template #contacts><p class="p-4">Contenu Contacts</p></template>
|
||||||
|
</MalioTabList>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref } from 'vue'
|
import { computed, ref } from 'vue'
|
||||||
|
|
||||||
|
// --- Bac à sable interactif ---
|
||||||
|
const sbCount = ref(9)
|
||||||
|
const sbMax = ref(0)
|
||||||
|
const sbIcons = ref(true)
|
||||||
|
const sbLong = ref(false)
|
||||||
|
|
||||||
|
const SB_LABELS = [
|
||||||
|
'Informations', 'Adresses', 'Contacts', 'Comptabilité', 'Documents',
|
||||||
|
'Historique', 'Paramètres', 'Qualité', 'Facturation', 'Accueil',
|
||||||
|
'Notifications', 'Statistiques', 'Équipe', 'Sécurité', 'Étiquettes',
|
||||||
|
]
|
||||||
|
const SB_ICONS = [
|
||||||
|
'mdi:information-outline', 'mdi:map-marker-outline', 'mdi:account-box-outline', 'mdi:web',
|
||||||
|
'mdi:file-document-outline', 'mdi:history', 'mdi:cog-outline', 'mdi:check-decagram-outline',
|
||||||
|
'mdi:receipt-text-outline', 'mdi:home', 'mdi:bell-outline', 'mdi:chart-bar',
|
||||||
|
'mdi:account-group-outline', 'mdi:lock-outline', 'mdi:tag-outline',
|
||||||
|
]
|
||||||
|
|
||||||
|
const sbTabs = computed(() =>
|
||||||
|
Array.from({ length: sbCount.value }, (_, i) => ({
|
||||||
|
key: `sb${i}`,
|
||||||
|
label: sbLong.value ? `${SB_LABELS[i % SB_LABELS.length]} détaillé` : SB_LABELS[i % SB_LABELS.length],
|
||||||
|
icon: sbIcons.value ? SB_ICONS[i % SB_ICONS.length] : undefined,
|
||||||
|
})),
|
||||||
|
)
|
||||||
|
const sbMaxProp = computed(() => (sbMax.value > 0 ? sbMax.value : undefined))
|
||||||
|
const sbValue = ref('sb0')
|
||||||
|
|
||||||
const tabs = [
|
const tabs = [
|
||||||
{ key: 'qualimat', label: 'Qualimat', icon: 'mdi:certificate-outline' },
|
{ key: 'qualimat', label: 'Qualimat', icon: 'mdi:certificate-outline' },
|
||||||
@@ -60,7 +178,25 @@ const tabsTwo = [
|
|||||||
{ key: 'details', label: 'Détails', icon: 'mdi:cog-outline' },
|
{ key: 'details', label: 'Détails', icon: 'mdi:cog-outline' },
|
||||||
]
|
]
|
||||||
|
|
||||||
|
const manyTabs = [
|
||||||
|
{ key: 'infos', label: 'Informations', icon: 'mdi:information-outline' },
|
||||||
|
{ key: 'adresses', label: 'Adresses', icon: 'mdi:map-marker-outline' },
|
||||||
|
{ key: 'contacts', label: 'Contacts', icon: 'mdi:account-box-outline' },
|
||||||
|
{ key: 'compta', label: 'Comptabilité', icon: 'mdi:web' },
|
||||||
|
{ key: 'documents', label: 'Documents', icon: 'mdi:file-document-outline' },
|
||||||
|
{ key: 'historique', label: 'Historique', icon: 'mdi:history' },
|
||||||
|
{ key: 'parametres', label: 'Paramètres', icon: 'mdi:cog-outline' },
|
||||||
|
]
|
||||||
|
|
||||||
const simpleValue = ref('qualimat')
|
const simpleValue = ref('qualimat')
|
||||||
const noIconValue = ref('tab1')
|
const noIconValue = ref('tab1')
|
||||||
const twoTabValue = ref('general')
|
const twoTabValue = ref('general')
|
||||||
|
const manyValue = ref('infos')
|
||||||
|
|
||||||
|
const fewTabs = [
|
||||||
|
{ key: 'general', label: 'Général', icon: 'mdi:information-outline' },
|
||||||
|
{ key: 'adresses', label: 'Adresses', icon: 'mdi:map-marker-outline' },
|
||||||
|
{ key: 'contacts', label: 'Contacts', icon: 'mdi:account-box-outline' },
|
||||||
|
]
|
||||||
|
const fewValue = ref('general')
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -0,0 +1,56 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 md:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
|
<MalioTimePicker v-model="simpleValue" label="Heure" />
|
||||||
|
<p class="mt-2 text-sm text-m-muted">Valeur : {{ simpleValue || '—' }}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Valeur initiale</h2>
|
||||||
|
<MalioTimePicker v-model="initialValue" label="Heure de départ" hint="Format HH:MM" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
||||||
|
<MalioTimePicker v-model="disabledValue" label="Heure verrouillée" disabled />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioTimePicker v-model="errorValue" label="Heure de fermeture" error="Heure invalide" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioTimePicker v-model="successValue" label="Heure confirmée" success="Horaire enregistré" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Non effaçable</h2>
|
||||||
|
<MalioTimePicker v-model="noClearValue" label="Heure" :clearable="false" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly vide)</h2>
|
||||||
|
<MalioTimePicker label="Heure (readonly vide)" :readonly="true" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly (readonly rempli)</h2>
|
||||||
|
<MalioTimePicker v-model="readonlyFilledTime" label="Heure (readonly rempli)" :readonly="true" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
|
||||||
|
const readonlyFilledTime = ref('14:30')
|
||||||
|
const simpleValue = ref('')
|
||||||
|
const initialValue = ref('08:30')
|
||||||
|
const disabledValue = ref('14:15')
|
||||||
|
const errorValue = ref('25:90')
|
||||||
|
const successValue = ref('09:00')
|
||||||
|
const noClearValue = ref('10:00')
|
||||||
|
</script>
|
||||||
+7
-178
@@ -1,181 +1,10 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="flex min-h-screen">
|
<div class="mx-auto max-w-2xl py-16 text-center">
|
||||||
<aside class="w-72 bg-m-bg p-6 text-white">
|
<h1 class="text-3xl font-bold text-m-text">
|
||||||
<button
|
Playground @malio/layer-ui
|
||||||
type="button"
|
</h1>
|
||||||
class="text-xl text-black font-semibold"
|
<p class="mt-4 text-m-muted">
|
||||||
@click="clearSelection"
|
Sélectionne un composant dans la barre latérale pour afficher sa page de démonstration.
|
||||||
>
|
</p>
|
||||||
Liste des composants
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<nav class="mt-6 flex flex-col gap-1">
|
|
||||||
<div
|
|
||||||
v-for="group in groups"
|
|
||||||
:key="group.category"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="flex w-full items-center justify-between rounded px-3 py-2 text-left text-black font-bold hover:bg-m-primary/10"
|
|
||||||
@click="toggleCategory(group.category)"
|
|
||||||
>
|
|
||||||
{{ group.category }}
|
|
||||||
<span
|
|
||||||
class="text-xs transition-transform duration-200"
|
|
||||||
:class="openCategories.has(group.category) ? 'rotate-90' : ''"
|
|
||||||
>
|
|
||||||
▶
|
|
||||||
</span>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<div
|
|
||||||
v-if="openCategories.has(group.category)"
|
|
||||||
class="ml-3 flex flex-col gap-1 border-l border-gray-300 pl-2"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
v-for="item in group.items"
|
|
||||||
:key="item.name"
|
|
||||||
type="button"
|
|
||||||
class="rounded px-3 py-1.5 text-left text-sm text-black hover:bg-m-primary hover:text-white"
|
|
||||||
:class="selectedName === item.name ? 'bg-m-primary/50 text-white' : ''"
|
|
||||||
@click="selectItem(item.name)"
|
|
||||||
>
|
|
||||||
{{ item.label }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
</aside>
|
|
||||||
|
|
||||||
<main class="flex-1 p-6">
|
|
||||||
<component
|
|
||||||
:is="selectedDemoComponent"
|
|
||||||
v-if="selectedDemoComponent"
|
|
||||||
/>
|
|
||||||
<p
|
|
||||||
v-else-if="selectedName"
|
|
||||||
class="text-gray-700"
|
|
||||||
>
|
|
||||||
Page de demo introuvable:
|
|
||||||
<code>.playground/pages/composant/{{ selectedDemoFileName }}.vue</code>
|
|
||||||
</p>
|
|
||||||
<div v-else>
|
|
||||||
<h1 class="text-2xl font-semibold text-gray-900">
|
|
||||||
Playground composants
|
|
||||||
</h1>
|
|
||||||
<p class="mt-2 text-gray-600">
|
|
||||||
Selectionne un composant dans la liste pour afficher sa page de demo.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import {computed, reactive, ref, watch, shallowRef} from 'vue'
|
|
||||||
|
|
||||||
type LoadedModule = {
|
|
||||||
default: unknown
|
|
||||||
}
|
|
||||||
|
|
||||||
type Item = {
|
|
||||||
name: string
|
|
||||||
label: string
|
|
||||||
}
|
|
||||||
|
|
||||||
type Group = {
|
|
||||||
category: string
|
|
||||||
items: Item[]
|
|
||||||
}
|
|
||||||
|
|
||||||
const componentModules = import.meta.glob('../../app/components/malio/**/*.vue')
|
|
||||||
const demoModules = import.meta.glob('./composant/**/*.vue')
|
|
||||||
|
|
||||||
const demoByName: Record<string, () => Promise<LoadedModule>> =
|
|
||||||
Object.fromEntries(
|
|
||||||
Object.entries(demoModules).map(([file, loader]) => {
|
|
||||||
const name = file.split('/').pop()?.replace('.vue', '') ?? ''
|
|
||||||
return [name.toLowerCase(), loader as () => Promise<LoadedModule>]
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
|
|
||||||
const groups = computed<Group[]>(() => {
|
|
||||||
const categoryMap = new Map<string, Item[]>()
|
|
||||||
|
|
||||||
Object.keys(componentModules).forEach((file) => {
|
|
||||||
const parts = file.split('/')
|
|
||||||
const name = parts.pop()?.replace('.vue', '') ?? ''
|
|
||||||
const category = parts.pop() ?? ''
|
|
||||||
|
|
||||||
if (!categoryMap.has(category)) {
|
|
||||||
categoryMap.set(category, [])
|
|
||||||
}
|
|
||||||
categoryMap.get(category)!.push({name, label: name})
|
|
||||||
})
|
|
||||||
|
|
||||||
return Array.from(categoryMap.entries())
|
|
||||||
.sort(([a], [b]) => a.localeCompare(b))
|
|
||||||
.map(([category, items]) => ({
|
|
||||||
category: category.charAt(0).toUpperCase() + category.slice(1),
|
|
||||||
items: items.sort((a, b) => a.label.localeCompare(b.label)),
|
|
||||||
}))
|
|
||||||
})
|
|
||||||
|
|
||||||
const openCategories = reactive(new Set<string>())
|
|
||||||
const selectedName = ref('')
|
|
||||||
const hasInitializedSelection = ref(false)
|
|
||||||
|
|
||||||
watch(
|
|
||||||
groups,
|
|
||||||
(val) => {
|
|
||||||
if (!hasInitializedSelection.value && val.length > 0) {
|
|
||||||
openCategories.add(val[0].category)
|
|
||||||
if (val[0].items.length > 0) {
|
|
||||||
selectedName.value = val[0].items[0].name
|
|
||||||
}
|
|
||||||
hasInitializedSelection.value = true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{immediate: true},
|
|
||||||
)
|
|
||||||
|
|
||||||
function toggleCategory(category: string) {
|
|
||||||
if (openCategories.has(category)) {
|
|
||||||
openCategories.delete(category)
|
|
||||||
} else {
|
|
||||||
openCategories.add(category)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function selectItem(name: string) {
|
|
||||||
selectedName.value = selectedName.value === name ? '' : name
|
|
||||||
}
|
|
||||||
|
|
||||||
function clearSelection() {
|
|
||||||
selectedName.value = ''
|
|
||||||
}
|
|
||||||
|
|
||||||
const selectedDemoComponent = shallowRef<unknown>(null)
|
|
||||||
|
|
||||||
watch(selectedName, async (name) => {
|
|
||||||
if (!name) {
|
|
||||||
selectedDemoComponent.value = null
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const loader = demoByName[name.toLowerCase()]
|
|
||||||
if (!loader) {
|
|
||||||
selectedDemoComponent.value = null
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const mod = await loader()
|
|
||||||
selectedDemoComponent.value = mod.default
|
|
||||||
})
|
|
||||||
|
|
||||||
const selectedDemoFileName = computed(() => {
|
|
||||||
const name = selectedName.value
|
|
||||||
if (!name) return ''
|
|
||||||
return name.charAt(0).toLowerCase() + name.slice(1)
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|||||||
@@ -0,0 +1,80 @@
|
|||||||
|
import type {SidebarSection} from '../app/components/malio/sidebar/Sidebar.vue'
|
||||||
|
|
||||||
|
export const navSections: SidebarSection[] = [
|
||||||
|
{
|
||||||
|
label: 'BOUTONS',
|
||||||
|
icon: 'mdi:gesture-tap-button',
|
||||||
|
items: [
|
||||||
|
{label: 'Button', to: '/composant/button/button'},
|
||||||
|
{label: 'Button Icon', to: '/composant/button/buttonIcon'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'CHAMPS',
|
||||||
|
icon: 'mdi:form-textbox',
|
||||||
|
items: [
|
||||||
|
{label: 'Texte', to: '/composant/input/inputText'},
|
||||||
|
{label: 'Nombre', to: '/composant/input/inputNumber'},
|
||||||
|
{label: 'Montant', to: '/composant/input/inputAmount'},
|
||||||
|
{label: 'Email', to: '/composant/input/inputEmail'},
|
||||||
|
{label: 'Mot de passe', to: '/composant/input/inputPassword'},
|
||||||
|
{label: 'Téléphone', to: '/composant/input/inputPhone'},
|
||||||
|
{label: 'Zone de texte', to: '/composant/input/inputTextArea'},
|
||||||
|
{label: 'Saisie assistée', to: '/composant/input/inputAutocomplete'},
|
||||||
|
{label: 'Upload', to: '/composant/input/inputUpload'},
|
||||||
|
{label: 'Éditeur riche', to: '/composant/input/inputRichText'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'DATES & HEURES',
|
||||||
|
icon: 'mdi:calendar-clock',
|
||||||
|
items: [
|
||||||
|
{label: 'Date', to: '/composant/date/date'},
|
||||||
|
{label: 'Plage de dates', to: '/composant/date/dateRange'},
|
||||||
|
{label: 'Semaine', to: '/composant/date/dateWeek'},
|
||||||
|
{label: 'Date & heure', to: '/composant/date/datetime'},
|
||||||
|
{label: 'Heure', to: '/composant/time/time'},
|
||||||
|
{label: 'Sélecteur d\'heure', to: '/composant/time/timePicker'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'SÉLECTIONS',
|
||||||
|
icon: 'mdi:form-dropdown',
|
||||||
|
items: [
|
||||||
|
{label: 'Select', to: '/composant/select/select'},
|
||||||
|
{label: 'Select Checkbox', to: '/composant/select/selectCheckbox'},
|
||||||
|
{label: 'Checkbox', to: '/composant/checkbox/checkbox'},
|
||||||
|
{label: 'Radio', to: '/composant/radio/radioButton'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'NAVIGATION',
|
||||||
|
icon: 'mdi:navigation-variant',
|
||||||
|
items: [
|
||||||
|
{label: 'Sidebar', to: '/composant/sidebar/sidebar'},
|
||||||
|
{label: 'Drawer', to: '/composant/drawer/drawer'},
|
||||||
|
{label: 'Modal', to: '/composant/modal/modal'},
|
||||||
|
{label: 'Onglets', to: '/composant/tab/tabList'},
|
||||||
|
{label: 'Accordéon', to: '/composant/accordion/accordion'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'DONNÉES',
|
||||||
|
icon: 'mdi:table',
|
||||||
|
items: [
|
||||||
|
{label: 'DataTable', to: '/composant/datatable/datatable'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'DIVERS',
|
||||||
|
icon: 'mdi:dots-horizontal',
|
||||||
|
items: [
|
||||||
|
{label: 'Champs readonly', to: '/composant/divers/readonly'},
|
||||||
|
{label: 'Champs disabled', to: '/composant/divers/disabled'},
|
||||||
|
{label: 'Heure', to: '/composant/time/time'},
|
||||||
|
{label: 'Sélecteur de site', to: '/composant/site/siteSelector'},
|
||||||
|
{label: 'Formulaire client', to: '/composant/form/client'},
|
||||||
|
{label: 'Filtres', to: '/composant/filtre/filtres'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
]
|
||||||
@@ -27,9 +27,64 @@ Liste des évolutions de la librairie Malio layer UI
|
|||||||
* [#MUI-22] Création d'un composant datatable
|
* [#MUI-22] Création d'un composant datatable
|
||||||
* [#MUI-27] Création d'un composant sélection de site
|
* [#MUI-27] Création d'un composant sélection de site
|
||||||
* Création d'un composant rich text (TipTap) avec sortie markdown / HTML
|
* Création d'un composant rich text (TipTap) avec sortie markdown / HTML
|
||||||
|
* [#MUI-30] Création d'un composant email
|
||||||
|
* [#MUI-31] Création d'un composant téléphone
|
||||||
|
* [#MUI-32] Création d'un composant saisie assistée (autocomplete)
|
||||||
|
* [#MUI-34] Revoir le système de playground
|
||||||
|
* [#MUI-33] Développer le composant Datepicker
|
||||||
|
* [#MUI-33] Création du composant DateTime (date + heure, sélecteur d'heure natif intérimaire)
|
||||||
|
* [#MUI-36] Création d'un composant modal (dialogue centré, focus-trap, scroll-lock, footer fixe)
|
||||||
|
* [#MUI-37] Création d'un composant accordéon
|
||||||
|
* [#MUI-39] Création d'un sélecteur d'heure à molettes (MalioTimePicker) ; DateTime rebranché dessus (remplace l'input time natif intérimaire)
|
||||||
|
* InputAutocomplete : prop `localFilter` pour le filtrage côté client des listes statiques (case-insensitive `label.includes(query)`), sans avoir à brancher `@search`
|
||||||
|
* InputTextArea : la scrollbar passe en primary (bleu) au focus, comme la liste du Select
|
||||||
|
* Token Tailwind partagé `w-m-btn-action` (150px) exposé via `tailwind.config.ts` + CSS var `--m-btn-action-width` dans `malio.css` — utilisable côté consommateur pour les boutons d'action (`<MalioButton button-class="w-m-btn-action" />`), themable en redéfinissant la CSS var
|
||||||
|
* [#MUI-41] Prop `required` cohérente + astérisque rouge dans le label sur la famille formulaire (Select, SelectCheckbox, InputUpload, InputRichText gagnent la prop)
|
||||||
|
* [#MUI-41] InputEmail : sanitisation à la saisie (suppression des espaces, option `lowercase`)
|
||||||
|
* [#MUI-42] MalioDate : saisie clavier `JJ/MM/AAAA` optionnelle (prop `editable`, masque maska, validation au blur, message `invalidMessage`)
|
||||||
|
* InputEmail : bouton `+` d'ajout optionnel (prop `addable`, event `add`), calqué sur InputPhone ; l'icône email passe à gauche quand le bouton est actif
|
||||||
|
* InputAmount : affichage groupé des milliers à la française (`1 234 567,89`) en temps réel ; `modelValue` reste propre (`'1234567.89'`) ; `maxLength` borne la longueur du modèle
|
||||||
|
* [#MUI-42] Anneau de focus clavier standardisé (`outline` 2px `m-primary`, offset 2px) affiché **uniquement** à la navigation clavier (jamais au clic souris), sur l'ensemble des champs et contrôles : inputs (Text, Email, Password, Phone, Amount, Number + boutons ±, Upload, TextArea, Autocomplete), Select, SelectCheckbox, famille Date (Date, DateRange, DateTime, DateWeek), Button, ButtonIcon. Mécanique : composable `useKbdFocusRing` (détection de modalité clavier/souris) + utilitaires CSS `.m-focus-ring` (éléments à `:focus-visible` natif) et `.m-focus-ring-kbd` (champs texte, où `:focus-visible` se déclenche aussi à la souris)
|
||||||
|
* [#MUI-42] Anneau « combo » : quand un dropdown / calendrier est ouvert (Autocomplete, Select, SelectCheckbox, Date), l'anneau entoure le champ **et** la liste / le calendrier d'un seul tenant, adapté au sens d'ouverture (utilitaires `.m-combo-ring-top` / `.m-combo-ring-bottom`)
|
||||||
|
* [#MUI-42] Navigation clavier WAI-ARIA APG sur les listes déroulantes : Select et SelectCheckbox gagnent la navigation (flèches, Home/End, Entrée/Espace, Échap, Tab — absente jusque-là), avec scroll automatique de l'option active et `aria-activedescendant` ; InputAutocomplete complété (scroll auto, ArrowUp ouvre sur la dernière option, Home/End, Tab ferme)
|
||||||
|
* [#MUI-42] SelectCheckbox : la ligne « Tout sélectionner » est intégrée à la navigation clavier ; le clic sur toute la ligne d'option (et plus seulement le label) coche/décoche
|
||||||
|
* [#MUI-42] InputUpload : prop `clearable` (croix `mdi:close` focusable qui vide le champ + event `clear`) et ouverture du sélecteur de fichier au clavier (Entrée / Espace)
|
||||||
|
* [#MUI-42] Famille Date : ouverture du calendrier au clavier (Entrée / Espace), fermeture par Échap
|
||||||
|
* [#MUI-43] MalioDate : event `update:valid` (booléen) exposant l'état de validité de la saisie (`false` sur date malformée ou hors `min`/`max`, qui n'émet pas `modelValue`) — permet au parent de bloquer le submit ; la validité ne couvre pas `required` (champ vide = valide)
|
||||||
|
* [#MUI-43] MalioDateTime : saisie clavier `JJ/MM/AAAA HH:MM` optionnelle (prop `editable`, masque maska, `invalidMessage`) + même event `update:valid` que MalioDate (mêmes règles, émis dès le montage). Nouveau parseur `parseDisplayToIsoDateTime`.
|
||||||
|
* [#MUI-43] Famille Date editable (MalioDate, MalioDateTime) : gabarit fantôme progressif — le format (`JJ/MM/AAAA` / `JJ/MM/AAAA HH:MM`) s'affiche en gris et se remplit au fil de la saisie (tapé en noir, reste en gris) ; séparateurs (`/`, espace, `:`) posés automatiquement dès qu'un groupe est complet (maska `eager`). CalendarField : prop `placeholderTemplate` (le masque maska en est dérivé), remplace l'ancienne mécanique de masque codé en dur.
|
||||||
|
* [#MUI-43] CalendarField : la croix d'effacement réinitialise désormais la saisie clavier même après une date invalide (le `v-model` restant `null`, le champ se vidait pas).
|
||||||
|
* [#MUI-44] MalioDate / MalioDateTime : event `update:rawValue` (string) exposant la saisie brute sur un canal séparé pour la validation back-autoritative — saisie invalide (non parsable ou hors `min`/`max`) → texte trimmé tel que tapé, saisie valide/vide + clear + sélection au calendrier → `''`. `modelValue` reste `string` ISO `| null` (la saisie invalide n'y transite jamais) ; le parent construit son payload via `valid ? modelValue : rawValue`.
|
||||||
|
* [#MUI-45] MalioDate : prop `markedDates` (`Record<"YYYY-MM-DD", 'success' | 'danger'>`) appliquant un fond tokenisé par jour dans la grille (générique, fourni par le consommateur ; précédence sélection/`today` > variante marquée > défaut) + event `month-change` (`{ month: 0-11, year }`) émis à l'ouverture du popover et à chaque navigation de mois. Sert l'écran *Heures* de SIRH (jours validés en vert, chargement du mois visible à la volée).
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
* Cohérence du mode **`disabled`** sur toute la famille formulaire (calqué sur InputText : texte + label grisés, `cursor-not-allowed`, aucune affordance interactive). Concrètement, quand `disabled` : le **bouton « + »** d'ajout disparaît (InputPhone, InputEmail), l'**œil** de révélation disparaît (InputPassword), le **chevron** disparaît (Select, SelectCheckbox, InputAutocomplete), la **croix d'effacement** reste masquée (date, upload, time), le **label** passe en `text-m-muted` (Select, SelectCheckbox, famille Date via CalendarField, TimePicker), et les **tags** du SelectCheckbox + la valeur du Select passent en gris. (InputText, InputAmount, InputNumber, InputTextArea, InputRichText, Checkbox, RadioButton, InputUpload étaient déjà conformes.)
|
||||||
|
* TabList : le nombre d'onglets visibles en mode fenêtré s'**adapte automatiquement à la largeur réelle** (mesure via `ResizeObserver` + ligne de mesure cachée), au lieu d'un `maxVisibleTabs` fixe qui pouvait faire déborder les onglets sur les chevrons. Les chevrons restent fixés aux bords et le nombre affiché est choisi pour que les onglets tiennent (pas de chevauchement ni de rognage). `maxVisibleTabs` devient un **plafond optionnel**. Calcul isolé dans une fonction pure testable (`tabFit.ts`, basée sur les largeurs réelles des onglets). Sans layout (SSR), repli sur le plafond / tous les onglets. **Breaking** : la prop `maxWidth` est supprimée (la barre utilise désormais toute la largeur disponible au lieu d'être plafonnée à 1100px).
|
||||||
|
* TabList : au **survol** d'un onglet inactif, on applique désormais le même style que l'onglet actif — texte `m-primary` plein + barre soulignée `m-primary` (`hover:after:*`) — au lieu du discret `text-m-primary/70`, pour bien marquer la cible.
|
||||||
|
* Sidebar : états visuels des liens de navigation — **survol** : highlight pleine largeur entièrement porté par le `<li>` (fond `m-primary` à 10 % + texte `m-primary` + semi-bold, `hover:bg-m-primary/10 hover:text-m-primary hover:font-semibold`, espacement `pt-1 pb-1`). La couleur de base (`text-black`) est aussi sur le `<li>` et le `<a>` ne fige plus sa couleur (il hérite) : sinon, sur les bandes `pt-1`/`pb-1` situées hors du `<a>`, le fond devenait bleu mais le texte restait noir. **Lien actif** : texte `m-primary` + semi-bold, sans fond (`active-class="!text-m-primary font-semibold"` ; `!important` car `active-class` est hors `twMerge`).
|
||||||
|
* DataTable : libellés de pagination en français — `Préc.` / `Suiv.` (étaient `Prev` / `Next`) ; aria-labels déjà en français inchangés.
|
||||||
|
* MalioButton : dimensions par défaut `w-[180px]` / `h-[38px]` (étaient `w-[200px]` / `h-[40px]`).
|
||||||
|
* DataTable : tailles par défaut revues — texte header `16px` (était `20px`), texte body `14px` (était `18px`), sélecteur de lignes et boutons de pagination (Prev / numéros / Next) alignés à `30px` de haut, padding de `12px` entre le bas du tableau et la barre de pagination, texte header et body passés en noir (`text-black`, étaient `text-m-primary`).
|
||||||
|
* Select : nouvelle prop `fieldClass` pour surcharger les classes du field (notamment la hauteur `h-[40px]` jusqu'ici codée en dur) ; utilisée par le DataTable pour passer le sélecteur de perPage à `30px`.
|
||||||
|
* [#MUI-35] Refonte du composant drawer : slots `#header`/`#footer`, prop `side` (droite/gauche), `dismissable`, `closeOnEscape`, classes d'override, focus-trap, scroll-lock et fermeture au clavier. **Breaking** : la prop `title` est remplacée par le slot `#header`.
|
||||||
|
* [#MUI-42] Button / ButtonIcon : l'anneau de focus passe du halo `ring-2 ring-m-primary/50` à l'anneau standard `.m-focus-ring` (outline plein, offset 2px), pour l'homogénéité avec les autres composants.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
* Sidebar : le **lien actif** reste actif sur les **sous-routes** (match par préfixe via `useRoute().path` au lieu de l'`active-class` de NuxtLink qui dépendait de l'imbrication des routes) — ex. `/supplier` reste surligné sur `/supplier/1/edit`. Nouvelle option `exact: true` par item pour forcer le match strict.
|
||||||
|
* Famille Date (CalendarField) : le **clic sur le picto calendrier** ouvre désormais le popover (le `<Icon>` en overlay absolu interceptait le clic sans le traiter, et ne le laissait pas retomber sur l'input). Couvre Date, DateTime, DateRange, DateWeek. La croix d'effacement conserve son comportement (efface sans ouvrir).
|
||||||
|
* Famille Date editable (MalioDate, MalioDateTime) : la saisie clavier est désormais **bornée par champ** sur le premier **et** le second chiffre (jour `01-31`, mois `01-12`, heure `00-23`, minute `00-59`) — une valeur hors plage (`99/99/9999`, un jour `33`, un mois `19`…) ne peut plus être tapée (auparavant saisissable puis rejetée a posteriori par la validation). Les impossibilités calendaires fines (`31/02`, 29/02 non bissextile, hors `min`/`max`) restent captées par la validation. Implémenté via `buildBoundedMask(template)` (CalendarField) : un `preProcess` maska valide chaque champ progressivement (un chiffre n'est accepté que s'il reste une complétion valide dans la plage) ; il distingue le mois des minutes (même lettre `M`) selon la présence d'heures dans le gabarit.
|
||||||
|
* DataTable : pagination réalignée verticalement après l'introduction du `min-h-[1rem]` du Select — la barre pagination passe en `items-center`, et le MalioSelect du sélecteur de perPage est encapsulé dans un wrapper `h-12` qui borne sa taille flex à la hauteur du field (le slot vide déborde invisiblement en dessous). Span « Lignes : » et boutons Prev/Page/Next sont désormais centrés exactement sur le field (y=24)
|
||||||
|
* Drawer : le slot `#footer` est désormais rendu hors de la zone scrollable (épinglé en bas, comme la modal) ; seul le body défile et la scrollbar ne s'étend plus derrière le footer
|
||||||
* Hauteur des boutons de pagination du datatable alignée sur le select (40px)
|
* Hauteur des boutons de pagination du datatable alignée sur le select (40px)
|
||||||
* Distribution de `tailwind.config.ts` aux projets consommateurs avec paths `content` absolus
|
* Distribution de `tailwind.config.ts` aux projets consommateurs avec paths `content` absolus
|
||||||
|
* Espace réservé (`min-h-[1rem]`) pour le paragraphe hint/error/success de 15 composants (Input*, Select*, Time*, CalendarField, Checkbox) — l'apparition d'une erreur ne décale plus les cellules voisines dans une grille
|
||||||
|
* InputPhone : la croix `+` (add button) suit la même cascade d'état que les autres icônes du champ (muted / primary en focus / black quand rempli / danger / success) au lieu d'être figée en primary
|
||||||
|
* Select / SelectCheckbox : le chevron suit l'état du champ (muted par défaut, primary à l'ouverture, black avec une option sélectionnée, danger / success en cas d'erreur ou succès) au lieu de `text-current`
|
||||||
|
* InputTextArea : composant single-root (était multi-root) — le wrapper du message ne prend plus sa propre cellule de grille, `row-span-2` fonctionne à nouveau
|
||||||
|
* Label désactivé en `text-m-muted` (gris des bordures) au lieu de `text-black/60` sur les inputs à floating-label (InputText, Email, Password, Amount, Phone, Upload, Autocomplete, TextArea, RichText)
|
||||||
|
* InputAutocomplete : suppression de 4 sources de saut visuel au focus / ouverture (extra translate label, padding `grow-height:focus`, `focus:pl-[11px]`, `!border-b-0` remplacé par `!border-b-transparent`)
|
||||||
|
* Select / SelectCheckbox : mêmes correctifs anti-saut (suppression du padding `grow-height:focus` et remplacement de `!border-b-0` / `!border-t-0` par leurs variantes `transparent`)
|
||||||
|
* MalioButton : largeur par défaut alignée sur `w-[200px]` (au lieu de `w-[240px]`) pour correspondre au sizing des formulaires de l'app
|
||||||
|
* [#MUI-42] RadioButton : ajout d'un focus visible au clavier (`outline` 2px `m-primary`, offset 2px) — l'input en `appearance-none` n'avait aucun indicateur de focus, seul l'`outline: auto 1px` du navigateur restait, quasi invisible. La navigation native (Tab entre groupes, flèches dans le groupe) reste inchangée
|
||||||
|
* [#MUI-42] Checkbox : ajout d'un focus visible au clavier sur la case (`outline` 2px `m-primary`, offset 2px) — l'input réel est masqué (`clip-path`), aucun indicateur n'apparaissait à la tabulation
|
||||||
|
* [#MUI-42] Select : le focus reste sur le bouton après sélection (un `blur()` renvoyait le focus au `body`, cassant la tabulation clavier — un Tab repartait du haut de page)
|
||||||
|
|||||||
+629
-19
@@ -2,6 +2,10 @@
|
|||||||
|
|
||||||
Tous les composants sont auto-importés avec le préfixe `Malio`. Utiliser `v-model` pour le binding bidirectionnel sur les composants de formulaire.
|
Tous les composants sont auto-importés avec le préfixe `Malio`. Utiliser `v-model` pour le binding bidirectionnel sur les composants de formulaire.
|
||||||
|
|
||||||
|
> **Champ obligatoire :** sur les composants de formulaire, la prop `required` ajoute un astérisque rouge dans le label. C'est un repère visuel ; la sémantique « obligatoire » est portée par l'attribut natif `required` ou `aria-required`.
|
||||||
|
|
||||||
|
> **Focus clavier :** tous les champs et contrôles affichent un anneau de focus (`outline` 2px `m-primary`, offset 2px) **uniquement** à la navigation clavier (Tab), jamais au clic souris. Sur les composants à dropdown/calendrier ouverts, l'anneau entoure le champ et la liste d'un seul tenant. Voir la note « Clavier » de chaque composant pour la navigation détaillée.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## MalioInputText
|
## MalioInputText
|
||||||
@@ -15,10 +19,11 @@ Champ texte avec label, icône optionnelle et support de masque de saisie.
|
|||||||
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
||||||
| `disabled` | `boolean` | `false` | Désactive le champ |
|
| `disabled` | `boolean` | `false` | Désactive le champ |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule |
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
| `required` | `boolean` | `false` | Champ requis |
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `hint` | `string` | `''` | Message d'aide |
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
| `success` | `string` | `''` | Message de succès |
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
| `iconName` | `string` | `''` | Icône Iconify (ex: `mdi:magnify`) |
|
| `iconName` | `string` | `''` | Icône Iconify (ex: `mdi:magnify`) |
|
||||||
| `iconPosition` | `'left' \| 'right'` | `'right'` | Position de l'icône |
|
| `iconPosition` | `'left' \| 'right'` | `'right'` | Position de l'icône |
|
||||||
| `iconSize` | `string \| number` | `24` | Taille icône |
|
| `iconSize` | `string \| number` | `24` | Taille icône |
|
||||||
@@ -53,9 +58,11 @@ Champ mot de passe avec toggle visibilité.
|
|||||||
| `displayIcon` | `boolean` | `true` | Afficher l'icône toggle |
|
| `displayIcon` | `boolean` | `true` | Afficher l'icône toggle |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule |
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `hint` | `string` | `''` | Message d'aide |
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
| `success` | `string` | `''` | Message de succès |
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
@@ -66,23 +73,196 @@ Champ mot de passe avec toggle visibilité.
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## MalioInputEmail
|
||||||
|
|
||||||
|
Champ email (`type="email"` + `inputmode="email"`) avec icône `mdi:email-outline` à droite par défaut.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `label` | `string` | `''` | Label du champ |
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `autocomplete` | `string` | `'off'` | Autocomplétion (passer `'email'` pour suggérer l'email utilisateur) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactive le champ |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `lowercase` | `boolean` | `false` | Normalise la saisie en minuscules à la frappe |
|
||||||
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `iconName` | `string` | `'mdi:email-outline'` | Icône Iconify (chaîne vide pour masquer) |
|
||||||
|
| `iconPosition` | `'left' \| 'right'` | `'right'` | Position de l'icône |
|
||||||
|
| `iconSize` | `string \| number` | `24` | Taille icône |
|
||||||
|
| `iconColor` | `string` | `'text-m-muted'` | Classe couleur icône |
|
||||||
|
| `addable` | `boolean` | `false` | Affiche un bouton `+` à droite qui émet l'event `add` (l'icône email passe à gauche) |
|
||||||
|
| `addIconName` | `string` | `'mdi:plus'` | Icône Iconify du bouton d'ajout |
|
||||||
|
| `addButtonLabel` | `string` | `'Ajouter une adresse email'` | aria-label du bouton d'ajout |
|
||||||
|
| `inputClass` | `string` | `''` | Classes CSS input |
|
||||||
|
| `labelClass` | `string` | `''` | Classes CSS label |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS conteneur |
|
||||||
|
|
||||||
|
> **Sanitisation à la saisie :** tous les espaces sont supprimés automatiquement au fil de la frappe (sans masque). Avec `lowercase=true`, la valeur est également convertie en minuscules à la frappe. La validation du format (ex. présence d'un `@`) reste à la charge du parent via la prop `error` ou la couche de validation.
|
||||||
|
|
||||||
|
**Events :**
|
||||||
|
- `update:modelValue(value: string)`
|
||||||
|
- `add()` — émis au clic du bouton `+` (uniquement si `addable`, non `disabled`, non `readonly`)
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioInputEmail v-model="email" label="Adresse email" />
|
||||||
|
<MalioInputEmail v-model="email" label="Email" autocomplete="email" />
|
||||||
|
<MalioInputEmail v-model="email" label="Email" :icon-name="''" />
|
||||||
|
<MalioInputEmail v-model="email" label="Email" error="Adresse email invalide" />
|
||||||
|
<MalioInputEmail v-model="email" label="Email" addable @add="addEmailField" />
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioInputPhone
|
||||||
|
|
||||||
|
Champ téléphone (`type="tel"` + `inputmode="tel"`) avec icône `mdi:phone-outline` à gauche par défaut et bouton `+` optionnel à droite pour gérer une liste de numéros côté parent.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `label` | `string` | `''` | Label du champ |
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `autocomplete` | `string` | `'off'` | Autocomplétion (passer `'tel'` pour suggérer un numéro enregistré) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactive le champ et le bouton + |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule (désactive aussi le bouton +) |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `iconName` | `string` | `'mdi:phone-outline'` | Icône Iconify (chaîne vide pour masquer) |
|
||||||
|
| `iconPosition` | `'left' \| 'right'` | `'left'` | Position de l'icône |
|
||||||
|
| `iconSize` | `string \| number` | `24` | Taille icône |
|
||||||
|
| `iconColor` | `string` | `'text-m-muted'` | Classe couleur icône |
|
||||||
|
| `mask` | `string \| MaskInputOptions` | `undefined` | Masque maska (aucun par défaut, utile pour mono-pays) |
|
||||||
|
| `addable` | `boolean` | `false` | Affiche un bouton à droite qui émet l'event `add` |
|
||||||
|
| `addIconName` | `string` | `'mdi:plus'` | Icône Iconify du bouton d'ajout |
|
||||||
|
| `addButtonLabel` | `string` | `'Ajouter un numéro'` | aria-label du bouton d'ajout |
|
||||||
|
| `inputClass` | `string` | `''` | Classes CSS input |
|
||||||
|
| `labelClass` | `string` | `''` | Classes CSS label |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS conteneur |
|
||||||
|
|
||||||
|
**Events :**
|
||||||
|
- `update:modelValue(value: string)`
|
||||||
|
- `add()` — émis au clic du bouton `+` (uniquement si `addable`, non `disabled`, non `readonly`)
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioInputPhone v-model="phone" label="Téléphone" />
|
||||||
|
<MalioInputPhone v-model="phone" label="Téléphone (FR)" mask="+33 # ## ## ## ##" />
|
||||||
|
<MalioInputPhone v-model="phone" label="Téléphone" addable @add="addPhoneField" />
|
||||||
|
<MalioInputPhone v-model="phone" label="Téléphone" error="Numéro invalide" />
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioInputAutocomplete
|
||||||
|
|
||||||
|
Champ de saisie assistée (typeahead / combobox) : l'utilisateur tape pour filtrer une liste d'options, ou pour déclencher une recherche côté parent (API). Par défaut le parent alimente `options` et `loading` en réponse à l'event `search` — c'est lui qui gère l'appel API, l'auth, la transformation et le cache. Pour une liste **statique** courte, activer `localFilter` fait filtrer le composant lui-même (case-insensitive `label.includes(query)`) sans avoir à brancher `@search`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `modelValue` | `string \| number \| null` | `undefined` | Valeur sélectionnée (v-model) |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `options` | `{label: string; value: string\|number}[]` | `[]` | Liste affichée dans le dropdown |
|
||||||
|
| `loading` | `boolean` | `false` | Affiche un spinner + un message de chargement |
|
||||||
|
| `debounce` | `number` | `300` | Délai (ms) avant émission de `search` |
|
||||||
|
| `minSearchLength` | `number` | `0` | Caractères mini avant d'émettre `search` |
|
||||||
|
| `allowCreate` | `boolean` | `false` | Autorise la saisie libre validée par Entrée (émet `create`) |
|
||||||
|
| `localFilter` | `boolean` | `false` | Filtre `options` côté client par sous-chaîne du label (case-insensitive). À utiliser pour les listes statiques courtes ; en mode API on laisse `false` et le parent répond à `@search`. |
|
||||||
|
| `iconName` | `string` | `''` | Icône Iconify décorative |
|
||||||
|
| `iconPosition` | `'left' \| 'right'` | `'left'` | Position de l'icône décorative |
|
||||||
|
| `iconSize` | `string \| number` | `24` | Taille de l'icône |
|
||||||
|
| `iconColor` | `string` | `'text-m-muted'` | Classe couleur de l'icône |
|
||||||
|
| `noResultsText` | `string` | `'Aucun résultat'` | Texte affiché quand `options` est vide |
|
||||||
|
| `loadingText` | `string` | `'Chargement…'` | Texte affiché pendant le chargement |
|
||||||
|
| `minSearchText` | `string` | `'Tapez pour rechercher'` | Texte affiché tant que `minSearchLength` n'est pas atteint |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactive le champ et empêche l'ouverture |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule (n'ouvre pas le dropdown) |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur (prioritaire) |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` | `string` | `''` | Classes CSS input |
|
||||||
|
| `labelClass` | `string` | `''` | Classes CSS label |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS conteneur |
|
||||||
|
|
||||||
|
**Events :**
|
||||||
|
- `update:modelValue(value: string \| number \| null)` — valeur sélectionnée (v-model)
|
||||||
|
- `search(query: string)` — émis (après debounce + minSearchLength) avec le texte tapé ; le parent l'écoute pour lancer son fetch API
|
||||||
|
- `select(option: Option \| null)` — émis avec l'objet `Option` complet (utile pour récupérer aussi le `label`)
|
||||||
|
- `create(value: string)` — émis quand `allowCreate=true` et que l'utilisateur valide une valeur libre
|
||||||
|
|
||||||
|
**Clavier (WAI-ARIA APG) :** `↓` ouvre / option suivante, `↑` précédente (ou ouvre sur la dernière option si fermé), `Début`/`Fin`, scroll automatique de l'option active, `Entrée` sélection (ou création), `Échap` annule, `Tab` ferme. Anneau de focus clavier (combo champ + liste à l'ouverture).
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<!-- Usage statique (filtrage côté client via local-filter) -->
|
||||||
|
<MalioInputAutocomplete v-model="country" label="Pays" :options="countries" local-filter />
|
||||||
|
|
||||||
|
<!-- Usage API (parent gère le fetch) -->
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="clientId"
|
||||||
|
label="Client"
|
||||||
|
:options="clientOptions"
|
||||||
|
:loading="isFetching"
|
||||||
|
:min-search-length="2"
|
||||||
|
@search="onSearchClients"
|
||||||
|
@select="onSelectClient"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Avec création libre -->
|
||||||
|
<MalioInputAutocomplete
|
||||||
|
v-model="category"
|
||||||
|
label="Catégorie"
|
||||||
|
:options="categories"
|
||||||
|
allow-create
|
||||||
|
@create="onCreateCategory"
|
||||||
|
/>
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts
|
||||||
|
async function onSearchClients(query: string) {
|
||||||
|
isFetching.value = true
|
||||||
|
const res = await $fetch('/api/clients', {params: {q: query}})
|
||||||
|
clientOptions.value = res.map(c => ({label: c.name, value: c.id}))
|
||||||
|
isFetching.value = false
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## MalioInputAmount
|
## MalioInputAmount
|
||||||
|
|
||||||
Champ montant avec icône devise (euro par défaut).
|
Champ montant avec icône devise (euro par défaut).
|
||||||
|
|
||||||
|
L'affichage est groupé à la française (`1 234 567,89` : espace pour les milliers, virgule décimale), mis à jour en temps réel pendant la saisie. La valeur émise (`modelValue`) reste une **chaîne numérique propre** (point décimal, sans espaces, ex. `'1234567.89'`). `maxLength` borne la longueur de cette chaîne propre (pas de l'affichage).
|
||||||
|
|
||||||
| Prop | Type | Défaut | Description |
|
| Prop | Type | Défaut | Description |
|
||||||
|------|------|--------|-------------|
|
|------|------|--------|-------------|
|
||||||
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
| `modelValue` | `string \| null` | `undefined` | Valeur (v-model) |
|
||||||
| `label` | `string` | `''` | Label |
|
| `label` | `string` | `''` | Label |
|
||||||
| `iconName` | `string` | `'mdi:currency-eur'` | Icône devise |
|
| `iconName` | `string` | `'mdi:currency-eur'` | Icône devise |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioInputAmount v-model="montant" label="Montant TTC" />
|
<MalioInputAmount v-model="montant" label="Montant TTC" />
|
||||||
<MalioInputAmount v-model="prix" label="Prix" error="Montant invalide" />
|
<MalioInputAmount v-model="prix" label="Prix" error="Montant invalide" />
|
||||||
|
<MalioInputAmount v-model="gros" label="Budget" />
|
||||||
|
<!-- saisie 1234567.89 → affiché "1 234 567,89", modelValue "1234567.89" -->
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -98,7 +278,9 @@ Champ numérique avec boutons +/-.
|
|||||||
| `min` | `number \| string` | — | Valeur minimum |
|
| `min` | `number \| string` | — | Valeur minimum |
|
||||||
| `max` | `number \| string` | — | Valeur maximum |
|
| `max` | `number \| string` | — | Valeur maximum |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
@@ -121,7 +303,10 @@ Zone de texte multiligne avec compteur et redimensionnement.
|
|||||||
| `maxLength` | `number` | `800` | Longueur max |
|
| `maxLength` | `number` | `800` | Longueur max |
|
||||||
| `showCounter` | `boolean` | `false` | Afficher le compteur |
|
| `showCounter` | `boolean` | `false` | Afficher le compteur |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS sur la div conteneur (utile pour `row-span-*`, `col-span-*`, etc.) |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
@@ -148,9 +333,11 @@ Zone de texte multiligne avec compteur et redimensionnement.
|
|||||||
| `editable` | `boolean` | `true` | `false` → mode affichage seul (toolbar masquée) |
|
| `editable` | `boolean` | `true` | `false` → mode affichage seul (toolbar masquée) |
|
||||||
| `disabled` | `boolean` | `false` | Désactive l'édition et la toolbar |
|
| `disabled` | `boolean` | `false` | Désactive l'édition et la toolbar |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule (toolbar visible mais désactivée) |
|
| `readonly` | `boolean` | `false` | Lecture seule (toolbar visible mais désactivée) |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `hint` | `string` | `''` | Message d'aide |
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
| `success` | `string` | `''` | Message de succès |
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
| `outputFormat` | `'markdown' \| 'html'` | `'html'` | Format émis dans `update:modelValue` |
|
| `outputFormat` | `'markdown' \| 'html'` | `'html'` | Format émis dans `update:modelValue` |
|
||||||
| `groupClass` | `string` | `''` | Classes CSS conteneur (twMerge) |
|
| `groupClass` | `string` | `''` | Classes CSS conteneur (twMerge) |
|
||||||
| `labelClass` | `string` | `''` | Classes CSS label (twMerge) |
|
| `labelClass` | `string` | `''` | Classes CSS label (twMerge) |
|
||||||
@@ -177,13 +364,20 @@ Champ d'upload de fichier.
|
|||||||
| `label` | `string` | `''` | Label |
|
| `label` | `string` | `''` | Label |
|
||||||
| `accept` | `string` | `''` | Types de fichiers acceptés |
|
| `accept` | `string` | `''` | Types de fichiers acceptés |
|
||||||
| `displayIcon` | `boolean` | `true` | Afficher l'icône |
|
| `displayIcon` | `boolean` | `true` | Afficher l'icône |
|
||||||
|
| `clearable` | `boolean` | `false` | Affiche une croix (`mdi:close`) focusable qui vide le champ quand un fichier est sélectionné |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Champ en lecture seule (bordure noire, pas de focus bleu/grossissement, label/icône gris→noir selon rempli). |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`, `file-selected(file: File)`
|
**Events :** `update:modelValue(value: string)`, `file-selected(file: File)`, `clear()`
|
||||||
|
|
||||||
|
**Clavier :** `Entrée` / `Espace` ouvrent le sélecteur de fichier. La croix `clearable` est focusable (anneau clavier, `Entrée`/`Espace`).
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioInputUpload v-model="fileName" label="Document" accept=".pdf,.doc" @file-selected="onFile" />
|
<MalioInputUpload v-model="fileName" label="Document" accept=".pdf,.doc" @file-selected="onFile" />
|
||||||
|
<MalioInputUpload v-model="fileName" label="Document" clearable @clear="onClear" />
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -201,18 +395,23 @@ Liste déroulante.
|
|||||||
| `hint` | `string` | `''` | Message d'aide |
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
| `success` | `string` | `''` | Message de succès |
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Champ en lecture seule (bordure noire, pas de focus bleu/grossissement, label/icône gris→noir selon rempli). |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `groupClass` | `string` | `''` | Classes CSS conteneur (twMerge) |
|
| `groupClass` | `string` | `''` | Classes CSS conteneur (twMerge) |
|
||||||
| `minWidth` | `string` | `'w-96'` | Classe largeur minimum |
|
|
||||||
| `maxWidth` | `string` | `''` | Classe largeur maximum |
|
|
||||||
| `rounded` | `string` | `'rounded-md'` | Classe border-radius |
|
| `rounded` | `string` | `'rounded-md'` | Classe border-radius |
|
||||||
| `textField` | `string` | `'text-lg'` | Classe taille texte bouton |
|
| `textField` | `string` | `'text-lg'` | Classe taille texte bouton |
|
||||||
| `textValue` | `string` | `'text-lg'` | Classe taille texte valeur |
|
| `textValue` | `string` | `'text-lg'` | Classe taille texte valeur |
|
||||||
| `textLabel` | `string` | `'text-sm'` | Classe taille texte label |
|
| `textLabel` | `string` | `'text-sm'` | Classe taille texte label |
|
||||||
|
| `fieldClass` | `string` | `''` | Classes supplémentaires sur le field (override hauteur, ex. `h-[30px]`) |
|
||||||
|
| `noOptionsText` | `string` | `'Aucune option disponible'` | Message affiché dans la dropdown quand `options` est vide |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string | number | null)`
|
**Events :** `update:modelValue(value: string | number | null)`
|
||||||
**Slots :** `icon` (icône dropdown custom)
|
**Slots :** `icon` (icône dropdown custom)
|
||||||
|
|
||||||
|
**Clavier (WAI-ARIA APG) :** `↓`/`↑`/`Entrée`/`Espace` ouvrent ; liste ouverte → `↑↓` naviguent (scroll auto de l'option active), `Début`/`Fin`, `Entrée`/`Espace` sélectionnent, `Échap`/`Tab` ferment. Le focus reste sur le bouton après sélection. Anneau de focus clavier (combo bouton + liste à l'ouverture, adapté au sens haut/bas).
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioSelect v-model="pays" label="Pays" :options="[{ value: 'FR', text: 'France' }, { value: 'BE', text: 'Belgique' }]" />
|
<MalioSelect v-model="pays" label="Pays" :options="[{ value: 'FR', text: 'France' }, { value: 'BE', text: 'Belgique' }]" />
|
||||||
<MalioSelect v-model="ville" label="Ville" :options="villes" empty-option-label="Choisir..." />
|
<MalioSelect v-model="ville" label="Ville" :options="villes" empty-option-label="Choisir..." />
|
||||||
@@ -227,16 +426,22 @@ Liste déroulante multi-sélection avec checkboxes.
|
|||||||
|
|
||||||
| Prop | Type | Défaut | Description |
|
| Prop | Type | Défaut | Description |
|
||||||
|------|------|--------|-------------|
|
|------|------|--------|-------------|
|
||||||
| `modelValue` | `(string \| number)[]` | **requis** | Valeurs sélectionnées (v-model) |
|
| `modelValue` | `(string \| number)[]` | `[]` | Valeurs sélectionnées (v-model) |
|
||||||
| `options` | `{ value: string \| number, text: string }[]` | `[]` | Options |
|
| `options` | `{ value: string \| number, text: string }[]` | `[]` | Options |
|
||||||
| `displayTag` | `boolean` | `false` | Afficher les tags sélectionnés |
|
| `displayTag` | `boolean` | `false` | Afficher les tags sélectionnés |
|
||||||
| `displaySelectAll` | `boolean` | `false` | Afficher "Tout sélectionner" |
|
| `displaySelectAll` | `boolean` | `false` | Afficher "Tout sélectionner" |
|
||||||
| `selectAllLabel` | `string` | `'Tout sélectionner'` | Texte du sélecteur global |
|
| `selectAllLabel` | `string` | `'Tout sélectionner'` | Texte du sélecteur global |
|
||||||
| `label` | `string` | `''` | Label |
|
| `label` | `string` | `''` | Label |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Champ en lecture seule (bordure noire, pas de focus bleu/grossissement, label/icône gris→noir selon rempli). |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `noOptionsText` | `string` | `'Aucune option disponible'` | Message affiché dans la dropdown quand `options` est vide |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: (string | number)[])`
|
**Events :** `update:modelValue(value: (string | number)[])`
|
||||||
|
|
||||||
|
**Clavier (WAI-ARIA APG) :** `↓`/`↑`/`Entrée`/`Espace` ouvrent ; liste ouverte → `↑↓` naviguent (scroll auto), `Début`/`Fin`, `Entrée`/`Espace` cochent/décochent l'option active (la liste **reste ouverte**), `Échap`/`Tab` ferment. La ligne « Tout sélectionner » est navigable au clavier. Le clic sur toute la ligne (pas que le label) coche/décoche. Anneau de focus clavier (combo bouton + liste à l'ouverture).
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioSelectCheckbox v-model="competences" label="Compétences" :options="skills" :display-tag="true" />
|
<MalioSelectCheckbox v-model="competences" label="Compétences" :options="skills" :display-tag="true" />
|
||||||
<MalioSelectCheckbox v-model="sites" label="Sites" :options="sitesList" :display-select-all="true" />
|
<MalioSelectCheckbox v-model="sites" label="Sites" :options="sitesList" :display-select-all="true" />
|
||||||
@@ -254,10 +459,14 @@ Case à cocher.
|
|||||||
| `label` | `string` | `''` | Label |
|
| `label` | `string` | `''` | Label |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule |
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: boolean)`
|
**Events :** `update:modelValue(value: boolean)`
|
||||||
|
|
||||||
|
**Clavier :** `Espace` coche/décoche. Focus clavier visible sur la case (`outline` 2px `m-primary`).
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioCheckbox v-model="accepte" label="J'accepte les conditions" />
|
<MalioCheckbox v-model="accepte" label="J'accepte les conditions" />
|
||||||
<MalioCheckbox v-model="newsletter" label="Newsletter" disabled />
|
<MalioCheckbox v-model="newsletter" label="Newsletter" disabled />
|
||||||
@@ -277,9 +486,12 @@ Bouton radio (à utiliser en groupe avec le même `name`).
|
|||||||
| `name` | `string` | `''` | Nom du groupe radio |
|
| `name` | `string` | `''` | Nom du groupe radio |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule |
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string | number | boolean | null)`
|
**Events :** `update:modelValue(value: string | number | boolean | null)`
|
||||||
|
|
||||||
|
**Clavier :** comportement natif d'un groupe radio (options partageant le même `name`) — `Tab` / `Maj+Tab` entre/sort du groupe (1 seul arrêt par groupe), `↑↓←→` déplacent la sélection entre les options d'un même groupe. Focus clavier visible (`outline` 2px `m-primary`).
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioRadioButton v-model="civilite" name="civilite" value="M" label="Monsieur" />
|
<MalioRadioButton v-model="civilite" name="civilite" value="M" label="Monsieur" />
|
||||||
<MalioRadioButton v-model="civilite" name="civilite" value="Mme" label="Madame" />
|
<MalioRadioButton v-model="civilite" name="civilite" value="Mme" label="Madame" />
|
||||||
@@ -287,6 +499,136 @@ Bouton radio (à utiliser en groupe avec le même `name`).
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## MalioDate
|
||||||
|
|
||||||
|
Sélecteur de date unique avec popover (grille de calendrier + vue mois/année).
|
||||||
|
|
||||||
|
La valeur est une chaîne ISO `"YYYY-MM-DD"`. Cliquer un jour émet la date et ferme le popover.
|
||||||
|
|
||||||
|
Avec `editable`, l'utilisateur peut aussi taper la date au clavier. La saisie est **bornée par champ** (1er *et* 2e chiffre) : jour `01-31`, mois `01-12`, heure `00-23`, minute `00-59`, si bien qu'une valeur hors plage (`99/99/9999`, un jour `33`, un mois `19`…) ne peut pas être tapée. Les impossibilités calendaires fines (`31/02`, 29/02 non bissextile, dépassement `min`/`max`) restent captées par la validation, en filet de sécurité. La valeur n'est émise qu'au blur (ou sur Entrée) si elle est valide et dans les bornes ; sinon le texte est conservé et le champ passe en erreur (`invalidMessage`). Un **gabarit fantôme** affiche le format `JJ/MM/AAAA` en gris et se remplit au fur et à mesure de la saisie (caractères tapés en noir, reste du gabarit en gris).
|
||||||
|
|
||||||
|
L'event `update:valid` remonte l'état de validité de la saisie au parent (`true` = vide ou date valide dans les bornes ; `false` = saisie malformée ou hors `min`/`max`). Il est émis **dès le montage** (état d'un champ pré-rempli connu sans interaction) puis à chaque transition. Il permet d'agréger la validité des champs date dans la gate de submit d'un formulaire — une saisie invalide n'émettant pas `modelValue`, c'est le seul signal disponible côté parent. La validité ne couvre **pas** l'obligation `required` (un champ vide reste valide), qui reste à la charge du parent.
|
||||||
|
|
||||||
|
L'event `update:rawValue` expose la **saisie brute** sur un canal séparé, pour les formulaires en validation back-autoritative (le serveur tranche le format et renvoie un `422`). Il est émis à chaque commit : saisie invalide (non parsable ou hors `min`/`max`) → la chaîne trimmée telle que tapée (ex. `"32/13/2026"`) ; saisie valide ou vide, clear, sélection au calendrier → `''`. Le parent construit alors son payload via `valid ? modelValue : rawValue`. La saisie invalide **ne transite jamais** par `modelValue` (qui reste `string` ISO `| null` pour l'affichage et le round-trip) ; `valid` dit *qu'il y a* une erreur, `rawValue` dit *quoi* envoyer.
|
||||||
|
|
||||||
|
La prop `markedDates` permet d'afficher un **statut par jour** dans la grille : un objet `{ "YYYY-MM-DD": "success" | "danger" }` applique un fond tokenisé (`success` → vert clair, `danger` → rouge clair). C'est **purement générique** — aucune logique métier dans le layer : le consommateur fournit la liste des jours à marquer. **Précédence** : un jour sélectionné garde son style primary (fond plein, prime sur la variante marquée) ; le jour courant (`today`) **garde sa bordure** et reçoit **en plus** le fond marqué s'il est dans `markedDates` (vert/rouge bordé) ; sinon, fond marqué simple.
|
||||||
|
|
||||||
|
L'event `month-change` remonte le **mois affiché** dans le popover (`{ month: number /* 0-11 */, year: number }`). Il est émis **à l'ouverture** du popover (sur le mois de la valeur, ou le mois courant) **et à chaque navigation** (chevrons, sélection dans la vue mois). Couplé à `markedDates`, il permet à un consommateur (ex. l'écran *Heures* de SIRH) de charger les statuts du mois visible à la volée : on écoute `@month-change` pour fetch, puis on réinjecte le résultat dans `:marked-dates`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Date ISO `"YYYY-MM-DD"` (v-model) |
|
||||||
|
| `id` | `string` | `''` | Id du champ |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `placeholder` | `string` | `'JJ/MM/AAAA'` | Placeholder |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `hint` | `string` | `''` | Texte d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `min` | `string` | `undefined` | Date min `"YYYY-MM-DD"` (jours antérieurs désactivés) |
|
||||||
|
| `max` | `string` | `undefined` | Date max `"YYYY-MM-DD"` (jours postérieurs désactivés) |
|
||||||
|
| `markedDates` | `Record<string, 'success' \| 'danger'>` | `undefined` | Statut par jour : ISO `"YYYY-MM-DD"` → fond tokenisé. Générique (fourni par le consommateur). |
|
||||||
|
| `clearable` | `boolean` | `true` | Affiche la croix d'effacement |
|
||||||
|
| `editable` | `boolean` | `false` | Autorise la saisie clavier `JJ/MM/AAAA` (masque maska, validation au blur) en plus du calendrier |
|
||||||
|
| `invalidMessage` | `string` | `'Date invalide'` | Message affiché quand la saisie clavier est invalide ou hors `min`/`max` |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` / `labelClass` / `groupClass` | `string` | `''` | Override des classes |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string | null)`, `update:valid(value: boolean)`, `update:rawValue(value: string)`, `month-change(value: { month: number /* 0-11 */, year: number })`
|
||||||
|
|
||||||
|
**Clavier :** `Entrée` / `Espace` ouvrent le calendrier, `Échap` ferme. Anneau de focus clavier (combo champ + calendrier à l'ouverture). La croix d'effacement est focusable. _(Comportement partagé par DateRange, DateTime, DateWeek via le shell CalendarField.)_
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioDate v-model="date" label="Date de naissance" />
|
||||||
|
<!-- date === "2026-05-20" -->
|
||||||
|
<MalioDate v-model="rdv" label="Rendez-vous" :min="todayIso" :max="maxIso" />
|
||||||
|
<MalioDate v-model="date" label="Date de naissance" editable />
|
||||||
|
<MalioDate v-model="date" label="Date de naissance" editable @update:valid="dateValide = $event" />
|
||||||
|
<!-- Validation back-autoritative : on envoie la saisie brute si invalide -->
|
||||||
|
<MalioDate v-model="date" editable @update:valid="valide = $event" @update:rawValue="brut = $event" />
|
||||||
|
<!-- payload : valide ? date : brut -->
|
||||||
|
<!-- Statut par jour + chargement du mois visible (ex. SIRH « Heures ») -->
|
||||||
|
<MalioDate
|
||||||
|
v-model="date"
|
||||||
|
:marked-dates="statutsDuMois"
|
||||||
|
@month-change="({ month, year }) => chargerStatuts(month, year)"
|
||||||
|
/>
|
||||||
|
<!-- statutsDuMois === { "2026-05-05": "success", "2026-05-20": "danger" } -->
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioDateRange
|
||||||
|
|
||||||
|
Sélecteur de **plage de dates** (date de début → date de fin) dans un seul champ. Cliquer un premier jour démarre la plage, le second la termine ; un survol prévisualise la plage.
|
||||||
|
|
||||||
|
La valeur est un objet `{ start: string; end: string }` (dates ISO `"YYYY-MM-DD"`), ou `null`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `modelValue` | `{ start: string; end: string } \| null` | `undefined` | Plage de dates ISO (v-model) |
|
||||||
|
| `id` | `string` | `''` | Id du champ |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `placeholder` | `string` | `'JJ/MM/AAAA'` | Placeholder |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `hint` | `string` | `''` | Texte d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `min` | `string` | `undefined` | Date min `"YYYY-MM-DD"` |
|
||||||
|
| `max` | `string` | `undefined` | Date max `"YYYY-MM-DD"` |
|
||||||
|
| `clearable` | `boolean` | `true` | Affiche la croix d'effacement |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` / `labelClass` / `groupClass` | `string` | `''` | Override des classes |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: { start: string; end: string } | null)`
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioDateRange v-model="periode" label="Période de séjour" />
|
||||||
|
<!-- periode === { start: "2026-05-20", end: "2026-05-27" } -->
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioDateWeek
|
||||||
|
|
||||||
|
Sélecteur de **semaine ISO** : cliquer un jour (ou un numéro de semaine) sélectionne la semaine entière.
|
||||||
|
|
||||||
|
La valeur est une chaîne au format **semaine ISO native** `"YYYY-Www"` (ex. `"2026-W21"`), ou `null`. Le champ affiche `Semaine W (JJ/MM → JJ/MM/AAAA)`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Semaine ISO `"YYYY-Www"` (v-model) |
|
||||||
|
| `id` | `string` | `''` | Id du champ |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `placeholder` | `string` | `'JJ/MM/AAAA'` | Placeholder |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `hint` | `string` | `''` | Texte d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `min` | `string` | `undefined` | Date min `"YYYY-MM-DD"` |
|
||||||
|
| `max` | `string` | `undefined` | Date max `"YYYY-MM-DD"` |
|
||||||
|
| `clearable` | `boolean` | `true` | Affiche la croix d'effacement |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` / `labelClass` / `groupClass` | `string` | `''` | Override des classes |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string | null)`
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioDateWeek v-model="semaine" label="Semaine de livraison" />
|
||||||
|
<!-- semaine === "2026-W21" -->
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## MalioTime
|
## MalioTime
|
||||||
|
|
||||||
Sélecteur d'heure.
|
Sélecteur d'heure.
|
||||||
@@ -297,7 +639,9 @@ Sélecteur d'heure.
|
|||||||
| `label` | `string` | `''` | Label |
|
| `label` | `string` | `''` | Label |
|
||||||
| `disabled` | `boolean` | `false` | Désactivé |
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
| `readonly` | `boolean` | `false` | Lecture seule |
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
| `error` | `string` | `''` | Message d'erreur |
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
@@ -308,6 +652,80 @@ Sélecteur d'heure.
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## MalioTimePicker
|
||||||
|
|
||||||
|
Sélecteur d'heure à **molettes style iOS** (champ + popover). Deux colonnes infinies (heures `00–23`, minutes `00–59`, pas de 1) avec une bande de sélection centrale ; la valeur centrée est sélectionnée. Défilement, clic sur une valeur (recentrage) ou flèches clavier (`role="spinbutton"`). Pour une saisie clavier directe au format texte, voir plutôt `MalioTime`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Heure au format `"HH:MM"` (v-model) |
|
||||||
|
| `placeholder` | `string` | `'HH:MM'` | Placeholder |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactive le champ |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `clearable` | `boolean` | `true` | Affiche la croix d'effacement |
|
||||||
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` / `labelClass` / `groupClass` | `string` | `''` | Override des classes |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string | null)`
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioTimePicker v-model="heure" label="Heure" />
|
||||||
|
<MalioTimePicker v-model="heure" label="Départ" hint="Format HH:MM" />
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioDateTime
|
||||||
|
|
||||||
|
Champ unique combinant **date et heure** dans un popover (grille de calendrier + sélecteur d'heure sous la grille).
|
||||||
|
|
||||||
|
> Depuis MUI-39, le réglage de l'heure utilise le sélecteur à molettes (cf. `MalioTimePicker`), qui remplace l'ancien `<input type="time">` natif intérimaire.
|
||||||
|
|
||||||
|
La valeur est une chaîne **ISO naïve sans fuseau** au format `"YYYY-MM-DDTHH:MM:00"` (heure murale locale). Symfony (`DateTimeNormalizer`) parse ce format et applique son fuseau configuré côté back — pas de gestion de fuseau côté front.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Date + heure ISO naïve `"YYYY-MM-DDTHH:MM:00"` (v-model) |
|
||||||
|
| `id` | `string` | `''` | Id du champ |
|
||||||
|
| `name` | `string` | `''` | Attribut name |
|
||||||
|
| `label` | `string` | `''` | Label flottant |
|
||||||
|
| `placeholder` | `string` | `'JJ/MM/AAAA HH:MM'` | Placeholder |
|
||||||
|
| `required` | `boolean` | `false` | Champ requis (astérisque rouge dans le label) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactivé |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule |
|
||||||
|
| `hint` | `string` | `''` | Texte d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `min` | `string` | `undefined` | Borne min. Borne la grille sur la partie date ; en saisie `editable`, comparée au **datetime complet** (préférer une borne datetime, sinon les heures du jour `max` seraient rejetées). |
|
||||||
|
| `max` | `string` | `undefined` | Borne max (idem) |
|
||||||
|
| `clearable` | `boolean` | `true` | Affiche la croix d'effacement |
|
||||||
|
| `editable` | `boolean` | `false` | Autorise la saisie clavier `JJ/MM/AAAA HH:MM` (masque maska, validation au blur / sur Entrée) en plus du calendrier |
|
||||||
|
| `invalidMessage` | `string` | `'Date invalide'` | Message affiché quand la saisie clavier est invalide ou hors `min`/`max` |
|
||||||
|
| `reserveMessageSpace` | `boolean` | `true` | Réserve l'espace de la ligne message sous le champ (min-h) même sans message. `false` = la ligne ne prend de place que s'il y a un hint/error/success. |
|
||||||
|
| `inputClass` / `labelClass` / `groupClass` | `string` | `''` | Override des classes |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string | null)`, `update:valid(value: boolean)`, `update:rawValue(value: string)`
|
||||||
|
|
||||||
|
Flux : cliquer un jour fixe la date (heure par défaut `00:00`), régler l'heure met à jour l'heure ; le popover se ferme au clic extérieur. La valeur est émise en direct à chaque interaction.
|
||||||
|
|
||||||
|
Avec `editable`, l'utilisateur peut aussi taper `JJ/MM/AAAA HH:MM` au clavier. La valeur n'est émise qu'au blur (ou sur Entrée) si elle est valide et dans les bornes ; sinon le texte est conservé et le champ passe en erreur (`invalidMessage`). Un **gabarit fantôme** affiche le format en gris et se remplit au fil de la saisie (cf. MalioDate). L'event `update:valid` (booléen) — émis **dès le montage** puis à chaque transition — remonte l'état de validité au parent (`false` = saisie malformée ou hors `min`/`max`, qui n'émet pas `modelValue`), pour bloquer un submit. La validité ne couvre **pas** `required` (champ vide = valide), comme sur `MalioDate`. L'event `update:rawValue` expose la saisie brute pour la validation back-autoritative (mêmes règles que `MalioDate` : texte trimmé sur saisie invalide, `''` sinon — clear et sélection au calendrier compris).
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioDateTime v-model="rdv" label="Date et heure du rendez-vous" />
|
||||||
|
<!-- rdv === "2026-05-20T14:30:00" -->
|
||||||
|
<MalioDateTime v-model="rdv" label="Rendez-vous" editable @update:valid="rdvValide = $event" />
|
||||||
|
<MalioDateTime v-model="rdv" editable @update:valid="valide = $event" @update:rawValue="brut = $event" />
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## MalioButton
|
## MalioButton
|
||||||
|
|
||||||
Bouton d'action avec 4 variantes visuelles et icône optionnelle.
|
Bouton d'action avec 4 variantes visuelles et icône optionnelle.
|
||||||
@@ -331,8 +749,11 @@ Bouton d'action avec 4 variantes visuelles et icône optionnelle.
|
|||||||
<MalioButton label="Voir plus" variant="tertiary" />
|
<MalioButton label="Voir plus" variant="tertiary" />
|
||||||
<MalioButton label="Supprimer" variant="danger" icon-name="mdi:trash" icon-position="left" />
|
<MalioButton label="Supprimer" variant="danger" icon-name="mdi:trash" icon-position="left" />
|
||||||
<MalioButton label="Pleine largeur" button-class="w-full" />
|
<MalioButton label="Pleine largeur" button-class="w-full" />
|
||||||
|
<MalioButton label="Modifier" button-class="w-m-btn-action" /> <!-- 150px, format bouton d'action -->
|
||||||
```
|
```
|
||||||
|
|
||||||
|
> **Token de largeur partagé** : `w-m-btn-action` (150px) est exposé via `tailwind.config.ts` du layer, branché sur la CSS var `--m-btn-action-width`. Pour les boutons d'action (listes, lignes de tableau, footers denses…). Themable côté consommateur en redéfinissant `--m-btn-action-width` dans son propre CSS.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## MalioButtonIcon
|
## MalioButtonIcon
|
||||||
@@ -364,18 +785,93 @@ Navigation par onglets avec contenu dynamique.
|
|||||||
| Prop | Type | Défaut | Description |
|
| Prop | Type | Défaut | Description |
|
||||||
|------|------|--------|-------------|
|
|------|------|--------|-------------|
|
||||||
| `modelValue` | `string` | `undefined` | Onglet actif (v-model) |
|
| `modelValue` | `string` | `undefined` | Onglet actif (v-model) |
|
||||||
| `tabs` | `{ key: string, label: string, icon?: string }[]` | **requis** | Liste des onglets |
|
| `tabs` | `Tab[]` | **requis** | Liste des onglets (voir type ci-dessous) |
|
||||||
|
| `maxVisibleTabs` | `number` | `undefined` | **Plafond** optionnel du nombre d'onglets visibles. Non défini = uniquement limité par la largeur. |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: string)`
|
Le nombre d'onglets affichés s'**adapte automatiquement à la largeur disponible** (mesurée au runtime via `ResizeObserver`). Quand tous les onglets ne tiennent pas, la barre passe en mode fenêtré : les flèches gauche/droite (fixées aux bords) font défiler la fenêtre un onglet à la fois, et le nombre visible est choisi pour que les onglets tiennent (jamais de chevauchement ni de rognage). `maxVisibleTabs`, s'il est fourni, plafonne ce nombre.
|
||||||
|
|
||||||
|
Type `Tab` :
|
||||||
|
|
||||||
|
| Propriété | Type | Défaut | Description |
|
||||||
|
|-----------|------|--------|-------------|
|
||||||
|
| `key` | `string` | — | Identifiant unique (utilisé pour le slot et le v-model) |
|
||||||
|
| `label` | `string` | — | Texte de l'onglet |
|
||||||
|
| `icon` | `string` | — | Nom Iconify (optionnel) |
|
||||||
|
| `iconSize` | `string` | `24` | Taille de l'icône |
|
||||||
|
| `disabled` | `boolean` | `false` | Onglet désactivé : grisé et non cliquable. Le parent calcule cet état selon sa logique de validation |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string)` — émis uniquement quand l'onglet cible n'est pas `disabled`
|
||||||
**Slots :** Un slot nommé par `tab.key` pour le contenu de chaque onglet
|
**Slots :** Un slot nommé par `tab.key` pour le contenu de chaque onglet
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioTabList v-model="activeTab" :tabs="[{ key: 'infos', label: 'Informations' }, { key: 'docs', label: 'Documents', icon: 'mdi:file' }]">
|
<MalioTabList v-model="activeTab" :tabs="tabs">
|
||||||
<template #infos>Contenu infos</template>
|
<template #infos>Contenu infos</template>
|
||||||
<template #docs>Contenu docs</template>
|
<template #docs>Contenu docs</template>
|
||||||
</MalioTabList>
|
</MalioTabList>
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**Pattern de gating progressif** (déverrouille les onglets quand les précédents sont valides) :
|
||||||
|
|
||||||
|
```ts
|
||||||
|
const informationValid = computed(() => name.value && email.value)
|
||||||
|
const adressesValid = computed(() => /^\d{5}$/.test(codePostal.value))
|
||||||
|
|
||||||
|
const tabs = computed(() => [
|
||||||
|
{ key: 'information', label: 'Information' },
|
||||||
|
{ key: 'contacts', label: 'Contacts', disabled: !informationValid.value },
|
||||||
|
{ key: 'adresses', label: 'Adresses', disabled: !informationValid.value },
|
||||||
|
{ key: 'transport', label: 'Transport', disabled: !informationValid.value || !adressesValid.value },
|
||||||
|
])
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioAccordion
|
||||||
|
|
||||||
|
Accordéon compositionnel : `<MalioAccordion>` enveloppe des `<MalioAccordionItem>`. Plusieurs panneaux ouverts (`multiple`, défaut) ou un seul (`single`). Pensé pour les filtres en drawer et les FAQ.
|
||||||
|
|
||||||
|
### MalioAccordion
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `mode` | `'single' \| 'multiple'` | `'multiple'` | Un seul ou plusieurs panneaux ouverts |
|
||||||
|
| `modelValue` | `string \| string[]` | `undefined` | Clés ouvertes (v-model). `string[]` en `multiple`, `string` en `single` |
|
||||||
|
| `id` | `string` | auto | Préfixe des IDs d'accessibilité |
|
||||||
|
| `groupClass` | `string` | `''` | Classes du conteneur (twMerge) |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string | string[])`
|
||||||
|
|
||||||
|
### MalioAccordionItem
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `title` | `string` | — | Texte de l'en-tête |
|
||||||
|
| `value` | `string` | auto | Clé unique de la section |
|
||||||
|
| `defaultOpen` | `boolean` | `false` | Ouvert au montage (mode non contrôlé) |
|
||||||
|
| `disabled` | `boolean` | `false` | En-tête non cliquable |
|
||||||
|
| `headerClass` | `string` | `''` | Override classes en-tête (twMerge) |
|
||||||
|
| `panelClass` | `string` | `''` | Override classes panneau (twMerge) |
|
||||||
|
|
||||||
|
**Slot :** par défaut = contenu du panneau.
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<!-- Filtres : plusieurs sections ouvertes -->
|
||||||
|
<MalioAccordion v-model="ouverts">
|
||||||
|
<MalioAccordionItem title="Prix" value="prix">
|
||||||
|
<MalioInputAmount v-model="prix" />
|
||||||
|
</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Catégorie" value="cat">
|
||||||
|
<MalioCheckbox v-model="cats" />
|
||||||
|
</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
|
||||||
|
<!-- FAQ : une seule section ouverte -->
|
||||||
|
<MalioAccordion mode="single">
|
||||||
|
<MalioAccordionItem title="Question 1" value="q1">Réponse 1</MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Question 2" value="q2">Réponse 2</MalioAccordionItem>
|
||||||
|
</MalioAccordion>
|
||||||
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## MalioSidebar
|
## MalioSidebar
|
||||||
@@ -389,7 +885,10 @@ Barre latérale de navigation rétractable.
|
|||||||
| `sidebarClass` | `string` | `''` | Classes CSS sidebar |
|
| `sidebarClass` | `string` | `''` | Classes CSS sidebar |
|
||||||
| `toggleClass` | `string` | `''` | Classes CSS bouton toggle |
|
| `toggleClass` | `string` | `''` | Classes CSS bouton toggle |
|
||||||
|
|
||||||
**Type SidebarSection :** `{ title?: string, items: { label: string, icon?: string, to?: string, href?: string, active?: boolean }[] }`
|
**Type SidebarSection :** `{ label?: string, icon?: string, items: SidebarItem[] }`
|
||||||
|
**Type SidebarItem :** `{ label: string, to: string, exact?: boolean }`
|
||||||
|
|
||||||
|
**Lien actif :** un lien est marqué actif (texte `m-primary` + semi-bold) quand la route courante **est ce lien ou une de ses sous-routes** (match par préfixe) — ex. `/supplier` reste actif sur `/supplier/1/edit`. Mettre `exact: true` sur l'item force le match strict (actif uniquement sur la route exacte). Indépendant de l'imbrication des routes côté consommateur.
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: boolean)`
|
**Events :** `update:modelValue(value: boolean)`
|
||||||
**Slots :** `logo` (sidebar ouverte), `logo-collapsed` (sidebar fermée)
|
**Slots :** `logo` (sidebar ouverte), `logo-collapsed` (sidebar fermée)
|
||||||
@@ -405,29 +904,110 @@ Barre latérale de navigation rétractable.
|
|||||||
|
|
||||||
## MalioDrawer
|
## MalioDrawer
|
||||||
|
|
||||||
Panneau latéral (drawer) qui s'ouvre depuis la droite avec backdrop semi-transparent.
|
Panneau latéral (drawer) qui s'ouvre depuis la droite ou la gauche avec backdrop semi-transparent. Gère l'accessibilité (focus-trap, restitution du focus, `Échap`), le verrouillage du scroll de la page et un empilement correct de plusieurs drawers.
|
||||||
|
|
||||||
| Prop | Type | Défaut | Description |
|
| Prop | Type | Défaut | Description |
|
||||||
|------|------|--------|-------------|
|
|------|------|--------|-------------|
|
||||||
| `id` | `string` | auto | Identifiant HTML |
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
| `modelValue` | `boolean` | `undefined` | État ouvert/fermé (v-model) |
|
| `modelValue` | `boolean` | `undefined` | État ouvert/fermé (v-model) |
|
||||||
| `title` | `string` | `''` | Titre affiché dans le header |
|
| `side` | `'right' \| 'left'` | `'right'` | Côté d'apparition |
|
||||||
| `showClose` | `boolean` | `true` | Afficher le bouton de fermeture (croix) |
|
| `showClose` | `boolean` | `true` | Afficher le bouton de fermeture (croix) |
|
||||||
| `drawerClass` | `string` | `''` | Classes CSS panneau (twMerge) |
|
| `dismissable` | `boolean` | `true` | Fermer au clic sur le backdrop |
|
||||||
|
| `closeOnEscape` | `boolean` | `true` | Fermer avec la touche `Échap` |
|
||||||
|
| `ariaLabel` | `string` | `''` | Nom accessible de secours quand le slot `#header` est absent |
|
||||||
|
| `drawerClass` | `string` | `''` | Classes CSS panneau, ex. largeur `max-w-2xl` (twMerge) |
|
||||||
|
| `overlayClass` | `string` | `''` | Classes CSS backdrop (twMerge) |
|
||||||
|
| `headerClass` | `string` | `''` | Classes CSS barre header (twMerge) |
|
||||||
|
| `bodyClass` | `string` | `''` | Classes CSS zone scrollable (twMerge) |
|
||||||
|
| `footerClass` | `string` | `''` | Classes CSS du footer fixe (twMerge) |
|
||||||
|
|
||||||
**Events :** `update:modelValue(value: boolean)`
|
**Events :** `update:modelValue(value: boolean)`, `close()`
|
||||||
**Slots :** `default` (contenu du drawer)
|
|
||||||
|
**Slots :**
|
||||||
|
- `header` — en-tête (titre, etc.), fixe en haut. S'il est absent et que `showClose` est `true`, seule la croix est affichée.
|
||||||
|
- `default` — contenu (zone scrollable : seul le body défile).
|
||||||
|
- `footer` — actions (boutons). Rendu en bas du panneau, fixe, hors de la zone scrollable. N'apparaît que si le slot est fourni.
|
||||||
|
|
||||||
```vue
|
```vue
|
||||||
<MalioDrawer v-model="isOpen" title="Détails">
|
<MalioDrawer v-model="isOpen">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold">Détails</h2>
|
||||||
|
</template>
|
||||||
<p>Contenu du drawer</p>
|
<p>Contenu du drawer</p>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
<MalioDrawer v-model="isOpen" title="Sans croix" :show-close="false">
|
|
||||||
<p>Fermeture uniquement via backdrop</p>
|
<!-- Côté gauche, largeur custom -->
|
||||||
|
<MalioDrawer v-model="isOpen" side="left" drawer-class="max-w-2xl">
|
||||||
|
<template #header><h2>Navigation</h2></template>
|
||||||
|
<p>Drawer large depuis la gauche</p>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
<MalioDrawer v-model="isOpen" title="Large" drawer-class="max-w-2xl">
|
|
||||||
<p>Drawer plus large</p>
|
<!-- Footer d'actions (fixe en bas, hors zone scrollable) -->
|
||||||
|
<MalioDrawer v-model="isOpen">
|
||||||
|
<template #header><h2>Formulaire</h2></template>
|
||||||
|
<MalioInputText label="Nom" />
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Enregistrer" button-class="w-full" @click="isOpen = false" />
|
||||||
|
</template>
|
||||||
</MalioDrawer>
|
</MalioDrawer>
|
||||||
|
|
||||||
|
<!-- Non fermable au backdrop / Échap (croix uniquement) -->
|
||||||
|
<MalioDrawer v-model="isOpen" :dismissable="false" :close-on-escape="false">
|
||||||
|
<template #header><h2>Action requise</h2></template>
|
||||||
|
<p>Fermeture via la croix uniquement</p>
|
||||||
|
</MalioDrawer>
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioModal
|
||||||
|
|
||||||
|
Boîte de dialogue modale centrée avec backdrop semi-transparent. Gère l'accessibilité (focus-trap, restitution du focus, `Échap`), le verrouillage du scroll de la page et un empilement correct de plusieurs modals. Structure : header fixe, body scrollable (`max-h-[85vh]`), footer fixe.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `modelValue` | `boolean` | `undefined` | État ouvert/fermé (v-model) |
|
||||||
|
| `showClose` | `boolean` | `true` | Afficher le bouton de fermeture (croix) |
|
||||||
|
| `dismissable` | `boolean` | `true` | Fermer au clic sur le backdrop |
|
||||||
|
| `closeOnEscape` | `boolean` | `true` | Fermer avec la touche `Échap` |
|
||||||
|
| `ariaLabel` | `string` | `''` | Nom accessible de secours quand le slot `#header` est absent |
|
||||||
|
| `modalClass` | `string` | `''` | Classes CSS panneau, ex. largeur `max-w-lg` (twMerge) |
|
||||||
|
| `overlayClass` | `string` | `''` | Classes CSS backdrop (twMerge) |
|
||||||
|
| `headerClass` | `string` | `''` | Classes CSS barre header (twMerge) |
|
||||||
|
| `bodyClass` | `string` | `''` | Classes CSS zone scrollable (twMerge) |
|
||||||
|
| `footerClass` | `string` | `''` | Classes CSS footer fixe (twMerge) |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: boolean)`, `close()`
|
||||||
|
|
||||||
|
**Slots :**
|
||||||
|
- `header` — en-tête (titre, etc.). S'il est absent et que `showClose` est `true`, seule la croix est affichée.
|
||||||
|
- `default` — contenu (zone scrollable).
|
||||||
|
- `footer` — actions (boutons). Rendu en bas, fixe, séparé par une bordure. N'apparaît que si le slot est fourni.
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioModal v-model="isOpen">
|
||||||
|
<template #header>
|
||||||
|
<h2 class="text-[24px] font-bold">Détails</h2>
|
||||||
|
</template>
|
||||||
|
<p>Contenu de la modal</p>
|
||||||
|
</MalioModal>
|
||||||
|
|
||||||
|
<!-- Largeur custom + footer d'actions -->
|
||||||
|
<MalioModal v-model="isOpen" modal-class="max-w-lg">
|
||||||
|
<template #header><h2>Nouveau contact</h2></template>
|
||||||
|
<MalioInputText label="Nom" />
|
||||||
|
<template #footer>
|
||||||
|
<MalioButton label="Annuler" variant="secondary" button-class="flex-1" @click="isOpen = false" />
|
||||||
|
<MalioButton label="Enregistrer" button-class="flex-1" @click="isOpen = false" />
|
||||||
|
</template>
|
||||||
|
</MalioModal>
|
||||||
|
|
||||||
|
<!-- Non fermable au backdrop / Échap (croix uniquement) -->
|
||||||
|
<MalioModal v-model="isOpen" :dismissable="false" :close-on-escape="false">
|
||||||
|
<template #header><h2>Action requise</h2></template>
|
||||||
|
<p>Fermeture via la croix uniquement</p>
|
||||||
|
</MalioModal>
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -485,3 +1065,33 @@ Tableau de données presentational avec pagination, filtres par slots et lignes
|
|||||||
v-model:per-page="perPage"
|
v-model:per-page="perPage"
|
||||||
/>
|
/>
|
||||||
```
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## MalioSiteSelector
|
||||||
|
|
||||||
|
Sélecteur de site sous forme de tuiles segmentées (`role="radiogroup"`). Chaque site occupe une tuile de largeur égale ; la tuile active s'affiche pleine opacité dans sa couleur (`site.color`), les autres sont atténuées. Pattern contrôlé (`v-model`) ou non contrôlé (premier site sélectionné par défaut).
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `sites` | `{ id: string, name: string, color: string }[]` | **requis** | Liste des sites (la `color` colore la tuile active) |
|
||||||
|
| `modelValue` | `string` | `undefined` | `id` du site sélectionné (v-model) |
|
||||||
|
| `id` | `string` | auto | Identifiant HTML du conteneur |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS du conteneur (twMerge) |
|
||||||
|
| `tileClass` | `string` | `''` | Classes CSS de chaque tuile (twMerge) |
|
||||||
|
| `labelClass` | `string` | `''` | Classes CSS du label de tuile (twMerge) |
|
||||||
|
|
||||||
|
**Events :**
|
||||||
|
- `update:modelValue(value: string)` — `id` du site sélectionné (v-model)
|
||||||
|
- `change(site: Site)` — émis avec l'objet site complet sélectionné
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioSiteSelector
|
||||||
|
v-model="siteId"
|
||||||
|
:sites="[
|
||||||
|
{ id: 'paris', name: 'Paris', color: '#2563eb' },
|
||||||
|
{ id: 'lyon', name: 'Lyon', color: '#16a34a' },
|
||||||
|
]"
|
||||||
|
@change="onSiteChange"
|
||||||
|
/>
|
||||||
|
```
|
||||||
|
|||||||
@@ -2,10 +2,46 @@
|
|||||||
@tailwind components;
|
@tailwind components;
|
||||||
@tailwind utilities;
|
@tailwind utilities;
|
||||||
|
|
||||||
|
@layer components {
|
||||||
|
/* Anneau de focus clavier standard (navigation au Tab), invisible à la souris.
|
||||||
|
Deux déclencheurs, même rendu :
|
||||||
|
- .m-focus-ring → s'appuie sur :focus-visible natif. Pour les éléments
|
||||||
|
où :focus-visible se limite déjà au clavier (boutons,
|
||||||
|
onglets, tuiles, checkbox/radio…).
|
||||||
|
- .m-focus-ring-kbd → classe ajoutée en JS (via useKbdFocusRing) uniquement
|
||||||
|
quand le focus vient du clavier. Pour les champs texte,
|
||||||
|
où :focus-visible natif se déclenche aussi à la souris.
|
||||||
|
Le `:focus` sur .m-focus-ring-kbd élève la spécificité pour passer devant le
|
||||||
|
`outline-none` des inputs. */
|
||||||
|
.m-focus-ring:focus-visible,
|
||||||
|
.m-focus-ring-kbd:focus {
|
||||||
|
outline: 2px solid rgb(var(--m-primary) / 1);
|
||||||
|
outline-offset: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Anneau de focus clavier pour un combobox ouvert (input + liste) : l'anneau
|
||||||
|
entoure le bloc entier d'un seul tenant. L'input porte le contour haut+côtés,
|
||||||
|
la liste le contour côtés+bas ; la jonction (bas de l'input / haut de la liste)
|
||||||
|
reste sans contour pour un raccord sans couture. */
|
||||||
|
.m-combo-ring-top {
|
||||||
|
box-shadow:
|
||||||
|
-2px 0 0 0 rgb(var(--m-primary) / 1),
|
||||||
|
2px 0 0 0 rgb(var(--m-primary) / 1),
|
||||||
|
0 -2px 0 0 rgb(var(--m-primary) / 1);
|
||||||
|
}
|
||||||
|
.m-combo-ring-bottom {
|
||||||
|
box-shadow:
|
||||||
|
-2px 0 0 0 rgb(var(--m-primary) / 1),
|
||||||
|
2px 0 0 0 rgb(var(--m-primary) / 1),
|
||||||
|
0 2px 0 0 rgb(var(--m-primary) / 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
:root {
|
:root {
|
||||||
/* ── Globales ── */
|
/* ── Globales ── */
|
||||||
--m-primary: 34 39 131; /* #222783 - Bleu Malio */
|
--m-primary: 34 39 131; /* #222783 - Bleu Malio */
|
||||||
|
--m-primary-light: 239 239 253; /* #EFEFFD - Teinte claire du primary (fonds doux) */
|
||||||
--m-bg: 243 244 248; /* #F3F4F8 - Fond de page */
|
--m-bg: 243 244 248; /* #F3F4F8 - Fond de page */
|
||||||
--m-surface: 243 244 248; /* #F3F4F8 - Fond hover/cartes */
|
--m-surface: 243 244 248; /* #F3F4F8 - Fond hover/cartes */
|
||||||
--m-text: 15 23 42; /* #0F172A */
|
--m-text: 15 23 42; /* #0F172A */
|
||||||
@@ -30,6 +66,9 @@
|
|||||||
--m-btn-danger-hover: 234 151 151; /* #EA9797 */
|
--m-btn-danger-hover: 234 151 151; /* #EA9797 */
|
||||||
--m-btn-danger-active: 255 83 86; /* #FF5356 */
|
--m-btn-danger-active: 255 83 86; /* #FF5356 */
|
||||||
|
|
||||||
|
/* ── Largeurs Boutons ── */
|
||||||
|
--m-btn-action-width: 150px; /* Boutons d'action (liste, ligne tableau, footer dense…) */
|
||||||
|
|
||||||
/* ── Couleurs de site (usage ponctuel) ── */
|
/* ── Couleurs de site (usage ponctuel) ── */
|
||||||
--m-site-blue: 5 108 242; /* #056CF2 - Bleu Châtellerault */
|
--m-site-blue: 5 108 242; /* #056CF2 - Bleu Châtellerault */
|
||||||
--m-site-yellow: 243 203 0; /* #F3CB00 - Jaune Saint-Jean */
|
--m-site-yellow: 243 203 0; /* #F3CB00 - Jaune Saint-Jean */
|
||||||
|
|||||||
@@ -0,0 +1,256 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import {nextTick} from 'vue'
|
||||||
|
import Accordion from './Accordion.vue'
|
||||||
|
import AccordionItem from './AccordionItem.vue'
|
||||||
|
|
||||||
|
const TWO_ITEMS = `
|
||||||
|
<MalioAccordionItem title="Prix" value="prix"><p>Contenu prix</p></MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Catégorie" value="cat"><p>Contenu catégorie</p></MalioAccordionItem>
|
||||||
|
`
|
||||||
|
|
||||||
|
function mountAccordion(props: Record<string, unknown> = {}, slot: string = TWO_ITEMS, attachTo?: HTMLElement) {
|
||||||
|
return mount(Accordion, {
|
||||||
|
props,
|
||||||
|
slots: {default: slot},
|
||||||
|
attachTo,
|
||||||
|
global: {components: {MalioAccordionItem: AccordionItem}},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('MalioAccordion — rendu & mode multiple', () => {
|
||||||
|
it('renders each item header with its title', () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers).toHaveLength(2)
|
||||||
|
expect(headers[0].text()).toContain('Prix')
|
||||||
|
expect(headers[1].text()).toContain('Catégorie')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the slot content of each panel', () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
expect(wrapper.html()).toContain('Contenu prix')
|
||||||
|
expect(wrapper.html()).toContain('Contenu catégorie')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('all panels are collapsed by default', () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('false')
|
||||||
|
const regions = wrapper.findAll('[role="region"]')
|
||||||
|
expect(regions[0].classes()).toContain('grid-rows-[0fr]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens a panel on header click (multiple mode is default)', async () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('true')
|
||||||
|
const regions = wrapper.findAll('[role="region"]')
|
||||||
|
expect(regions[0].classes()).toContain('grid-rows-[1fr]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps multiple panels open simultaneously in multiple mode', async () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
await headers[1].trigger('click')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('true')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes an open panel when its header is clicked again', async () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wires aria-controls / aria-labelledby / role=region correctly', () => {
|
||||||
|
const wrapper = mountAccordion({id: 'acc'})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
const regions = wrapper.findAll('[role="region"]')
|
||||||
|
expect(headers[0].attributes('id')).toBe('acc-header-prix')
|
||||||
|
expect(headers[0].attributes('aria-controls')).toBe('acc-panel-prix')
|
||||||
|
expect(regions[0].attributes('id')).toBe('acc-panel-prix')
|
||||||
|
expect(regions[0].attributes('aria-labelledby')).toBe('acc-header-prix')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue with an array in multiple mode', async () => {
|
||||||
|
const wrapper = mountAccordion()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([['prix']])
|
||||||
|
await nextTick()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioAccordion — mode single & contrôlé', () => {
|
||||||
|
it('opening a panel closes the others in single mode', async () => {
|
||||||
|
const wrapper = mountAccordion({mode: 'single'})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
await headers[1].trigger('click')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits a string in single mode', async () => {
|
||||||
|
const wrapper = mountAccordion({mode: 'single'})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[1].trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['cat'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits empty string when closing the open panel in single mode', async () => {
|
||||||
|
const wrapper = mountAccordion({mode: 'single', modelValue: 'prix'})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('respects modelValue array in controlled multiple mode', () => {
|
||||||
|
const wrapper = mountAccordion({modelValue: ['cat']})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('respects modelValue string in controlled single mode', () => {
|
||||||
|
const wrapper = mountAccordion({mode: 'single', modelValue: 'prix'})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('true')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not mutate local state in controlled mode (emits only)', async () => {
|
||||||
|
const wrapper = mountAccordion({modelValue: []})
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[0].trigger('click')
|
||||||
|
// état piloté par le parent : sans mise à jour de la prop, reste fermé
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([['prix']])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioAccordion — defaultOpen, disabled & clavier', () => {
|
||||||
|
const WITH_DEFAULT_OPEN = `
|
||||||
|
<MalioAccordionItem title="Prix" value="prix"><p>P</p></MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Catégorie" value="cat" :default-open="true"><p>C</p></MalioAccordionItem>
|
||||||
|
`
|
||||||
|
const WITH_DISABLED = `
|
||||||
|
<MalioAccordionItem title="Prix" value="prix"><p>P</p></MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="Catégorie" value="cat" :disabled="true"><p>C</p></MalioAccordionItem>
|
||||||
|
`
|
||||||
|
|
||||||
|
it('opens defaultOpen items initially in uncontrolled mode', async () => {
|
||||||
|
const wrapper = mountAccordion({}, WITH_DEFAULT_OPEN)
|
||||||
|
await nextTick()
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers[0].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets disabled and aria-disabled on a disabled item', () => {
|
||||||
|
const wrapper = mountAccordion({}, WITH_DISABLED)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
expect(headers[1].attributes('disabled')).toBeDefined()
|
||||||
|
expect(headers[1].attributes('aria-disabled')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not toggle a disabled item on click', async () => {
|
||||||
|
const wrapper = mountAccordion({}, WITH_DISABLED)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
await headers[1].trigger('click')
|
||||||
|
expect(headers[1].attributes('aria-expanded')).toBe('false')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves focus to the next header on ArrowDown', async () => {
|
||||||
|
const root = document.createElement('div')
|
||||||
|
document.body.appendChild(root)
|
||||||
|
const wrapper = mountAccordion({}, TWO_ITEMS, root)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
;(headers[0].element as HTMLElement).focus()
|
||||||
|
await headers[0].trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
expect(document.activeElement).toBe(headers[1].element)
|
||||||
|
wrapper.unmount()
|
||||||
|
root.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wraps focus to the first header on ArrowDown from the last', async () => {
|
||||||
|
const root = document.createElement('div')
|
||||||
|
document.body.appendChild(root)
|
||||||
|
const wrapper = mountAccordion({}, TWO_ITEMS, root)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
;(headers[1].element as HTMLElement).focus()
|
||||||
|
await headers[1].trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
expect(document.activeElement).toBe(headers[0].element)
|
||||||
|
wrapper.unmount()
|
||||||
|
root.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves focus to the previous header on ArrowUp', async () => {
|
||||||
|
const root = document.createElement('div')
|
||||||
|
document.body.appendChild(root)
|
||||||
|
const wrapper = mountAccordion({}, TWO_ITEMS, root)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
;(headers[1].element as HTMLElement).focus()
|
||||||
|
await headers[1].trigger('keydown', {key: 'ArrowUp'})
|
||||||
|
expect(document.activeElement).toBe(headers[0].element)
|
||||||
|
wrapper.unmount()
|
||||||
|
root.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('skips disabled headers during keyboard navigation', async () => {
|
||||||
|
const root = document.createElement('div')
|
||||||
|
document.body.appendChild(root)
|
||||||
|
const slot = `
|
||||||
|
<MalioAccordionItem title="A" value="a"><p>A</p></MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="B" value="b" :disabled="true"><p>B</p></MalioAccordionItem>
|
||||||
|
<MalioAccordionItem title="C" value="c"><p>C</p></MalioAccordionItem>
|
||||||
|
`
|
||||||
|
const wrapper = mountAccordion({}, slot, root)
|
||||||
|
const headers = wrapper.findAll('button[aria-expanded]')
|
||||||
|
;(headers[0].element as HTMLElement).focus()
|
||||||
|
await headers[0].trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
// saute le header désactivé (B) pour aller directement à C
|
||||||
|
expect(document.activeElement).toBe(headers[2].element)
|
||||||
|
wrapper.unmount()
|
||||||
|
root.remove()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioAccordion — overflow du panneau (popovers enfants)', () => {
|
||||||
|
const ONE = `<MalioAccordionItem title="A" value="a"><p>contenu</p></MalioAccordionItem>`
|
||||||
|
const ONE_OPEN = `<MalioAccordionItem title="A" value="a" :default-open="true"><p>contenu</p></MalioAccordionItem>`
|
||||||
|
|
||||||
|
it('clips the panel (overflow-hidden) while collapsed', () => {
|
||||||
|
const wrapper = mountAccordion({}, ONE)
|
||||||
|
const inner = wrapper.find('[role="region"] > div')
|
||||||
|
expect(inner.classes()).toContain('overflow-hidden')
|
||||||
|
expect(inner.classes()).not.toContain('overflow-visible')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('lets the panel overflow once open at mount (defaultOpen)', async () => {
|
||||||
|
const wrapper = mountAccordion({}, ONE_OPEN)
|
||||||
|
await nextTick()
|
||||||
|
expect(wrapper.find('[role="region"] > div').classes()).toContain('overflow-visible')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('switches to overflow-visible after the open transition ends', async () => {
|
||||||
|
const wrapper = mountAccordion({}, ONE)
|
||||||
|
await wrapper.find('button[aria-expanded]').trigger('click')
|
||||||
|
await wrapper.find('[role="region"]').trigger('transitionend', {propertyName: 'grid-template-rows'})
|
||||||
|
expect(wrapper.find('[role="region"] > div').classes()).toContain('overflow-visible')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('re-clips (overflow-hidden) as soon as it closes', async () => {
|
||||||
|
const wrapper = mountAccordion({}, ONE_OPEN)
|
||||||
|
await nextTick()
|
||||||
|
await wrapper.find('button[aria-expanded]').trigger('click')
|
||||||
|
expect(wrapper.find('[role="region"] > div').classes()).toContain('overflow-hidden')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,109 @@
|
|||||||
|
<template>
|
||||||
|
<div v-bind="$attrs" :class="rootClass">
|
||||||
|
<slot />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, provide, ref, useId} from 'vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import {accordionContextKey, type AccordionItemRegistration} from './context'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioAccordion', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<{
|
||||||
|
mode?: 'single' | 'multiple'
|
||||||
|
modelValue?: string | string[]
|
||||||
|
id?: string
|
||||||
|
groupClass?: string
|
||||||
|
}>(), {
|
||||||
|
mode: 'multiple',
|
||||||
|
modelValue: undefined,
|
||||||
|
id: '',
|
||||||
|
groupClass: '',
|
||||||
|
})
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'update:modelValue', value: string | string[]): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const generatedId = useId()
|
||||||
|
const baseId = computed(() => props.id || `malio-accordion-${generatedId}`)
|
||||||
|
const mode = computed(() => props.mode)
|
||||||
|
|
||||||
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const localOpen = ref<string[]>([])
|
||||||
|
|
||||||
|
const items = ref<AccordionItemRegistration[]>([])
|
||||||
|
|
||||||
|
const openKeys = computed<string[]>(() => {
|
||||||
|
if (isControlled.value) {
|
||||||
|
const v = props.modelValue
|
||||||
|
if (props.mode === 'single') return v ? [v as string] : []
|
||||||
|
if (Array.isArray(v)) return v
|
||||||
|
return v ? [v as string] : []
|
||||||
|
}
|
||||||
|
return localOpen.value
|
||||||
|
})
|
||||||
|
|
||||||
|
function isOpen(value: string) {
|
||||||
|
return openKeys.value.includes(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggle(value: string) {
|
||||||
|
const current = openKeys.value
|
||||||
|
let next: string[]
|
||||||
|
if (props.mode === 'single') {
|
||||||
|
next = current.includes(value) ? [] : [value]
|
||||||
|
} else {
|
||||||
|
next = current.includes(value)
|
||||||
|
? current.filter(v => v !== value)
|
||||||
|
: [...current, value]
|
||||||
|
}
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localOpen.value = next
|
||||||
|
}
|
||||||
|
emit('update:modelValue', props.mode === 'single' ? (next[0] ?? '') : next)
|
||||||
|
}
|
||||||
|
|
||||||
|
function register(item: AccordionItemRegistration, defaultOpen: boolean) {
|
||||||
|
items.value.push(item)
|
||||||
|
if (defaultOpen && !isControlled.value) {
|
||||||
|
if (props.mode === 'single') {
|
||||||
|
if (localOpen.value.length === 0) localOpen.value = [item.value]
|
||||||
|
} else if (!localOpen.value.includes(item.value)) {
|
||||||
|
localOpen.value.push(item.value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function unregister(value: string) {
|
||||||
|
items.value = items.value.filter(i => i.value !== value)
|
||||||
|
}
|
||||||
|
|
||||||
|
// `items` est ordonné par ordre de montage (= ordre du DOM pour des sections
|
||||||
|
// statiques/ajoutées en fin). Si un consommateur réordonne dynamiquement les
|
||||||
|
// items, cet ordre peut diverger de l'ordre visuel ; trier par position DOM
|
||||||
|
// serait alors nécessaire (hors périmètre v1).
|
||||||
|
function focusSibling(value: string, offset: 1 | -1) {
|
||||||
|
const enabled = items.value.filter(i => !i.isDisabled())
|
||||||
|
const idx = enabled.findIndex(i => i.value === value)
|
||||||
|
if (idx === -1) return
|
||||||
|
const next = enabled[(idx + offset + enabled.length) % enabled.length]
|
||||||
|
next?.getHeaderEl()?.focus()
|
||||||
|
}
|
||||||
|
|
||||||
|
const rootClass = computed(() =>
|
||||||
|
twMerge('divide-y divide-black border-y border-black', props.groupClass),
|
||||||
|
)
|
||||||
|
|
||||||
|
provide(accordionContextKey, {
|
||||||
|
mode,
|
||||||
|
baseId,
|
||||||
|
isOpen,
|
||||||
|
toggle,
|
||||||
|
register,
|
||||||
|
unregister,
|
||||||
|
focusSibling,
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
import {describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import Accordion from './Accordion.vue'
|
||||||
|
import AccordionItem from './AccordionItem.vue'
|
||||||
|
|
||||||
|
function mountInAccordion(slot: string, accordionProps: Record<string, unknown> = {}) {
|
||||||
|
return mount(Accordion, {
|
||||||
|
props: accordionProps,
|
||||||
|
slots: {default: slot},
|
||||||
|
global: {components: {MalioAccordionItem: AccordionItem}},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('MalioAccordionItem', () => {
|
||||||
|
it('throws when used outside MalioAccordion', () => {
|
||||||
|
const spy = vi.spyOn(console, 'warn').mockImplementation(() => {})
|
||||||
|
expect(() => mount(AccordionItem, {props: {title: 'Solo'}})).toThrow(
|
||||||
|
/à l'intérieur de MalioAccordion/,
|
||||||
|
)
|
||||||
|
spy.mockRestore()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an auto id-based value and still toggles when value prop is omitted', async () => {
|
||||||
|
const wrapper = mountInAccordion(
|
||||||
|
`<MalioAccordionItem title="Sans value"><p>X</p></MalioAccordionItem>`,
|
||||||
|
)
|
||||||
|
const header = wrapper.find('button[aria-expanded]')
|
||||||
|
expect(header.attributes('aria-controls')).toMatch(/-panel-malio-accordion-item-/)
|
||||||
|
await header.trigger('click')
|
||||||
|
expect(header.attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies headerClass and panelClass overrides via twMerge', () => {
|
||||||
|
const wrapper = mountInAccordion(
|
||||||
|
`<MalioAccordionItem title="T" value="t" header-class="bg-red-500" panel-class="text-lg"><p>X</p></MalioAccordionItem>`,
|
||||||
|
)
|
||||||
|
const header = wrapper.find('button[aria-expanded]')
|
||||||
|
expect(header.classes()).toContain('bg-red-500')
|
||||||
|
expect(wrapper.find('[role="region"]').html()).toContain('text-lg')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders a rotating chevron icon', () => {
|
||||||
|
const wrapper = mountInAccordion(
|
||||||
|
`<MalioAccordionItem title="T" value="t"><p>X</p></MalioAccordionItem>`,
|
||||||
|
)
|
||||||
|
expect(wrapper.find('button[aria-expanded] svg').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,126 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<h3 class="m-0">
|
||||||
|
<button
|
||||||
|
:id="headerId"
|
||||||
|
ref="headerRef"
|
||||||
|
type="button"
|
||||||
|
:class="headerClasses"
|
||||||
|
:aria-expanded="open"
|
||||||
|
:aria-controls="panelId"
|
||||||
|
:disabled="disabled"
|
||||||
|
:aria-disabled="disabled || undefined"
|
||||||
|
@click="onToggle"
|
||||||
|
@keydown.down.prevent="ctx.focusSibling(value, 1)"
|
||||||
|
@keydown.up.prevent="ctx.focusSibling(value, -1)"
|
||||||
|
>
|
||||||
|
<span>{{ title }}</span>
|
||||||
|
<IconifyIcon
|
||||||
|
icon="mdi:chevron-down"
|
||||||
|
:width="24"
|
||||||
|
class="shrink-0 transition-transform duration-200"
|
||||||
|
:class="open ? 'rotate-180' : ''"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</h3>
|
||||||
|
<div
|
||||||
|
:id="panelId"
|
||||||
|
role="region"
|
||||||
|
:aria-labelledby="headerId"
|
||||||
|
class="grid transition-[grid-template-rows] duration-200 ease-out"
|
||||||
|
:class="open ? 'grid-rows-[1fr]' : 'grid-rows-[0fr]'"
|
||||||
|
@transitionend="onPanelTransitionEnd"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
:class="overflowVisible ? 'overflow-visible' : 'overflow-hidden'"
|
||||||
|
:inert="!open || undefined"
|
||||||
|
>
|
||||||
|
<div :class="panelInnerClass">
|
||||||
|
<slot />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, inject, onBeforeUnmount, onMounted, ref, useId, watch} from 'vue'
|
||||||
|
import {Icon as IconifyIcon} from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import {accordionContextKey} from './context'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioAccordionItem', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<{
|
||||||
|
title: string
|
||||||
|
value?: string
|
||||||
|
defaultOpen?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
headerClass?: string
|
||||||
|
panelClass?: string
|
||||||
|
}>(), {
|
||||||
|
value: '',
|
||||||
|
defaultOpen: false,
|
||||||
|
disabled: false,
|
||||||
|
headerClass: '',
|
||||||
|
panelClass: '',
|
||||||
|
})
|
||||||
|
|
||||||
|
const ctx = inject(accordionContextKey)
|
||||||
|
if (!ctx) {
|
||||||
|
throw new Error('MalioAccordionItem doit être utilisé à l\'intérieur de MalioAccordion')
|
||||||
|
}
|
||||||
|
|
||||||
|
const generatedId = useId()
|
||||||
|
const value = computed(() => props.value || `malio-accordion-item-${generatedId}`)
|
||||||
|
const headerRef = ref<HTMLButtonElement | null>(null)
|
||||||
|
const headerId = computed(() => `${ctx.baseId.value}-header-${value.value}`)
|
||||||
|
const panelId = computed(() => `${ctx.baseId.value}-panel-${value.value}`)
|
||||||
|
const open = computed(() => ctx.isOpen(value.value))
|
||||||
|
|
||||||
|
// Le panneau garde `overflow-hidden` pendant l'animation (clipping requis par
|
||||||
|
// la transition grid-template-rows), puis passe en `overflow-visible` une fois
|
||||||
|
// complètement ouvert pour qu'un popover enfant (datepicker, select…) ne soit
|
||||||
|
// pas rogné. On re-clippe dès le début de la fermeture.
|
||||||
|
const overflowVisible = ref(false)
|
||||||
|
|
||||||
|
watch(open, (isOpen) => {
|
||||||
|
if (!isOpen) overflowVisible.value = false
|
||||||
|
})
|
||||||
|
|
||||||
|
function onPanelTransitionEnd(e: TransitionEvent) {
|
||||||
|
if (e.propertyName === 'grid-template-rows' && open.value) {
|
||||||
|
overflowVisible.value = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onToggle() {
|
||||||
|
if (props.disabled) return
|
||||||
|
ctx.toggle(value.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
const headerClasses = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'flex w-full items-center justify-between gap-4 px-7 pt-[28px] pb-[20px] text-left font-[600] text-[20px] transition-colors',
|
||||||
|
props.disabled ? 'cursor-not-allowed text-m-muted' : 'cursor-pointer hover:bg-m-surface',
|
||||||
|
props.headerClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const panelInnerClass = computed(() => twMerge('px-7 pt-[10px] pb-[20px]', props.panelClass))
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
ctx.register(
|
||||||
|
{
|
||||||
|
value: value.value,
|
||||||
|
getHeaderEl: () => headerRef.value,
|
||||||
|
isDisabled: () => props.disabled,
|
||||||
|
},
|
||||||
|
props.defaultOpen,
|
||||||
|
)
|
||||||
|
// Ouvert au montage (defaultOpen / contrôlé) : pas d'animation, overflow visible direct.
|
||||||
|
if (open.value) overflowVisible.value = true
|
||||||
|
})
|
||||||
|
|
||||||
|
onBeforeUnmount(() => ctx.unregister(value.value))
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
import type {ComputedRef, InjectionKey} from 'vue'
|
||||||
|
|
||||||
|
export interface AccordionItemRegistration {
|
||||||
|
value: string
|
||||||
|
getHeaderEl: () => HTMLElement | null
|
||||||
|
isDisabled: () => boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AccordionContext {
|
||||||
|
mode: ComputedRef<'single' | 'multiple'>
|
||||||
|
baseId: ComputedRef<string>
|
||||||
|
isOpen: (value: string) => boolean
|
||||||
|
toggle: (value: string) => void
|
||||||
|
register: (item: AccordionItemRegistration, defaultOpen: boolean) => void
|
||||||
|
unregister: (value: string) => void
|
||||||
|
focusSibling: (value: string, offset: 1 | -1) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export const accordionContextKey: InjectionKey<AccordionContext> = Symbol('MalioAccordion')
|
||||||
@@ -162,8 +162,8 @@ describe('MalioButton', () => {
|
|||||||
it('applies correct dimensions', () => {
|
it('applies correct dimensions', () => {
|
||||||
const wrapper = mountComponent()
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
expect(wrapper.get('button').classes()).toContain('w-[240px]')
|
expect(wrapper.get('button').classes()).toContain('w-[180px]')
|
||||||
expect(wrapper.get('button').classes()).toContain('h-[40px]')
|
expect(wrapper.get('button').classes()).toContain('h-[38px]')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('applies font styles', () => {
|
it('applies font styles', () => {
|
||||||
|
|||||||
@@ -84,7 +84,7 @@ const variantClasses = computed(() => {
|
|||||||
|
|
||||||
const mergedButtonClass = computed(() =>
|
const mergedButtonClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'inline-flex w-[240px] h-[40px] items-center justify-center gap-1 p-[10px] rounded-md text-base font-bold leading-[150%] transition-colors duration-150 focus:outline-none focus-visible:ring-2 focus-visible:ring-m-primary/50',
|
'inline-flex w-[180px] h-[38px] items-center justify-center gap-1 p-[10px] rounded-md text-base font-bold leading-[150%] transition-colors duration-150 m-focus-ring',
|
||||||
variantClasses.value,
|
variantClasses.value,
|
||||||
props.buttonClass,
|
props.buttonClass,
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ const isFilled = computed(() => props.variant === 'filled')
|
|||||||
|
|
||||||
const mergedButtonClass = computed(() =>
|
const mergedButtonClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'inline-flex items-center justify-center rounded-md p-1 transition-colors duration-150 focus:outline-none focus-visible:ring-2 focus-visible:ring-m-primary/50',
|
'inline-flex items-center justify-center rounded-md p-1 transition-colors duration-150 m-focus-ring',
|
||||||
isFilled.value
|
isFilled.value
|
||||||
? props.disabled
|
? props.disabled
|
||||||
? 'bg-m-disabled text-white cursor-not-allowed'
|
? 'bg-m-disabled text-white cursor-not-allowed'
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ type CheckboxProps = {
|
|||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const CheckboxForTest = Checkbox as DefineComponent<CheckboxProps>
|
const CheckboxForTest = Checkbox as DefineComponent<CheckboxProps>
|
||||||
@@ -114,7 +115,7 @@ describe('MalioCheckbox', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
expect(wrapper.get('input').attributes('aria-invalid')).toBe('true')
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('true')
|
||||||
expect(wrapper.get('label').classes()).toContain('text-m-error')
|
expect(wrapper.get('label').classes()).toContain('text-m-danger')
|
||||||
expect(wrapper.get('p').text()).toBe('You must accept')
|
expect(wrapper.get('p').text()).toBe('You must accept')
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -125,7 +126,7 @@ describe('MalioCheckbox', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
expect(wrapper.get('p').text()).toBe('Invalid')
|
expect(wrapper.get('p').text()).toBe('Invalid')
|
||||||
expect(wrapper.get('p').classes()).toContain('text-m-error')
|
expect(wrapper.get('p').classes()).toContain('text-m-danger')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('shows success styles and message when there is no error', () => {
|
it('shows success styles and message when there is no error', () => {
|
||||||
@@ -139,4 +140,55 @@ describe('MalioCheckbox', () => {
|
|||||||
expect(wrapper.get('p').text()).toBe('Valid')
|
expect(wrapper.get('p').text()).toBe('Valid')
|
||||||
expect(wrapper.get('p').classes()).toContain('text-m-success')
|
expect(wrapper.get('p').classes()).toContain('text-m-success')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('uses muted label color when unchecked', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Accept terms', modelValue: false})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses black label color when checked', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Accept terms', modelValue: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('updates label color when toggled without v-model (uncontrolled)', async () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Accept terms'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue(true)
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountCheckbox({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -25,12 +25,12 @@
|
|||||||
</svg>
|
</svg>
|
||||||
</span>
|
</span>
|
||||||
<span>
|
<span>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="mergedMessageClass"
|
:class="mergedMessageClass"
|
||||||
>
|
>
|
||||||
@@ -40,8 +40,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
|
||||||
defineOptions({name: 'MalioCheckbox', inheritAttrs: false})
|
defineOptions({name: 'MalioCheckbox', inheritAttrs: false})
|
||||||
|
|
||||||
@@ -60,6 +61,7 @@ const props = withDefaults(
|
|||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -75,14 +77,17 @@ const props = withDefaults(
|
|||||||
hint: '',
|
hint: '',
|
||||||
error: '',
|
error: '',
|
||||||
success: '',
|
success: '',
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
const attrs = useAttrs()
|
const attrs = useAttrs()
|
||||||
const generatedId = useId()
|
const generatedId = useId()
|
||||||
|
const localChecked = ref(false)
|
||||||
|
|
||||||
const inputId = computed(() => props.id?.toString() || `malio-checkbox-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-checkbox-${generatedId}`)
|
||||||
const isChecked = computed(() => !!props.modelValue)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const isChecked = computed(() => (isControlled.value ? !!props.modelValue : localChecked.value))
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
const disabled = computed(() => props.disabled)
|
const disabled = computed(() => props.disabled)
|
||||||
@@ -101,16 +106,17 @@ const mergedGroupClass = computed(() =>
|
|||||||
|
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'inp-cbx peer',
|
'inp-cbx peer ',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
const mergedLabelClass = computed(() =>
|
const mergedLabelClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'cbx text-black',
|
'cbx text-lg',
|
||||||
|
isChecked.value ? 'text-black' : 'text-m-muted',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60' : '',
|
disabled.value ? 'cursor-not-allowed text-black/60' : '',
|
||||||
hasError.value ? 'text-m-error' : '',
|
hasError.value ? 'text-m-danger' : '',
|
||||||
hasSuccess.value ? 'text-m-success' : '',
|
hasSuccess.value ? 'text-m-success' : '',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
@@ -119,8 +125,9 @@ const mergedLabelClass = computed(() =>
|
|||||||
const mergedMessageClass = computed(() =>
|
const mergedMessageClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'text-xs',
|
'text-xs',
|
||||||
|
props.reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'text-m-error'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
@@ -139,6 +146,10 @@ const onChange = (event: Event) => {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localChecked.value = target.checked
|
||||||
|
}
|
||||||
|
|
||||||
emit('update:modelValue', target.checked)
|
emit('update:modelValue', target.checked)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -161,10 +172,19 @@ const onChange = (event: Event) => {
|
|||||||
height: 18px;
|
height: 18px;
|
||||||
flex: 0 0 18px;
|
flex: 0 0 18px;
|
||||||
transform: scale(1);
|
transform: scale(1);
|
||||||
border: 2px solid rgb(0, 0, 0);
|
border: 2px solid rgb(var(--m-muted) / 1);
|
||||||
transition: all 0.1s ease;
|
transition: all 0.1s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.inp-cbx:checked + .cbx span:first-child {
|
||||||
|
border-color: rgb(0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.inp-cbx:focus-visible + .cbx span:first-child {
|
||||||
|
outline: 2px solid rgb(var(--m-primary) / 1);
|
||||||
|
outline-offset: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
.cbx span:first-child svg {
|
.cbx span:first-child svg {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 2px;
|
top: 2px;
|
||||||
@@ -200,14 +220,14 @@ const onChange = (event: Event) => {
|
|||||||
stroke-dashoffset: 0;
|
stroke-dashoffset: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.inp-cbx + .cbx.text-m-error span:first-child {
|
.inp-cbx + .cbx.text-m-danger span:first-child {
|
||||||
border-color: rgb(var(--m-error) / 1);
|
border-color: rgb(var(--m-danger) / 1);
|
||||||
}
|
}
|
||||||
.cbx.text-m-error span:first-child svg {
|
.cbx.text-m-danger span:first-child svg {
|
||||||
stroke: rgb(var(--m-error) / 1);
|
stroke: rgb(var(--m-danger) / 1);
|
||||||
}
|
}
|
||||||
.inp-cbx:checked + .cbx.text-m-error span:first-child {
|
.inp-cbx:checked + .cbx.text-m-danger span:first-child {
|
||||||
border-color: rgb(var(--m-error) / 1);
|
border-color: rgb(var(--m-danger) / 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
.inp-cbx + .cbx.text-m-success span:first-child {
|
.inp-cbx + .cbx.text-m-success span:first-child {
|
||||||
|
|||||||
@@ -7,14 +7,14 @@
|
|||||||
v-for="col in columns"
|
v-for="col in columns"
|
||||||
:key="col.key"
|
:key="col.key"
|
||||||
scope="col"
|
scope="col"
|
||||||
class="border-b border-black px-3 py-3 text-left align-middle text-[20px]"
|
class="border-b border-black px-3 py-3 text-left align-middle text-[16px]"
|
||||||
>
|
>
|
||||||
<slot
|
<slot
|
||||||
v-if="$slots[`header-${col.key}`]"
|
v-if="$slots[`header-${col.key}`]"
|
||||||
:name="`header-${col.key}`"
|
:name="`header-${col.key}`"
|
||||||
:column="col"
|
:column="col"
|
||||||
/>
|
/>
|
||||||
<span v-else class="font-semibold text-m-primary">{{ col.label }}</span>
|
<span v-else class="font-semibold text-black">{{ col.label }}</span>
|
||||||
</th>
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
@@ -32,7 +32,7 @@
|
|||||||
<td
|
<td
|
||||||
v-for="col in columns"
|
v-for="col in columns"
|
||||||
:key="col.key"
|
:key="col.key"
|
||||||
class="px-3 py-4 text-[18px] text-m-primary"
|
class="px-3 py-4 text-[14px] text-black"
|
||||||
:class="index < items.length - 1 ? 'border-b border-black' : ''"
|
:class="index < items.length - 1 ? 'border-b border-black' : ''"
|
||||||
>
|
>
|
||||||
<slot
|
<slot
|
||||||
@@ -57,30 +57,33 @@
|
|||||||
|
|
||||||
<div
|
<div
|
||||||
v-if="totalItems > 0"
|
v-if="totalItems > 0"
|
||||||
class="flex justify-between pt-2"
|
class="flex items-center justify-between pt-3"
|
||||||
data-test="pagination"
|
data-test="pagination"
|
||||||
>
|
>
|
||||||
<div class="flex gap-4">
|
<div class="flex items-center gap-4">
|
||||||
<span class="whitespace-nowrap text-[16px] text-black self-center">Lignes :</span>
|
<span class="whitespace-nowrap text-[16px] text-black">Lignes :</span>
|
||||||
<MalioSelect
|
<div class="h-[30px]">
|
||||||
:model-value="perPage"
|
<MalioSelect
|
||||||
:options="perPageSelectOptions"
|
:model-value="perPage"
|
||||||
min-width="w-20 !mt-0"
|
:options="perPageSelectOptions"
|
||||||
rounded="rounded"
|
group-class="w-20 h-[30px]"
|
||||||
text-field="text-sm"
|
field-class="h-[30px]"
|
||||||
text-value="text-sm"
|
rounded="rounded"
|
||||||
text-label="text-xs"
|
text-field="text-sm"
|
||||||
data-test="per-page-select"
|
text-value="text-sm"
|
||||||
@update:model-value="onPerPageChange"
|
text-label="text-xs"
|
||||||
/>
|
data-test="per-page-select"
|
||||||
|
@update:model-value="onPerPageChange"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<nav aria-label="Pagination" class="flex gap-1" data-test="pagination-nav">
|
<nav aria-label="Pagination" class="flex items-center gap-1" data-test="pagination-nav">
|
||||||
<MalioButton
|
<MalioButton
|
||||||
variant="tertiary"
|
variant="tertiary"
|
||||||
label="Prev"
|
label="Préc."
|
||||||
:disabled="page <= 1"
|
:disabled="page <= 1"
|
||||||
button-class="h-10 w-auto min-w-0 px-3 text-sm"
|
button-class="h-[30px] w-auto min-w-0 px-3 text-sm"
|
||||||
aria-label="Page précédente"
|
aria-label="Page précédente"
|
||||||
data-test="prev-button"
|
data-test="prev-button"
|
||||||
@click="goToPage(page - 1)"
|
@click="goToPage(page - 1)"
|
||||||
@@ -95,7 +98,7 @@
|
|||||||
<button
|
<button
|
||||||
v-else
|
v-else
|
||||||
type="button"
|
type="button"
|
||||||
class="h-10 min-w-[2.5rem] rounded px-2 text-sm transition-colors"
|
class="inline-flex h-[30px] min-w-[2.5rem] items-center justify-center rounded px-2 text-sm transition-colors"
|
||||||
:class="p === page
|
:class="p === page
|
||||||
? 'bg-m-btn-primary text-white font-semibold'
|
? 'bg-m-btn-primary text-white font-semibold'
|
||||||
: 'text-m-text hover:bg-m-bg'"
|
: 'text-m-text hover:bg-m-bg'"
|
||||||
@@ -109,9 +112,9 @@
|
|||||||
|
|
||||||
<MalioButton
|
<MalioButton
|
||||||
variant="tertiary"
|
variant="tertiary"
|
||||||
label="Next"
|
label="Suiv."
|
||||||
:disabled="page >= totalPages"
|
:disabled="page >= totalPages"
|
||||||
button-class="h-10 w-auto min-w-0 px-3 text-sm"
|
button-class="h-[30px] w-auto min-w-0 px-3 text-sm"
|
||||||
aria-label="Page suivante"
|
aria-label="Page suivante"
|
||||||
data-test="next-button"
|
data-test="next-button"
|
||||||
@click="goToPage(page + 1)"
|
@click="goToPage(page + 1)"
|
||||||
|
|||||||
@@ -0,0 +1,624 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import Date_ from './Date.vue'
|
||||||
|
|
||||||
|
type DateProps = {
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
markedDates?: Record<string, 'success' | 'danger'>
|
||||||
|
clearable?: boolean
|
||||||
|
editable?: boolean
|
||||||
|
invalidMessage?: string
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const DateForTest = Date_ as DefineComponent<DateProps>
|
||||||
|
const mountDate = (props: DateProps = {}) => mount(DateForTest, {props, attachTo: document.body})
|
||||||
|
|
||||||
|
describe('MalioDate', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
describe('rendu', () => {
|
||||||
|
it('renders the label and the calendar icon', () => {
|
||||||
|
const wrapper = mountDate({label: 'Date de naissance'})
|
||||||
|
expect(wrapper.get('label').text()).toBe('Date de naissance')
|
||||||
|
expect(wrapper.find('[data-test="calendar-icon"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountDate({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountDate({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('displays the formatted value in the field', () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('19/05/2026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not show the popover initially', () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('popover', () => {
|
||||||
|
it('opens on field click', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on calendar icon click', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="calendar-icon"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on calendar icon click in editable mode', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
await wrapper.get('[data-test="calendar-icon"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on the current month when there is no value', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Mai 2026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on the value month when a value is set', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2025-12-25'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Décembre 2025')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes on outside mousedown', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
document.body.dispatchEvent(new MouseEvent('mousedown', {bubbles: true}))
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('navigation jours', () => {
|
||||||
|
it('goes to the next month on the right chevron', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-next"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Juin 2026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rolls December to January and bumps the year', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-12-15'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-next"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Janvier 2027')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('month-change', () => {
|
||||||
|
it('émet month-change à l\'ouverture avec le mois courant', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('month-change')?.at(-1)).toEqual([{month: 4, year: 2026}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet month-change sur le mois de la valeur à l\'ouverture', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2025-12-25'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('month-change')?.at(-1)).toEqual([{month: 11, year: 2025}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet month-change à chaque navigation de mois', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-next"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('month-change')?.at(-1)).toEqual([{month: 5, year: 2026}])
|
||||||
|
await wrapper.get('[data-test="header-prev"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-prev"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('month-change')?.at(-1)).toEqual([{month: 3, year: 2026}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('ne ré-émet pas month-change après fermeture', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const countOpen = wrapper.emitted('month-change')?.length ?? 0
|
||||||
|
document.body.dispatchEvent(new MouseEvent('mousedown', {bubbles: true}))
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(wrapper.emitted('month-change')?.length ?? 0).toBe(countOpen)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('markedDates', () => {
|
||||||
|
it('transmet markedDates à la grille (fond tokenisé)', async () => {
|
||||||
|
const wrapper = mountDate({markedDates: {'2026-05-20': 'success'}})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const pill = wrapper.get('[data-iso="2026-05-20"]').get('span.rounded-full')
|
||||||
|
expect(pill.classes()).toContain('bg-m-success/15')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('sélection', () => {
|
||||||
|
it('emits the ISO date and closes on day click', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('bornes min/max', () => {
|
||||||
|
it('disables days outside the range', async () => {
|
||||||
|
const wrapper = mountDate({min: '2026-05-10', max: '2026-05-20'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const outside = wrapper.get('[data-test="day"][data-iso="2026-05-05"]')
|
||||||
|
expect((outside.element as HTMLButtonElement).disabled).toBe(true)
|
||||||
|
await outside.trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('vue mois', () => {
|
||||||
|
it('switches to month view on header toggle', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-toggle"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="month-picker"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('navigates the year with chevrons in month view', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-toggle"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-next"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('2027')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns to day view on month click', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="header-toggle"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="month"][data-month="0"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="month-picker"]').exists()).toBe(false)
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Janvier 2026')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('effacement', () => {
|
||||||
|
it('shows the clear button when there is a value', () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
expect(wrapper.find('[data-test="clear"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the clear button when empty', () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
expect(wrapper.find('[data-test="clear"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits null and does not open the popover on clear', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('états', () => {
|
||||||
|
it('does not open when disabled', async () => {
|
||||||
|
const wrapper = mountDate({disabled: true})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disabled : label grisé', () => {
|
||||||
|
const wrapper = mountDate({disabled: true, label: 'Date'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disabled : pas de croix d\'effacement même avec une valeur', () => {
|
||||||
|
const wrapper = mountDate({disabled: true, modelValue: '2026-05-19'})
|
||||||
|
expect(wrapper.find('[data-test="clear"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disabled + rempli : icône calendrier grisée (pas noire)', () => {
|
||||||
|
const wrapper = mountDate({disabled: true, modelValue: '2026-05-19'})
|
||||||
|
const icon = wrapper.get('[data-test="calendar-icon"]')
|
||||||
|
expect(icon.classes()).toContain('text-m-muted')
|
||||||
|
expect(icon.classes()).not.toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open when readonly', async () => {
|
||||||
|
const wrapper = mountDate({readonly: true, modelValue: '2026-05-19'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : bordure noire sans bleu', () => {
|
||||||
|
const wrapper = mountDate({readonly: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
expect(input.classes()).toContain('border-black')
|
||||||
|
expect(input.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(input.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label muted sans bleu', () => {
|
||||||
|
const wrapper = mountDate({readonly: true, label: 'Date'})
|
||||||
|
const label = wrapper.get('label')
|
||||||
|
expect(label.classes()).toContain('text-m-muted')
|
||||||
|
expect(label.classes()).not.toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : icône calendrier en text-m-muted', () => {
|
||||||
|
const wrapper = mountDate({readonly: true, label: 'Date'})
|
||||||
|
expect(wrapper.get('[data-test="calendar-icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label et icône en noir, bordure noire', () => {
|
||||||
|
const wrapper = mountDate({readonly: true, label: 'Date', modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
const label = wrapper.get('label')
|
||||||
|
const icon = wrapper.get('[data-test="calendar-icon"]')
|
||||||
|
expect(input.classes()).toContain('border-black')
|
||||||
|
expect(input.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(label.classes()).toContain('text-black')
|
||||||
|
expect(icon.classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('accessibilité', () => {
|
||||||
|
it('sets aria-invalid and describedby on error', () => {
|
||||||
|
const wrapper = mountDate({error: 'Date requise'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
expect(input.attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(input.attributes('aria-describedby')).toBeTruthy()
|
||||||
|
expect(wrapper.text()).toContain('Date requise')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('synchronisation externe', () => {
|
||||||
|
it('updates the displayed value when modelValue changes', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
await wrapper.setProps({modelValue: '2026-12-25'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('25/12/2026')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('reserveMessageSpace', () => {
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountDate({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountDate({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountDate({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('saisie manuelle (editable)', () => {
|
||||||
|
it('efface l\'erreur de saisie quand modelValue change de l\'extérieur', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
await wrapper.setProps({modelValue: '2026-05-19'})
|
||||||
|
expect(wrapper.text()).not.toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('par défaut (editable=false) l\'input reste readonly et affiche la valeur', () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
expect(input.attributes('readonly')).toBeDefined()
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('19/05/2026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('editable=true : l\'input n\'est plus readonly', () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
expect(wrapper.get('[data-test="date-input"]').attributes('readonly')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet l\'ISO sur saisie clavier valide au blur', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('19/05/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('garde le texte et affiche « Date invalide » sur saisie invalide au blur', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('31/02/2026')
|
||||||
|
expect(input.attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('passe en erreur si la date saisie est hors min/max', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, min: '2026-05-10', max: '2026-05-20'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('25/12/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet null sur saisie vidée au blur', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('efface l\'erreur de saisie quand on sélectionne une date au calendrier', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
await input.trigger('focus')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19'])
|
||||||
|
expect(wrapper.text()).not.toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('valide et ferme le popover sur Entrée', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.trigger('focus')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
await input.setValue('19/05/2026')
|
||||||
|
// Valeur DOM réelle de la touche Entrée ('Enter') ; `trigger('keydown.enter')`
|
||||||
|
// produirait `key: 'enter'`, qui ne matche pas le handler manuel `e.key === 'Enter'`.
|
||||||
|
await input.trigger('keydown', {key: 'Enter'})
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('utilise le message invalidMessage personnalisé', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, invalidMessage: 'Format incorrect'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
// 31/02/2026 : champs valides (jour ≤ 31, mois ≤ 12) mais le 31 février n'existe pas.
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.text()).toContain('Format incorrect')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('empêche la frappe d\'une date absurde (99/99/9999 borné par le masque)', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('99/99/9999')
|
||||||
|
await input.trigger('blur')
|
||||||
|
// Le bornage refuse « 9 » dès le 1er chiffre du jour/mois : la saisie absurde
|
||||||
|
// ne s'inscrit jamais et aucune date réelle n'est émise.
|
||||||
|
expect((input.element as HTMLInputElement).value).not.toContain('99')
|
||||||
|
const emitted = wrapper.emitted('update:modelValue') ?? []
|
||||||
|
expect(emitted.every(([value]) => value === null)).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('empêche un jour > 31 ou un mois > 12 (exemple métier 33/19, 2e chiffre borné)', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
// 33 (jour) : le 2e « 3 » est refusé → seul « 3 » subsiste.
|
||||||
|
await input.setValue('33')
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('3')
|
||||||
|
// 19 en mois : après un jour valide, le 2e chiffre du mois (« 9 ») est refusé.
|
||||||
|
await input.setValue('15/19')
|
||||||
|
expect((input.element as HTMLInputElement).value).not.toContain('19')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('gabarit de saisie (editable)', () => {
|
||||||
|
it('affiche le gabarit complet en gris quand editable + focus + vide', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('focus')
|
||||||
|
const ghost = wrapper.get('[data-test="format-ghost"]')
|
||||||
|
expect(ghost.text()).toBe('JJ/MM/AAAA')
|
||||||
|
expect(wrapper.get('[data-test="ghost-remaining"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('remplit le gabarit au fur et à mesure de la saisie', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.trigger('focus')
|
||||||
|
await input.setValue('19')
|
||||||
|
// eager : le séparateur se pose dès que le groupe est complet (« 19 » → « 19/ »)
|
||||||
|
expect(wrapper.get('[data-test="format-ghost"]').text()).toBe('19/MM/AAAA')
|
||||||
|
expect(wrapper.get('[data-test="ghost-typed"]').text()).toBe('19/')
|
||||||
|
expect(wrapper.get('[data-test="ghost-typed"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('pose le séparateur automatiquement dès qu\'un groupe est complet (eager)', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('1905')
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('19/05/')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas de gabarit en mode non editable', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="format-ghost"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas de gabarit quand editable mais vide et non focus', () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
expect(wrapper.find('[data-test="format-ghost"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('vide le champ au clic sur la croix même après une saisie invalide (modelValue déjà null)', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('31/02/2026')
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('état de validité (update:valid)', () => {
|
||||||
|
it('émet valid=true au montage avec une valeur valide', () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true au montage quand le champ est vide', () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur saisie clavier valide', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('19/05/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=false sur saisie malformée sans émettre modelValue', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([false])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=false sur saisie hors min/max', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, min: '2026-05-10', max: '2026-05-20'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('25/12/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([false])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur saisie vidée même si le champ est requis', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, required: true, modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur clear', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true quand on sélectionne une date au calendrier', async () => {
|
||||||
|
const wrapper = mountDate()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('repasse valid=true quand modelValue change de l\'extérieur après une saisie invalide', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([false])
|
||||||
|
await wrapper.setProps({modelValue: '2026-05-19'})
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('saisie brute (update:rawValue)', () => {
|
||||||
|
it('émet le texte brut trimmé sur saisie malformée, sans émettre modelValue', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['31/02/2026'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet le texte brut trimmé sur saisie hors min/max', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, min: '2026-05-10', max: '2026-05-20'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('25/12/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['25/12/2026'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide et l\'ISO sur saisie clavier valide', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('19/05/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide sur saisie vidée au blur', async () => {
|
||||||
|
const wrapper = mountDate({editable: true, modelValue: '2026-05-19'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide sur clear', async () => {
|
||||||
|
const wrapper = mountDate({modelValue: '2026-05-19'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide quand on sélectionne une date au calendrier', async () => {
|
||||||
|
const wrapper = mountDate({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['31/02/2026'])
|
||||||
|
await input.trigger('focus')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,159 @@
|
|||||||
|
<template>
|
||||||
|
<CalendarField
|
||||||
|
:id="id"
|
||||||
|
:display-value="displayValue"
|
||||||
|
:sync-to="modelValue ?? null"
|
||||||
|
:name="name"
|
||||||
|
:label="label"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:readonly="readonly"
|
||||||
|
:hint="hint"
|
||||||
|
:error="mergedError"
|
||||||
|
:success="success"
|
||||||
|
:clearable="clearable"
|
||||||
|
:editable="editable"
|
||||||
|
:input-class="inputClass"
|
||||||
|
:label-class="labelClass"
|
||||||
|
:group-class="groupClass"
|
||||||
|
v-bind="$attrs"
|
||||||
|
@clear="onClear"
|
||||||
|
@commit="onCommit"
|
||||||
|
@month-change="(payload) => emit('month-change', payload)"
|
||||||
|
>
|
||||||
|
<template #default="{ currentMonth, currentYear, close }">
|
||||||
|
<MonthGrid
|
||||||
|
:month="currentMonth"
|
||||||
|
:year="currentYear"
|
||||||
|
:selected-date="modelValue ?? null"
|
||||||
|
:marked-dates="markedDates"
|
||||||
|
:min="min"
|
||||||
|
:max="max"
|
||||||
|
@select="(iso) => onSelect(iso, close)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</CalendarField>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref, watch} from 'vue'
|
||||||
|
import CalendarField from './internal/CalendarField.vue'
|
||||||
|
import MonthGrid from './internal/MonthGrid.vue'
|
||||||
|
import {formatIsoToDisplay, isDateInRange, isValidIso, parseDisplayToIso} from './composables/dateFormat'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDate', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
// Statut générique par jour, ISO yyyy-mm-dd → variante de fond. Aucune
|
||||||
|
// logique métier dans le layer : le consommateur fournit la liste.
|
||||||
|
markedDates?: Record<string, 'success' | 'danger'>
|
||||||
|
clearable?: boolean
|
||||||
|
editable?: boolean
|
||||||
|
invalidMessage?: string
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
label: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
placeholder: 'JJ/MM/AAAA',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
min: undefined,
|
||||||
|
max: undefined,
|
||||||
|
markedDates: undefined,
|
||||||
|
clearable: true,
|
||||||
|
editable: false,
|
||||||
|
invalidMessage: 'Date invalide',
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'update:modelValue', value: string | null): void
|
||||||
|
(e: 'update:valid', value: boolean): void
|
||||||
|
// Canal séparé pour la saisie invalide (validation back-autoritative) : texte brut
|
||||||
|
// tel que tapé sur saisie non parsable/hors plage, '' sinon. Ne JAMAIS transiter
|
||||||
|
// par modelValue, qui doit rester ISO|null pour l'affichage et le round-trip.
|
||||||
|
(e: 'update:rawValue', value: string): void
|
||||||
|
// Mois affiché dans le popover (month 0-11) : à l'ouverture et à chaque nav.
|
||||||
|
(e: 'month-change', value: {month: number, year: number}): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const displayValue = computed(() => formatIsoToDisplay(props.modelValue ?? null))
|
||||||
|
|
||||||
|
const internalError = ref('')
|
||||||
|
const mergedError = computed(() => props.error || internalError.value)
|
||||||
|
|
||||||
|
// La validité ne reflète que la saisie : malformée/hors plage → false. Un champ
|
||||||
|
// vide est valide (l'obligation `required` reste à la charge du parent).
|
||||||
|
const setError = (message: string) => {
|
||||||
|
internalError.value = message
|
||||||
|
emit('update:valid', message === '')
|
||||||
|
}
|
||||||
|
|
||||||
|
const onCommit = (text: string) => {
|
||||||
|
const trimmed = text.trim()
|
||||||
|
if (trimmed === '') {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const iso = parseDisplayToIso(trimmed)
|
||||||
|
if (iso && isDateInRange(iso, props.min, props.max)) {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', iso)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setError(props.invalidMessage)
|
||||||
|
emit('update:rawValue', trimmed)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClear = () => {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSelect = (iso: string, close: () => void) => {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', iso)
|
||||||
|
close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// immediate : émet aussi la validité au montage, pour que le parent connaisse
|
||||||
|
// l'état d'un champ pré-rempli (formulaire d'édition) sans interaction préalable.
|
||||||
|
watch(() => props.modelValue, (val) => {
|
||||||
|
setError('')
|
||||||
|
if (val && !isValidIso(val) && import.meta.dev) {
|
||||||
|
console.warn(`[MalioDate] modelValue invalide ignoré : "${val}"`)
|
||||||
|
}
|
||||||
|
}, {immediate: true})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,155 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import DateRange from './DateRange.vue'
|
||||||
|
|
||||||
|
type RangeValue = {start: string; end: string}
|
||||||
|
type DateRangeProps = {
|
||||||
|
modelValue?: RangeValue | null
|
||||||
|
label?: string
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
error?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
clearable?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const DateRangeForTest = DateRange as DefineComponent<DateRangeProps>
|
||||||
|
const mountRange = (props: DateRangeProps = {}) =>
|
||||||
|
mount(DateRangeForTest, {props, attachTo: document.body})
|
||||||
|
|
||||||
|
const openAndClickDays = async (wrapper: ReturnType<typeof mountRange>, isos: string[]) => {
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
for (const iso of isos) {
|
||||||
|
await wrapper.get(`[data-test="day"][data-iso="${iso}"]`).trigger('click')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('MalioDateRange', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
it('renders the label and calendar icon', () => {
|
||||||
|
const wrapper = mountRange({label: 'Période'})
|
||||||
|
expect(wrapper.get('label').text()).toBe('Période')
|
||||||
|
expect(wrapper.find('[data-test="calendar-icon"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('displays the formatted range when modelValue is set', () => {
|
||||||
|
const wrapper = mountRange({modelValue: {start: '2026-05-19', end: '2026-05-25'}})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('19/05/2026 - 25/05/2026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows an empty field without a value', () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on the start month when a range is set', async () => {
|
||||||
|
const wrapper = mountRange({modelValue: {start: '2025-12-10', end: '2025-12-20'}})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Décembre 2025')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not emit on the first click', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-19'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits the range and closes on the second click', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-19', '2026-05-25'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([{start: '2026-05-19', end: '2026-05-25'}])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('auto-inverts when the second click is before the first', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-25', '2026-05-19'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([{start: '2026-05-19', end: '2026-05-25'}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows a single-day range', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-19', '2026-05-19'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([{start: '2026-05-19', end: '2026-05-19'}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('restarts a new range on the third click', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-19', '2026-05-25'])
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-10"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toHaveLength(1)
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-12"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([{start: '2026-05-10', end: '2026-05-12'}])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('previews the range on hover while selecting', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-22"]').trigger('mouseenter')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-20"]').attributes('data-range-role')).toBe('in-range')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not preview before selecting', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-22"]').trigger('mouseenter')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-20"]').attributes('data-range-role')).toBe('none')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('marks start, end and in-range roles for a committed range', async () => {
|
||||||
|
const wrapper = mountRange({modelValue: {start: '2026-05-19', end: '2026-05-25'}})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-19"]').attributes('data-range-role')).toBe('start')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-25"]').attributes('data-range-role')).toBe('end')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-22"]').attributes('data-range-role')).toBe('in-range')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('cancels an in-progress selection on outside click', async () => {
|
||||||
|
const wrapper = mountRange()
|
||||||
|
await openAndClickDays(wrapper, ['2026-05-19'])
|
||||||
|
document.body.dispatchEvent(new MouseEvent('mousedown', {bubbles: true}))
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-19"]').attributes('data-range-role')).toBe('none')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits null on clear', async () => {
|
||||||
|
const wrapper = mountRange({modelValue: {start: '2026-05-19', end: '2026-05-25'}})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables days outside min/max', async () => {
|
||||||
|
const wrapper = mountRange({min: '2026-05-10', max: '2026-05-20'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const outside = wrapper.get('[data-test="day"][data-iso="2026-05-05"]')
|
||||||
|
expect((outside.element as HTMLButtonElement).disabled).toBe(true)
|
||||||
|
await outside.trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-invalid on error', () => {
|
||||||
|
const wrapper = mountRange({error: 'Période requise'})
|
||||||
|
expect(wrapper.get('[data-test="date-input"]').attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(wrapper.text()).toContain('Période requise')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open when disabled', async () => {
|
||||||
|
const wrapper = mountRange({disabled: true})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,140 @@
|
|||||||
|
<template>
|
||||||
|
<CalendarField
|
||||||
|
:id="id"
|
||||||
|
:display-value="displayValue"
|
||||||
|
:sync-to="validRange?.start ?? null"
|
||||||
|
:name="name"
|
||||||
|
:label="label"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:readonly="readonly"
|
||||||
|
:hint="hint"
|
||||||
|
:error="error"
|
||||||
|
:success="success"
|
||||||
|
:clearable="clearable"
|
||||||
|
:input-class="inputClass"
|
||||||
|
:label-class="labelClass"
|
||||||
|
:group-class="groupClass"
|
||||||
|
v-bind="$attrs"
|
||||||
|
@clear="onClear"
|
||||||
|
@close="onClose"
|
||||||
|
>
|
||||||
|
<template #default="{ currentMonth, currentYear, close }">
|
||||||
|
<MonthGrid
|
||||||
|
:month="currentMonth"
|
||||||
|
:year="currentYear"
|
||||||
|
:range-start="rangeStart"
|
||||||
|
:range-end="rangeEnd"
|
||||||
|
:preview-date="previewDate"
|
||||||
|
:min="min"
|
||||||
|
:max="max"
|
||||||
|
@select="(iso) => onSelectDay(iso, close)"
|
||||||
|
@hover="onHover"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</CalendarField>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref} from 'vue'
|
||||||
|
import CalendarField from './internal/CalendarField.vue'
|
||||||
|
import MonthGrid from './internal/MonthGrid.vue'
|
||||||
|
import {formatIsoToDisplay, isValidIso} from './composables/dateFormat'
|
||||||
|
import {normalizeRange, type DateRangeValue} from './composables/dateRange'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDateRange', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: DateRangeValue | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
clearable?: boolean
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
label: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
placeholder: 'JJ/MM/AAAA',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
min: undefined,
|
||||||
|
max: undefined,
|
||||||
|
clearable: true,
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{(e: 'update:modelValue', value: DateRangeValue | null): void}>()
|
||||||
|
|
||||||
|
const pendingStart = ref<string | null>(null)
|
||||||
|
const hoverDate = ref<string | null>(null)
|
||||||
|
const isSelecting = computed(() => pendingStart.value !== null)
|
||||||
|
|
||||||
|
const validRange = computed<DateRangeValue | null>(() => {
|
||||||
|
const v = props.modelValue
|
||||||
|
if (v && isValidIso(v.start) && isValidIso(v.end)) return v
|
||||||
|
return null
|
||||||
|
})
|
||||||
|
|
||||||
|
const rangeStart = computed(() =>
|
||||||
|
isSelecting.value ? pendingStart.value : (validRange.value?.start ?? null),
|
||||||
|
)
|
||||||
|
const rangeEnd = computed(() =>
|
||||||
|
isSelecting.value ? null : (validRange.value?.end ?? null),
|
||||||
|
)
|
||||||
|
const previewDate = computed(() => (isSelecting.value ? hoverDate.value : null))
|
||||||
|
|
||||||
|
const displayValue = computed(() => {
|
||||||
|
if (isSelecting.value || !validRange.value) return ''
|
||||||
|
return `${formatIsoToDisplay(validRange.value.start)} - ${formatIsoToDisplay(validRange.value.end)}`
|
||||||
|
})
|
||||||
|
|
||||||
|
const onSelectDay = (iso: string, close: () => void) => {
|
||||||
|
if (pendingStart.value === null) {
|
||||||
|
pendingStart.value = iso
|
||||||
|
hoverDate.value = null
|
||||||
|
return
|
||||||
|
}
|
||||||
|
emit('update:modelValue', normalizeRange(pendingStart.value, iso))
|
||||||
|
pendingStart.value = null
|
||||||
|
hoverDate.value = null
|
||||||
|
close()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onHover = (iso: string | null) => {
|
||||||
|
if (isSelecting.value) hoverDate.value = iso
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClose = () => {
|
||||||
|
pendingStart.value = null
|
||||||
|
hoverDate.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClear = () => {
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
pendingStart.value = null
|
||||||
|
hoverDate.value = null
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,354 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import DateTime_ from './DateTime.vue'
|
||||||
|
import MalioTimePicker from '../time/TimePicker.vue'
|
||||||
|
|
||||||
|
type DateTimeProps = {
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
clearable?: boolean
|
||||||
|
editable?: boolean
|
||||||
|
invalidMessage?: string
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const DateTimeForTest = DateTime_ as DefineComponent<DateTimeProps>
|
||||||
|
const mountDateTime = (props: DateTimeProps = {}) =>
|
||||||
|
mount(DateTimeForTest, {props, attachTo: document.body})
|
||||||
|
|
||||||
|
describe('MalioDateTime', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19, 9, 5, 0)) // 19 mai 2026, 09:05
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
describe('rendu', () => {
|
||||||
|
it('affiche le label et l\'icône calendrier', () => {
|
||||||
|
const wrapper = mountDateTime({label: 'Rendez-vous'})
|
||||||
|
expect(wrapper.get('label').text()).toBe('Rendez-vous')
|
||||||
|
expect(wrapper.find('[data-test="calendar-icon"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche la valeur formatée date + heure dans le champ', () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('20/05/2026 14:30')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('popover', () => {
|
||||||
|
it('ouvre la grille et le champ sélecteur d\'heure au clic', async () => {
|
||||||
|
const wrapper = mountDateTime()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="month-grid"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.findComponent(MalioTimePicker).exists()).toBe(true)
|
||||||
|
expect(wrapper.find('[data-test="time-field"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('sélection', () => {
|
||||||
|
it('émet le jour à l\'heure actuelle (si aucune heure choisie) et garde le popover ouvert', async () => {
|
||||||
|
const wrapper = mountDateTime()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
// heure système figée à 09:05
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19T09:05:00'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applique l\'heure réglée avant le clic du jour', async () => {
|
||||||
|
const wrapper = mountDateTime()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
wrapper.findComponent(MalioTimePicker).vm.$emit('update:modelValue', '09:15')
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-19T09:15:00'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('met à jour l\'heure quand une date est déjà choisie', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
wrapper.findComponent(MalioTimePicker).vm.$emit('update:modelValue', '08:45')
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-20T08:45:00'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('initialise le champ heure depuis la valeur', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.findComponent(MalioTimePicker).props('modelValue')).toBe('14:30')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('bornes min/max', () => {
|
||||||
|
it('désactive les jours hors bornes (datetime borné sur la date)', async () => {
|
||||||
|
const wrapper = mountDateTime({min: '2026-05-10T00:00:00', max: '2026-05-20T00:00:00'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const outside = wrapper.get('[data-test="day"][data-iso="2026-05-05"]')
|
||||||
|
expect((outside.element as HTMLButtonElement).disabled).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('effacement', () => {
|
||||||
|
it('émet null au clic sur la croix', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('accessibilité', () => {
|
||||||
|
it('positionne aria-invalid et describedby sur erreur', () => {
|
||||||
|
const wrapper = mountDateTime({error: 'Date requise'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
expect(input.attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(input.attributes('aria-describedby')).toBeTruthy()
|
||||||
|
expect(wrapper.text()).toContain('Date requise')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('saisie manuelle (editable)', () => {
|
||||||
|
it('par défaut (editable=false) l\'input reste readonly', () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
expect(wrapper.get('[data-test="date-input"]').attributes('readonly')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('editable=true : l\'input n\'est plus readonly', () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
expect(wrapper.get('[data-test="date-input"]').attributes('readonly')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet le datetime ISO sur saisie clavier valide au blur', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('20/05/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-20T14:30:00'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('garde le texte et affiche « Date invalide » sur saisie invalide au blur', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect((input.element as HTMLInputElement).value).toBe('31/02/2026 14:30')
|
||||||
|
expect(input.attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('empêche la frappe d\'un datetime absurde (99/99/9999 99:99 borné par le masque)', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('99/99/9999 99:99')
|
||||||
|
await input.trigger('blur')
|
||||||
|
// Le masque borne le 1er chiffre de chaque champ (jour 0-3, mois 0-1,
|
||||||
|
// heure 0-2, minute 0-5) : « 9 » est rejeté partout, rien ne s'inscrit
|
||||||
|
// et aucun datetime réel n'est émis.
|
||||||
|
expect((input.element as HTMLInputElement).value).not.toContain('99')
|
||||||
|
const emitted = wrapper.emitted('update:modelValue') ?? []
|
||||||
|
expect(emitted.every(([value]) => value === null)).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('passe en erreur si le datetime saisi est hors min/max', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, min: '2026-05-10T00:00:00', max: '2026-05-20T00:00:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('25/12/2026 10:00')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet null sur saisie vidée au blur', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, modelValue: '2026-05-20T14:30:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('valide et ferme le popover sur Entrée', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.trigger('focus')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(true)
|
||||||
|
await input.setValue('20/05/2026 14:30')
|
||||||
|
await input.trigger('keydown', {key: 'Enter'})
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-20T14:30:00'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('utilise le message invalidMessage personnalisé', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, invalidMessage: 'Format incorrect'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
// 31/02 : champs valides mais date inexistante (le masque la laisse passer, la validation la rejette).
|
||||||
|
await input.setValue('31/02/2026 10:00')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.text()).toContain('Format incorrect')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('efface l\'erreur de saisie quand modelValue change de l\'extérieur', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.text()).toContain('Date invalide')
|
||||||
|
await wrapper.setProps({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
expect(wrapper.text()).not.toContain('Date invalide')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('gabarit de saisie (editable)', () => {
|
||||||
|
it('affiche le gabarit date+heure complet en gris quand editable + focus + vide', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('focus')
|
||||||
|
expect(wrapper.get('[data-test="format-ghost"]').text().replace(/\xa0/g, ' ')).toBe('JJ/MM/AAAA HH:MM')
|
||||||
|
expect(wrapper.get('[data-test="ghost-remaining"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('remplit le gabarit au fur et à mesure de la saisie', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.trigger('focus')
|
||||||
|
await input.setValue('190520')
|
||||||
|
expect(wrapper.get('[data-test="format-ghost"]').text().replace(/\xa0/g, ' ')).toBe('19/05/20AA HH:MM')
|
||||||
|
expect(wrapper.get('[data-test="ghost-typed"]').text()).toBe('19/05/20')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas de gabarit en mode non editable', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="format-ghost"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('état de validité (update:valid)', () => {
|
||||||
|
it('émet valid=true au montage avec une valeur valide', () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true au montage quand le champ est vide', () => {
|
||||||
|
const wrapper = mountDateTime()
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur saisie clavier valide', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('20/05/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=false sur saisie malformée sans émettre modelValue', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([false])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur saisie vidée même si le champ est requis', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, required: true, modelValue: '2026-05-20T14:30:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true sur clear', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet valid=true quand on sélectionne une date au calendrier', async () => {
|
||||||
|
const wrapper = mountDateTime()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('repasse valid=true quand modelValue change de l\'extérieur après une saisie invalide', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([false])
|
||||||
|
await wrapper.setProps({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
expect(wrapper.emitted('update:valid')?.at(-1)).toEqual([true])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('saisie brute (update:rawValue)', () => {
|
||||||
|
it('émet le texte brut trimmé sur saisie malformée, sans émettre modelValue', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['31/02/2026 14:30'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet le texte brut trimmé sur saisie hors min/max', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, min: '2026-05-10T00:00:00', max: '2026-05-20T00:00:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('25/12/2026 10:00')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['25/12/2026 10:00'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide et l\'ISO sur saisie clavier valide', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('20/05/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-05-20T14:30:00'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide sur saisie vidée au blur', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true, modelValue: '2026-05-20T14:30:00'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide sur clear', async () => {
|
||||||
|
const wrapper = mountDateTime({modelValue: '2026-05-20T14:30:00'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet rawValue vide quand on sélectionne une date au calendrier', async () => {
|
||||||
|
const wrapper = mountDateTime({editable: true})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]')
|
||||||
|
await input.setValue('31/02/2026 14:30')
|
||||||
|
await input.trigger('blur')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual(['31/02/2026 14:30'])
|
||||||
|
await input.trigger('focus')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-19"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:rawValue')?.at(-1)).toEqual([''])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,186 @@
|
|||||||
|
<template>
|
||||||
|
<CalendarField
|
||||||
|
:id="id"
|
||||||
|
:display-value="displayValue"
|
||||||
|
:sync-to="datePart"
|
||||||
|
:name="name"
|
||||||
|
:label="label"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:readonly="readonly"
|
||||||
|
:hint="hint"
|
||||||
|
:error="mergedError"
|
||||||
|
:success="success"
|
||||||
|
:clearable="clearable"
|
||||||
|
:editable="editable"
|
||||||
|
placeholder-template="JJ/MM/AAAA HH:MM"
|
||||||
|
:input-class="inputClass"
|
||||||
|
:label-class="labelClass"
|
||||||
|
:group-class="groupClass"
|
||||||
|
v-bind="$attrs"
|
||||||
|
@clear="onClear"
|
||||||
|
@commit="onCommit"
|
||||||
|
>
|
||||||
|
<template #default="{ currentMonth, currentYear }">
|
||||||
|
<MonthGrid
|
||||||
|
:month="currentMonth"
|
||||||
|
:year="currentYear"
|
||||||
|
:selected-date="datePart"
|
||||||
|
:min="min?.slice(0, 10)"
|
||||||
|
:max="max?.slice(0, 10)"
|
||||||
|
@select="onSelectDay"
|
||||||
|
/>
|
||||||
|
<div class="mt-4">
|
||||||
|
<MalioTimePicker
|
||||||
|
:model-value="timeValue || null"
|
||||||
|
label="Heure"
|
||||||
|
:clearable="false"
|
||||||
|
static-popover
|
||||||
|
@update:model-value="onTimeChange"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</CalendarField>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref, watch} from 'vue'
|
||||||
|
import CalendarField from './internal/CalendarField.vue'
|
||||||
|
import MonthGrid from './internal/MonthGrid.vue'
|
||||||
|
import MalioTimePicker from '../time/TimePicker.vue'
|
||||||
|
import {formatTime} from '../time/composables/timeFormat'
|
||||||
|
import {isDateInRange} from './composables/dateFormat'
|
||||||
|
import {composeDateTime, formatIsoDateTimeToDisplay, isValidIsoDateTime, parseDisplayToIsoDateTime, splitDateTime} from './composables/datetimeFormat'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDateTime', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
clearable?: boolean
|
||||||
|
editable?: boolean
|
||||||
|
invalidMessage?: string
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
label: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
placeholder: 'JJ/MM/AAAA HH:MM',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
min: undefined,
|
||||||
|
max: undefined,
|
||||||
|
clearable: true,
|
||||||
|
editable: false,
|
||||||
|
invalidMessage: 'Date invalide',
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'update:modelValue', value: string | null): void
|
||||||
|
(e: 'update:valid', value: boolean): void
|
||||||
|
// Canal séparé pour la saisie invalide (validation back-autoritative) : texte brut
|
||||||
|
// tel que tapé sur saisie non parsable/hors plage, '' sinon. Ne JAMAIS transiter
|
||||||
|
// par modelValue, qui doit rester ISO|null pour l'affichage et le round-trip.
|
||||||
|
(e: 'update:rawValue', value: string): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
// pendingTime : heure réglée avant qu'un jour ne soit choisi (sinon on ne peut pas émettre).
|
||||||
|
const pendingTime = ref('')
|
||||||
|
|
||||||
|
const parts = computed(() => splitDateTime(props.modelValue ?? null))
|
||||||
|
const datePart = computed(() => parts.value.date)
|
||||||
|
const displayValue = computed(() => formatIsoDateTimeToDisplay(props.modelValue ?? null))
|
||||||
|
const timeValue = computed(() => parts.value.time || pendingTime.value)
|
||||||
|
|
||||||
|
const internalError = ref('')
|
||||||
|
const mergedError = computed(() => props.error || internalError.value)
|
||||||
|
|
||||||
|
// La validité ne reflète que la saisie clavier : malformée/hors plage → false. Un
|
||||||
|
// champ vide est valide (l'obligation `required` reste à la charge du parent).
|
||||||
|
const setError = (message: string) => {
|
||||||
|
internalError.value = message
|
||||||
|
emit('update:valid', message === '')
|
||||||
|
}
|
||||||
|
|
||||||
|
function onSelectDay(iso: string) {
|
||||||
|
// Si aucune heure n'a été choisie, on prend l'heure actuelle (pas 00:00).
|
||||||
|
// (heure courante au moment du clic)
|
||||||
|
const now = new Date()
|
||||||
|
const time = parts.value.time || pendingTime.value || formatTime(now.getHours(), now.getMinutes())
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', composeDateTime(iso, time))
|
||||||
|
}
|
||||||
|
|
||||||
|
function onTimeChange(value: string | null) {
|
||||||
|
if (!value) return
|
||||||
|
if (datePart.value) {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', composeDateTime(datePart.value, value))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
pendingTime.value = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCommit(text: string) {
|
||||||
|
const trimmed = text.trim()
|
||||||
|
if (trimmed === '') {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const iso = parseDisplayToIsoDateTime(trimmed)
|
||||||
|
if (iso && isDateInRange(iso, props.min, props.max)) {
|
||||||
|
setError('')
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', iso)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setError(props.invalidMessage)
|
||||||
|
emit('update:rawValue', trimmed)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onClear() {
|
||||||
|
setError('')
|
||||||
|
pendingTime.value = ''
|
||||||
|
emit('update:rawValue', '')
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
}
|
||||||
|
|
||||||
|
// immediate : émet aussi la validité au montage, pour que le parent connaisse
|
||||||
|
// l'état d'un champ pré-rempli (formulaire d'édition) sans interaction préalable.
|
||||||
|
watch(() => props.modelValue, (val) => {
|
||||||
|
setError('')
|
||||||
|
if (val && !isValidIsoDateTime(val) && import.meta.dev) {
|
||||||
|
console.warn(`[MalioDateTime] modelValue invalide ignoré : "${val}"`)
|
||||||
|
}
|
||||||
|
}, {immediate: true})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,122 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import DateWeek from './DateWeek.vue'
|
||||||
|
|
||||||
|
type DateWeekProps = {
|
||||||
|
modelValue?: string | null
|
||||||
|
label?: string
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
error?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const DateWeekForTest = DateWeek as DefineComponent<DateWeekProps>
|
||||||
|
const mountWeek = (props: DateWeekProps = {}) =>
|
||||||
|
mount(DateWeekForTest, {props, attachTo: document.body})
|
||||||
|
|
||||||
|
describe('MalioDateWeek', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
it('renders the label and calendar icon', () => {
|
||||||
|
const wrapper = mountWeek({label: 'Semaine'})
|
||||||
|
expect(wrapper.get('label').text()).toBe('Semaine')
|
||||||
|
expect(wrapper.find('[data-test="calendar-icon"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('displays the formatted week when modelValue is set', () => {
|
||||||
|
const wrapper = mountWeek({modelValue: '2026-W21'})
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('Semaine 21 (18/05 → 24/05/2026)')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows an empty field without a value', () => {
|
||||||
|
const wrapper = mountWeek()
|
||||||
|
const input = wrapper.get('[data-test="date-input"]').element as HTMLInputElement
|
||||||
|
expect(input.value).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens on the month of the selected week', async () => {
|
||||||
|
const wrapper = mountWeek({modelValue: '2026-W01'}) // lundi 2025-12-29
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="header-toggle"]').text()).toContain('Décembre 2025')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('selects the week when a day is clicked', async () => {
|
||||||
|
const wrapper = mountWeek()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-20"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-W21'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('selects the week when the week number is clicked', async () => {
|
||||||
|
const wrapper = mountWeek()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="week-number"][data-week-start="2026-05-18"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['2026-W21'])
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('previews the whole week on day hover', async () => {
|
||||||
|
const wrapper = mountWeek()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="day"][data-iso="2026-05-20"]').trigger('mouseenter')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-18"]').attributes('data-range-role')).toBe('start')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-24"]').attributes('data-range-role')).toBe('end')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-20"]').attributes('data-range-role')).toBe('in-range')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('previews the whole week on week-number hover', async () => {
|
||||||
|
const wrapper = mountWeek()
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
await wrapper.get('[data-test="week-number"][data-week-start="2026-05-18"]').trigger('mouseenter')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-22"]').attributes('data-range-role')).toBe('in-range')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('marks the committed week number', async () => {
|
||||||
|
const wrapper = mountWeek({modelValue: '2026-W21'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.get('[data-test="week-number"][data-week-start="2026-05-18"]').attributes('data-marked')).toBe('true')
|
||||||
|
expect(wrapper.get('[data-test="day"][data-iso="2026-05-18"]').attributes('data-range-role')).toBe('start')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits null on clear', async () => {
|
||||||
|
const wrapper = mountWeek({modelValue: '2026-W21'})
|
||||||
|
await wrapper.get('[data-test="clear"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables a week fully outside min/max', async () => {
|
||||||
|
const wrapper = mountWeek({min: '2026-05-18', max: '2026-05-31'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
const earlyWeek = wrapper.get('[data-test="week-number"][data-week-start="2026-05-11"]')
|
||||||
|
expect((earlyWeek.element as HTMLButtonElement).disabled).toBe(true)
|
||||||
|
const selectableWeek = wrapper.get('[data-test="week-number"][data-week-start="2026-05-18"]')
|
||||||
|
expect((selectableWeek.element as HTMLButtonElement).disabled).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open when disabled', async () => {
|
||||||
|
const wrapper = mountWeek({disabled: true})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open when readonly', async () => {
|
||||||
|
const wrapper = mountWeek({readonly: true, modelValue: '2026-W21'})
|
||||||
|
await wrapper.get('[data-test="date-input"]').trigger('click')
|
||||||
|
expect(wrapper.find('[data-test="popover"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-invalid on error', () => {
|
||||||
|
const wrapper = mountWeek({error: 'Semaine requise'})
|
||||||
|
expect(wrapper.get('[data-test="date-input"]').attributes('aria-invalid')).toBe('true')
|
||||||
|
expect(wrapper.text()).toContain('Semaine requise')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,123 @@
|
|||||||
|
<template>
|
||||||
|
<CalendarField
|
||||||
|
:id="id"
|
||||||
|
:display-value="displayValue"
|
||||||
|
:sync-to="validWeek?.monday ?? null"
|
||||||
|
:name="name"
|
||||||
|
:label="label"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:readonly="readonly"
|
||||||
|
:hint="hint"
|
||||||
|
:error="error"
|
||||||
|
:success="success"
|
||||||
|
:clearable="clearable"
|
||||||
|
:input-class="inputClass"
|
||||||
|
:label-class="labelClass"
|
||||||
|
:group-class="groupClass"
|
||||||
|
v-bind="$attrs"
|
||||||
|
@clear="onClear"
|
||||||
|
@close="onClose"
|
||||||
|
>
|
||||||
|
<template #default="{ currentMonth, currentYear, close }">
|
||||||
|
<MonthGrid
|
||||||
|
:month="currentMonth"
|
||||||
|
:year="currentYear"
|
||||||
|
:range-start="activeMonday"
|
||||||
|
:range-end="activeSunday"
|
||||||
|
:marked-week-start="validWeek?.monday ?? null"
|
||||||
|
interactive-week-number
|
||||||
|
:min="min"
|
||||||
|
:max="max"
|
||||||
|
@select="(iso) => onSelect(iso, close)"
|
||||||
|
@hover="onHover"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</CalendarField>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref} from 'vue'
|
||||||
|
import CalendarField from './internal/CalendarField.vue'
|
||||||
|
import MonthGrid from './internal/MonthGrid.vue'
|
||||||
|
import {formatWeekDisplay, isValidIsoWeek, isoWeekToMonday, mondayOf, sundayOf, toIsoWeek} from './composables/dateWeek'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDateWeek', inheritAttrs: false})
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
clearable?: boolean
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
label: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
placeholder: 'JJ/MM/AAAA',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
min: undefined,
|
||||||
|
max: undefined,
|
||||||
|
clearable: true,
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{(e: 'update:modelValue', value: string | null): void}>()
|
||||||
|
|
||||||
|
const hoverWeekStart = ref<string | null>(null)
|
||||||
|
|
||||||
|
const validWeek = computed(() => {
|
||||||
|
if (props.modelValue && isValidIsoWeek(props.modelValue)) {
|
||||||
|
return {monday: isoWeekToMonday(props.modelValue) as string}
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
})
|
||||||
|
|
||||||
|
const activeMonday = computed(() => hoverWeekStart.value ?? validWeek.value?.monday ?? null)
|
||||||
|
const activeSunday = computed(() => (activeMonday.value ? sundayOf(activeMonday.value) : null))
|
||||||
|
|
||||||
|
const displayValue = computed(() => (validWeek.value ? formatWeekDisplay(props.modelValue as string) : ''))
|
||||||
|
|
||||||
|
const onSelect = (iso: string, close: () => void) => {
|
||||||
|
emit('update:modelValue', toIsoWeek(iso))
|
||||||
|
hoverWeekStart.value = null
|
||||||
|
close()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onHover = (iso: string | null) => {
|
||||||
|
hoverWeekStart.value = iso ? mondayOf(iso) : null
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClose = () => {
|
||||||
|
hoverWeekStart.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClear = () => {
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
hoverWeekStart.value = null
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,62 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {formatIsoToDisplay, isDateInRange, isValidIso, parseDisplayToIso} from './dateFormat'
|
||||||
|
|
||||||
|
describe('dateFormat', () => {
|
||||||
|
describe('isValidIso', () => {
|
||||||
|
it('accepts a real ISO date', () => {
|
||||||
|
expect(isValidIso('2026-05-19')).toBe(true)
|
||||||
|
})
|
||||||
|
it('rejects a malformed string', () => {
|
||||||
|
expect(isValidIso('19/05/2026')).toBe(false)
|
||||||
|
expect(isValidIso('2026-5-9')).toBe(false)
|
||||||
|
expect(isValidIso('')).toBe(false)
|
||||||
|
})
|
||||||
|
it('rejects an impossible date', () => {
|
||||||
|
expect(isValidIso('2026-02-30')).toBe(false)
|
||||||
|
expect(isValidIso('2026-13-01')).toBe(false)
|
||||||
|
})
|
||||||
|
it('accepts Feb 29 on a leap year and rejects it otherwise', () => {
|
||||||
|
expect(isValidIso('2024-02-29')).toBe(true)
|
||||||
|
expect(isValidIso('2026-02-29')).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('formatIsoToDisplay', () => {
|
||||||
|
it('formats ISO to DD/MM/YYYY', () => {
|
||||||
|
expect(formatIsoToDisplay('2026-05-19')).toBe('19/05/2026')
|
||||||
|
})
|
||||||
|
it('returns empty string for null or invalid input', () => {
|
||||||
|
expect(formatIsoToDisplay(null)).toBe('')
|
||||||
|
expect(formatIsoToDisplay('nope')).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('parseDisplayToIso', () => {
|
||||||
|
it('parses DD/MM/YYYY to ISO', () => {
|
||||||
|
expect(parseDisplayToIso('19/05/2026')).toBe('2026-05-19')
|
||||||
|
})
|
||||||
|
it('returns null for malformed or impossible input', () => {
|
||||||
|
expect(parseDisplayToIso('2026-05-19')).toBeNull()
|
||||||
|
expect(parseDisplayToIso('31/02/2026')).toBeNull()
|
||||||
|
expect(parseDisplayToIso('')).toBeNull()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('isDateInRange', () => {
|
||||||
|
it('returns true when no bounds are given', () => {
|
||||||
|
expect(isDateInRange('2026-05-19')).toBe(true)
|
||||||
|
})
|
||||||
|
it('respects the min bound (inclusive)', () => {
|
||||||
|
expect(isDateInRange('2026-05-19', '2026-05-19')).toBe(true)
|
||||||
|
expect(isDateInRange('2026-05-18', '2026-05-19')).toBe(false)
|
||||||
|
})
|
||||||
|
it('respects the max bound (inclusive)', () => {
|
||||||
|
expect(isDateInRange('2026-05-19', undefined, '2026-05-19')).toBe(true)
|
||||||
|
expect(isDateInRange('2026-05-20', undefined, '2026-05-19')).toBe(false)
|
||||||
|
})
|
||||||
|
it('respects both bounds', () => {
|
||||||
|
expect(isDateInRange('2026-05-15', '2026-05-10', '2026-05-20')).toBe(true)
|
||||||
|
expect(isDateInRange('2026-05-25', '2026-05-10', '2026-05-20')).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
export function isValidIso(iso: string): boolean {
|
||||||
|
if (!/^\d{4}-\d{2}-\d{2}$/.test(iso)) return false
|
||||||
|
const [y, m, d] = iso.split('-').map(Number)
|
||||||
|
const date = new Date(y, m - 1, d)
|
||||||
|
return date.getFullYear() === y && date.getMonth() === m - 1 && date.getDate() === d
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatIsoToDisplay(iso: string | null): string {
|
||||||
|
if (!iso || !isValidIso(iso)) return ''
|
||||||
|
const [y, m, d] = iso.split('-')
|
||||||
|
return `${d}/${m}/${y}`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseDisplayToIso(display: string): string | null {
|
||||||
|
const match = /^(\d{2})\/(\d{2})\/(\d{4})$/.exec(display.trim())
|
||||||
|
if (!match) return null
|
||||||
|
const [, dd, mm, yyyy] = match
|
||||||
|
const iso = `${yyyy}-${mm}-${dd}`
|
||||||
|
return isValidIso(iso) ? iso : null
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isDateInRange(iso: string, min?: string, max?: string): boolean {
|
||||||
|
if (min && iso < min) return false
|
||||||
|
if (max && iso > max) return false
|
||||||
|
return true
|
||||||
|
}
|
||||||
@@ -0,0 +1,57 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {dayRangeRole, normalizeRange, resolveRangeBounds} from './dateRange'
|
||||||
|
|
||||||
|
describe('dateRange', () => {
|
||||||
|
describe('normalizeRange', () => {
|
||||||
|
it('keeps an already ordered pair', () => {
|
||||||
|
expect(normalizeRange('2026-05-19', '2026-05-25')).toEqual({start: '2026-05-19', end: '2026-05-25'})
|
||||||
|
})
|
||||||
|
it('swaps a reversed pair', () => {
|
||||||
|
expect(normalizeRange('2026-05-25', '2026-05-19')).toEqual({start: '2026-05-19', end: '2026-05-25'})
|
||||||
|
})
|
||||||
|
it('handles an equal pair', () => {
|
||||||
|
expect(normalizeRange('2026-05-19', '2026-05-19')).toEqual({start: '2026-05-19', end: '2026-05-19'})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('resolveRangeBounds', () => {
|
||||||
|
it('returns null without a start', () => {
|
||||||
|
expect(resolveRangeBounds(null, null, null)).toBeNull()
|
||||||
|
})
|
||||||
|
it('returns a single-point range when only start is set', () => {
|
||||||
|
expect(resolveRangeBounds('2026-05-19', null, null)).toEqual({lo: '2026-05-19', hi: '2026-05-19'})
|
||||||
|
})
|
||||||
|
it('orders start and committed end', () => {
|
||||||
|
expect(resolveRangeBounds('2026-05-19', '2026-05-25', null)).toEqual({lo: '2026-05-19', hi: '2026-05-25'})
|
||||||
|
})
|
||||||
|
it('uses preview when end is not set', () => {
|
||||||
|
expect(resolveRangeBounds('2026-05-19', null, '2026-05-22')).toEqual({lo: '2026-05-19', hi: '2026-05-22'})
|
||||||
|
})
|
||||||
|
it('inverts when preview is before start', () => {
|
||||||
|
expect(resolveRangeBounds('2026-05-19', null, '2026-05-10')).toEqual({lo: '2026-05-10', hi: '2026-05-19'})
|
||||||
|
})
|
||||||
|
it('prioritises committed end over preview', () => {
|
||||||
|
expect(resolveRangeBounds('2026-05-19', '2026-05-25', '2026-05-30')).toEqual({lo: '2026-05-19', hi: '2026-05-25'})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('dayRangeRole', () => {
|
||||||
|
const bounds = {lo: '2026-05-19', hi: '2026-05-25'}
|
||||||
|
it('returns none without bounds', () => {
|
||||||
|
expect(dayRangeRole('2026-05-20', null)).toBe('none')
|
||||||
|
})
|
||||||
|
it('returns single when lo === hi and matches', () => {
|
||||||
|
expect(dayRangeRole('2026-05-19', {lo: '2026-05-19', hi: '2026-05-19'})).toBe('single')
|
||||||
|
expect(dayRangeRole('2026-05-20', {lo: '2026-05-19', hi: '2026-05-19'})).toBe('none')
|
||||||
|
})
|
||||||
|
it('returns start, end and in-range', () => {
|
||||||
|
expect(dayRangeRole('2026-05-19', bounds)).toBe('start')
|
||||||
|
expect(dayRangeRole('2026-05-25', bounds)).toBe('end')
|
||||||
|
expect(dayRangeRole('2026-05-22', bounds)).toBe('in-range')
|
||||||
|
})
|
||||||
|
it('returns none outside the bounds', () => {
|
||||||
|
expect(dayRangeRole('2026-05-10', bounds)).toBe('none')
|
||||||
|
expect(dayRangeRole('2026-05-30', bounds)).toBe('none')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
export type DateRangeValue = {start: string; end: string}
|
||||||
|
|
||||||
|
export function normalizeRange(a: string, b: string): DateRangeValue {
|
||||||
|
return a <= b ? {start: a, end: b} : {start: b, end: a}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function resolveRangeBounds(
|
||||||
|
start: string | null,
|
||||||
|
end: string | null,
|
||||||
|
preview: string | null,
|
||||||
|
): {lo: string; hi: string} | null {
|
||||||
|
if (!start) return null
|
||||||
|
const other = end ?? preview
|
||||||
|
if (!other) return {lo: start, hi: start}
|
||||||
|
return start <= other ? {lo: start, hi: other} : {lo: other, hi: start}
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DayRangeRole = 'none' | 'single' | 'start' | 'end' | 'in-range'
|
||||||
|
|
||||||
|
export function dayRangeRole(
|
||||||
|
iso: string,
|
||||||
|
bounds: {lo: string; hi: string} | null,
|
||||||
|
): DayRangeRole {
|
||||||
|
if (!bounds) return 'none'
|
||||||
|
const {lo, hi} = bounds
|
||||||
|
if (lo === hi) return iso === lo ? 'single' : 'none'
|
||||||
|
if (iso === lo) return 'start'
|
||||||
|
if (iso === hi) return 'end'
|
||||||
|
if (iso > lo && iso < hi) return 'in-range'
|
||||||
|
return 'none'
|
||||||
|
}
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {
|
||||||
|
formatWeekDisplay,
|
||||||
|
isValidIsoWeek,
|
||||||
|
isoWeekToMonday,
|
||||||
|
mondayOf,
|
||||||
|
sundayOf,
|
||||||
|
toIsoWeek,
|
||||||
|
} from './dateWeek'
|
||||||
|
|
||||||
|
describe('dateWeek', () => {
|
||||||
|
describe('mondayOf / sundayOf', () => {
|
||||||
|
it('returns Monday and Sunday of a midweek date', () => {
|
||||||
|
expect(mondayOf('2026-05-20')).toBe('2026-05-18') // mercredi
|
||||||
|
expect(sundayOf('2026-05-20')).toBe('2026-05-24')
|
||||||
|
})
|
||||||
|
it('keeps Monday on a Monday', () => {
|
||||||
|
expect(mondayOf('2026-05-18')).toBe('2026-05-18')
|
||||||
|
})
|
||||||
|
it('returns the preceding Monday for a Sunday', () => {
|
||||||
|
expect(mondayOf('2026-05-24')).toBe('2026-05-18')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('toIsoWeek', () => {
|
||||||
|
it('returns the ISO week of a date', () => {
|
||||||
|
expect(toIsoWeek('2026-05-20')).toBe('2026-W21')
|
||||||
|
})
|
||||||
|
it('handles year boundaries', () => {
|
||||||
|
expect(toIsoWeek('2026-01-01')).toBe('2026-W01')
|
||||||
|
expect(toIsoWeek('2025-12-31')).toBe('2026-W01')
|
||||||
|
expect(toIsoWeek('2027-01-01')).toBe('2026-W53')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('isoWeekToMonday', () => {
|
||||||
|
it('returns the Monday of a week string', () => {
|
||||||
|
expect(isoWeekToMonday('2026-W21')).toBe('2026-05-18')
|
||||||
|
})
|
||||||
|
it('round-trips with toIsoWeek', () => {
|
||||||
|
for (const w of ['2026-W01', '2026-W21', '2026-W53', '2024-W09']) {
|
||||||
|
const monday = isoWeekToMonday(w)
|
||||||
|
expect(monday).not.toBeNull()
|
||||||
|
expect(toIsoWeek(monday as string)).toBe(w)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
it('returns null for invalid input', () => {
|
||||||
|
expect(isoWeekToMonday('2026-21')).toBeNull()
|
||||||
|
expect(isoWeekToMonday('2026-W00')).toBeNull()
|
||||||
|
expect(isoWeekToMonday('2026-W54')).toBeNull()
|
||||||
|
expect(isoWeekToMonday('2025-W53')).toBeNull() // 2025 n'a que 52 semaines ISO
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('isValidIsoWeek', () => {
|
||||||
|
it('accepts a real ISO week', () => {
|
||||||
|
expect(isValidIsoWeek('2026-W21')).toBe(true)
|
||||||
|
})
|
||||||
|
it('rejects malformed or impossible weeks', () => {
|
||||||
|
expect(isValidIsoWeek('2026-21')).toBe(false)
|
||||||
|
expect(isValidIsoWeek('2026-W00')).toBe(false)
|
||||||
|
expect(isValidIsoWeek('2026-W54')).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('formatWeekDisplay', () => {
|
||||||
|
it('formats a week as a human label', () => {
|
||||||
|
expect(formatWeekDisplay('2026-W21')).toBe('Semaine 21 (18/05 → 24/05/2026)')
|
||||||
|
})
|
||||||
|
it('returns empty string for invalid input', () => {
|
||||||
|
expect(formatWeekDisplay('2026-W54')).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import {formatIsoToDisplay} from './dateFormat'
|
||||||
|
|
||||||
|
const parseUtc = (iso: string): Date => {
|
||||||
|
const [y, m, d] = iso.split('-').map(Number)
|
||||||
|
return new Date(Date.UTC(y, m - 1, d))
|
||||||
|
}
|
||||||
|
|
||||||
|
const toIso = (d: Date): string => {
|
||||||
|
const y = d.getUTCFullYear()
|
||||||
|
const m = String(d.getUTCMonth() + 1).padStart(2, '0')
|
||||||
|
const day = String(d.getUTCDate()).padStart(2, '0')
|
||||||
|
return `${y}-${m}-${day}`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mondayOf(iso: string): string {
|
||||||
|
const d = parseUtc(iso)
|
||||||
|
const dayNum = d.getUTCDay() || 7 // dimanche = 7
|
||||||
|
d.setUTCDate(d.getUTCDate() - (dayNum - 1))
|
||||||
|
return toIso(d)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function sundayOf(iso: string): string {
|
||||||
|
const d = parseUtc(mondayOf(iso))
|
||||||
|
d.setUTCDate(d.getUTCDate() + 6)
|
||||||
|
return toIso(d)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function toIsoWeek(iso: string): string {
|
||||||
|
const d = parseUtc(iso)
|
||||||
|
const dayNum = d.getUTCDay() || 7
|
||||||
|
d.setUTCDate(d.getUTCDate() + 4 - dayNum) // jeudi de la semaine
|
||||||
|
const isoYear = d.getUTCFullYear()
|
||||||
|
const yearStart = new Date(Date.UTC(isoYear, 0, 1))
|
||||||
|
const week = Math.ceil((((d.getTime() - yearStart.getTime()) / 86400000) + 1) / 7)
|
||||||
|
return `${isoYear}-W${String(week).padStart(2, '0')}`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isoWeekToMonday(week: string): string | null {
|
||||||
|
const m = /^(\d{4})-W(\d{2})$/.exec(week)
|
||||||
|
if (!m) return null
|
||||||
|
const year = Number(m[1])
|
||||||
|
const w = Number(m[2])
|
||||||
|
if (w < 1 || w > 53) return null
|
||||||
|
// Lundi de la semaine 1 = lundi de la semaine contenant le 4 janvier
|
||||||
|
const jan4 = new Date(Date.UTC(year, 0, 4))
|
||||||
|
const jan4Day = jan4.getUTCDay() || 7
|
||||||
|
const monday = new Date(jan4)
|
||||||
|
monday.setUTCDate(jan4.getUTCDate() - (jan4Day - 1) + (w - 1) * 7)
|
||||||
|
const iso = toIso(monday)
|
||||||
|
// Garde-fou : la semaine 53 n'existe pas pour toutes les années
|
||||||
|
if (toIsoWeek(iso) !== week) return null
|
||||||
|
return iso
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isValidIsoWeek(week: string): boolean {
|
||||||
|
return isoWeekToMonday(week) !== null
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatWeekDisplay(week: string): string {
|
||||||
|
const monday = isoWeekToMonday(week)
|
||||||
|
if (!monday) return ''
|
||||||
|
const sunday = sundayOf(monday)
|
||||||
|
const w = Number(week.slice(6))
|
||||||
|
const startDdMm = formatIsoToDisplay(monday).slice(0, 5) // "18/05"
|
||||||
|
const endFull = formatIsoToDisplay(sunday) // "24/05/2026"
|
||||||
|
return `Semaine ${w} (${startDdMm} → ${endFull})`
|
||||||
|
}
|
||||||
@@ -0,0 +1,90 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {
|
||||||
|
composeDateTime,
|
||||||
|
formatIsoDateTimeToDisplay,
|
||||||
|
isValidIsoDateTime,
|
||||||
|
parseDisplayToIsoDateTime,
|
||||||
|
splitDateTime,
|
||||||
|
} from './datetimeFormat'
|
||||||
|
|
||||||
|
describe('datetimeFormat', () => {
|
||||||
|
describe('isValidIsoDateTime', () => {
|
||||||
|
it('accepte un datetime ISO complet valide', () => {
|
||||||
|
expect(isValidIsoDateTime('2026-05-20T14:30:00')).toBe(true)
|
||||||
|
expect(isValidIsoDateTime('2026-01-01T00:00:00')).toBe(true)
|
||||||
|
expect(isValidIsoDateTime('2026-12-31T23:59:59')).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rejette une date seule, des composants invalides ou une chaîne vide', () => {
|
||||||
|
expect(isValidIsoDateTime('2026-05-20')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('2026-13-01T00:00:00')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('2026-05-20T24:00:00')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('2026-05-20T14:60:00')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('2026-05-20T14:30:60')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('2026-05-20T14:30')).toBe(false)
|
||||||
|
expect(isValidIsoDateTime('')).toBe(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('formatIsoDateTimeToDisplay', () => {
|
||||||
|
it('formate un datetime valide en JJ/MM/AAAA HH:MM', () => {
|
||||||
|
expect(formatIsoDateTimeToDisplay('2026-05-20T14:30:00')).toBe('20/05/2026 14:30')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renvoie une chaîne vide pour nul ou invalide', () => {
|
||||||
|
expect(formatIsoDateTimeToDisplay(null)).toBe('')
|
||||||
|
expect(formatIsoDateTimeToDisplay('2026-05-20')).toBe('')
|
||||||
|
expect(formatIsoDateTimeToDisplay('nope')).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('splitDateTime', () => {
|
||||||
|
it('découpe un datetime valide', () => {
|
||||||
|
expect(splitDateTime('2026-05-20T14:30:00')).toEqual({date: '2026-05-20', time: '14:30'})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renvoie date null et time vide pour nul, date seule ou invalide', () => {
|
||||||
|
expect(splitDateTime(null)).toEqual({date: null, time: ''})
|
||||||
|
expect(splitDateTime('2026-05-20')).toEqual({date: null, time: ''})
|
||||||
|
expect(splitDateTime('nope')).toEqual({date: null, time: ''})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('parseDisplayToIsoDateTime', () => {
|
||||||
|
it('parse un JJ/MM/AAAA HH:MM valide en datetime ISO', () => {
|
||||||
|
expect(parseDisplayToIsoDateTime('20/05/2026 14:30')).toBe('2026-05-20T14:30:00')
|
||||||
|
expect(parseDisplayToIsoDateTime('01/01/2026 00:00')).toBe('2026-01-01T00:00:00')
|
||||||
|
expect(parseDisplayToIsoDateTime('31/12/2026 23:59')).toBe('2026-12-31T23:59:00')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('tolère les espaces autour', () => {
|
||||||
|
expect(parseDisplayToIsoDateTime(' 20/05/2026 14:30 ')).toBe('2026-05-20T14:30:00')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rejette une date malformée', () => {
|
||||||
|
expect(parseDisplayToIsoDateTime('32/01/2026 10:00')).toBeNull()
|
||||||
|
expect(parseDisplayToIsoDateTime('10/13/2026 10:00')).toBeNull()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rejette une heure hors bornes', () => {
|
||||||
|
expect(parseDisplayToIsoDateTime('20/05/2026 24:00')).toBeNull()
|
||||||
|
expect(parseDisplayToIsoDateTime('20/05/2026 12:60')).toBeNull()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rejette un format incomplet ou sans heure', () => {
|
||||||
|
expect(parseDisplayToIsoDateTime('20/05/2026')).toBeNull()
|
||||||
|
expect(parseDisplayToIsoDateTime('20/05/2026 14')).toBeNull()
|
||||||
|
expect(parseDisplayToIsoDateTime('')).toBeNull()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('composeDateTime', () => {
|
||||||
|
it('recompose un datetime ISO avec secondes à 00', () => {
|
||||||
|
expect(composeDateTime('2026-05-20', '14:30')).toBe('2026-05-20T14:30:00')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('utilise 00:00 quand l\'heure est vide', () => {
|
||||||
|
expect(composeDateTime('2026-05-20', '')).toBe('2026-05-20T00:00:00')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import {isValidIso, parseDisplayToIso} from './dateFormat'
|
||||||
|
|
||||||
|
const DATETIME_RE = /^(\d{4}-\d{2}-\d{2})T(\d{2}):(\d{2}):(\d{2})$/
|
||||||
|
|
||||||
|
export function isValidIsoDateTime(s: string): boolean {
|
||||||
|
const m = DATETIME_RE.exec(s)
|
||||||
|
if (!m) return false
|
||||||
|
const [, date, hh, mm, ss] = m
|
||||||
|
if (!isValidIso(date)) return false
|
||||||
|
const h = Number(hh)
|
||||||
|
const min = Number(mm)
|
||||||
|
const sec = Number(ss)
|
||||||
|
return h >= 0 && h <= 23 && min >= 0 && min <= 59 && sec >= 0 && sec <= 59
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatIsoDateTimeToDisplay(s: string | null): string {
|
||||||
|
if (!s || !isValidIsoDateTime(s)) return ''
|
||||||
|
const [date, time] = s.split('T')
|
||||||
|
const [y, mo, d] = date.split('-')
|
||||||
|
const [hh, mm] = time.split(':')
|
||||||
|
return `${d}/${mo}/${y} ${hh}:${mm}`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function splitDateTime(s: string | null): {date: string | null; time: string} {
|
||||||
|
if (!s || !isValidIsoDateTime(s)) return {date: null, time: ''}
|
||||||
|
const [date, time] = s.split('T')
|
||||||
|
return {date, time: time.slice(0, 5)}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseDisplayToIsoDateTime(display: string): string | null {
|
||||||
|
const match = /^(\d{2}\/\d{2}\/\d{4}) (\d{2}):(\d{2})$/.exec(display.trim())
|
||||||
|
if (!match) return null
|
||||||
|
const [, datePart, hh, mm] = match
|
||||||
|
const iso = parseDisplayToIso(datePart)
|
||||||
|
if (!iso) return null
|
||||||
|
if (Number(hh) > 23 || Number(mm) > 59) return null
|
||||||
|
return `${iso}T${hh}:${mm}:00`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function composeDateTime(date: string, time: string): string {
|
||||||
|
const t = time || '00:00'
|
||||||
|
return `${date}T${t}:00`
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import {describe, it, expect} from 'vitest'
|
||||||
|
import {buildBoundedMask} from './maskTemplate'
|
||||||
|
|
||||||
|
describe('buildBoundedMask', () => {
|
||||||
|
it('dérive le masque structurel du gabarit (séparateurs conservés)', () => {
|
||||||
|
expect(buildBoundedMask('JJ/MM/AAAA').mask).toBe('##/##/####')
|
||||||
|
expect(buildBoundedMask('JJ/MM/AAAA HH:MM').mask).toBe('##/##/#### ##:##')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('preProcess — bornage de la saisie (1er ET 2e chiffre)', () => {
|
||||||
|
const pre = (template: string, value: string) => buildBoundedMask(template).preProcess!(value)
|
||||||
|
|
||||||
|
it('jour : refuse > 31 et 00, accepte 01-31', () => {
|
||||||
|
expect(pre('JJ/MM/AAAA', '32')).toBe('3') // 32 impossible → 2e chiffre refusé
|
||||||
|
expect(pre('JJ/MM/AAAA', '33')).toBe('3') // exemple métier : 33 refusé
|
||||||
|
expect(pre('JJ/MM/AAAA', '31')).toBe('31')
|
||||||
|
expect(pre('JJ/MM/AAAA', '00')).toBe('0') // 00 impossible
|
||||||
|
expect(pre('JJ/MM/AAAA', '09')).toBe('09')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('mois : refuse > 12 et 00 (après un jour valide) — cas 33/19', () => {
|
||||||
|
expect(pre('JJ/MM/AAAA', '0119')).toBe('011') // 19 (mois) refusé
|
||||||
|
expect(pre('JJ/MM/AAAA', '0113')).toBe('011')
|
||||||
|
expect(pre('JJ/MM/AAAA', '0112')).toBe('0112')
|
||||||
|
expect(pre('JJ/MM/AAAA', '0100')).toBe('010')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('laisse l’année libre', () => {
|
||||||
|
expect(pre('JJ/MM/AAAA', '01012026')).toBe('01012026')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('heure 00-23 et minute 00-59 (datetime), sans confondre minute et mois', () => {
|
||||||
|
const t = 'JJ/MM/AAAA HH:MM'
|
||||||
|
expect(pre(t, '010120262300')).toBe('010120262300') // 23:00 ok
|
||||||
|
expect(pre(t, '010120262460')).toBe('010120262') // heure 24 refusée
|
||||||
|
expect(pre(t, '010120261259')).toBe('010120261259') // minute 59 ok (≠ mois)
|
||||||
|
expect(pre(t, '010120261260')).toBe('0101202612') // minute 60 refusée
|
||||||
|
})
|
||||||
|
|
||||||
|
it('stoppe à la première saisie invalide (99/99/9999 → rien)', () => {
|
||||||
|
expect(pre('JJ/MM/AAAA', '99/99/9999')).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
import type {MaskInputOptions} from 'maska'
|
||||||
|
|
||||||
|
// Un champ numérique du gabarit : sa longueur et la plage de valeurs autorisée.
|
||||||
|
interface Field {
|
||||||
|
length: number
|
||||||
|
min: number
|
||||||
|
max: number
|
||||||
|
}
|
||||||
|
|
||||||
|
// Découpe un gabarit (ex. `JJ/MM/AAAA HH:MM`) en champs numériques bornés.
|
||||||
|
// Le `M` désigne le mois avant les heures, et les minutes après — d'où le suivi
|
||||||
|
// de `seenHour`, pour ne pas borner les minutes comme un mois (0-59 vs 1-12).
|
||||||
|
function parseFields(template: string): Field[] {
|
||||||
|
const fields: Field[] = []
|
||||||
|
let seenHour = false
|
||||||
|
let i = 0
|
||||||
|
|
||||||
|
while (i < template.length) {
|
||||||
|
const ch = template[i]!
|
||||||
|
if (!/[A-Za-z]/.test(ch)) {
|
||||||
|
i++ // séparateur (/, espace, :)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
let j = i
|
||||||
|
while (j < template.length && template[j] === ch) j++
|
||||||
|
const length = j - i
|
||||||
|
const letter = ch.toUpperCase()
|
||||||
|
if (letter === 'H') seenHour = true
|
||||||
|
|
||||||
|
if (letter === 'J') fields.push({length, min: 1, max: 31})
|
||||||
|
else if (letter === 'M') fields.push(seenHour ? {length, min: 0, max: 59} : {length, min: 1, max: 12})
|
||||||
|
else if (letter === 'H') fields.push({length, min: 0, max: 23})
|
||||||
|
else fields.push({length, min: 0, max: 10 ** length - 1}) // année (ou autre) : libre
|
||||||
|
|
||||||
|
i = j
|
||||||
|
}
|
||||||
|
|
||||||
|
return fields
|
||||||
|
}
|
||||||
|
|
||||||
|
// Un chiffre est accepté tant qu'il existe encore une complétion valide du champ :
|
||||||
|
// on borne la valeur partielle [min possible (padding 0), max possible (padding 9)]
|
||||||
|
// et on vérifie qu'elle croise la plage autorisée [field.min, field.max].
|
||||||
|
function canComplete(partial: string, field: Field): boolean {
|
||||||
|
const low = Number(partial.padEnd(field.length, '0'))
|
||||||
|
const high = Number(partial.padEnd(field.length, '9'))
|
||||||
|
return high >= field.min && low <= field.max
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ne conserve que les chiffres qui gardent chaque champ complétable, et s'arrête
|
||||||
|
// au premier chiffre invalide (rien de ce qui suit n'est réinterprété). maska
|
||||||
|
// réinsère ensuite les séparateurs via le masque structurel.
|
||||||
|
function clampDigits(rawDigits: string, fields: Field[]): string {
|
||||||
|
let result = ''
|
||||||
|
let di = 0
|
||||||
|
|
||||||
|
for (const field of fields) {
|
||||||
|
let fieldDigits = ''
|
||||||
|
while (fieldDigits.length < field.length) {
|
||||||
|
if (di >= rawDigits.length) return result + fieldDigits // plus de saisie
|
||||||
|
const candidate = fieldDigits + rawDigits[di]
|
||||||
|
if (!canComplete(candidate, field)) return result + fieldDigits // 1er chiffre invalide → stop
|
||||||
|
fieldDigits = candidate
|
||||||
|
di++
|
||||||
|
}
|
||||||
|
result += fieldDigits
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construit les options maska d'un champ date/heure à partir d'un gabarit
|
||||||
|
* d'affichage (ex. `JJ/MM/AAAA`, `JJ/MM/AAAA HH:MM`).
|
||||||
|
*
|
||||||
|
* - `mask` : masque structurel (chiffres + séparateurs), pour le formatage/eager.
|
||||||
|
* - `preProcess` : borne la saisie AVANT masquage, sur le 1er **et** le 2e chiffre
|
||||||
|
* de chaque champ (jour 1-31, mois 1-12, heure 0-23, minute 0-59), si bien
|
||||||
|
* qu'une valeur impossible (99/99/9999, 33, 19 en mois…) ne peut pas être tapée.
|
||||||
|
* Les impossibilités calendaires fines (31/02, 29/02 non bissextile) et les
|
||||||
|
* bornes `min`/`max` restent du ressort de la validation, en filet.
|
||||||
|
*/
|
||||||
|
export function buildBoundedMask(template: string): Pick<MaskInputOptions, 'mask' | 'preProcess'> {
|
||||||
|
const mask = template.replace(/[A-Za-z]/g, '#')
|
||||||
|
const fields = parseFields(template)
|
||||||
|
return {
|
||||||
|
mask,
|
||||||
|
preProcess: (value: string) => clampDigits(value.replace(/\D/g, ''), fields),
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {defineComponent, h, ref} from 'vue'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import {useCalendarPopover} from './useCalendarPopover'
|
||||||
|
|
||||||
|
const mountHost = () => {
|
||||||
|
const api: ReturnType<typeof useCalendarPopover> = {} as never
|
||||||
|
const Host = defineComponent({
|
||||||
|
setup() {
|
||||||
|
const root = ref<HTMLElement | null>(null)
|
||||||
|
Object.assign(api, useCalendarPopover(root))
|
||||||
|
return () => h('div', {ref: root}, 'host')
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const wrapper = mount(Host, {attachTo: document.body})
|
||||||
|
return {wrapper, api}
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('useCalendarPopover', () => {
|
||||||
|
it('starts closed in days view', () => {
|
||||||
|
const {api} = mountHost()
|
||||||
|
expect(api.isOpen.value).toBe(false)
|
||||||
|
expect(api.viewMode.value).toBe('days')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('open() opens in days view', () => {
|
||||||
|
const {api} = mountHost()
|
||||||
|
api.open()
|
||||||
|
expect(api.isOpen.value).toBe(true)
|
||||||
|
expect(api.viewMode.value).toBe('days')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('toggleView() switches between days and months', () => {
|
||||||
|
const {api} = mountHost()
|
||||||
|
api.open()
|
||||||
|
api.toggleView()
|
||||||
|
expect(api.viewMode.value).toBe('months')
|
||||||
|
api.toggleView()
|
||||||
|
expect(api.viewMode.value).toBe('days')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('close() resets isOpen and viewMode', () => {
|
||||||
|
const {api} = mountHost()
|
||||||
|
api.open()
|
||||||
|
api.toggleView()
|
||||||
|
api.close()
|
||||||
|
expect(api.isOpen.value).toBe(false)
|
||||||
|
expect(api.viewMode.value).toBe('days')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes on outside mousedown', () => {
|
||||||
|
const {api} = mountHost()
|
||||||
|
api.open()
|
||||||
|
document.body.dispatchEvent(new MouseEvent('mousedown', {bubbles: true}))
|
||||||
|
expect(api.isOpen.value).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('stays open on inside mousedown', () => {
|
||||||
|
const {wrapper, api} = mountHost()
|
||||||
|
api.open()
|
||||||
|
wrapper.element.dispatchEvent(new MouseEvent('mousedown', {bubbles: true}))
|
||||||
|
expect(api.isOpen.value).toBe(true)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import {onBeforeUnmount, onMounted, ref, type Ref} from 'vue'
|
||||||
|
|
||||||
|
export function useCalendarPopover(rootRef: Ref<HTMLElement | null>) {
|
||||||
|
const isOpen = ref(false)
|
||||||
|
const viewMode = ref<'days' | 'months'>('days')
|
||||||
|
|
||||||
|
const open = () => {
|
||||||
|
isOpen.value = true
|
||||||
|
viewMode.value = 'days'
|
||||||
|
}
|
||||||
|
const close = () => {
|
||||||
|
isOpen.value = false
|
||||||
|
viewMode.value = 'days'
|
||||||
|
}
|
||||||
|
const toggleView = () => {
|
||||||
|
viewMode.value = viewMode.value === 'days' ? 'months' : 'days'
|
||||||
|
}
|
||||||
|
|
||||||
|
const onMouseDown = (event: MouseEvent) => {
|
||||||
|
if (!isOpen.value || !rootRef.value) return
|
||||||
|
if (!rootRef.value.contains(event.target as Node)) close()
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => document.addEventListener('mousedown', onMouseDown))
|
||||||
|
onBeforeUnmount(() => document.removeEventListener('mousedown', onMouseDown))
|
||||||
|
|
||||||
|
return {isOpen, viewMode, open, close, toggleView}
|
||||||
|
}
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {ref} from 'vue'
|
||||||
|
import {useCalendarView} from './useCalendarView'
|
||||||
|
|
||||||
|
describe('useCalendarView', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
it('initialises to the current month and year', () => {
|
||||||
|
const {currentMonth, currentYear} = useCalendarView(ref('days'))
|
||||||
|
expect(currentMonth.value).toBe(4)
|
||||||
|
expect(currentYear.value).toBe(2026)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('goToNext advances the month in days view', () => {
|
||||||
|
const {currentMonth, goToNext} = useCalendarView(ref('days'))
|
||||||
|
goToNext()
|
||||||
|
expect(currentMonth.value).toBe(5)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rolls December to January and bumps the year', () => {
|
||||||
|
const {currentMonth, currentYear, goToNext} = useCalendarView(ref('days'))
|
||||||
|
currentMonth.value = 11
|
||||||
|
goToNext()
|
||||||
|
expect(currentMonth.value).toBe(0)
|
||||||
|
expect(currentYear.value).toBe(2027)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rolls January to December backwards', () => {
|
||||||
|
const {currentMonth, currentYear, goToPrev} = useCalendarView(ref('days'))
|
||||||
|
currentMonth.value = 0
|
||||||
|
goToPrev()
|
||||||
|
expect(currentMonth.value).toBe(11)
|
||||||
|
expect(currentYear.value).toBe(2025)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('navigates the year in months view', () => {
|
||||||
|
const {currentYear, goToNext, goToPrev} = useCalendarView(ref('months'))
|
||||||
|
goToNext()
|
||||||
|
expect(currentYear.value).toBe(2027)
|
||||||
|
goToPrev()
|
||||||
|
expect(currentYear.value).toBe(2026)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('selectMonth sets the current month', () => {
|
||||||
|
const {currentMonth, selectMonth} = useCalendarView(ref('days'))
|
||||||
|
selectMonth(0)
|
||||||
|
expect(currentMonth.value).toBe(0)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('syncToIso sets month/year from a valid ISO', () => {
|
||||||
|
const {currentMonth, currentYear, syncToIso} = useCalendarView(ref('days'))
|
||||||
|
syncToIso('2025-12-25')
|
||||||
|
expect(currentMonth.value).toBe(11)
|
||||||
|
expect(currentYear.value).toBe(2025)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('syncToIso falls back to today for null/invalid', () => {
|
||||||
|
const {currentMonth, currentYear, syncToIso} = useCalendarView(ref('days'))
|
||||||
|
syncToIso('2025-12-25')
|
||||||
|
syncToIso(null)
|
||||||
|
expect(currentMonth.value).toBe(4)
|
||||||
|
expect(currentYear.value).toBe(2026)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
import {ref, type Ref} from 'vue'
|
||||||
|
import {isValidIso} from './dateFormat'
|
||||||
|
|
||||||
|
export function useCalendarView(viewMode: Ref<'days' | 'months'>) {
|
||||||
|
const today = new Date()
|
||||||
|
const currentMonth = ref(today.getMonth())
|
||||||
|
const currentYear = ref(today.getFullYear())
|
||||||
|
|
||||||
|
const goToPrev = () => {
|
||||||
|
if (viewMode.value === 'months') {
|
||||||
|
currentYear.value -= 1
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (currentMonth.value === 0) {
|
||||||
|
currentMonth.value = 11
|
||||||
|
currentYear.value -= 1
|
||||||
|
} else {
|
||||||
|
currentMonth.value -= 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const goToNext = () => {
|
||||||
|
if (viewMode.value === 'months') {
|
||||||
|
currentYear.value += 1
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (currentMonth.value === 11) {
|
||||||
|
currentMonth.value = 0
|
||||||
|
currentYear.value += 1
|
||||||
|
} else {
|
||||||
|
currentMonth.value += 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectMonth = (m: number) => {
|
||||||
|
currentMonth.value = m
|
||||||
|
}
|
||||||
|
|
||||||
|
const syncToIso = (iso: string | null) => {
|
||||||
|
if (iso && isValidIso(iso)) {
|
||||||
|
currentMonth.value = Number(iso.slice(5, 7)) - 1
|
||||||
|
currentYear.value = Number(iso.slice(0, 4))
|
||||||
|
} else {
|
||||||
|
const now = new Date()
|
||||||
|
currentMonth.value = now.getMonth()
|
||||||
|
currentYear.value = now.getFullYear()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {currentMonth, currentYear, goToPrev, goToNext, selectMonth, syncToIso}
|
||||||
|
}
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {ref} from 'vue'
|
||||||
|
import {useMonthMatrix} from './useMonthMatrix'
|
||||||
|
|
||||||
|
describe('useMonthMatrix', () => {
|
||||||
|
it('always produces 6 weeks of 7 days', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(4), ref(2026)) // mai 2026
|
||||||
|
expect(weeks.value).toHaveLength(6)
|
||||||
|
weeks.value.forEach(week => expect(week.days).toHaveLength(7))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('starts every week on a Monday', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(4), ref(2026))
|
||||||
|
weeks.value.forEach(week => {
|
||||||
|
const first = new Date(`${week.days[0].isoDate}T00:00:00`)
|
||||||
|
expect(first.getDay()).toBe(1) // 1 = lundi
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('flags exactly the days of the current month', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(4), ref(2026)) // mai = 31 jours
|
||||||
|
const currentMonthDays = weeks.value
|
||||||
|
.flatMap(w => w.days)
|
||||||
|
.filter(d => d.isCurrentMonth)
|
||||||
|
expect(currentMonthDays).toHaveLength(31)
|
||||||
|
expect(currentMonthDays.every(d => d.isoDate.startsWith('2026-05'))).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles leap year February (29 days)', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(1), ref(2024)) // février 2024
|
||||||
|
const days = weeks.value.flatMap(w => w.days).filter(d => d.isCurrentMonth)
|
||||||
|
expect(days).toHaveLength(29)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('assigns ISO week 1 to the week containing Jan 4th', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(0), ref(2026)) // janvier 2026
|
||||||
|
const weekWithJan4 = weeks.value.find(w =>
|
||||||
|
w.days.some(d => d.isoDate === '2026-01-04'),
|
||||||
|
)
|
||||||
|
expect(weekWithJan4?.weekNumber).toBe(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reacts to month/year changes', () => {
|
||||||
|
const month = ref(4)
|
||||||
|
const year = ref(2026)
|
||||||
|
const {weeks} = useMonthMatrix(month, year)
|
||||||
|
const mayCount = weeks.value.flatMap(w => w.days).filter(d => d.isCurrentMonth).length
|
||||||
|
month.value = 1 // février
|
||||||
|
year.value = 2024
|
||||||
|
const febCount = weeks.value.flatMap(w => w.days).filter(d => d.isCurrentMonth).length
|
||||||
|
expect(mayCount).toBe(31)
|
||||||
|
expect(febCount).toBe(29)
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('isToday', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
it('flags only today', () => {
|
||||||
|
const {weeks} = useMonthMatrix(ref(4), ref(2026))
|
||||||
|
const todays = weeks.value.flatMap(w => w.days).filter(d => d.isToday)
|
||||||
|
expect(todays).toHaveLength(1)
|
||||||
|
expect(todays[0].isoDate).toBe('2026-05-19')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
import {computed, type ComputedRef, type Ref} from 'vue'
|
||||||
|
|
||||||
|
export type DayCell = {
|
||||||
|
isoDate: string
|
||||||
|
day: number
|
||||||
|
isCurrentMonth: boolean
|
||||||
|
isToday: boolean
|
||||||
|
}
|
||||||
|
export type WeekRow = {
|
||||||
|
weekNumber: number
|
||||||
|
days: DayCell[]
|
||||||
|
}
|
||||||
|
|
||||||
|
const toIso = (d: Date): string => {
|
||||||
|
const y = d.getFullYear()
|
||||||
|
const m = String(d.getMonth() + 1).padStart(2, '0')
|
||||||
|
const day = String(d.getDate()).padStart(2, '0')
|
||||||
|
return `${y}-${m}-${day}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const isoWeek = (d: Date): number => {
|
||||||
|
const target = new Date(Date.UTC(d.getFullYear(), d.getMonth(), d.getDate()))
|
||||||
|
const dayNum = target.getUTCDay() || 7 // dimanche = 7
|
||||||
|
target.setUTCDate(target.getUTCDate() + 4 - dayNum) // jeudi de la semaine
|
||||||
|
const yearStart = new Date(Date.UTC(target.getUTCFullYear(), 0, 1))
|
||||||
|
return Math.ceil((((target.getTime() - yearStart.getTime()) / 86400000) + 1) / 7)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useMonthMatrix(
|
||||||
|
month: Ref<number>,
|
||||||
|
year: Ref<number>,
|
||||||
|
): {weeks: ComputedRef<WeekRow[]>} {
|
||||||
|
const weeks = computed<WeekRow[]>(() => {
|
||||||
|
const todayIso = toIso(new Date())
|
||||||
|
const first = new Date(year.value, month.value, 1)
|
||||||
|
// recule jusqu'au lundi (getDay : 0 = dimanche)
|
||||||
|
const offset = (first.getDay() + 6) % 7
|
||||||
|
const start = new Date(year.value, month.value, 1 - offset)
|
||||||
|
|
||||||
|
const rows: WeekRow[] = []
|
||||||
|
const cursor = new Date(start)
|
||||||
|
for (let w = 0; w < 6; w++) {
|
||||||
|
const days: DayCell[] = []
|
||||||
|
for (let i = 0; i < 7; i++) {
|
||||||
|
const iso = toIso(cursor)
|
||||||
|
days.push({
|
||||||
|
isoDate: iso,
|
||||||
|
day: cursor.getDate(),
|
||||||
|
isCurrentMonth: cursor.getMonth() === month.value,
|
||||||
|
isToday: iso === todayIso,
|
||||||
|
})
|
||||||
|
cursor.setDate(cursor.getDate() + 1)
|
||||||
|
}
|
||||||
|
rows.push({weekNumber: isoWeek(new Date(`${days[0].isoDate}T00:00:00`)), days})
|
||||||
|
}
|
||||||
|
return rows
|
||||||
|
})
|
||||||
|
|
||||||
|
return {weeks}
|
||||||
|
}
|
||||||
@@ -0,0 +1,388 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
ref="root"
|
||||||
|
:class="mergedGroupClass"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
:id="inputId"
|
||||||
|
v-maska="maskaOptions"
|
||||||
|
:name="name"
|
||||||
|
data-test="date-input"
|
||||||
|
:readonly="inputReadonly"
|
||||||
|
autocomplete="off"
|
||||||
|
:class="mergedInputClass"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:value="editable ? draft : displayValue"
|
||||||
|
:aria-invalid="!!error"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
:aria-expanded="isOpen"
|
||||||
|
aria-haspopup="dialog"
|
||||||
|
v-bind="attrs"
|
||||||
|
placeholder="_"
|
||||||
|
type="text"
|
||||||
|
@click="onFieldClick"
|
||||||
|
@focus="onFocus(); onKbdFocus()"
|
||||||
|
@input="onInput"
|
||||||
|
@blur="onBlur(); onKbdBlur()"
|
||||||
|
@keydown="onKeydown"
|
||||||
|
>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="showGhost"
|
||||||
|
data-test="format-ghost"
|
||||||
|
aria-hidden="true"
|
||||||
|
class="pointer-events-none absolute left-0 right-0 top-1/2 flex h-10 -translate-y-1/2 items-center overflow-hidden whitespace-nowrap rounded-md border border-transparent pl-3 pr-10 text-lg"
|
||||||
|
><span
|
||||||
|
data-test="ghost-typed"
|
||||||
|
class="text-black"
|
||||||
|
>{{ ghostTyped }}</span><span
|
||||||
|
data-test="ghost-remaining"
|
||||||
|
class="text-m-muted"
|
||||||
|
>{{ ghostRemaining }}</span></div>
|
||||||
|
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="inputId"
|
||||||
|
:class="mergedLabelClass"
|
||||||
|
>
|
||||||
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<div class="absolute right-3 top-1/2 flex -translate-y-1/2 items-center gap-1">
|
||||||
|
<button
|
||||||
|
v-if="showClear"
|
||||||
|
type="button"
|
||||||
|
data-test="clear"
|
||||||
|
class="m-focus-ring rounded-malio text-m-muted hover:text-m-primary"
|
||||||
|
aria-label="Effacer la date"
|
||||||
|
@click.stop="onClearClick"
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
icon="mdi:close"
|
||||||
|
:width="16"
|
||||||
|
:height="16"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<Icon
|
||||||
|
data-test="calendar-icon"
|
||||||
|
icon="mdi:calendar-blank"
|
||||||
|
:width="24"
|
||||||
|
:height="24"
|
||||||
|
:class="[iconStateClass, (disabled || readonly) ? 'cursor-not-allowed' : 'cursor-pointer']"
|
||||||
|
@click="onFieldClick"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="isOpen"
|
||||||
|
data-test="popover"
|
||||||
|
role="dialog"
|
||||||
|
class="absolute left-0 right-0 top-full z-20 box-border w-full rounded-b-md bg-white p-[10px] shadow-[0_4px_4px_0_rgba(0,0,0,0.25)]"
|
||||||
|
:class="keyboardFocused ? 'm-combo-ring-bottom' : ''"
|
||||||
|
>
|
||||||
|
<CalendarHeader
|
||||||
|
:view-mode="viewMode"
|
||||||
|
:current-month="currentMonth"
|
||||||
|
:current-year="currentYear"
|
||||||
|
@prev="goToPrev"
|
||||||
|
@next="goToNext"
|
||||||
|
@toggle-view="toggleView"
|
||||||
|
/>
|
||||||
|
<slot
|
||||||
|
v-if="viewMode === 'days'"
|
||||||
|
:current-month="currentMonth"
|
||||||
|
:current-year="currentYear"
|
||||||
|
:close="closePopover"
|
||||||
|
/>
|
||||||
|
<MonthPicker
|
||||||
|
v-else
|
||||||
|
:selected-month="currentMonth"
|
||||||
|
@select="onSelectMonth"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p
|
||||||
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
|
:id="`${inputId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError ? 'text-m-danger' : hasSuccess ? 'text-m-success' : 'text-m-muted',
|
||||||
|
'mt-1 ml-[2px] text-xs',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ error || success || hint }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, ref, useAttrs, useId, watch} from 'vue'
|
||||||
|
import {Icon} from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import {vMaska} from 'maska/vue'
|
||||||
|
import type {MaskInputOptions} from 'maska'
|
||||||
|
import MalioRequiredMark from '../../shared/RequiredMark.vue'
|
||||||
|
import CalendarHeader from './CalendarHeader.vue'
|
||||||
|
import MonthPicker from './MonthPicker.vue'
|
||||||
|
import {useCalendarPopover} from '../composables/useCalendarPopover'
|
||||||
|
import {useCalendarView} from '../composables/useCalendarView'
|
||||||
|
import {buildBoundedMask} from '../composables/maskTemplate'
|
||||||
|
import {useKbdFocusRing} from '../../shared/useKbdFocusRing'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioCalendarField', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
displayValue: string
|
||||||
|
syncTo: string | null
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
label?: string
|
||||||
|
placeholder?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
clearable?: boolean
|
||||||
|
editable?: boolean
|
||||||
|
placeholderTemplate?: string
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
label: '',
|
||||||
|
placeholder: 'JJ/MM/AAAA',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
clearable: true,
|
||||||
|
editable: false,
|
||||||
|
placeholderTemplate: 'JJ/MM/AAAA',
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
reserveMessageSpace: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'clear' | 'close'): void
|
||||||
|
(e: 'commit', value: string): void
|
||||||
|
// Mois affiché (month 0-11) : émis à l'ouverture du popover et à chaque
|
||||||
|
// navigation, pour qu'un consommateur (ex. SIRH) charge les données du mois.
|
||||||
|
(e: 'month-change', value: {month: number, year: number}): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const attrs = useAttrs()
|
||||||
|
const generatedId = useId()
|
||||||
|
const root = ref<HTMLElement | null>(null)
|
||||||
|
|
||||||
|
const draft = ref(props.displayValue)
|
||||||
|
// Le masque maska est dérivé du gabarit : masque structurel pour le formatage,
|
||||||
|
// + preProcess qui borne la saisie (1er ET 2e chiffre : jour 1-31, mois 1-12,
|
||||||
|
// heure 0-23, minute 0-59) afin qu'une valeur impossible (99/99/9999, 33, mois 19…)
|
||||||
|
// ne puisse pas être tapée. eager : pose les séparateurs dès qu'un groupe est complet.
|
||||||
|
const maskaOptions = computed<MaskInputOptions>(() => {
|
||||||
|
if (!props.editable) return {eager: false}
|
||||||
|
const {mask, preProcess} = buildBoundedMask(props.placeholderTemplate)
|
||||||
|
return {mask, preProcess, eager: true}
|
||||||
|
})
|
||||||
|
const inputReadonly = computed(() => !props.editable || props.readonly || props.disabled)
|
||||||
|
|
||||||
|
// Gabarit fantôme : la partie saisie (noire) + le reste du gabarit (gris), affiché
|
||||||
|
// par-dessus l'input (dont le texte est rendu transparent en mode editable).
|
||||||
|
// Espaces → insécables : un espace en bord de span (flex-item) serait sinon rogné,
|
||||||
|
// collant la suite du gabarit à la date (« 12/12/1999HH:MM »).
|
||||||
|
const nbsp = (s: string) => s.replace(/ /g, ' ')
|
||||||
|
const ghostTyped = computed(() => nbsp(draft.value))
|
||||||
|
const ghostRemaining = computed(() => nbsp(props.placeholderTemplate.slice(draft.value.length)))
|
||||||
|
|
||||||
|
watch(() => props.displayValue, (value) => {
|
||||||
|
draft.value = value
|
||||||
|
})
|
||||||
|
|
||||||
|
const {isOpen, viewMode, open, close: closePopover, toggleView} = useCalendarPopover(root)
|
||||||
|
const {currentMonth, currentYear, goToPrev, goToNext, selectMonth, syncToIso} = useCalendarView(viewMode)
|
||||||
|
|
||||||
|
const inputId = computed(() => props.id?.toString() || `malio-date-${generatedId}`)
|
||||||
|
const hasError = computed(() => !!props.error)
|
||||||
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
|
const isFilled = computed(() =>
|
||||||
|
(props.editable ? draft.value.length : props.displayValue.length) > 0,
|
||||||
|
)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const showGhost = computed(() => props.editable && (isOpen.value || isFilled.value))
|
||||||
|
const showClear = computed(() =>
|
||||||
|
props.clearable && isFilled.value && !props.disabled && !props.readonly,
|
||||||
|
)
|
||||||
|
const describedBy = computed(() =>
|
||||||
|
(props.hint || hasError.value || hasSuccess.value) ? `${inputId.value}-describedby` : undefined,
|
||||||
|
)
|
||||||
|
|
||||||
|
watch(isOpen, (value) => {
|
||||||
|
if (!value) emit('close')
|
||||||
|
})
|
||||||
|
|
||||||
|
// Émet le mois affiché tant que le popover est ouvert : une fois à l'ouverture
|
||||||
|
// (isOpen → true, après syncToIso), puis à chaque changement de mois/année.
|
||||||
|
watch([isOpen, currentMonth, currentYear], () => {
|
||||||
|
if (isOpen.value) emit('month-change', {month: currentMonth.value, year: currentYear.value})
|
||||||
|
})
|
||||||
|
|
||||||
|
const onFieldClick = () => {
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
if (props.editable) {
|
||||||
|
if (!isOpen.value) {
|
||||||
|
syncToIso(props.syncTo)
|
||||||
|
open()
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (isOpen.value) {
|
||||||
|
closePopover()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
syncToIso(props.syncTo)
|
||||||
|
open()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onFocus = () => {
|
||||||
|
if (props.disabled || props.readonly || !props.editable) return
|
||||||
|
if (!isOpen.value) {
|
||||||
|
syncToIso(props.syncTo)
|
||||||
|
open()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onInput = (event: Event) => {
|
||||||
|
draft.value = (event.target as HTMLInputElement).value
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset local immédiat : sur saisie invalide, modelValue est déjà null, donc le
|
||||||
|
// watch(displayValue) ne se redéclenche pas — il faut vider le draft soi-même.
|
||||||
|
const onClearClick = () => {
|
||||||
|
draft.value = ''
|
||||||
|
emit('clear')
|
||||||
|
}
|
||||||
|
|
||||||
|
const onBlur = () => {
|
||||||
|
if (!props.editable) return
|
||||||
|
emit('commit', draft.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onEnter = () => {
|
||||||
|
if (!props.editable) return
|
||||||
|
emit('commit', draft.value)
|
||||||
|
closePopover()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onKeydown = (e: KeyboardEvent) => {
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
|
||||||
|
if (e.key === 'Escape') {
|
||||||
|
if (isOpen.value) {
|
||||||
|
e.preventDefault()
|
||||||
|
closePopover()
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.editable) {
|
||||||
|
// En mode éditable, Entrée valide la saisie (Espace = caractère normal)
|
||||||
|
if (e.key === 'Enter') {
|
||||||
|
e.preventDefault()
|
||||||
|
onEnter()
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mode non éditable : Entrée / Espace ouvre ou ferme le calendrier
|
||||||
|
if (e.key === 'Enter' || e.key === ' ') {
|
||||||
|
e.preventDefault()
|
||||||
|
onFieldClick()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(() => props.syncTo, (value) => {
|
||||||
|
if (isOpen.value) syncToIso(value)
|
||||||
|
})
|
||||||
|
|
||||||
|
const onSelectMonth = (m: number) => {
|
||||||
|
selectMonth(m)
|
||||||
|
toggleView()
|
||||||
|
}
|
||||||
|
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
twMerge('relative flex h-12 w-full items-center', props.groupClass),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedInputClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-input peer min-h-[40px] w-full cursor-pointer rounded-md border bg-white py-1 pl-3 pr-10 text-lg outline-none transition-[padding] duration-150 placeholder:text-transparent',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
|
props.disabled ? 'cursor-not-allowed text-black/60 border-m-muted' : '',
|
||||||
|
hasError.value
|
||||||
|
? 'border-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'border-m-success'
|
||||||
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
(!isReadonly.value && isOpen.value) ? 'border-m-primary !py-[9px] !rounded-b-none' : '',
|
||||||
|
keyboardFocused.value ? (isOpen.value ? 'm-combo-ring-top' : 'm-focus-ring-kbd') : '',
|
||||||
|
// En mode editable, le texte réel est masqué : c'est le gabarit fantôme qui l'affiche.
|
||||||
|
props.editable ? 'text-transparent caret-black' : '',
|
||||||
|
props.inputClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedLabelClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-label absolute left-3 top-2 mt-[5px] inline-block origin-left font-medium text-sm transition-transform duration-150',
|
||||||
|
(isReadonly.value ? isFilled.value : (isFilled.value || isOpen.value)) ? '-translate-y-[1.25rem] scale-90' : '',
|
||||||
|
hasError.value
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'text-m-success'
|
||||||
|
: props.disabled
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: isOpen.value
|
||||||
|
? 'text-m-primary'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted text-black',
|
||||||
|
props.labelClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (props.disabled) return 'text-m-muted'
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isOpen.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return 'text-m-muted'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.floating-label {
|
||||||
|
background: white;
|
||||||
|
padding: 0 0.25rem;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
<template>
|
||||||
|
<div class="flex h-[36px] justify-between border-b border-black/60 mb-3">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-test="header-prev"
|
||||||
|
class="ml-2 flex self-start rounded"
|
||||||
|
:aria-label="viewMode === 'days' ? 'Mois précédent' : 'Année précédente'"
|
||||||
|
@click="emit('prev')"
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
icon="mdi:chevron-left"
|
||||||
|
:width="25"
|
||||||
|
:height="25"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-test="header-toggle"
|
||||||
|
class="flex gap-1 rounded text-base font-medium"
|
||||||
|
@click="emit('toggle-view')"
|
||||||
|
>
|
||||||
|
<span class="mt-[2px]">{{ label }}</span>
|
||||||
|
<Icon
|
||||||
|
icon="mdi:chevron-down"
|
||||||
|
:width="25"
|
||||||
|
:height="25"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-test="header-next"
|
||||||
|
class="mr-2 flex self-start rounded"
|
||||||
|
:aria-label="viewMode === 'days' ? 'Mois suivant' : 'Année suivante'"
|
||||||
|
@click="emit('next')"
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
icon="mdi:chevron-right"
|
||||||
|
:width="25"
|
||||||
|
:height="25"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed} from 'vue'
|
||||||
|
import {Icon} from '@iconify/vue'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDateCalendarHeader'})
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
viewMode: 'days' | 'months'
|
||||||
|
currentMonth: number
|
||||||
|
currentYear: number
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'prev' | 'next' | 'toggle-view'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const monthsLong = ['janvier', 'février', 'mars', 'avril', 'mai', 'juin',
|
||||||
|
'juillet', 'août', 'septembre', 'octobre', 'novembre', 'décembre']
|
||||||
|
|
||||||
|
const label = computed(() => {
|
||||||
|
const name = monthsLong[props.currentMonth]
|
||||||
|
return `${name.charAt(0).toUpperCase()}${name.slice(1)} ${props.currentYear}`
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import {afterEach, beforeEach, describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import MonthGrid from './MonthGrid.vue'
|
||||||
|
|
||||||
|
type MonthGridProps = {
|
||||||
|
month: number
|
||||||
|
year: number
|
||||||
|
selectedDate?: string | null
|
||||||
|
markedDates?: Record<string, 'success' | 'danger'>
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const Grid = MonthGrid as DefineComponent<MonthGridProps>
|
||||||
|
const mountGrid = (props: MonthGridProps) => mount(Grid, {props, attachTo: document.body})
|
||||||
|
|
||||||
|
// Récupère la pastille (span rond) qui porte les classes de `cellClass` pour un jour donné.
|
||||||
|
const pill = (wrapper: ReturnType<typeof mountGrid>, iso: string) =>
|
||||||
|
wrapper.get(`[data-iso="${iso}"]`).get('span.rounded-full')
|
||||||
|
|
||||||
|
describe('MalioDateMonthGrid — markedDates', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
vi.setSystemTime(new Date(2026, 4, 19)) // 19 mai 2026
|
||||||
|
})
|
||||||
|
afterEach(() => vi.useRealTimers())
|
||||||
|
|
||||||
|
it('applique un fond success sur un jour marqué', () => {
|
||||||
|
const wrapper = mountGrid({month: 4, year: 2026, markedDates: {'2026-05-20': 'success'}})
|
||||||
|
expect(pill(wrapper, '2026-05-20').classes()).toContain('bg-m-success/15')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applique un fond danger sur un jour marqué', () => {
|
||||||
|
const wrapper = mountGrid({month: 4, year: 2026, markedDates: {'2026-05-21': 'danger'}})
|
||||||
|
expect(pill(wrapper, '2026-05-21').classes()).toContain('bg-m-danger/15')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('ne marque pas les jours absents de markedDates', () => {
|
||||||
|
const wrapper = mountGrid({month: 4, year: 2026, markedDates: {'2026-05-20': 'success'}})
|
||||||
|
const classes = pill(wrapper, '2026-05-22').classes()
|
||||||
|
expect(classes).not.toContain('bg-m-success/15')
|
||||||
|
expect(classes).not.toContain('bg-m-danger/15')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('précédence : la sélection (primary) prime sur la variante marquée', () => {
|
||||||
|
const wrapper = mountGrid({
|
||||||
|
month: 4,
|
||||||
|
year: 2026,
|
||||||
|
selectedDate: '2026-05-22',
|
||||||
|
markedDates: {'2026-05-22': 'success'},
|
||||||
|
})
|
||||||
|
const classes = pill(wrapper, '2026-05-22').classes()
|
||||||
|
expect(classes).toContain('bg-m-primary')
|
||||||
|
expect(classes).toContain('text-white')
|
||||||
|
expect(classes).not.toContain('bg-m-success/15')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('today marqué : garde sa bordure ET reçoit le fond marqué', () => {
|
||||||
|
const wrapper = mountGrid({month: 4, year: 2026, markedDates: {'2026-05-19': 'success'}})
|
||||||
|
const classes = pill(wrapper, '2026-05-19').classes()
|
||||||
|
expect(classes).toContain('border-m-primary')
|
||||||
|
expect(classes).toContain('bg-m-success/15')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('today non marqué : bordure sans fond marqué', () => {
|
||||||
|
const wrapper = mountGrid({month: 4, year: 2026, markedDates: {'2026-05-20': 'success'}})
|
||||||
|
const classes = pill(wrapper, '2026-05-19').classes()
|
||||||
|
expect(classes).toContain('border-m-primary')
|
||||||
|
expect(classes).not.toContain('bg-m-success/15')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,192 @@
|
|||||||
|
<template>
|
||||||
|
<div
|
||||||
|
data-test="month-grid"
|
||||||
|
@mouseleave="emit('hover', null)"
|
||||||
|
>
|
||||||
|
<div class="grid grid-cols-[auto_repeat(7,minmax(0,1fr))]">
|
||||||
|
<div class="mr-[12px] flex h-8 w-[35px] items-center justify-center text-[14px] font-medium opacity-[60%]">
|
||||||
|
S
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-for="d in dayLabels"
|
||||||
|
:key="d"
|
||||||
|
class="flex h-8 items-center justify-center text-[14px] font-medium opacity-[60%]"
|
||||||
|
>
|
||||||
|
{{ d }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<template
|
||||||
|
v-for="(week, wIndex) in weeks"
|
||||||
|
:key="week.days[0].isoDate"
|
||||||
|
>
|
||||||
|
<component
|
||||||
|
:is="interactiveWeekNumber ? 'button' : 'div'"
|
||||||
|
data-test="week-number"
|
||||||
|
:data-week-start="week.days[0].isoDate"
|
||||||
|
:data-marked="markedWeekStart === week.days[0].isoDate"
|
||||||
|
:type="interactiveWeekNumber ? 'button' : undefined"
|
||||||
|
:disabled="interactiveWeekNumber ? !weekSelectable(week) : undefined"
|
||||||
|
class="mr-[12px] flex h-[45px] w-[35px] shrink-0 items-center justify-center p-[10px] text-sm"
|
||||||
|
:class="[
|
||||||
|
weekNumberClass(week),
|
||||||
|
wIndex === 0 ? 'rounded-t-md' : '',
|
||||||
|
wIndex === weeks.length - 1 ? 'rounded-b-md' : '',
|
||||||
|
]"
|
||||||
|
@click="onWeekNumberClick(week)"
|
||||||
|
@mouseenter="onWeekNumberHover(week)"
|
||||||
|
>
|
||||||
|
{{ week.weekNumber }}
|
||||||
|
</component>
|
||||||
|
<button
|
||||||
|
v-for="cell in week.days"
|
||||||
|
:key="cell.isoDate"
|
||||||
|
type="button"
|
||||||
|
data-test="day"
|
||||||
|
:data-iso="cell.isoDate"
|
||||||
|
:data-range-role="roleOf(cell)"
|
||||||
|
:disabled="!inRange(cell.isoDate)"
|
||||||
|
:aria-label="ariaLabel(cell)"
|
||||||
|
:aria-disabled="!inRange(cell.isoDate)"
|
||||||
|
class="relative flex h-[45px] w-full items-center justify-center"
|
||||||
|
:class="inRange(cell.isoDate) ? 'cursor-pointer' : 'cursor-not-allowed'"
|
||||||
|
@click="onSelect(cell.isoDate)"
|
||||||
|
@mouseenter="emit('hover', cell.isoDate)"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="roleOf(cell) === 'in-range'"
|
||||||
|
class="absolute inset-x-0 top-1/2 h-10 -translate-y-1/2 bg-m-primary-light"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
v-else-if="roleOf(cell) === 'start'"
|
||||||
|
class="absolute inset-x-0 top-1/2 h-10 -translate-y-1/2 rounded-l-full bg-m-primary-light"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
v-else-if="roleOf(cell) === 'end'"
|
||||||
|
class="absolute inset-x-0 top-1/2 h-10 -translate-y-1/2 rounded-r-full bg-m-primary-light"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
class="relative flex h-10 w-10 items-center justify-center rounded-full text-sm font-medium transition-colors duration-100"
|
||||||
|
:class="cellClass(cell)"
|
||||||
|
>
|
||||||
|
{{ cell.day }}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, toRef} from 'vue'
|
||||||
|
import {useMonthMatrix, type DayCell, type WeekRow} from '../composables/useMonthMatrix'
|
||||||
|
import {isDateInRange} from '../composables/dateFormat'
|
||||||
|
import {dayRangeRole, resolveRangeBounds, type DayRangeRole} from '../composables/dateRange'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioDateMonthGrid'})
|
||||||
|
|
||||||
|
// Statut générique par jour : aucune sémantique métier dans le layer, juste un
|
||||||
|
// fond tokenisé. `success` et `danger` suffisent pour l'instant (MUI-45).
|
||||||
|
type MarkedVariant = 'success' | 'danger'
|
||||||
|
const markedBg: Record<MarkedVariant, string> = {
|
||||||
|
success: 'bg-m-success/15',
|
||||||
|
danger: 'bg-m-danger/15',
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
month: number
|
||||||
|
year: number
|
||||||
|
selectedDate?: string | null
|
||||||
|
rangeStart?: string | null
|
||||||
|
rangeEnd?: string | null
|
||||||
|
previewDate?: string | null
|
||||||
|
interactiveWeekNumber?: boolean
|
||||||
|
markedWeekStart?: string | null
|
||||||
|
markedDates?: Record<string, MarkedVariant>
|
||||||
|
min?: string
|
||||||
|
max?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
selectedDate: null,
|
||||||
|
rangeStart: undefined,
|
||||||
|
rangeEnd: undefined,
|
||||||
|
previewDate: undefined,
|
||||||
|
interactiveWeekNumber: false,
|
||||||
|
markedWeekStart: null,
|
||||||
|
markedDates: undefined,
|
||||||
|
min: undefined,
|
||||||
|
max: undefined,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'select', iso: string): void
|
||||||
|
(e: 'hover', iso: string | null): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const dayLabels = ['Lun', 'Mar', 'Mer', 'Jeu', 'Ven', 'Sam', 'Dim']
|
||||||
|
const monthsLong = ['janvier', 'février', 'mars', 'avril', 'mai', 'juin',
|
||||||
|
'juillet', 'août', 'septembre', 'octobre', 'novembre', 'décembre']
|
||||||
|
|
||||||
|
const {weeks} = useMonthMatrix(toRef(props, 'month'), toRef(props, 'year'))
|
||||||
|
|
||||||
|
const inRange = (iso: string) => isDateInRange(iso, props.min, props.max)
|
||||||
|
|
||||||
|
const weekSelectable = (week: WeekRow) => week.days.some(d => inRange(d.isoDate))
|
||||||
|
|
||||||
|
const weekNumberClass = (week: WeekRow) => {
|
||||||
|
if (props.markedWeekStart === week.days[0].isoDate) return 'bg-m-primary text-white'
|
||||||
|
const parts = ['bg-m-primary-light']
|
||||||
|
parts.push(week.days.some(d => d.isToday) ? 'text-black' : 'text-black/60')
|
||||||
|
if (props.interactiveWeekNumber && weekSelectable(week)) parts.push('cursor-pointer')
|
||||||
|
return parts.join(' ')
|
||||||
|
}
|
||||||
|
|
||||||
|
const onWeekNumberClick = (week: WeekRow) => {
|
||||||
|
if (!props.interactiveWeekNumber || !weekSelectable(week)) return
|
||||||
|
emit('select', week.days[0].isoDate)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onWeekNumberHover = (week: WeekRow) => {
|
||||||
|
if (!props.interactiveWeekNumber || !weekSelectable(week)) return
|
||||||
|
emit('hover', week.days[0].isoDate)
|
||||||
|
}
|
||||||
|
|
||||||
|
const isRangeMode = computed(() => props.rangeStart !== undefined)
|
||||||
|
const bounds = computed(() =>
|
||||||
|
isRangeMode.value
|
||||||
|
? resolveRangeBounds(props.rangeStart ?? null, props.rangeEnd ?? null, props.previewDate ?? null)
|
||||||
|
: null,
|
||||||
|
)
|
||||||
|
|
||||||
|
const roleOf = (cell: DayCell): DayRangeRole => {
|
||||||
|
if (isRangeMode.value) return dayRangeRole(cell.isoDate, bounds.value)
|
||||||
|
return props.selectedDate === cell.isoDate ? 'single' : 'none'
|
||||||
|
}
|
||||||
|
|
||||||
|
const ariaLabel = (cell: DayCell) => {
|
||||||
|
const [, m, d] = cell.isoDate.split('-')
|
||||||
|
return `${Number(d)} ${monthsLong[Number(m) - 1]} ${cell.isoDate.slice(0, 4)}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const cellClass = (cell: DayCell) => {
|
||||||
|
if (!inRange(cell.isoDate)) return 'text-m-muted/30'
|
||||||
|
const role = roleOf(cell)
|
||||||
|
if (role === 'start' || role === 'end' || role === 'single') return 'bg-m-primary text-white'
|
||||||
|
if (role === 'in-range') return 'text-black'
|
||||||
|
const parts = ['hover:bg-m-primary/10']
|
||||||
|
// Précédence : sélection/range (primary, return ci-dessus) > variante marquée > défaut.
|
||||||
|
// `today` n'est pas exclusif : il garde sa bordure ET peut recevoir le fond marqué.
|
||||||
|
const marked = props.markedDates?.[cell.isoDate]
|
||||||
|
if (marked) parts.push(markedBg[marked])
|
||||||
|
if (cell.isToday) parts.push('border border-m-primary text-m-primary')
|
||||||
|
else if (cell.isCurrentMonth) parts.push('text-black')
|
||||||
|
else parts.push('opacity-[60%]')
|
||||||
|
return parts.join(' ')
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSelect = (iso: string) => {
|
||||||
|
if (!inRange(iso)) return
|
||||||
|
emit('select', iso)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
<template>
|
||||||
|
<div
|
||||||
|
data-test="month-picker"
|
||||||
|
class="grid grid-cols-3 gap-3"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
v-for="(name, index) in months"
|
||||||
|
:key="name"
|
||||||
|
type="button"
|
||||||
|
data-test="month"
|
||||||
|
:data-month="index"
|
||||||
|
class="flex h-[45px] w-full items-center justify-center"
|
||||||
|
@click="emit('select', index)"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
class="flex h-[30px] w-full items-center justify-center rounded text-sm transition-colors duration-100"
|
||||||
|
:class="index === selectedMonth
|
||||||
|
? 'bg-m-primary text-white'
|
||||||
|
: 'text-black hover:bg-m-primary/10'"
|
||||||
|
>
|
||||||
|
{{ name }}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
defineOptions({name: 'MalioDateMonthPicker'})
|
||||||
|
|
||||||
|
defineProps<{selectedMonth?: number}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{(e: 'select', month: number): void}>()
|
||||||
|
|
||||||
|
const months = ['Janvier', 'Février', 'Mars', 'Avril', 'Mai', 'Juin',
|
||||||
|
'Juillet', 'Août', 'Septembre', 'Octobre', 'Novembre', 'Décembre']
|
||||||
|
</script>
|
||||||
@@ -1,15 +1,22 @@
|
|||||||
import { describe, expect, it } from 'vitest'
|
import { afterEach, describe, expect, it } from 'vitest'
|
||||||
import { mount } from '@vue/test-utils'
|
import { enableAutoUnmount, mount } from '@vue/test-utils'
|
||||||
import type { DefineComponent } from 'vue'
|
import type { DefineComponent } from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import Drawer from './Drawer.vue'
|
import Drawer from './Drawer.vue'
|
||||||
|
|
||||||
type DrawerProps = {
|
type DrawerProps = {
|
||||||
modelValue?: boolean
|
|
||||||
title?: string
|
|
||||||
showClose?: boolean
|
|
||||||
id?: string
|
id?: string
|
||||||
|
modelValue?: boolean
|
||||||
|
side?: 'right' | 'left'
|
||||||
|
showClose?: boolean
|
||||||
|
dismissable?: boolean
|
||||||
|
closeOnEscape?: boolean
|
||||||
|
ariaLabel?: string
|
||||||
drawerClass?: string
|
drawerClass?: string
|
||||||
|
overlayClass?: string
|
||||||
|
headerClass?: string
|
||||||
|
bodyClass?: string
|
||||||
|
footerClass?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const DrawerForTest = Drawer as DefineComponent<DrawerProps>
|
const DrawerForTest = Drawer as DefineComponent<DrawerProps>
|
||||||
@@ -18,64 +25,38 @@ function mountComponent(props: DrawerProps = {}, slots?: Record<string, string>)
|
|||||||
return mount(DrawerForTest, {
|
return mount(DrawerForTest, {
|
||||||
props,
|
props,
|
||||||
slots,
|
slots,
|
||||||
global: {
|
global: { stubs: { Teleport: true } },
|
||||||
stubs: {
|
|
||||||
Teleport: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('MalioDrawer', () => {
|
describe('MalioDrawer', () => {
|
||||||
|
enableAutoUnmount(afterEach)
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
})
|
||||||
|
|
||||||
it('does not render when modelValue is false', () => {
|
it('does not render when modelValue is false', () => {
|
||||||
const wrapper = mountComponent({ modelValue: false })
|
const wrapper = mountComponent({ modelValue: false })
|
||||||
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('renders when modelValue is true', () => {
|
it('renders the panel when modelValue is true', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
expect(wrapper.find('[data-test="panel"]').exists()).toBe(true)
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(true)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('renders the title', () => {
|
it('renders default slot in the body', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true, title: 'Mon tiroir' })
|
|
||||||
expect(wrapper.find('h2').text()).toBe('Mon tiroir')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('renders slot content', () => {
|
|
||||||
const wrapper = mountComponent(
|
const wrapper = mountComponent(
|
||||||
{ modelValue: true },
|
{ modelValue: true },
|
||||||
{ default: '<p data-test="content">Contenu du drawer</p>' },
|
{ default: '<p data-test="content">Contenu</p>' },
|
||||||
)
|
)
|
||||||
expect(wrapper.find('[data-test="content"]').text()).toBe('Contenu du drawer')
|
expect(wrapper.find('[data-test="body"] [data-test="content"]').text()).toBe('Contenu')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('emits update:modelValue false on backdrop click', async () => {
|
it('works in uncontrolled mode (defaults closed)', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
const wrapper = mountComponent()
|
||||||
await wrapper.find('[data-test="backdrop"]').trigger('click')
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('emits update:modelValue false on close button click', async () => {
|
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
|
||||||
await wrapper.find('[data-test="close-button"]').trigger('click')
|
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('shows close button by default', () => {
|
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
|
||||||
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(true)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('hides close button when showClose is false', () => {
|
|
||||||
const wrapper = mountComponent({ modelValue: true, showClose: false })
|
|
||||||
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(false)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('close button renders mdi:close icon', () => {
|
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
|
||||||
const icon = wrapper.findComponent(IconifyIcon)
|
|
||||||
expect(icon.props('icon')).toBe('mdi:close')
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it('uses custom id when provided', () => {
|
it('uses custom id when provided', () => {
|
||||||
@@ -85,38 +66,275 @@ describe('MalioDrawer', () => {
|
|||||||
|
|
||||||
it('generates an id when not provided', () => {
|
it('generates an id when not provided', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
const id = wrapper.find('.fixed').attributes('id')
|
expect(wrapper.find('.fixed').attributes('id')).toMatch(/^malio-drawer-/)
|
||||||
expect(id).toMatch(/^malio-drawer-/)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it('has role="dialog" and aria-modal on panel', () => {
|
it('has role="dialog" and aria-modal on the panel', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
const panel = wrapper.find('[data-test="panel"]')
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
expect(panel.attributes('role')).toBe('dialog')
|
expect(panel.attributes('role')).toBe('dialog')
|
||||||
expect(panel.attributes('aria-modal')).toBe('true')
|
expect(panel.attributes('aria-modal')).toBe('true')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('aria-labelledby links to title id', () => {
|
|
||||||
const wrapper = mountComponent({ modelValue: true, id: 'test-drawer' })
|
|
||||||
const panel = wrapper.find('[data-test="panel"]')
|
|
||||||
expect(panel.attributes('aria-labelledby')).toBe('test-drawer-title')
|
|
||||||
expect(wrapper.find('h2').attributes('id')).toBe('test-drawer-title')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('applies drawerClass to the panel', () => {
|
it('applies drawerClass to the panel', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true, drawerClass: 'max-w-lg' })
|
const wrapper = mountComponent({ modelValue: true, drawerClass: 'max-w-2xl' })
|
||||||
const panel = wrapper.find('[data-test="panel"]')
|
expect(wrapper.find('[data-test="panel"]').classes()).toContain('max-w-2xl')
|
||||||
expect(panel.classes()).toContain('max-w-lg')
|
|
||||||
})
|
})
|
||||||
|
|
||||||
it('works in uncontrolled mode', () => {
|
it('renders the #header slot inside the header bar', () => {
|
||||||
const wrapper = mountComponent()
|
const wrapper = mountComponent(
|
||||||
// Without modelValue, defaults to closed
|
{ modelValue: true },
|
||||||
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
{ header: '<h2 data-test="title">Titre</h2>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="header"] [data-test="title"]').text()).toBe('Titre')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the header bar when showClose is true even without #header', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="header"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render the header bar when no #header and showClose is false', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, showClose: false })
|
||||||
|
expect(wrapper.find('[data-test="header"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows the close button by default', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the close button when showClose is false', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, showClose: false },
|
||||||
|
{ header: '<h2>Titre</h2>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('close button renders mdi:cancel-bold icon', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
const icon = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(icon.props('icon')).toBe('mdi:cancel-bold')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('close button has aria-label "Fermer"', () => {
|
it('close button has aria-label "Fermer"', () => {
|
||||||
const wrapper = mountComponent({ modelValue: true })
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
expect(wrapper.find('[data-test="close-button"]').attributes('aria-label')).toBe('Fermer')
|
expect(wrapper.find('[data-test="close-button"]').attributes('aria-label')).toBe('Fermer')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue false and close on close button click', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="close-button"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-labelledby to the header id when #header is provided', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, id: 'test-drawer' },
|
||||||
|
{ header: '<h2>Titre</h2>' },
|
||||||
|
)
|
||||||
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
|
expect(panel.attributes('aria-labelledby')).toBe('test-drawer-header')
|
||||||
|
expect(wrapper.find('[data-test="header-content"]').attributes('id')).toBe('test-drawer-header')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-label from ariaLabel when no #header is provided', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, ariaLabel: 'Panneau latéral' })
|
||||||
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
|
expect(panel.attributes('aria-label')).toBe('Panneau latéral')
|
||||||
|
expect(panel.attributes('aria-labelledby')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies headerClass to the header bar', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, headerClass: 'bg-m-primary' })
|
||||||
|
expect(wrapper.find('[data-test="header"]').classes()).toContain('bg-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the #footer slot in a footer pinned below the body', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true },
|
||||||
|
{ footer: '<button data-test="save">Enregistrer</button>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="body"] [data-test="footer"]').exists()).toBe(false)
|
||||||
|
expect(wrapper.find('[data-test="footer"] [data-test="save"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render the footer wrapper when no #footer slot', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="footer"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies bodyClass to the body', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, bodyClass: 'px-10' })
|
||||||
|
expect(wrapper.find('[data-test="body"]').classes()).toContain('px-10')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies footerClass to the footer', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, footerClass: 'justify-end' },
|
||||||
|
{ footer: '<span>pied</span>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="footer"]').classes()).toContain('justify-end')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aligns to the right by default', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('.fixed').classes()).toContain('justify-end')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aligns to the left when side is "left"', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, side: 'left' })
|
||||||
|
expect(wrapper.find('.fixed').classes()).toContain('justify-start')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue false and close on backdrop click (dismissable)', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="backdrop"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not close on backdrop click when dismissable is false', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, dismissable: false })
|
||||||
|
await wrapper.find('[data-test="backdrop"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies overlayClass to the backdrop', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, overlayClass: 'bg-black/70' })
|
||||||
|
expect(wrapper.find('[data-test="backdrop"]').classes()).toContain('bg-black/70')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes on Escape key when closeOnEscape is true', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Escape' })
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not close on Escape when closeOnEscape is false', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, closeOnEscape: false })
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Escape' })
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('locks body scroll when opened and restores it when closed', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
await wrapper.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves focus into the panel when opened', async () => {
|
||||||
|
const wrapper = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: false, showClose: false },
|
||||||
|
slots: { default: '<button data-test="first">OK</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const first = wrapper.find('[data-test="first"]').element
|
||||||
|
expect(document.activeElement).toBe(first)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('restores focus to the trigger when closed', async () => {
|
||||||
|
const trigger = document.createElement('button')
|
||||||
|
document.body.appendChild(trigger)
|
||||||
|
trigger.focus()
|
||||||
|
expect(document.activeElement).toBe(trigger)
|
||||||
|
|
||||||
|
const wrapper = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
slots: { default: '<button>OK</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
await wrapper.setProps({ modelValue: false })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(document.activeElement).toBe(trigger)
|
||||||
|
|
||||||
|
wrapper.unmount()
|
||||||
|
trigger.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves focus to the close button on open (default showClose)', async () => {
|
||||||
|
const wrapper = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: false, showClose: true },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(document.activeElement).toBe(wrapper.find('[data-test="close-button"]').element)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wraps focus to the first element when Tab is pressed on the last element', async () => {
|
||||||
|
const wrapper = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: true, showClose: false },
|
||||||
|
slots: { default: '<button data-test="btn1">First</button><button data-test="btn2">Last</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const last = wrapper.find('[data-test="btn2"]').element as HTMLElement
|
||||||
|
last.focus()
|
||||||
|
expect(document.activeElement).toBe(last)
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Tab' })
|
||||||
|
expect(document.activeElement).toBe(wrapper.find('[data-test="btn1"]').element)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wraps focus to the last element when Shift+Tab is pressed on the first element', async () => {
|
||||||
|
const wrapper = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: true, showClose: false },
|
||||||
|
slots: { default: '<button data-test="btn1">First</button><button data-test="btn2">Last</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const first = wrapper.find('[data-test="btn1"]').element as HTMLElement
|
||||||
|
first.focus()
|
||||||
|
expect(document.activeElement).toBe(first)
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Tab', shiftKey: true })
|
||||||
|
expect(document.activeElement).toBe(wrapper.find('[data-test="btn2"]').element)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not release body scroll-lock when one stacked drawer closes while another is still open', async () => {
|
||||||
|
const wrapperA = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
const wrapperB = mount(DrawerForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
|
||||||
|
// Open drawer A → scroll locked
|
||||||
|
await wrapperA.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
// Open drawer B → still locked
|
||||||
|
await wrapperB.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
// Close drawer B → A is still open, scroll must remain locked
|
||||||
|
await wrapperB.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
// Close drawer A → both closed, scroll-lock released
|
||||||
|
await wrapperA.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,60 +1,77 @@
|
|||||||
<template>
|
<template>
|
||||||
<Teleport to="body">
|
<Teleport to="body">
|
||||||
<Transition
|
<Transition
|
||||||
name="drawer"
|
:name="`drawer-${side}`"
|
||||||
appear
|
appear
|
||||||
@after-leave="isRendered = false"
|
@after-leave="isRendered = false"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
v-if="isRendered && isOpen"
|
v-if="isRendered && isOpen"
|
||||||
:id="componentId"
|
:id="componentId"
|
||||||
class="fixed inset-0 z-50 flex justify-end"
|
class="fixed inset-0 z-50 flex"
|
||||||
|
:class="side === 'right' ? 'justify-end' : 'justify-start'"
|
||||||
v-bind="attrs"
|
v-bind="attrs"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
class="absolute inset-0 bg-black/40"
|
:class="twMerge('absolute inset-0 bg-black/40', overlayClass)"
|
||||||
data-test="backdrop"
|
data-test="backdrop"
|
||||||
@click="close"
|
@click="onBackdropClick"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
|
ref="panelRef"
|
||||||
:class="twMerge(
|
:class="twMerge(
|
||||||
'relative z-50 flex h-full w-full max-w-md flex-col bg-white shadow-xl',
|
'relative z-50 flex h-full w-full max-w-md flex-col bg-white',
|
||||||
drawerClass,
|
drawerClass,
|
||||||
)"
|
)"
|
||||||
role="dialog"
|
role="dialog"
|
||||||
:aria-modal="true"
|
aria-modal="true"
|
||||||
:aria-labelledby="titleId"
|
:aria-labelledby="hasHeader ? headerId : undefined"
|
||||||
|
:aria-label="hasHeader ? undefined : (ariaLabel || undefined)"
|
||||||
|
tabindex="-1"
|
||||||
data-test="panel"
|
data-test="panel"
|
||||||
|
@keydown="onKeydown"
|
||||||
>
|
>
|
||||||
<div class="flex items-center justify-between px-5 pb-8 pt-8">
|
<div
|
||||||
<h2
|
v-if="hasHeader || showClose"
|
||||||
:id="titleId"
|
:class="twMerge('flex items-center justify-between gap-4 px-5 py-[25px]', headerClass)"
|
||||||
class="text-[32px] font-semibold text-m-primary"
|
data-test="header"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
:id="headerId"
|
||||||
|
class="min-w-0 flex-1"
|
||||||
|
data-test="header-content"
|
||||||
>
|
>
|
||||||
{{ title }}
|
<slot name="header" />
|
||||||
</h2>
|
</div>
|
||||||
<button
|
<button
|
||||||
v-if="showClose"
|
v-if="showClose"
|
||||||
type="button"
|
type="button"
|
||||||
aria-label="Fermer"
|
aria-label="Fermer"
|
||||||
class="flex h-8 w-8 cursor-pointer items-center justify-center rounded-full transition-colors hover:bg-m-surface"
|
class="flex h-8 w-8 shrink-0 cursor-pointer items-center justify-center rounded-full transition-colors hover:bg-m-surface"
|
||||||
data-test="close-button"
|
data-test="close-button"
|
||||||
@click="close"
|
@click="close"
|
||||||
>
|
>
|
||||||
<IconifyIcon
|
<IconifyIcon
|
||||||
icon="mdi:close"
|
icon="mdi:cancel-bold"
|
||||||
:width="24"
|
:width="16"
|
||||||
:height="24"
|
:height="16"
|
||||||
/>
|
/>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
class="flex-1 overflow-y-auto px-5"
|
:class="twMerge('flex-1 overflow-y-auto px-5', bodyClass)"
|
||||||
|
data-test="body"
|
||||||
>
|
>
|
||||||
<slot />
|
<slot />
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="$slots.footer"
|
||||||
|
:class="twMerge('flex shrink-0 items-center gap-3 px-5 py-4', footerClass)"
|
||||||
|
data-test="footer"
|
||||||
|
>
|
||||||
|
<slot name="footer" />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Transition>
|
</Transition>
|
||||||
@@ -62,7 +79,17 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, ref, useAttrs, useId, watch } from 'vue'
|
import {
|
||||||
|
computed,
|
||||||
|
nextTick,
|
||||||
|
onBeforeUnmount,
|
||||||
|
onMounted,
|
||||||
|
ref,
|
||||||
|
useAttrs,
|
||||||
|
useId,
|
||||||
|
useSlots,
|
||||||
|
watch,
|
||||||
|
} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import { twMerge } from 'tailwind-merge'
|
import { twMerge } from 'tailwind-merge'
|
||||||
|
|
||||||
@@ -72,68 +99,195 @@ const props = withDefaults(
|
|||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
modelValue?: boolean
|
modelValue?: boolean
|
||||||
title?: string
|
side?: 'right' | 'left'
|
||||||
showClose?: boolean
|
showClose?: boolean
|
||||||
|
dismissable?: boolean
|
||||||
|
closeOnEscape?: boolean
|
||||||
|
ariaLabel?: string
|
||||||
drawerClass?: string
|
drawerClass?: string
|
||||||
|
overlayClass?: string
|
||||||
|
headerClass?: string
|
||||||
|
bodyClass?: string
|
||||||
|
footerClass?: string
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
modelValue: undefined,
|
modelValue: undefined,
|
||||||
title: '',
|
side: 'right',
|
||||||
showClose: true,
|
showClose: true,
|
||||||
|
dismissable: true,
|
||||||
|
closeOnEscape: true,
|
||||||
|
ariaLabel: '',
|
||||||
drawerClass: '',
|
drawerClass: '',
|
||||||
|
overlayClass: '',
|
||||||
|
headerClass: '',
|
||||||
|
bodyClass: '',
|
||||||
|
footerClass: '',
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(e: 'update:modelValue', value: boolean): void
|
(e: 'update:modelValue', value: boolean): void
|
||||||
|
(e: 'close'): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const attrs = useAttrs()
|
const attrs = useAttrs()
|
||||||
const generatedId = useId()
|
const generatedId = useId()
|
||||||
|
|
||||||
const componentId = computed(() => props.id || `malio-drawer-${generatedId}`)
|
const componentId = computed(() => props.id || `malio-drawer-${generatedId}`)
|
||||||
const titleId = computed(() => `${componentId.value}-title`)
|
|
||||||
|
const slots = useSlots()
|
||||||
|
const headerId = computed(() => `${componentId.value}-header`)
|
||||||
|
const hasHeader = computed(() => !!slots.header)
|
||||||
|
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const localValue = ref(false)
|
const localValue = ref(false)
|
||||||
|
|
||||||
const isOpen = computed(() =>
|
const isOpen = computed(() =>
|
||||||
isControlled.value ? props.modelValue! : localValue.value,
|
isControlled.value ? props.modelValue! : localValue.value,
|
||||||
)
|
)
|
||||||
|
|
||||||
const isRendered = ref(isOpen.value)
|
const isRendered = ref(isOpen.value)
|
||||||
|
|
||||||
|
const panelRef = ref<HTMLElement | null>(null)
|
||||||
|
|
||||||
|
let previouslyFocused: HTMLElement | null = null
|
||||||
|
// Per-instance flag: true while this drawer holds a scroll-lock count slot.
|
||||||
|
let lockedByThisInstance = false
|
||||||
|
|
||||||
|
function getFocusable(container: HTMLElement): HTMLElement[] {
|
||||||
|
return Array.from(
|
||||||
|
container.querySelectorAll<HTMLElement>(
|
||||||
|
'a[href], button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled]), [tabindex]:not([tabindex="-1"]), [contenteditable]:not([contenteditable="false"])',
|
||||||
|
),
|
||||||
|
).filter((el) => el.tabIndex !== -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onOpen() {
|
||||||
|
previouslyFocused = (document.activeElement as HTMLElement | null) ?? null
|
||||||
|
if (!lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = true
|
||||||
|
openDrawerCount++
|
||||||
|
if (openDrawerCount === 1) {
|
||||||
|
document.body.style.overflow = 'hidden'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nextTick(() => {
|
||||||
|
const panel = panelRef.value
|
||||||
|
if (!panel) return
|
||||||
|
const focusable = getFocusable(panel)
|
||||||
|
;(focusable[0] ?? panel).focus()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function onClose() {
|
||||||
|
if (lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = false
|
||||||
|
openDrawerCount = Math.max(0, openDrawerCount - 1)
|
||||||
|
if (openDrawerCount === 0) {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
previouslyFocused?.focus?.()
|
||||||
|
previouslyFocused = null
|
||||||
|
}
|
||||||
|
|
||||||
watch(isOpen, (val) => {
|
watch(isOpen, (val) => {
|
||||||
if (val) isRendered.value = true
|
if (val) {
|
||||||
|
isRendered.value = true
|
||||||
|
onOpen()
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
onClose()
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
function close() {
|
onMounted(() => {
|
||||||
if (!isControlled.value) {
|
if (isOpen.value) onOpen()
|
||||||
localValue.value = false
|
})
|
||||||
|
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
// If this instance is still holding a scroll-lock slot, release it.
|
||||||
|
if (lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = false
|
||||||
|
openDrawerCount = Math.max(0, openDrawerCount - 1)
|
||||||
|
if (openDrawerCount === 0) {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
function onBackdropClick() {
|
||||||
|
if (props.dismissable) close()
|
||||||
|
}
|
||||||
|
|
||||||
|
function onKeydown(e: KeyboardEvent) {
|
||||||
|
if (e.key === 'Escape' && props.closeOnEscape) {
|
||||||
|
e.stopPropagation()
|
||||||
|
close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (e.key !== 'Tab') return
|
||||||
|
|
||||||
|
const panel = panelRef.value
|
||||||
|
if (!panel) return
|
||||||
|
const focusable = getFocusable(panel)
|
||||||
|
if (focusable.length === 0) {
|
||||||
|
e.preventDefault()
|
||||||
|
panel.focus()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const first = focusable[0]!
|
||||||
|
const last = focusable[focusable.length - 1]!
|
||||||
|
if (e.shiftKey && document.activeElement === first) {
|
||||||
|
e.preventDefault()
|
||||||
|
last.focus()
|
||||||
|
}
|
||||||
|
else if (!e.shiftKey && document.activeElement === last) {
|
||||||
|
e.preventDefault()
|
||||||
|
first.focus()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function close() {
|
||||||
|
if (!isControlled.value) localValue.value = false
|
||||||
emit('update:modelValue', false)
|
emit('update:modelValue', false)
|
||||||
|
emit('close')
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
// Shared across all MalioDrawer instances: only the last open drawer releases the body scroll-lock.
|
||||||
|
let openDrawerCount = 0
|
||||||
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.drawer-enter-active,
|
.drawer-right-enter-active,
|
||||||
.drawer-leave-active {
|
.drawer-right-leave-active,
|
||||||
|
.drawer-left-enter-active,
|
||||||
|
.drawer-left-leave-active {
|
||||||
transition: opacity 0.2s ease;
|
transition: opacity 0.2s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.drawer-enter-active > div:last-child,
|
.drawer-right-enter-active > div:last-child,
|
||||||
.drawer-leave-active > div:last-child {
|
.drawer-right-leave-active > div:last-child,
|
||||||
|
.drawer-left-enter-active > div:last-child,
|
||||||
|
.drawer-left-leave-active > div:last-child {
|
||||||
transition: transform 0.3s ease;
|
transition: transform 0.3s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.drawer-enter-from,
|
.drawer-right-enter-from,
|
||||||
.drawer-leave-to {
|
.drawer-right-leave-to,
|
||||||
|
.drawer-left-enter-from,
|
||||||
|
.drawer-left-leave-to {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.drawer-enter-from > div:last-child,
|
.drawer-right-enter-from > div:last-child,
|
||||||
.drawer-leave-to > div:last-child {
|
.drawer-right-leave-to > div:last-child {
|
||||||
transform: translateX(100%);
|
transform: translateX(100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.drawer-left-enter-from > div:last-child,
|
||||||
|
.drawer-left-leave-to > div:last-child {
|
||||||
|
transform: translateX(-100%);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ type InputProps = {
|
|||||||
iconPosition?: 'left' | 'right'
|
iconPosition?: 'left' | 'right'
|
||||||
iconSize?: string | number
|
iconSize?: string | number
|
||||||
iconColor?: string
|
iconColor?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputForTest = Input as DefineComponent<InputProps>
|
const InputForTest = Input as DefineComponent<InputProps>
|
||||||
@@ -53,6 +54,16 @@ describe('MalioInputText', () => {
|
|||||||
expect(wrapper.get('label').text()).toBe('labelTest')
|
expect(wrapper.get('label').text()).toBe('labelTest')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
it('applies the name attribute', () => {
|
it('applies the name attribute', () => {
|
||||||
const wrapper = mountInput({name: 'nameTest'})
|
const wrapper = mountInput({name: 'nameTest'})
|
||||||
|
|
||||||
@@ -126,6 +137,13 @@ describe('MalioInputText', () => {
|
|||||||
expect(wrapper.get('input').classes()).toContain('text-black/60')
|
expect(wrapper.get('input').classes()).toContain('text-black/60')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('shows muted label color when disabled (matches border color)', () => {
|
||||||
|
const wrapper = mountInput({label: 'Email', disabled: true, modelValue: 'foo@bar.com'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('text-black/60')
|
||||||
|
})
|
||||||
|
|
||||||
it('emits update:modelValue on input change', async () => {
|
it('emits update:modelValue on input change', async () => {
|
||||||
const wrapper = mountInput({modelValue: ''})
|
const wrapper = mountInput({modelValue: ''})
|
||||||
|
|
||||||
@@ -253,6 +271,34 @@ describe('MalioInputText', () => {
|
|||||||
expect(wrapper.get('p.text-m-muted').text()).toBe('Hint message test')
|
expect(wrapper.get('p.text-m-muted').text()).toBe('Hint message test')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('reserves space for the message even when no hint/error/success is set', () => {
|
||||||
|
const wrapper = mountInput({})
|
||||||
|
|
||||||
|
const p = wrapper.find('p')
|
||||||
|
expect(p.exists()).toBe(true)
|
||||||
|
expect(p.text()).toBe('')
|
||||||
|
expect(p.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
it('does not render label when label prop is missing', () => {
|
it('does not render label when label prop is missing', () => {
|
||||||
const wrapper = mountInput({labelClass: 'text-red-500'})
|
const wrapper = mountInput({labelClass: 'text-red-500'})
|
||||||
|
|
||||||
@@ -279,7 +325,7 @@ describe('MalioInputText', () => {
|
|||||||
|
|
||||||
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
||||||
expect(wrapper.get('input').classes()).toContain('!pl-11')
|
expect(wrapper.get('input').classes()).toContain('!pl-11')
|
||||||
expect(wrapper.get('label').classes()).toContain('left-8')
|
expect(wrapper.get('label').classes()).toContain('left-11')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('passes icon size props to icon component', () => {
|
it('passes icon size props to icon component', () => {
|
||||||
@@ -294,4 +340,39 @@ describe('MalioInputText', () => {
|
|||||||
|
|
||||||
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountInput({iconName: 'mdi:key-outline'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountInput({iconName: 'mdi:key-outline', modelValue: 'hello'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir et icône noire', () => {
|
||||||
|
const wrapper = mountInput({label: 'Champ', readonly: true, modelValue: 'hello', iconName: 'mdi:key-outline'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ type InputAmountProps = {
|
|||||||
iconPosition?: 'left' | 'right'
|
iconPosition?: 'left' | 'right'
|
||||||
iconSize?: string | number
|
iconSize?: string | number
|
||||||
iconColor?: string
|
iconColor?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputAmountForTest = InputAmount as DefineComponent<InputAmountProps>
|
const InputAmountForTest = InputAmount as DefineComponent<InputAmountProps>
|
||||||
@@ -96,7 +97,7 @@ describe('MalioInputAmount', () => {
|
|||||||
await wrapper.get('input').setValue('12.5')
|
await wrapper.get('input').setValue('12.5')
|
||||||
|
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['12.5'])
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['12.5'])
|
||||||
expect(wrapper.get('input').element.value).toBe('12.5')
|
expect(wrapper.get('input').element.value).toBe('12,5')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('accepts commas but normalizes them to dots', async () => {
|
it('accepts commas but normalizes them to dots', async () => {
|
||||||
@@ -105,7 +106,7 @@ describe('MalioInputAmount', () => {
|
|||||||
await wrapper.get('input').setValue('0012,345abc')
|
await wrapper.get('input').setValue('0012,345abc')
|
||||||
|
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['12.34'])
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['12.34'])
|
||||||
expect(wrapper.get('input').element.value).toBe('12.34')
|
expect(wrapper.get('input').element.value).toBe('12,34')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('normalizes a leading decimal separator', async () => {
|
it('normalizes a leading decimal separator', async () => {
|
||||||
@@ -114,7 +115,7 @@ describe('MalioInputAmount', () => {
|
|||||||
await wrapper.get('input').setValue(',5')
|
await wrapper.get('input').setValue(',5')
|
||||||
|
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['0.5'])
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['0.5'])
|
||||||
expect(wrapper.get('input').element.value).toBe('0.5')
|
expect(wrapper.get('input').element.value).toBe('0,5')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('keeps the normalized decimal value on blur', async () => {
|
it('keeps the normalized decimal value on blur', async () => {
|
||||||
@@ -125,7 +126,7 @@ describe('MalioInputAmount', () => {
|
|||||||
await input.trigger('blur')
|
await input.trigger('blur')
|
||||||
|
|
||||||
expect(wrapper.emitted('update:modelValue')).toEqual([['12.5']])
|
expect(wrapper.emitted('update:modelValue')).toEqual([['12.5']])
|
||||||
expect(input.element.value).toBe('12.5')
|
expect(input.element.value).toBe('12,5')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('keeps integer values unchanged on blur', async () => {
|
it('keeps integer values unchanged on blur', async () => {
|
||||||
@@ -158,6 +159,123 @@ describe('MalioInputAmount', () => {
|
|||||||
|
|
||||||
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
||||||
expect(wrapper.get('input').classes()).toContain('!pl-11')
|
expect(wrapper.get('input').classes()).toContain('!pl-11')
|
||||||
expect(wrapper.get('label').classes()).toContain('left-8')
|
expect(wrapper.get('label').classes()).toContain('left-11')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountInputAmount()
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: '12,50'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : icône en text-m-muted', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir et icône noire', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', readonly: true, modelValue: '12.50'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountInputAmount({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('groupe les milliers à l\'affichage tout en émettant la valeur propre', async () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: ''})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('1234567')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['1234567'])
|
||||||
|
expect(wrapper.get('input').element.value).toBe('1 234 567')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('groupe un grand montant avec décimales', async () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: ''})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('1234567,89')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['1234567.89'])
|
||||||
|
expect(wrapper.get('input').element.value).toBe('1 234 567,89')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('formate la valeur initiale (modelValue) en groupé', () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: '1234567.89'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('1 234 567,89')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('maxLength borne la longueur du modèle : un dépassement est ignoré', async () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: '', maxLength: 4})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('12345')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
expect(wrapper.get('input').element.value).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('maxLength autorise une valeur à la limite', async () => {
|
||||||
|
const wrapper = mountInputAmount({modelValue: '', maxLength: 4})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('1234')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.at(-1)).toEqual(['1234'])
|
||||||
|
expect(wrapper.get('input').element.value).toBe('1 234')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'a plus d\'attribut maxlength natif sur l\'input', () => {
|
||||||
|
const wrapper = mountInputAmount({maxLength: 4})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('maxlength')).toBeUndefined()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -9,10 +9,9 @@
|
|||||||
:autocomplete="autocomplete"
|
:autocomplete="autocomplete"
|
||||||
:class="mergedInputClass"
|
:class="mergedInputClass"
|
||||||
:required="required"
|
:required="required"
|
||||||
:maxlength="maxLength"
|
|
||||||
:minlength="minLength"
|
:minlength="minLength"
|
||||||
:disabled="disabled"
|
:disabled="disabled"
|
||||||
:value="currentValue"
|
:value="formattedValue"
|
||||||
:readonly="readonly"
|
:readonly="readonly"
|
||||||
:aria-invalid="!!error"
|
:aria-invalid="!!error"
|
||||||
:aria-describedby="describedBy"
|
:aria-describedby="describedBy"
|
||||||
@@ -21,7 +20,7 @@
|
|||||||
inputmode="decimal"
|
inputmode="decimal"
|
||||||
placeholder="_"
|
placeholder="_"
|
||||||
@input="onInput"
|
@input="onInput"
|
||||||
@focus="isFocused = true"
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
@blur="onBlur"
|
@blur="onBlur"
|
||||||
>
|
>
|
||||||
|
|
||||||
@@ -30,7 +29,7 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<IconifyIcon
|
<IconifyIcon
|
||||||
@@ -39,18 +38,12 @@
|
|||||||
:width="iconSize"
|
:width="iconSize"
|
||||||
:height="iconSize"
|
:height="iconSize"
|
||||||
data-test="icon"
|
data-test="icon"
|
||||||
:class="[
|
:class="[iconStateClass, iconPositionClass]"
|
||||||
hasError
|
|
||||||
? 'text-m-danger'
|
|
||||||
: hasSuccess
|
|
||||||
? 'text-m-success' : iconColor,
|
|
||||||
iconPositionClass,
|
|
||||||
]"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -58,7 +51,8 @@
|
|||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'mt-1 text-xs ml-[2px] ',
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ hint || error || success }}
|
{{ hint || error || success }}
|
||||||
@@ -70,9 +64,14 @@
|
|||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
import {normalizeAmount, formatGroupedAmount, countSignificant, caretFromSignificant} from './composables/amountFormat'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputAmount', inheritAttrs: false})
|
defineOptions({name: 'MalioInputAmount', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -95,6 +94,7 @@ const props = withDefaults(
|
|||||||
iconPosition?: 'left' | 'right'
|
iconPosition?: 'left' | 'right'
|
||||||
iconSize?: string | number
|
iconSize?: string | number
|
||||||
iconColor?: string
|
iconColor?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -115,8 +115,9 @@ const props = withDefaults(
|
|||||||
hint: '',
|
hint: '',
|
||||||
error: '',
|
error: '',
|
||||||
success: '',
|
success: '',
|
||||||
iconSize: 24,
|
iconSize: 20,
|
||||||
iconColor: 'text-m-muted',
|
iconColor: 'text-m-muted',
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -128,10 +129,16 @@ const isFocused = ref(false)
|
|||||||
const inputId = computed(() => props.id?.toString() || `malio-input-amount-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-input-amount-${generatedId}`)
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
const shouldFloatLabel = computed(() => isFocused.value || currentValue.value.length > 0)
|
const formattedValue = computed(() => formatGroupedAmount(currentValue.value))
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success)
|
const hasSuccess = computed(() => !!props.success)
|
||||||
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
|
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
@@ -141,30 +148,40 @@ const mergedGroupClass = computed(() =>
|
|||||||
)
|
)
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-input grow-height peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent focus:border-2 text-lg rounded-md',
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
isFilled.value ? 'border-black' : 'border-m-muted',
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
: 'focus:border-m-primary',
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
iconInputPaddingClass.value,
|
iconInputPaddingClass.value,
|
||||||
focusPaddingClass.value,
|
isReadonly.value ? '' : focusPaddingClass.value,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
const mergedLabelClass = computed(() =>
|
const mergedLabelClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
labelPositionClass.value,
|
labelPositionClass.value,
|
||||||
shouldFloatLabel.value ? '-translate-y-[1.25rem] peer-focus:-translate-y-[1.55rem] scale-90' : '',
|
shouldFloatLabel.value
|
||||||
disabled.value ? 'peer-[&:not(:placeholder-shown):not(:focus)]:text-black/60' : '',
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -178,40 +195,37 @@ const emit = defineEmits<{
|
|||||||
(event: 'update:modelValue', value: string): void
|
(event: 'update:modelValue', value: string): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const normalizeAmount = (value: string) => {
|
// À la frappe : parse vers le modèle propre (émis), reformate l'affichage groupé, repositionne le curseur.
|
||||||
const sanitizedValue = value
|
|
||||||
.replace(/\s+/g, '')
|
|
||||||
.replace(/,/g, '.')
|
|
||||||
.replace(/[^\d.]/g, '')
|
|
||||||
const [integerPartRaw = '', ...decimalParts] = sanitizedValue.split('.')
|
|
||||||
const integerPart = integerPartRaw.replace(/^0+(?=\d)/, '')
|
|
||||||
const decimalPart = decimalParts.join('').slice(0, 2)
|
|
||||||
|
|
||||||
if (sanitizedValue.includes('.')) {
|
|
||||||
return `${integerPart || '0'}.${decimalPart}`
|
|
||||||
}
|
|
||||||
|
|
||||||
return integerPart
|
|
||||||
}
|
|
||||||
|
|
||||||
// Keep the DOM input value, local state, and v-model emission in sync.
|
|
||||||
const updateValue = (target: HTMLInputElement, value: string) => {
|
|
||||||
target.value = value
|
|
||||||
if (!isControlled.value) {
|
|
||||||
localValue.value = value
|
|
||||||
}
|
|
||||||
emit('update:modelValue', value)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Normalize while typing so the field never keeps invalid amount characters.
|
|
||||||
const onInput = (event: Event) => {
|
const onInput = (event: Event) => {
|
||||||
const target = event.target as HTMLInputElement
|
const target = event.target as HTMLInputElement
|
||||||
updateValue(target, normalizeAmount(target.value))
|
const rawText = target.value
|
||||||
|
const caret = target.selectionStart ?? rawText.length
|
||||||
|
const model = normalizeAmount(rawText)
|
||||||
|
|
||||||
|
// maxLength borne la longueur du MODÈLE (pas l'affichage) : on ignore le keystroke en dépassement.
|
||||||
|
if (props.maxLength != null && model.length > Number(props.maxLength)) {
|
||||||
|
target.value = formattedValue.value
|
||||||
|
const restored = Math.min(Math.max(0, caret - 1), formattedValue.value.length)
|
||||||
|
target.setSelectionRange(restored, restored)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const display = formatGroupedAmount(model)
|
||||||
|
const sig = countSignificant(rawText, caret)
|
||||||
|
target.value = display
|
||||||
|
const newCaret = caretFromSignificant(display, sig)
|
||||||
|
target.setSelectionRange(newCaret, newCaret)
|
||||||
|
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localValue.value = model
|
||||||
|
}
|
||||||
|
emit('update:modelValue', model)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Keep the blur handler only for focus-driven UI state.
|
// Keep the blur handler only for focus-driven UI state.
|
||||||
const onBlur = () => {
|
const onBlur = () => {
|
||||||
isFocused.value = false
|
isFocused.value = false
|
||||||
|
onKbdBlur()
|
||||||
}
|
}
|
||||||
|
|
||||||
const iconInputPaddingClass = computed(() => {
|
const iconInputPaddingClass = computed(() => {
|
||||||
@@ -222,7 +236,7 @@ const iconInputPaddingClass = computed(() => {
|
|||||||
const disabled = computed(() => props.disabled)
|
const disabled = computed(() => props.disabled)
|
||||||
|
|
||||||
const labelPositionClass = computed(() => {
|
const labelPositionClass = computed(() => {
|
||||||
if (props.iconName && props.iconPosition === 'left') return 'left-8'
|
if (props.iconName && props.iconPosition === 'left') return 'left-11'
|
||||||
return 'left-3'
|
return 'left-3'
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -235,6 +249,16 @@ const iconPositionClass = computed(() => {
|
|||||||
const sideClass = props.iconPosition === 'left' ? 'left-[10px]' : 'right-[10px]'
|
const sideClass = props.iconPosition === 'left' ? 'left-[10px]' : 'right-[10px]'
|
||||||
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return props.iconColor
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return props.iconColor
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
@@ -0,0 +1,572 @@
|
|||||||
|
import {describe, expect, it, vi} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import {Icon as IconifyIcon} from '@iconify/vue'
|
||||||
|
import InputAutocomplete from './InputAutocomplete.vue'
|
||||||
|
|
||||||
|
type Option = {
|
||||||
|
label: string
|
||||||
|
value: string | number
|
||||||
|
}
|
||||||
|
|
||||||
|
type InputAutocompleteProps = {
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
modelValue?: string | number | null
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
options?: Option[]
|
||||||
|
loading?: boolean
|
||||||
|
debounce?: number
|
||||||
|
minSearchLength?: number
|
||||||
|
allowCreate?: boolean
|
||||||
|
localFilter?: boolean
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
noResultsText?: string
|
||||||
|
loadingText?: string
|
||||||
|
minSearchText?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const InputAutocompleteForTest = InputAutocomplete as DefineComponent<InputAutocompleteProps>
|
||||||
|
|
||||||
|
const options: Option[] = [
|
||||||
|
{label: 'France', value: 'fr'},
|
||||||
|
{label: 'Belgique', value: 'be'},
|
||||||
|
{label: 'Canada', value: 'ca'},
|
||||||
|
]
|
||||||
|
|
||||||
|
const mountComponent = (props: InputAutocompleteProps = {}) =>
|
||||||
|
mount(InputAutocompleteForTest, {
|
||||||
|
props,
|
||||||
|
global: {
|
||||||
|
stubs: {
|
||||||
|
IconifyIcon: {
|
||||||
|
template: '<span data-test="icon" v-bind="$attrs" />',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioInputAutocomplete', () => {
|
||||||
|
it('renders the label text', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Pays'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').text()).toBe('Pays')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders with type combobox role', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('role')).toBe('combobox')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders input with provided modelValue label when option matches', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'fr', options})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('France')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens dropdown on focus', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.get('input').attributes('aria-expanded')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open dropdown on focus when disabled', async () => {
|
||||||
|
const wrapper = mountComponent({options, disabled: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not open dropdown on focus when readonly', async () => {
|
||||||
|
const wrapper = mountComponent({options, readonly: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders all options in dropdown', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('[data-test="option"]')
|
||||||
|
expect(items).toHaveLength(3)
|
||||||
|
expect(items[0].text()).toBe('France')
|
||||||
|
expect(items[1].text()).toBe('Belgique')
|
||||||
|
expect(items[2].text()).toBe('Canada')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue with option value when option is selected', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.findAll('[data-test="option"]')[1].trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['be'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits select with full option object', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.findAll('[data-test="option"]')[0].trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('select')?.[0]).toEqual([options[0]])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes dropdown after selecting an option', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.findAll('[data-test="option"]')[0].trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('fills input with selected option label after selection', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: null})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.findAll('[data-test="option"]')[1].trigger('click')
|
||||||
|
|
||||||
|
await wrapper.setProps({modelValue: 'be'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('Belgique')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits search after debounce when user types', async () => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
const wrapper = mountComponent({options, debounce: 300})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('fra')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('search')).toBeUndefined()
|
||||||
|
|
||||||
|
vi.advanceTimersByTime(300)
|
||||||
|
|
||||||
|
expect(wrapper.emitted('search')?.[0]).toEqual(['fra'])
|
||||||
|
vi.useRealTimers()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not emit search until minSearchLength is reached', async () => {
|
||||||
|
vi.useFakeTimers()
|
||||||
|
const wrapper = mountComponent({minSearchLength: 3, debounce: 300})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('fr')
|
||||||
|
vi.advanceTimersByTime(300)
|
||||||
|
|
||||||
|
expect(wrapper.emitted('search')).toBeUndefined()
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('fra')
|
||||||
|
vi.advanceTimersByTime(300)
|
||||||
|
|
||||||
|
expect(wrapper.emitted('search')?.[0]).toEqual(['fra'])
|
||||||
|
vi.useRealTimers()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows minSearch text in dropdown when minSearchLength not reached', async () => {
|
||||||
|
const wrapper = mountComponent({minSearchLength: 3, minSearchText: 'Tapez 3 caractères'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="min-search-text"]').text()).toBe('Tapez 3 caractères')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows loading text in dropdown when loading', async () => {
|
||||||
|
const wrapper = mountComponent({loading: true, loadingText: 'En cours…'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="loading-text"]').text()).toBe('En cours…')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows loading icon when loading', async () => {
|
||||||
|
const wrapper = mountComponent({loading: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="loading-icon"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('[data-test="chevron"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows no results text when options is empty', async () => {
|
||||||
|
const wrapper = mountComponent({options: [], noResultsText: 'Rien trouvé'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="no-results-text"]').text()).toBe('Rien trouvé')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clears selection when typing different value', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: 'fr'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('belg')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([null])
|
||||||
|
expect(wrapper.emitted('select')?.[0]).toEqual([null])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits create event with typed value when allowCreate and Enter pressed', async () => {
|
||||||
|
const wrapper = mountComponent({options, allowCreate: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('Custom')
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'Enter'})
|
||||||
|
|
||||||
|
expect(wrapper.emitted('create')?.[0]).toEqual(['Custom'])
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.some(e => e[0] === 'Custom')).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not emit create when allowCreate is false', async () => {
|
||||||
|
const wrapper = mountComponent({options, allowCreate: false})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('Custom')
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'Enter'})
|
||||||
|
|
||||||
|
expect(wrapper.emitted('create')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('selects option on Enter with active index', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'Enter'})
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['fr'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('navigates options with ArrowDown', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'ArrowDown'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('aria-activedescendant')).toContain('-option-1')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes dropdown on Escape', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(true)
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'Escape'})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="dropdown"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reverts input value on Escape', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: 'fr'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('xyz')
|
||||||
|
await wrapper.get('input').trigger('keydown', {key: 'Escape'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('France')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error message and styles', () => {
|
||||||
|
const wrapper = mountComponent({error: 'Champ invalide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Champ invalide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-danger')
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success message and styles', () => {
|
||||||
|
const wrapper = mountComponent({success: 'Champ valide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-success').text()).toBe('Champ valide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows hint message', () => {
|
||||||
|
const wrapper = mountComponent({hint: 'Tapez pour rechercher'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-muted').text()).toBe('Tapez pour rechercher')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders left icon when iconName provided with left position', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:magnify', iconPosition: 'left'})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon-left"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('[data-test="icon-right"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders right icon when iconName provided with right position', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:magnify', iconPosition: 'right'})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon-right"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('[data-test="icon-left"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render icon when iconName is empty', () => {
|
||||||
|
const wrapper = mountComponent({iconName: ''})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon-left"]').exists()).toBe(false)
|
||||||
|
expect(wrapper.find('[data-test="icon-right"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses left padding when icon is left', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:magnify', iconPosition: 'left'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').classes()).toContain('!pl-11')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses extra right padding when icon is right', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:magnify', iconPosition: 'right'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').classes()).toContain('!pr-16')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the chevron with default icon', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
const icons = wrapper.findAllComponents(IconifyIcon)
|
||||||
|
const chevron = icons[icons.length - 1]
|
||||||
|
expect(chevron.props('icon')).toBe('mdi:chevron-down')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('rotates the chevron when dropdown is open', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('rotate-0')
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('rotate-180')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets disabled attribute', () => {
|
||||||
|
const wrapper = mountComponent({disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('disabled')).toBeDefined()
|
||||||
|
expect(wrapper.get('input').classes()).toContain('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the chevron when disabled', () => {
|
||||||
|
const wrapper = mountComponent({disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="chevron"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets readonly attribute', () => {
|
||||||
|
const wrapper = mountComponent({readonly: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('readonly')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('links label to input via for/id', () => {
|
||||||
|
const wrapper = mountComponent({id: 'country', label: 'Pays'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('id')).toBe('country')
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe('country')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an id when missing and reuses it on label', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Pays'})
|
||||||
|
|
||||||
|
const inputId = wrapper.get('input').attributes('id')
|
||||||
|
|
||||||
|
expect(inputId?.startsWith('malio-input-autocomplete-')).toBe(true)
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe(inputId)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aria-invalid is false when no error', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('marks the option matching modelValue as aria-selected', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: 'be'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('[data-test="option"]')
|
||||||
|
expect(items[0].attributes('aria-selected')).toBe('false')
|
||||||
|
expect(items[1].attributes('aria-selected')).toBe('true')
|
||||||
|
expect(items[2].attributes('aria-selected')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('updates inputValue when modelValue changes externally', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: 'fr'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('France')
|
||||||
|
|
||||||
|
await wrapper.setProps({modelValue: 'ca'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('Canada')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clears inputValue when modelValue is cleared externally', async () => {
|
||||||
|
const wrapper = mountComponent({options, modelValue: 'fr'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('France')
|
||||||
|
|
||||||
|
await wrapper.setProps({modelValue: null})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses allowCreate modelValue as inputValue when no match in options', async () => {
|
||||||
|
const wrapper = mountComponent({options, allowCreate: true, modelValue: 'Custom'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('Custom')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not filter options when localFilter is false (default)', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('fr')
|
||||||
|
|
||||||
|
expect(wrapper.findAll('[data-test="option"]')).toHaveLength(3)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('filters options client-side when localFilter is true', async () => {
|
||||||
|
const wrapper = mountComponent({options, localFilter: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('fr')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('[data-test="option"]')
|
||||||
|
expect(items).toHaveLength(1)
|
||||||
|
expect(items[0].text()).toBe('France')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('localFilter is case-insensitive and matches substrings', async () => {
|
||||||
|
const wrapper = mountComponent({options, localFilter: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('GIQ')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('[data-test="option"]')
|
||||||
|
expect(items).toHaveLength(1)
|
||||||
|
expect(items[0].text()).toBe('Belgique')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('localFilter shows all options when input is empty', async () => {
|
||||||
|
const wrapper = mountComponent({options, localFilter: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.findAll('[data-test="option"]')).toHaveLength(3)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('localFilter shows the no-results state when nothing matches', async () => {
|
||||||
|
const wrapper = mountComponent({options, localFilter: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
await wrapper.get('input').setValue('zzzzz')
|
||||||
|
|
||||||
|
expect(wrapper.findAll('[data-test="option"]')).toHaveLength(0)
|
||||||
|
expect(wrapper.find('[data-test="no-results-text"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps the floating label at the same position whether focused or not (no jump)', async () => {
|
||||||
|
const wrapper = mountComponent({options, label: 'Pays', modelValue: 'fr'})
|
||||||
|
|
||||||
|
// when a value is selected and the field is not focused, the label is already floated
|
||||||
|
const labelClasses = wrapper.get('label').classes()
|
||||||
|
expect(labelClasses).toContain('-translate-y-[1.25rem]')
|
||||||
|
// and there is no extra peer-focus translate that would make it jump on click
|
||||||
|
expect(labelClasses).not.toContain('peer-focus:-translate-y-[1.55rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not shift inner text horizontally on focus (no focus:pl change)', () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
const inputClasses = wrapper.get('input').classes()
|
||||||
|
expect(inputClasses).not.toContain('focus:pl-[11px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps the bottom border allocation when open (transparent, not zero)', async () => {
|
||||||
|
const wrapper = mountComponent({options})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
const inputClasses = wrapper.get('input').classes()
|
||||||
|
// border-b-0 would shrink the bottom border to 0px and grow content area by 1px;
|
||||||
|
// border-b-transparent keeps the 1px allocation but hides the line
|
||||||
|
expect(inputClasses).not.toContain('!border-b-0')
|
||||||
|
expect(inputClasses).toContain('!border-b-transparent')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : chevron en text-m-muted', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir et icône noire', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true, modelValue: 'fr', options, iconName: 'mdi:magnify', iconPosition: 'left'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon-left"]').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', options})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', options, reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', options, reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,569 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
ref="root"
|
||||||
|
:class="mergedGroupClass"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
:id="inputId"
|
||||||
|
:name="name"
|
||||||
|
autocomplete="off"
|
||||||
|
:class="mergedInputClass"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:readonly="readonly"
|
||||||
|
:value="inputValue"
|
||||||
|
:aria-invalid="!!error"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
:aria-expanded="isOpen"
|
||||||
|
:aria-controls="listboxId"
|
||||||
|
:aria-activedescendant="activeOptionId"
|
||||||
|
role="combobox"
|
||||||
|
v-bind="attrs"
|
||||||
|
placeholder="_"
|
||||||
|
type="text"
|
||||||
|
@input="onInput"
|
||||||
|
@focus="onFocus"
|
||||||
|
@blur="onKbdBlur"
|
||||||
|
@click="onInputClick"
|
||||||
|
@keydown="onKeydown"
|
||||||
|
>
|
||||||
|
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="inputId"
|
||||||
|
:class="mergedLabelClass"
|
||||||
|
>
|
||||||
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="iconName && iconPosition === 'left'"
|
||||||
|
:icon="iconName"
|
||||||
|
:width="iconSize"
|
||||||
|
:height="iconSize"
|
||||||
|
data-test="icon-left"
|
||||||
|
:class="[iconStateClass, 'pointer-events-none absolute left-[10px] top-1/2 -translate-y-1/2']"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div class="pointer-events-none absolute right-3 top-1/2 flex -translate-y-1/2 items-center gap-1">
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="iconName && iconPosition === 'right'"
|
||||||
|
:icon="iconName"
|
||||||
|
:width="iconSize"
|
||||||
|
:height="iconSize"
|
||||||
|
data-test="icon-right"
|
||||||
|
:class="[iconStateClass]"
|
||||||
|
/>
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="loading"
|
||||||
|
icon="mdi:loading"
|
||||||
|
:width="20"
|
||||||
|
:height="20"
|
||||||
|
data-test="loading-icon"
|
||||||
|
class="animate-spin text-m-primary"
|
||||||
|
/>
|
||||||
|
<IconifyIcon
|
||||||
|
v-else-if="!disabled"
|
||||||
|
icon="mdi:chevron-down"
|
||||||
|
:width="20"
|
||||||
|
:height="20"
|
||||||
|
data-test="chevron"
|
||||||
|
class="transition-transform duration-300"
|
||||||
|
:class="[
|
||||||
|
isOpen ? 'rotate-180' : 'rotate-0',
|
||||||
|
chevronColorClass,
|
||||||
|
]"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ul
|
||||||
|
v-if="isOpen"
|
||||||
|
:id="listboxId"
|
||||||
|
ref="listRef"
|
||||||
|
data-test="dropdown"
|
||||||
|
role="listbox"
|
||||||
|
:aria-labelledby="inputId"
|
||||||
|
class="absolute left-0 right-0 top-[calc(100%-4px)] z-20 max-h-60 w-full overflow-auto rounded-b-md border border-t-0 bg-white"
|
||||||
|
:class="[
|
||||||
|
hasError
|
||||||
|
? 'border-m-danger select-scrollbar-error'
|
||||||
|
: hasSuccess
|
||||||
|
? 'border-m-success select-scrollbar-success'
|
||||||
|
: 'border-m-primary select-scrollbar-primary',
|
||||||
|
keyboardFocused ? 'm-combo-ring-bottom' : '',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
<li
|
||||||
|
v-if="loading"
|
||||||
|
class="px-3 py-2 text-m-muted"
|
||||||
|
data-test="loading-text"
|
||||||
|
>
|
||||||
|
{{ loadingText }}
|
||||||
|
</li>
|
||||||
|
<li
|
||||||
|
v-else-if="showMinSearch"
|
||||||
|
class="px-3 py-2 text-m-muted"
|
||||||
|
data-test="min-search-text"
|
||||||
|
>
|
||||||
|
{{ minSearchText }}
|
||||||
|
</li>
|
||||||
|
<li
|
||||||
|
v-else-if="filteredOptions.length === 0"
|
||||||
|
class="px-3 py-2 text-m-muted"
|
||||||
|
data-test="no-results-text"
|
||||||
|
>
|
||||||
|
{{ noResultsText }}
|
||||||
|
</li>
|
||||||
|
<template v-else>
|
||||||
|
<li
|
||||||
|
v-for="(opt, index) in filteredOptions"
|
||||||
|
:id="optionId(index)"
|
||||||
|
:key="String(opt.value)"
|
||||||
|
data-test="option"
|
||||||
|
role="option"
|
||||||
|
:aria-selected="opt.value === modelValue"
|
||||||
|
class="cursor-pointer px-3 py-2 text-black"
|
||||||
|
:class="[
|
||||||
|
index === activeIndex ? 'bg-m-muted/10' : '',
|
||||||
|
opt.value === modelValue ? 'bg-m-muted/10 font-semibold' : '',
|
||||||
|
]"
|
||||||
|
@mouseenter="activeIndex = index"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="onSelect(opt)"
|
||||||
|
>
|
||||||
|
{{ opt.label || '\u00A0' }}
|
||||||
|
</li>
|
||||||
|
</template>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<p
|
||||||
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
|
:id="`${inputId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError ? 'text-m-danger' : hasSuccess ? 'text-m-success' : 'text-m-muted',
|
||||||
|
'mt-1 ml-[2px] text-xs',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ hint || error || success }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {computed, nextTick, onBeforeUnmount, onMounted, ref, useAttrs, useId, watch} from 'vue'
|
||||||
|
import {Icon as IconifyIcon} from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioInputAutocomplete', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
|
type Option = {
|
||||||
|
label: string
|
||||||
|
value: string | number
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
modelValue?: string | number | null
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
options?: Option[]
|
||||||
|
loading?: boolean
|
||||||
|
debounce?: number
|
||||||
|
minSearchLength?: number
|
||||||
|
allowCreate?: boolean
|
||||||
|
localFilter?: boolean
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
noResultsText?: string
|
||||||
|
loadingText?: string
|
||||||
|
minSearchText?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
label: '',
|
||||||
|
required: false,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
options: () => [],
|
||||||
|
loading: false,
|
||||||
|
debounce: 300,
|
||||||
|
minSearchLength: 0,
|
||||||
|
allowCreate: false,
|
||||||
|
localFilter: false,
|
||||||
|
iconName: '',
|
||||||
|
iconPosition: 'left',
|
||||||
|
iconSize: 24,
|
||||||
|
iconColor: 'text-m-muted',
|
||||||
|
noResultsText: 'Aucun résultat',
|
||||||
|
loadingText: 'Chargement…',
|
||||||
|
minSearchText: 'Tapez pour rechercher',
|
||||||
|
reserveMessageSpace: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'update:modelValue', value: string | number | null): void
|
||||||
|
(e: 'search' | 'create', value: string): void
|
||||||
|
(e: 'select', option: Option | null): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const attrs = useAttrs()
|
||||||
|
const generatedId = useId()
|
||||||
|
const root = ref<HTMLElement | null>(null)
|
||||||
|
const listRef = ref<HTMLElement | null>(null)
|
||||||
|
const inputValue = ref<string>('')
|
||||||
|
const isFocused = ref(false)
|
||||||
|
const isOpen = ref(false)
|
||||||
|
const activeIndex = ref(-1)
|
||||||
|
let debounceTimer: ReturnType<typeof setTimeout> | null = null
|
||||||
|
|
||||||
|
const inputId = computed(() => props.id?.toString() || `malio-input-autocomplete-${generatedId}`)
|
||||||
|
const listboxId = computed(() => `${inputId.value}-listbox`)
|
||||||
|
|
||||||
|
const selectedOption = computed(() =>
|
||||||
|
props.options.find(o => o.value === props.modelValue) ?? null,
|
||||||
|
)
|
||||||
|
|
||||||
|
const hasSelection = computed(() =>
|
||||||
|
props.modelValue !== null && props.modelValue !== undefined && props.modelValue !== '',
|
||||||
|
)
|
||||||
|
|
||||||
|
const hasError = computed(() => !!props.error)
|
||||||
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
|
const isFilled = computed(() => inputValue.value.trim().length > 0 || hasSelection.value)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || inputValue.value.length > 0,
|
||||||
|
)
|
||||||
|
|
||||||
|
const showMinSearch = computed(() =>
|
||||||
|
props.minSearchLength > 0 && inputValue.value.length < props.minSearchLength,
|
||||||
|
)
|
||||||
|
|
||||||
|
const filteredOptions = computed(() => {
|
||||||
|
if (!props.localFilter) return props.options
|
||||||
|
const query = inputValue.value.trim().toLowerCase()
|
||||||
|
if (query === '') return props.options
|
||||||
|
return props.options.filter(opt =>
|
||||||
|
opt.label.toLowerCase().includes(query),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
const optionId = (index: number) => `${inputId.value}-option-${index}`
|
||||||
|
const activeOptionId = computed(() =>
|
||||||
|
activeIndex.value >= 0 && filteredOptions.value[activeIndex.value]
|
||||||
|
? optionId(activeIndex.value)
|
||||||
|
: undefined,
|
||||||
|
)
|
||||||
|
|
||||||
|
const describedBy = computed(() =>
|
||||||
|
(props.hint || hasError.value || hasSuccess.value) ? `${inputId.value}-describedby` : undefined,
|
||||||
|
)
|
||||||
|
|
||||||
|
watch(
|
||||||
|
[() => props.modelValue, () => props.options],
|
||||||
|
() => {
|
||||||
|
if (isFocused.value) return
|
||||||
|
if (selectedOption.value) {
|
||||||
|
inputValue.value = selectedOption.value.label
|
||||||
|
} else if (props.allowCreate && typeof props.modelValue === 'string' && props.modelValue !== '') {
|
||||||
|
inputValue.value = props.modelValue
|
||||||
|
} else if (!hasSelection.value) {
|
||||||
|
inputValue.value = ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{immediate: true},
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
twMerge('relative flex h-12 w-full items-center', props.groupClass),
|
||||||
|
)
|
||||||
|
|
||||||
|
const iconInputPaddingClass = computed(() => {
|
||||||
|
const parts: string[] = []
|
||||||
|
if (props.iconName && props.iconPosition === 'left') parts.push('!pl-11')
|
||||||
|
|
||||||
|
const hasCustomRight = !!props.iconName && props.iconPosition === 'right'
|
||||||
|
if (hasCustomRight) parts.push('!pr-16')
|
||||||
|
else parts.push('!pr-10')
|
||||||
|
|
||||||
|
return parts.join(' ')
|
||||||
|
})
|
||||||
|
|
||||||
|
const labelPositionClass = computed(() =>
|
||||||
|
props.iconName && props.iconPosition === 'left' ? 'left-11' : 'left-3',
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedInputClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
|
keyboardFocused.value ? (isOpen.value ? 'm-combo-ring-top' : 'm-focus-ring-kbd') : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
|
props.disabled
|
||||||
|
? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted'
|
||||||
|
: 'cursor-text',
|
||||||
|
hasError.value
|
||||||
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
|
isOpen.value ? '!rounded-b-none !border-b-transparent' : '',
|
||||||
|
props.inputClass,
|
||||||
|
iconInputPaddingClass.value,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedLabelClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
|
labelPositionClass.value,
|
||||||
|
shouldFloatLabel.value ? '-translate-y-[1.25rem] scale-90' : '',
|
||||||
|
hasError.value
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'text-m-success'
|
||||||
|
: props.disabled
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
|
props.labelClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (props.disabled) return props.iconColor
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return props.iconColor
|
||||||
|
})
|
||||||
|
|
||||||
|
const chevronColorClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isOpen.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return 'text-m-muted'
|
||||||
|
})
|
||||||
|
|
||||||
|
const scheduleSearch = () => {
|
||||||
|
if (debounceTimer) clearTimeout(debounceTimer)
|
||||||
|
if (showMinSearch.value) return
|
||||||
|
debounceTimer = setTimeout(() => {
|
||||||
|
emit('search', inputValue.value)
|
||||||
|
}, props.debounce)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onInput = (event: Event) => {
|
||||||
|
const target = event.target as HTMLInputElement
|
||||||
|
inputValue.value = target.value
|
||||||
|
if (!isOpen.value) isOpen.value = true
|
||||||
|
activeIndex.value = -1
|
||||||
|
|
||||||
|
if (hasSelection.value && target.value !== selectedOption.value?.label) {
|
||||||
|
emit('update:modelValue', null)
|
||||||
|
emit('select', null)
|
||||||
|
}
|
||||||
|
|
||||||
|
scheduleSearch()
|
||||||
|
}
|
||||||
|
|
||||||
|
const onFocus = () => {
|
||||||
|
onKbdFocus()
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
isFocused.value = true
|
||||||
|
isOpen.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const onInputClick = () => {
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
isOpen.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSelect = (option: Option) => {
|
||||||
|
inputValue.value = option.label
|
||||||
|
activeIndex.value = -1
|
||||||
|
emit('update:modelValue', option.value)
|
||||||
|
emit('select', option)
|
||||||
|
isOpen.value = false
|
||||||
|
isFocused.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const closeAndCommit = () => {
|
||||||
|
if (
|
||||||
|
props.allowCreate
|
||||||
|
&& inputValue.value !== ''
|
||||||
|
&& inputValue.value !== selectedOption.value?.label
|
||||||
|
) {
|
||||||
|
emit('update:modelValue', inputValue.value)
|
||||||
|
emit('create', inputValue.value)
|
||||||
|
} else if (selectedOption.value) {
|
||||||
|
inputValue.value = selectedOption.value.label
|
||||||
|
} else if (!props.allowCreate) {
|
||||||
|
inputValue.value = ''
|
||||||
|
}
|
||||||
|
isOpen.value = false
|
||||||
|
isFocused.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const closeAndRevert = () => {
|
||||||
|
if (selectedOption.value) {
|
||||||
|
inputValue.value = selectedOption.value.label
|
||||||
|
} else {
|
||||||
|
inputValue.value = ''
|
||||||
|
}
|
||||||
|
isOpen.value = false
|
||||||
|
isFocused.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Garde l'option active visible dans la liste défilante quand on navigue au clavier
|
||||||
|
watch(activeIndex, async (index) => {
|
||||||
|
if (index < 0 || !isOpen.value) return
|
||||||
|
await nextTick()
|
||||||
|
document.getElementById(optionId(index))?.scrollIntoView({block: 'nearest'})
|
||||||
|
})
|
||||||
|
|
||||||
|
const onKeydown = (event: KeyboardEvent) => {
|
||||||
|
// Tab : laisse le focus partir mais ferme la liste (et valide la saisie courante)
|
||||||
|
if (event.key === 'Tab') {
|
||||||
|
if (isOpen.value) closeAndCommit()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === 'Escape') {
|
||||||
|
event.preventDefault()
|
||||||
|
closeAndRevert()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === 'Enter') {
|
||||||
|
event.preventDefault()
|
||||||
|
if (activeIndex.value >= 0 && filteredOptions.value[activeIndex.value]) {
|
||||||
|
onSelect(filteredOptions.value[activeIndex.value])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (props.allowCreate && inputValue.value !== '') {
|
||||||
|
emit('update:modelValue', inputValue.value)
|
||||||
|
emit('create', inputValue.value)
|
||||||
|
isOpen.value = false
|
||||||
|
isFocused.value = false
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === 'ArrowDown') {
|
||||||
|
event.preventDefault()
|
||||||
|
if (!isOpen.value) {
|
||||||
|
isOpen.value = true
|
||||||
|
}
|
||||||
|
activeIndex.value = Math.min(activeIndex.value + 1, filteredOptions.value.length - 1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === 'ArrowUp') {
|
||||||
|
event.preventDefault()
|
||||||
|
// Liste fermée : ouvre et place sur la dernière option (APG)
|
||||||
|
if (!isOpen.value) {
|
||||||
|
isOpen.value = true
|
||||||
|
activeIndex.value = filteredOptions.value.length - 1
|
||||||
|
return
|
||||||
|
}
|
||||||
|
activeIndex.value = Math.max(activeIndex.value - 1, 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Home / End : première / dernière option quand la liste est ouverte
|
||||||
|
if (isOpen.value && event.key === 'Home') {
|
||||||
|
event.preventDefault()
|
||||||
|
activeIndex.value = 0
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (isOpen.value && event.key === 'End') {
|
||||||
|
event.preventDefault()
|
||||||
|
activeIndex.value = filteredOptions.value.length - 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClickOutside = (event: MouseEvent) => {
|
||||||
|
if (!root.value) return
|
||||||
|
if (!root.value.contains(event.target as Node)) {
|
||||||
|
closeAndCommit()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => document.addEventListener('mousedown', onClickOutside))
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
document.removeEventListener('mousedown', onClickOutside)
|
||||||
|
if (debounceTimer) clearTimeout(debounceTimer)
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.floating-label {
|
||||||
|
background: white;
|
||||||
|
padding: 0 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow-height {
|
||||||
|
transition: border-color 160ms ease, box-shadow 160ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-reduced-motion: reduce) {
|
||||||
|
.grow-height {
|
||||||
|
transition: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(ul[role="listbox"]) {
|
||||||
|
scrollbar-width: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.select-scrollbar-primary) {
|
||||||
|
scrollbar-color: rgb(var(--m-primary)) transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.select-scrollbar-error) {
|
||||||
|
scrollbar-color: #000000 transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.select-scrollbar-success) {
|
||||||
|
scrollbar-color: #000000 transparent;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,387 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import InputEmail from './InputEmail.vue'
|
||||||
|
|
||||||
|
type InputEmailProps = {
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
autocomplete?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
lowercase?: boolean
|
||||||
|
addable?: boolean
|
||||||
|
addIconName?: string
|
||||||
|
addButtonLabel?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const InputEmailForTest = InputEmail as DefineComponent<InputEmailProps>
|
||||||
|
|
||||||
|
const mountComponent = (props: InputEmailProps = {}) =>
|
||||||
|
mount(InputEmailForTest, {
|
||||||
|
props,
|
||||||
|
global: {
|
||||||
|
stubs: {
|
||||||
|
IconifyIcon: {
|
||||||
|
template: '<span data-test="icon" v-bind="$attrs" />',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioInputEmail', () => {
|
||||||
|
it('renders the initial input value', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'user@example.com'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('user@example.com')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the label text', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Adresse email'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').text()).toBe('Adresse email')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has type email', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('type')).toBe('email')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has inputmode email', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('inputmode')).toBe('email')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the default email icon', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
const iconComponent = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(iconComponent.props('icon')).toBe('mdi:email-outline')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding the icon', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:at'})
|
||||||
|
|
||||||
|
const iconComponent = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(iconComponent.props('icon')).toBe('mdi:at')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render icon when iconName is empty', () => {
|
||||||
|
const wrapper = mountComponent({iconName: ''})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('places icon on the right by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('right-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('places icon on the left when iconPosition is left', () => {
|
||||||
|
const wrapper = mountComponent({iconPosition: 'left'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue on input change', async () => {
|
||||||
|
const wrapper = mountComponent({modelValue: ''})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('new@example.com')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['new@example.com'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets disabled styles when true', () => {
|
||||||
|
const wrapper = mountComponent({disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('disabled')).toBeDefined()
|
||||||
|
expect(wrapper.get('input').classes()).toContain('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets readonly when true', () => {
|
||||||
|
const wrapper = mountComponent({readonly: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('readonly')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error message and styles', () => {
|
||||||
|
const wrapper = mountComponent({error: 'Email invalide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Email invalide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-danger')
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error style on icon', () => {
|
||||||
|
const wrapper = mountComponent({error: 'Error'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success message and styles', () => {
|
||||||
|
const wrapper = mountComponent({success: 'Email valide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-success').text()).toBe('Email valide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success style on icon', () => {
|
||||||
|
const wrapper = mountComponent({success: 'Success'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows default icon color when empty and unfocused', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'user@example.com'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps primary icon color when filled and focused', async () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'user@example.com'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps default icon color when disabled, even if filled', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'user@example.com', disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('error overrides focus color on icon', async () => {
|
||||||
|
const wrapper = mountComponent({error: 'Email invalide'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows hint message', () => {
|
||||||
|
const wrapper = mountComponent({hint: 'ex: prenom.nom@malio.fr'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-muted').text()).toBe('ex: prenom.nom@malio.fr')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('links label to input via for/id', () => {
|
||||||
|
const wrapper = mountComponent({id: 'email-field', label: 'Email'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('id')).toBe('email-field')
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe('email-field')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an id when missing and reuses it on label', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Email'})
|
||||||
|
|
||||||
|
const inputId = wrapper.get('input').attributes('id')
|
||||||
|
|
||||||
|
expect(inputId?.startsWith('malio-input-email-')).toBe(true)
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe(inputId)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aria-invalid is false when no error', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses autocomplete off by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('autocomplete')).toBe('off')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding autocomplete', () => {
|
||||||
|
const wrapper = mountComponent({autocomplete: 'email'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('autocomplete')).toBe('email')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('supprime tous les espaces saisis', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
await wrapper.get('input').setValue(' a b @ c.com ')
|
||||||
|
const emits = wrapper.emitted('update:modelValue')!
|
||||||
|
expect(emits[emits.length - 1]).toEqual(['ab@c.com'])
|
||||||
|
expect(wrapper.get('input').element.value).toBe('ab@c.com')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('conserve la casse par défaut', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
await wrapper.get('input').setValue('User@Example.COM')
|
||||||
|
const emits = wrapper.emitted('update:modelValue')!
|
||||||
|
expect(emits[emits.length - 1]).toEqual(['User@Example.COM'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('met en minuscules quand lowercase est vrai', async () => {
|
||||||
|
const wrapper = mountComponent({lowercase: true})
|
||||||
|
await wrapper.get('input').setValue('User@Example.COM')
|
||||||
|
const emits = wrapper.emitted('update:modelValue')!
|
||||||
|
expect(emits[emits.length - 1]).toEqual(['user@example.com'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('émet la valeur sanitisée en mode contrôlé', async () => {
|
||||||
|
const wrapper = mountComponent({modelValue: ''})
|
||||||
|
await wrapper.get('input').setValue(' a b @ c.com ')
|
||||||
|
expect(wrapper.emitted('update:modelValue')!.at(-1)).toEqual(['ab@c.com'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('resynchronise le DOM en mode contrôlé même quand la valeur sanitisée égale déjà modelValue', async () => {
|
||||||
|
// L'utilisateur ajoute un espace en fin alors que la valeur nettoyée vaut déjà modelValue.
|
||||||
|
// Le parent ne « changera » pas modelValue → Vue ne re-patche pas le DOM ; l'écriture
|
||||||
|
// manuelle target.value = sanitized est donc indispensable pour retirer l'espace affiché.
|
||||||
|
const wrapper = mountComponent({modelValue: 'ab@c.com'})
|
||||||
|
const input = wrapper.get('input')
|
||||||
|
await input.setValue('ab@c.com ')
|
||||||
|
expect(input.element.value).toBe('ab@c.com')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir et icône noire', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true, modelValue: 'user@example.com'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render add button by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders add button when addable is true', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits add event when add button is clicked', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
await wrapper.get('[data-test="add-button"]').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('add')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the add button when disabled', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not emit add when readonly', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true, readonly: true})
|
||||||
|
|
||||||
|
await wrapper.get('[data-test="add-button"]').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('add')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('add button is not natively disabled in readonly (onAdd guard blocks the action)', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, readonly: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').attributes('disabled')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves the email icon to the left automatically when addable', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
const icon = wrapper.get('[data-test="icon"]')
|
||||||
|
expect(icon.classes()).toContain('left-[10px]')
|
||||||
|
expect(icon.classes()).not.toContain('right-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps the email icon on the right when addable is false', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('right-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses the default add button aria-label', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').attributes('aria-label')).toBe('Ajouter une adresse email')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding the add button aria-label', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, addButtonLabel: 'Ajouter un destinataire'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').attributes('aria-label')).toBe('Ajouter un destinataire')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,323 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
:class="mergedGroupClass"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
:id="inputId"
|
||||||
|
:name="name"
|
||||||
|
:autocomplete="autocomplete"
|
||||||
|
:class="mergedInputClass"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:value="currentValue"
|
||||||
|
:readonly="readonly"
|
||||||
|
:aria-invalid="!!error"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
v-bind="attrs"
|
||||||
|
placeholder="_"
|
||||||
|
type="email"
|
||||||
|
inputmode="email"
|
||||||
|
@input="onInput"
|
||||||
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
|
>
|
||||||
|
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="inputId"
|
||||||
|
:class="mergedLabelClass"
|
||||||
|
>
|
||||||
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="iconName"
|
||||||
|
:icon="iconName"
|
||||||
|
:width="iconSize"
|
||||||
|
:height="iconSize"
|
||||||
|
data-test="icon"
|
||||||
|
:class="[iconStateClass, iconPositionClass]"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<button
|
||||||
|
v-if="addable && !disabled"
|
||||||
|
type="button"
|
||||||
|
:aria-label="addButtonLabel"
|
||||||
|
data-test="add-button"
|
||||||
|
:class="mergedAddButtonClass"
|
||||||
|
@click="onAdd"
|
||||||
|
>
|
||||||
|
<IconifyIcon
|
||||||
|
:icon="addIconName"
|
||||||
|
:width="24"
|
||||||
|
:height="24"
|
||||||
|
data-test="add-icon"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<p
|
||||||
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
|
:id="`${inputId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess
|
||||||
|
? 'text-m-success'
|
||||||
|
: 'text-m-muted',
|
||||||
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ hint || error || success }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
|
||||||
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioInputEmail', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
autocomplete?: string
|
||||||
|
modelValue?: string | null | undefined
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
addable?: boolean
|
||||||
|
addIconName?: string
|
||||||
|
addButtonLabel?: string
|
||||||
|
lowercase?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
autocomplete: 'off',
|
||||||
|
modelValue: undefined,
|
||||||
|
iconName: 'mdi:email-outline',
|
||||||
|
iconPosition: 'right',
|
||||||
|
label: '',
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
required: false,
|
||||||
|
readonly: false,
|
||||||
|
disabled: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
iconSize: 24,
|
||||||
|
iconColor: 'text-m-muted',
|
||||||
|
addable: false,
|
||||||
|
addIconName: 'mdi:plus',
|
||||||
|
addButtonLabel: 'Ajouter une adresse email',
|
||||||
|
lowercase: false,
|
||||||
|
reserveMessageSpace: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const attrs = useAttrs()
|
||||||
|
const generatedId = useId()
|
||||||
|
const localValue = ref('')
|
||||||
|
const isFocused = ref(false)
|
||||||
|
|
||||||
|
const inputId = computed(() => props.id?.toString() || `malio-input-email-${generatedId}`)
|
||||||
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
|
const hasError = computed(() => !!props.error)
|
||||||
|
const hasSuccess = computed(() => !!props.success)
|
||||||
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'relative flex h-12 w-full items-center',
|
||||||
|
props.groupClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
const mergedInputClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
|
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
||||||
|
hasError.value
|
||||||
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
|
props.inputClass,
|
||||||
|
iconInputPaddingClass.value,
|
||||||
|
isReadonly.value ? '' : focusPaddingClass.value,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
const mergedLabelClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
|
labelPositionClass.value,
|
||||||
|
shouldFloatLabel.value
|
||||||
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
|
hasError.value
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'text-m-success'
|
||||||
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
|
props.labelClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedAddButtonClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'absolute right-[10px] top-1/2 -translate-y-1/2 cursor-pointer transition-opacity hover:opacity-70',
|
||||||
|
iconStateClass.value,
|
||||||
|
props.disabled ? 'cursor-not-allowed opacity-40 hover:opacity-40' : '',
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const describedBy = computed(() => {
|
||||||
|
const ids: string[] = []
|
||||||
|
if (props.hint && !hasSuccess.value && !hasError.value) ids.push(`${inputId.value}-hint`)
|
||||||
|
if (hasError.value) ids.push(`${inputId.value}-error`)
|
||||||
|
if (hasSuccess.value && !hasError.value) ids.push(`${inputId.value}-success`)
|
||||||
|
return ids.length ? ids.join(' ') : undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(event: 'update:modelValue', value: string): void
|
||||||
|
(event: 'add'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const sanitizeEmail = (v: string) => {
|
||||||
|
let out = v.replace(/\s+/g, '')
|
||||||
|
if (props.lowercase) out = out.toLowerCase()
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
const onInput = (event: Event) => {
|
||||||
|
const target = event.target as HTMLInputElement
|
||||||
|
const raw = target.value
|
||||||
|
const sanitized = sanitizeEmail(raw)
|
||||||
|
|
||||||
|
if (sanitized !== raw) {
|
||||||
|
// `<input type="email">` ne supporte pas l'API de sélection :
|
||||||
|
// selectionStart vaut null et setSelectionRange lève en navigateur.
|
||||||
|
// (En jsdom selectionStart peut renvoyer un nombre, d'où le code gardé ci-dessous.)
|
||||||
|
const caret = target.selectionStart
|
||||||
|
target.value = sanitized
|
||||||
|
if (caret !== null) {
|
||||||
|
const newCaret = sanitizeEmail(raw.slice(0, caret)).length
|
||||||
|
try {
|
||||||
|
target.setSelectionRange(newCaret, newCaret)
|
||||||
|
} catch {
|
||||||
|
/* type d'input sans support de sélection — ignore */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localValue.value = sanitized
|
||||||
|
}
|
||||||
|
emit('update:modelValue', sanitized)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onAdd = () => {
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
emit('add')
|
||||||
|
}
|
||||||
|
|
||||||
|
const effectiveIconPosition = computed(() =>
|
||||||
|
props.addable && props.iconName ? 'left' : props.iconPosition,
|
||||||
|
)
|
||||||
|
|
||||||
|
const iconInputPaddingClass = computed(() => {
|
||||||
|
const leftIcon = props.iconName && effectiveIconPosition.value === 'left'
|
||||||
|
const rightIcon = props.iconName && effectiveIconPosition.value === 'right'
|
||||||
|
const parts: string[] = []
|
||||||
|
if (leftIcon) parts.push('!pl-11')
|
||||||
|
if (rightIcon || props.addable) parts.push('!pr-10')
|
||||||
|
return parts.join(' ')
|
||||||
|
})
|
||||||
|
|
||||||
|
const disabled = computed(() => props.disabled)
|
||||||
|
|
||||||
|
const labelPositionClass = computed(() => {
|
||||||
|
if (props.iconName && effectiveIconPosition.value === 'left') return 'left-11'
|
||||||
|
return 'left-3'
|
||||||
|
})
|
||||||
|
|
||||||
|
const focusPaddingClass = computed(() => {
|
||||||
|
if (props.iconName && effectiveIconPosition.value === 'left') return 'focus:!pl-11'
|
||||||
|
return 'focus:pl-[11px]'
|
||||||
|
})
|
||||||
|
|
||||||
|
const iconPositionClass = computed(() => {
|
||||||
|
const sideClass = effectiveIconPosition.value === 'left' ? 'left-[10px]' : 'right-[10px]'
|
||||||
|
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
||||||
|
})
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return props.iconColor
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return props.iconColor
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.floating-label {
|
||||||
|
background: white;
|
||||||
|
padding: 0 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow-height {
|
||||||
|
transition: border-color 160ms ease, box-shadow 160ms ease, padding-top 160ms ease, padding-bottom 160ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow-height:focus {
|
||||||
|
padding-top: 0.625rem;
|
||||||
|
padding-bottom: 0.625rem;
|
||||||
|
}
|
||||||
|
@media (prefers-reduced-motion: reduce) {
|
||||||
|
.grow-height { transition: none; }
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -6,9 +6,13 @@ import InputNumber from './InputNumber.vue'
|
|||||||
type InputNumberProps = {
|
type InputNumberProps = {
|
||||||
modelValue?: string | null
|
modelValue?: string | null
|
||||||
label?: string
|
label?: string
|
||||||
|
required?: boolean
|
||||||
readonly?: boolean
|
readonly?: boolean
|
||||||
min?: number | string
|
min?: number | string
|
||||||
max?: number | string
|
max?: number | string
|
||||||
|
error?: string
|
||||||
|
hint?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputNumberForTest = InputNumber as DefineComponent<InputNumberProps>
|
const InputNumberForTest = InputNumber as DefineComponent<InputNumberProps>
|
||||||
@@ -162,4 +166,33 @@ describe('MalioInputNumber', () => {
|
|||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['5'])
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['5'])
|
||||||
expect(input.element.value).toBe('5')
|
expect(input.element.value).toBe('5')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountInputNumber({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountInputNumber({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountInputNumber({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountInputNumber({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountInputNumber({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -6,10 +6,11 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
class="m-focus-ring rounded-malio"
|
||||||
:disabled="isMinusDisabled"
|
:disabled="isMinusDisabled"
|
||||||
@click="decrement"
|
@click="decrement"
|
||||||
>
|
>
|
||||||
@@ -35,11 +36,12 @@
|
|||||||
inputmode="numeric"
|
inputmode="numeric"
|
||||||
placeholder="_"
|
placeholder="_"
|
||||||
@input="onInput"
|
@input="onInput"
|
||||||
@focus="isFocused = true"
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
@blur="isFocused = false"
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
|
class="m-focus-ring rounded-malio"
|
||||||
:disabled="isPlusDisabled"
|
:disabled="isPlusDisabled"
|
||||||
@click="increment"
|
@click="increment"
|
||||||
>
|
>
|
||||||
@@ -51,7 +53,7 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -59,7 +61,8 @@
|
|||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'text-xs ml-[2px] ',
|
'text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ hint || error || success }}
|
{{ hint || error || success }}
|
||||||
@@ -71,9 +74,13 @@
|
|||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import {Icon as IconifyIcon} from '@iconify/vue'
|
import {Icon as IconifyIcon} from '@iconify/vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputNumber', inheritAttrs: false})
|
defineOptions({name: 'MalioInputNumber', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -91,6 +98,7 @@ const props = withDefaults(
|
|||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -108,6 +116,7 @@ const props = withDefaults(
|
|||||||
hint: '',
|
hint: '',
|
||||||
error: '',
|
error: '',
|
||||||
success: '',
|
success: '',
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -180,6 +189,7 @@ const mergedGroupClass = computed(() =>
|
|||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
' peer h-[22px] min-w-0 border bg-white text-center outline-none placeholder:text-transparent text-lg border-x-0 border-black',
|
' peer h-[22px] min-w-0 border bg-white text-center outline-none placeholder:text-transparent text-lg border-x-0 border-black',
|
||||||
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
props.disabled ? 'cursor-not-allowed text-black/60' : 'cursor-text',
|
props.disabled ? 'cursor-not-allowed text-black/60' : 'cursor-text',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
@@ -195,7 +205,7 @@ const mergedLabelClass = computed(() =>
|
|||||||
'cursor-pointer text-black mr-4 text-[18px]',
|
'cursor-pointer text-black mr-4 text-[18px]',
|
||||||
hasError.value ? 'text-m-danger' : '',
|
hasError.value ? 'text-m-danger' : '',
|
||||||
hasSuccess.value ? 'text-m-success' : '',
|
hasSuccess.value ? 'text-m-success' : '',
|
||||||
props.disabled ? 'cursor-not-allowed text-black/60' : '',
|
props.disabled ? 'cursor-not-allowed text-m-muted' : '',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ type InputPasswordProps = {
|
|||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
displayIcon?: boolean
|
displayIcon?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputPasswordForTest = InputPassword as DefineComponent<InputPasswordProps>
|
const InputPasswordForTest = InputPassword as DefineComponent<InputPasswordProps>
|
||||||
@@ -51,6 +52,16 @@ describe('MalioInputPassword', () => {
|
|||||||
expect(wrapper.get('label').text()).toBe('Mot de passe')
|
expect(wrapper.get('label').text()).toBe('Mot de passe')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
it('has type password by default', () => {
|
it('has type password by default', () => {
|
||||||
const wrapper = mountComponent()
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
@@ -80,6 +91,12 @@ describe('MalioInputPassword', () => {
|
|||||||
expect(wrapper.find('[data-test="icon"]').exists()).toBe(false)
|
expect(wrapper.find('[data-test="icon"]').exists()).toBe(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('hides the eye icon when disabled', () => {
|
||||||
|
const wrapper = mountComponent({disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
it('renders icon by default', () => {
|
it('renders icon by default', () => {
|
||||||
const wrapper = mountComponent()
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
@@ -171,4 +188,69 @@ describe('MalioInputPassword', () => {
|
|||||||
|
|
||||||
expect(wrapper.get('input').attributes('aria-invalid')).toBe('false')
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('false')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'secret'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : icône en text-m-muted', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir et icône noire', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true, modelValue: 'secret'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : eye toggle reste cliquable', async () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
await wrapper.get('[data-test="icon"]').trigger('click')
|
||||||
|
expect(wrapper.get('input').attributes('type')).toBe('text')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -20,8 +20,8 @@
|
|||||||
placeholder="_"
|
placeholder="_"
|
||||||
:type="isPasswordVisible ? 'text' : 'password'"
|
:type="isPasswordVisible ? 'text' : 'password'"
|
||||||
@input="onInput"
|
@input="onInput"
|
||||||
@focus="isFocused = true"
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
@blur="isFocused = false"
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
>
|
>
|
||||||
|
|
||||||
<label
|
<label
|
||||||
@@ -29,20 +29,17 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<IconifyIcon
|
<IconifyIcon
|
||||||
v-if="displayIcon"
|
v-if="displayIcon && !disabled"
|
||||||
:icon="isPasswordVisible ? 'mdi:eye-outline' : 'mdi:eye-off-outline'"
|
:icon="isPasswordVisible ? 'mdi:eye-outline' : 'mdi:eye-off-outline'"
|
||||||
:width="24"
|
:width="24"
|
||||||
:height="24"
|
:height="24"
|
||||||
data-test="icon"
|
data-test="icon"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
iconStateClass,
|
||||||
? 'text-m-danger'
|
|
||||||
: hasSuccess
|
|
||||||
? 'text-m-success' : 'text-m-muted',
|
|
||||||
'cursor-pointer absolute right-[10px] top-1/2 -translate-y-1/2',
|
'cursor-pointer absolute right-[10px] top-1/2 -translate-y-1/2',
|
||||||
]"
|
]"
|
||||||
@click="toggleVisibility"
|
@click="toggleVisibility"
|
||||||
@@ -50,7 +47,7 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -58,7 +55,8 @@
|
|||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'mt-1 text-xs ml-[2px] ',
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ hint || error || success }}
|
{{ hint || error || success }}
|
||||||
@@ -71,9 +69,13 @@
|
|||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputPassword', inheritAttrs: false})
|
defineOptions({name: 'MalioInputPassword', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -93,6 +95,7 @@ const props = withDefaults(
|
|||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
displayIcon?: boolean
|
displayIcon?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -112,6 +115,7 @@ const props = withDefaults(
|
|||||||
error: '',
|
error: '',
|
||||||
success: '',
|
success: '',
|
||||||
displayIcon: true,
|
displayIcon: true,
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -128,10 +132,15 @@ const toggleVisibility = () => {
|
|||||||
const inputId = computed(() => props.id?.toString() || `malio-input-password-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-input-password-${generatedId}`)
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
const shouldFloatLabel = computed(() => isFocused.value || currentValue.value.length > 0)
|
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success)
|
const hasSuccess = computed(() => !!props.success)
|
||||||
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'relative flex h-12 w-full items-center',
|
'relative flex h-12 w-full items-center',
|
||||||
@@ -140,16 +149,21 @@ const mergedGroupClass = computed(() =>
|
|||||||
)
|
)
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-input grow-height peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent focus:border-2 text-lg rounded-md',
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
isFilled.value ? 'border-black' : 'border-m-muted',
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
: 'focus:border-m-primary',
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
props.displayIcon ? '!pr-10' : '',
|
props.displayIcon ? '!pr-10' : '',
|
||||||
'focus:pl-[11px]',
|
isReadonly.value ? '' : 'focus:pl-[11px]',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -157,13 +171,18 @@ const mergedLabelClass = computed(() =>
|
|||||||
twMerge(
|
twMerge(
|
||||||
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
'left-3',
|
'left-3',
|
||||||
shouldFloatLabel.value ? '-translate-y-[1.25rem] peer-focus:-translate-y-[1.55rem] scale-90' : '',
|
shouldFloatLabel.value
|
||||||
disabled.value ? 'peer-[&:not(:placeholder-shown):not(:focus)]:text-black/60' : '',
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -189,6 +208,16 @@ const onInput = (event: Event) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const disabled = computed(() => props.disabled)
|
const disabled = computed(() => props.disabled)
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return 'text-m-muted'
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return 'text-m-muted'
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
@@ -0,0 +1,398 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import InputPhone from './InputPhone.vue'
|
||||||
|
|
||||||
|
type InputPhoneProps = {
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
autocomplete?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
mask?: string
|
||||||
|
addable?: boolean
|
||||||
|
addIconName?: string
|
||||||
|
addButtonLabel?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const InputPhoneForTest = InputPhone as DefineComponent<InputPhoneProps>
|
||||||
|
|
||||||
|
const mountComponent = (props: InputPhoneProps = {}) =>
|
||||||
|
mount(InputPhoneForTest, {
|
||||||
|
props,
|
||||||
|
global: {
|
||||||
|
stubs: {
|
||||||
|
IconifyIcon: {
|
||||||
|
template: '<span data-test="icon" v-bind="$attrs" />',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioInputPhone', () => {
|
||||||
|
it('renders the initial input value', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: '+33 6 12 34 56 78'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').element.value).toBe('+33 6 12 34 56 78')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the label text', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Téléphone'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').text()).toBe('Téléphone')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has type tel', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('type')).toBe('tel')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has inputmode tel', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('inputmode')).toBe('tel')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the default phone icon', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
const iconComponent = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(iconComponent.props('icon')).toBe('mdi:phone-outline')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding the icon', () => {
|
||||||
|
const wrapper = mountComponent({iconName: 'mdi:cellphone'})
|
||||||
|
|
||||||
|
const iconComponent = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(iconComponent.props('icon')).toBe('mdi:cellphone')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render icon when iconName is empty', () => {
|
||||||
|
const wrapper = mountComponent({iconName: ''})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="icon"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('places icon on the left by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('left-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('places icon on the right when iconPosition is right', () => {
|
||||||
|
const wrapper = mountComponent({iconPosition: 'right'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('right-[10px]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue on input change', async () => {
|
||||||
|
const wrapper = mountComponent({modelValue: ''})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('+33612345678')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['+33612345678'])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets disabled styles when true', () => {
|
||||||
|
const wrapper = mountComponent({disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('disabled')).toBeDefined()
|
||||||
|
expect(wrapper.get('input').classes()).toContain('cursor-not-allowed')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets readonly when true', () => {
|
||||||
|
const wrapper = mountComponent({readonly: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('readonly')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error message and styles', () => {
|
||||||
|
const wrapper = mountComponent({error: 'Numéro invalide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Numéro invalide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-danger')
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error style on icon', () => {
|
||||||
|
const wrapper = mountComponent({error: 'Error'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success message and styles', () => {
|
||||||
|
const wrapper = mountComponent({success: 'Numéro valide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-success').text()).toBe('Numéro valide')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success style on icon', () => {
|
||||||
|
const wrapper = mountComponent({success: 'Success'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows default icon color when empty and unfocused', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: '+33612345678'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps default icon color when disabled, even if filled', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: '+33612345678', disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('error overrides focus color on icon', async () => {
|
||||||
|
const wrapper = mountComponent({error: 'Numéro invalide'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows hint message', () => {
|
||||||
|
const wrapper = mountComponent({hint: 'Format international recommandé'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-muted').text()).toBe('Format international recommandé')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('links label to input via for/id', () => {
|
||||||
|
const wrapper = mountComponent({id: 'phone-field', label: 'Téléphone'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('id')).toBe('phone-field')
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe('phone-field')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an id when missing and reuses it on label', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Téléphone'})
|
||||||
|
|
||||||
|
const inputId = wrapper.get('input').attributes('id')
|
||||||
|
|
||||||
|
expect(inputId?.startsWith('malio-input-phone-')).toBe(true)
|
||||||
|
expect(wrapper.get('label').attributes('for')).toBe(inputId)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aria-invalid is false when no error', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('aria-invalid')).toBe('false')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses autocomplete off by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('autocomplete')).toBe('off')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding autocomplete', () => {
|
||||||
|
const wrapper = mountComponent({autocomplete: 'tel'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').attributes('autocomplete')).toBe('tel')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render add button by default', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders add button when addable is true', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits add event when add button is clicked', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
await wrapper.get('[data-test="add-button"]').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('add')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the add button when disabled', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, disabled: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="add-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not emit add when readonly', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true, readonly: true})
|
||||||
|
|
||||||
|
await wrapper.get('[data-test="add-button"]').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('add')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('add button is not natively disabled in readonly (onAdd guard blocks the action)', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, readonly: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').attributes('disabled')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : border-black appliqué sur l\'input', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', readonly: true})
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : icône en text-m-muted quand vide', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', readonly: true})
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : icône en text-black quand rempli', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', readonly: true, modelValue: '+33612345678'})
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : pas d\'apparence désactivée (pas opacity-40)', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', addable: true, readonly: true})
|
||||||
|
// opacity-40 was only ever applied to the add button, not the input
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).not.toContain('opacity-40')
|
||||||
|
// and the input is not natively disabled in readonly:
|
||||||
|
expect(wrapper.get('input').attributes('disabled')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label en text-m-muted', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', readonly: true})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label en text-black', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Tel', readonly: true, modelValue: '+33612345678'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the default add icon (mdi:plus)', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
const icons = wrapper.findAllComponents(IconifyIcon)
|
||||||
|
const addIcon = icons[icons.length - 1]
|
||||||
|
expect(addIcon.props('icon')).toBe('mdi:plus')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('allows overriding the add icon', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, addIconName: 'mdi:phone-plus'})
|
||||||
|
|
||||||
|
const icons = wrapper.findAllComponents(IconifyIcon)
|
||||||
|
const addIcon = icons[icons.length - 1]
|
||||||
|
expect(addIcon.props('icon')).toBe('mdi:phone-plus')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('exposes aria-label on add button', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, addButtonLabel: 'Ajouter un autre numéro'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').attributes('aria-label')).toBe('Ajouter un autre numéro')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('adds right padding to input when addable', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').classes()).toContain('!pr-10')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows default add button color when empty and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).toContain('text-m-muted')
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).not.toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows primary add button color on focus', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black add button color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, modelValue: '+33612345678'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('error overrides focus color on add button', async () => {
|
||||||
|
const wrapper = mountComponent({addable: true, error: 'Numéro invalide'})
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('success applies to add button', () => {
|
||||||
|
const wrapper = mountComponent({addable: true, success: 'Numéro valide'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="add-button"]').classes()).toContain('text-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies mask via maska directive', async () => {
|
||||||
|
const wrapper = mountComponent({mask: '+## # ## ## ## ##'})
|
||||||
|
|
||||||
|
await wrapper.get('input').setValue('33612345678')
|
||||||
|
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,297 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
:class="mergedGroupClass"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
:id="inputId"
|
||||||
|
v-maska="mask"
|
||||||
|
:name="name"
|
||||||
|
:autocomplete="autocomplete"
|
||||||
|
:class="mergedInputClass"
|
||||||
|
:required="required"
|
||||||
|
:disabled="disabled"
|
||||||
|
:value="currentValue"
|
||||||
|
:readonly="readonly"
|
||||||
|
:aria-invalid="!!error"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
v-bind="attrs"
|
||||||
|
placeholder="_"
|
||||||
|
type="tel"
|
||||||
|
inputmode="tel"
|
||||||
|
@input="onInput"
|
||||||
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
|
>
|
||||||
|
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="inputId"
|
||||||
|
:class="mergedLabelClass"
|
||||||
|
>
|
||||||
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="iconName"
|
||||||
|
:icon="iconName"
|
||||||
|
:width="iconSize"
|
||||||
|
:height="iconSize"
|
||||||
|
data-test="icon"
|
||||||
|
:class="[iconStateClass, iconPositionClass]"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<button
|
||||||
|
v-if="addable && !disabled"
|
||||||
|
type="button"
|
||||||
|
:aria-label="addButtonLabel"
|
||||||
|
data-test="add-button"
|
||||||
|
:class="mergedAddButtonClass"
|
||||||
|
@click="onAdd"
|
||||||
|
>
|
||||||
|
<IconifyIcon
|
||||||
|
:icon="addIconName"
|
||||||
|
:width="24"
|
||||||
|
:height="24"
|
||||||
|
data-test="add-icon"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<p
|
||||||
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
|
:id="`${inputId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess
|
||||||
|
? 'text-m-success'
|
||||||
|
: 'text-m-muted',
|
||||||
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ hint || error || success }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
|
||||||
|
import type {MaskInputOptions} from 'maska'
|
||||||
|
import {vMaska} from 'maska/vue'
|
||||||
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
|
defineOptions({name: 'MalioInputPhone', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
name?: string
|
||||||
|
autocomplete?: string
|
||||||
|
modelValue?: string | null | undefined
|
||||||
|
inputClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
groupClass?: string
|
||||||
|
required?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
iconName?: string
|
||||||
|
iconPosition?: 'left' | 'right'
|
||||||
|
iconSize?: string | number
|
||||||
|
iconColor?: string
|
||||||
|
mask?: string | MaskInputOptions
|
||||||
|
addable?: boolean
|
||||||
|
addIconName?: string
|
||||||
|
addButtonLabel?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
name: '',
|
||||||
|
autocomplete: 'off',
|
||||||
|
modelValue: undefined,
|
||||||
|
iconName: 'mdi:phone-outline',
|
||||||
|
iconPosition: 'left',
|
||||||
|
label: '',
|
||||||
|
inputClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
groupClass: '',
|
||||||
|
required: false,
|
||||||
|
readonly: false,
|
||||||
|
disabled: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
iconSize: 24,
|
||||||
|
iconColor: 'text-m-muted',
|
||||||
|
mask: undefined,
|
||||||
|
addable: false,
|
||||||
|
addIconName: 'mdi:plus',
|
||||||
|
addButtonLabel: 'Ajouter un numéro',
|
||||||
|
reserveMessageSpace: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const attrs = useAttrs()
|
||||||
|
const generatedId = useId()
|
||||||
|
const localValue = ref('')
|
||||||
|
const isFocused = ref(false)
|
||||||
|
|
||||||
|
const inputId = computed(() => props.id?.toString() || `malio-input-phone-${generatedId}`)
|
||||||
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
|
const hasError = computed(() => !!props.error)
|
||||||
|
const hasSuccess = computed(() => !!props.success)
|
||||||
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'relative flex h-12 w-full items-center',
|
||||||
|
props.groupClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
const mergedInputClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
|
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
||||||
|
hasError.value
|
||||||
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
|
props.inputClass,
|
||||||
|
iconInputPaddingClass.value,
|
||||||
|
isReadonly.value ? '' : focusPaddingClass.value,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
const mergedLabelClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
|
labelPositionClass.value,
|
||||||
|
shouldFloatLabel.value
|
||||||
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
|
hasError.value
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'text-m-success'
|
||||||
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
|
props.labelClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedAddButtonClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'absolute right-[10px] top-1/2 -translate-y-1/2 cursor-pointer transition-opacity hover:opacity-70',
|
||||||
|
iconStateClass.value,
|
||||||
|
props.disabled ? 'cursor-not-allowed opacity-40 hover:opacity-40' : '',
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const describedBy = computed(() => {
|
||||||
|
const ids: string[] = []
|
||||||
|
if (props.hint && !hasSuccess.value && !hasError.value) ids.push(`${inputId.value}-hint`)
|
||||||
|
if (hasError.value) ids.push(`${inputId.value}-error`)
|
||||||
|
if (hasSuccess.value && !hasError.value) ids.push(`${inputId.value}-success`)
|
||||||
|
return ids.length ? ids.join(' ') : undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(event: 'update:modelValue', value: string): void
|
||||||
|
(event: 'add'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const onInput = (event: Event) => {
|
||||||
|
const target = event.target as HTMLInputElement
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localValue.value = target.value
|
||||||
|
}
|
||||||
|
emit('update:modelValue', target.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onAdd = () => {
|
||||||
|
if (props.disabled || props.readonly) return
|
||||||
|
emit('add')
|
||||||
|
}
|
||||||
|
|
||||||
|
const iconInputPaddingClass = computed(() => {
|
||||||
|
const leftIcon = props.iconName && props.iconPosition === 'left'
|
||||||
|
const rightIcon = props.iconName && props.iconPosition === 'right'
|
||||||
|
const parts: string[] = []
|
||||||
|
if (leftIcon) parts.push('!pl-11')
|
||||||
|
if (rightIcon || props.addable) parts.push('!pr-10')
|
||||||
|
return parts.join(' ')
|
||||||
|
})
|
||||||
|
|
||||||
|
const disabled = computed(() => props.disabled)
|
||||||
|
|
||||||
|
const labelPositionClass = computed(() => {
|
||||||
|
if (props.iconName && props.iconPosition === 'left') return 'left-11'
|
||||||
|
return 'left-3'
|
||||||
|
})
|
||||||
|
|
||||||
|
const focusPaddingClass = computed(() => {
|
||||||
|
if (props.iconName && props.iconPosition === 'left') return 'focus:!pl-11'
|
||||||
|
return 'focus:pl-[11px]'
|
||||||
|
})
|
||||||
|
|
||||||
|
const iconPositionClass = computed(() => {
|
||||||
|
const sideClass = props.iconPosition === 'left' ? 'left-[10px]' : 'right-[10px]'
|
||||||
|
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
||||||
|
})
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return props.iconColor
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return props.iconColor
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.floating-label {
|
||||||
|
background: white;
|
||||||
|
padding: 0 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow-height {
|
||||||
|
transition: border-color 160ms ease, box-shadow 160ms ease, padding-top 160ms ease, padding-bottom 160ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grow-height:focus {
|
||||||
|
padding-top: 0.625rem;
|
||||||
|
padding-bottom: 0.625rem;
|
||||||
|
}
|
||||||
|
@media (prefers-reduced-motion: reduce) {
|
||||||
|
.grow-height { transition: none; }
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -19,6 +19,8 @@ type InputRichTextProps = {
|
|||||||
groupClass?: string
|
groupClass?: string
|
||||||
labelClass?: string
|
labelClass?: string
|
||||||
editorClass?: string
|
editorClass?: string
|
||||||
|
required?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputRichTextForTest = InputRichText as DefineComponent<InputRichTextProps>
|
const InputRichTextForTest = InputRichText as DefineComponent<InputRichTextProps>
|
||||||
@@ -155,6 +157,18 @@ describe('MalioInputRichText', () => {
|
|||||||
expect(editorContent.attributes('aria-describedby')).toBe('rt-aria-describedby')
|
expect(editorContent.attributes('aria-describedby')).toBe('rt-aria-describedby')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('expose aria-required quand required est vrai', async () => {
|
||||||
|
const wrapper = await mountComponent({required: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[aria-required="true"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'expose pas aria-required par défaut', async () => {
|
||||||
|
const wrapper = await mountComponent()
|
||||||
|
|
||||||
|
expect(wrapper.find('[aria-required="true"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
it('renders initial markdown content visually', async () => {
|
it('renders initial markdown content visually', async () => {
|
||||||
const wrapper = await mountComponent({modelValue: '## Mon titre\n\nUn paragraphe.'})
|
const wrapper = await mountComponent({modelValue: '## Mon titre\n\nUn paragraphe.'})
|
||||||
|
|
||||||
@@ -162,4 +176,35 @@ describe('MalioInputRichText', () => {
|
|||||||
expect(html).toContain('Mon titre')
|
expect(html).toContain('Mon titre')
|
||||||
expect(html).toContain('Un paragraphe.')
|
expect(html).toContain('Un paragraphe.')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Champ', required: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Champ'})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
:for="editorId"
|
:for="editorId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<!-- Mode lecture seule (rendu uniquement) -->
|
<!-- Mode lecture seule (rendu uniquement) -->
|
||||||
@@ -22,6 +22,7 @@
|
|||||||
v-else
|
v-else
|
||||||
:id="editorId"
|
:id="editorId"
|
||||||
:class="mergedEditorWrapperClass"
|
:class="mergedEditorWrapperClass"
|
||||||
|
:aria-required="required || undefined"
|
||||||
@click="focusEditor"
|
@click="focusEditor"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@@ -184,7 +185,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${editorId}-describedby`"
|
:id="`${editorId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -193,6 +194,7 @@
|
|||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'mt-1 text-xs ml-[2px]',
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ error || success || hint }}
|
{{ error || success || hint }}
|
||||||
@@ -211,6 +213,7 @@ import Color from '@tiptap/extension-color'
|
|||||||
import Highlight from '@tiptap/extension-highlight'
|
import Highlight from '@tiptap/extension-highlight'
|
||||||
import { Markdown } from 'tiptap-markdown'
|
import { Markdown } from 'tiptap-markdown'
|
||||||
import { twMerge } from 'tailwind-merge'
|
import { twMerge } from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
|
||||||
defineOptions({ name: 'MalioInputRichText', inheritAttrs: false })
|
defineOptions({ name: 'MalioInputRichText', inheritAttrs: false })
|
||||||
|
|
||||||
@@ -233,6 +236,8 @@ const props = withDefaults(
|
|||||||
groupClass?: string
|
groupClass?: string
|
||||||
labelClass?: string
|
labelClass?: string
|
||||||
editorClass?: string
|
editorClass?: string
|
||||||
|
required?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -250,6 +255,8 @@ const props = withDefaults(
|
|||||||
groupClass: '',
|
groupClass: '',
|
||||||
labelClass: '',
|
labelClass: '',
|
||||||
editorClass: '',
|
editorClass: '',
|
||||||
|
required: false,
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -279,10 +286,11 @@ const mergedLabelClass = computed(() =>
|
|||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: isFocused.value
|
: props.disabled
|
||||||
? 'text-m-primary'
|
? 'text-m-muted'
|
||||||
: 'text-m-text',
|
: isFocused.value
|
||||||
props.disabled ? 'text-black/60' : '',
|
? 'text-m-primary'
|
||||||
|
: 'text-m-text',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -308,6 +316,7 @@ const mergedReadonlyClass = computed(() =>
|
|||||||
'prose-headings:font-semibold prose-a:text-m-primary',
|
'prose-headings:font-semibold prose-a:text-m-primary',
|
||||||
'prose-code:rounded prose-code:bg-m-bg prose-code:px-1.5 prose-code:py-0.5 prose-code:before:content-none prose-code:after:content-none',
|
'prose-code:rounded prose-code:bg-m-bg prose-code:px-1.5 prose-code:py-0.5 prose-code:before:content-none prose-code:after:content-none',
|
||||||
'prose-pre:bg-m-text prose-pre:text-white',
|
'prose-pre:bg-m-text prose-pre:text-white',
|
||||||
|
'[&_pre_code]:bg-transparent [&_pre_code]:p-0 [&_pre_code]:text-inherit',
|
||||||
props.editorClass,
|
props.editorClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -486,7 +495,7 @@ onMounted(() => {
|
|||||||
],
|
],
|
||||||
editorProps: {
|
editorProps: {
|
||||||
attributes: {
|
attributes: {
|
||||||
class: 'prose prose-sm max-w-none w-full p-3 focus:outline-none prose-headings:font-semibold prose-a:text-m-primary prose-code:rounded prose-code:bg-m-bg prose-code:px-1.5 prose-code:py-0.5 prose-code:before:content-none prose-code:after:content-none prose-pre:bg-m-text prose-pre:text-white',
|
class: 'prose prose-sm max-w-none w-full p-3 focus:outline-none prose-headings:font-semibold prose-a:text-m-primary prose-code:rounded prose-code:bg-m-bg prose-code:px-1.5 prose-code:py-0.5 prose-code:before:content-none prose-code:after:content-none prose-pre:bg-m-text prose-pre:text-white [&_pre_code]:bg-transparent [&_pre_code]:p-0 [&_pre_code]:text-inherit',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
onUpdate: () => {
|
onUpdate: () => {
|
||||||
|
|||||||
@@ -21,8 +21,8 @@
|
|||||||
placeholder="_"
|
placeholder="_"
|
||||||
type="text"
|
type="text"
|
||||||
@input="onInput"
|
@input="onInput"
|
||||||
@focus="isFocused = true"
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
@blur="isFocused = false"
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
>
|
>
|
||||||
|
|
||||||
<label
|
<label
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<IconifyIcon
|
<IconifyIcon
|
||||||
@@ -39,18 +39,12 @@
|
|||||||
:width="iconSize"
|
:width="iconSize"
|
||||||
:height="iconSize"
|
:height="iconSize"
|
||||||
data-test="icon"
|
data-test="icon"
|
||||||
:class="[
|
:class="[iconStateClass, iconPositionClass]"
|
||||||
hasError
|
|
||||||
? 'text-m-danger'
|
|
||||||
: hasSuccess
|
|
||||||
? 'text-m-success' : iconColor,
|
|
||||||
iconPositionClass,
|
|
||||||
]"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -58,7 +52,8 @@
|
|||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'mt-1 text-xs ml-[2px] ',
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ hint || error || success }}
|
{{ hint || error || success }}
|
||||||
@@ -73,9 +68,13 @@ import {vMaska} from 'maska/vue'
|
|||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputText', inheritAttrs: false})
|
defineOptions({name: 'MalioInputText', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -100,6 +99,7 @@ const props = withDefaults(
|
|||||||
iconSize?: string | number
|
iconSize?: string | number
|
||||||
iconColor?: string
|
iconColor?: string
|
||||||
mask?: string | MaskInputOptions
|
mask?: string | MaskInputOptions
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -123,6 +123,7 @@ const props = withDefaults(
|
|||||||
iconSize: 24,
|
iconSize: 24,
|
||||||
iconColor: 'text-m-muted',
|
iconColor: 'text-m-muted',
|
||||||
mask: undefined,
|
mask: undefined,
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -134,10 +135,15 @@ const isFocused = ref(false)
|
|||||||
const inputId = computed(() => props.id?.toString() || `malio-input-text-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-input-text-${generatedId}`)
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
const shouldFloatLabel = computed(() => isFocused.value || currentValue.value.length > 0)
|
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success)
|
const hasSuccess = computed(() => !!props.success)
|
||||||
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'relative flex h-12 w-full items-center',
|
'relative flex h-12 w-full items-center',
|
||||||
@@ -146,30 +152,40 @@ const mergedGroupClass = computed(() =>
|
|||||||
)
|
)
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-input grow-height peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent focus:border-2 text-lg rounded-md',
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md',
|
||||||
isFilled.value ? 'border-black' : 'border-m-muted',
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-text',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
: 'focus:border-m-primary',
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
iconInputPaddingClass.value,
|
iconInputPaddingClass.value,
|
||||||
focusPaddingClass.value,
|
isReadonly.value ? '' : focusPaddingClass.value,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
const mergedLabelClass = computed(() =>
|
const mergedLabelClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
labelPositionClass.value,
|
labelPositionClass.value,
|
||||||
shouldFloatLabel.value ? '-translate-y-[1.25rem] peer-focus:-translate-y-[1.55rem] scale-90' : '',
|
shouldFloatLabel.value
|
||||||
disabled.value ? 'peer-[&:not(:placeholder-shown):not(:focus)]:text-black/60' : '',
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -202,7 +218,7 @@ const iconInputPaddingClass = computed(() => {
|
|||||||
const disabled = computed(() => props.disabled)
|
const disabled = computed(() => props.disabled)
|
||||||
|
|
||||||
const labelPositionClass = computed(() => {
|
const labelPositionClass = computed(() => {
|
||||||
if (props.iconName && props.iconPosition === 'left') return 'left-8'
|
if (props.iconName && props.iconPosition === 'left') return 'left-11'
|
||||||
return 'left-3'
|
return 'left-3'
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -215,6 +231,16 @@ const iconPositionClass = computed(() => {
|
|||||||
const sideClass = props.iconPosition === 'left' ? 'left-[10px]' : 'right-[10px]'
|
const sideClass = props.iconPosition === 'left' ? 'left-[10px]' : 'right-[10px]'
|
||||||
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
return `pointer-events-none absolute ${sideClass} top-1/2 -translate-y-1/2`
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return props.iconColor
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return props.iconColor
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ type InputTextAreaProps = {
|
|||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
rounded?: string
|
rounded?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputTextAreaForTest = InputTextArea as DefineComponent<InputTextAreaProps>
|
const InputTextAreaForTest = InputTextArea as DefineComponent<InputTextAreaProps>
|
||||||
@@ -149,4 +150,87 @@ describe('MalioInputTextArea', () => {
|
|||||||
expect(wrapper.find('p.text-m-success').exists()).toBe(false)
|
expect(wrapper.find('p.text-m-success').exists()).toBe(false)
|
||||||
expect(wrapper.get('p.text-m-danger').text()).toBe('Textarea error')
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Textarea error')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('renders as a single root element (works as a single grid item)', () => {
|
||||||
|
const host = document.createElement('div')
|
||||||
|
document.body.appendChild(host)
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {
|
||||||
|
attachTo: host,
|
||||||
|
})
|
||||||
|
|
||||||
|
// host > div[data-v-app] > component roots
|
||||||
|
const app = host.firstElementChild as HTMLElement
|
||||||
|
expect(app.children.length).toBe(1)
|
||||||
|
|
||||||
|
wrapper.unmount()
|
||||||
|
host.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies primary scrollbar class on focus', async () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest)
|
||||||
|
|
||||||
|
expect(wrapper.get('textarea').classes()).not.toContain('textarea-scrollbar-primary')
|
||||||
|
|
||||||
|
await wrapper.get('textarea').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('textarea').classes()).toContain('textarea-scrollbar-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('removes primary scrollbar class on blur', async () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest)
|
||||||
|
|
||||||
|
await wrapper.get('textarea').trigger('focus')
|
||||||
|
await wrapper.get('textarea').trigger('blur')
|
||||||
|
|
||||||
|
expect(wrapper.get('textarea').classes()).not.toContain('textarea-scrollbar-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', required: true}})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ'}})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de bleu', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', readonly: true}})
|
||||||
|
const field = wrapper.get('textarea')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu focus', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', readonly: true}})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
// En readonly, pas de couleur primary sur le label
|
||||||
|
expect(wrapper.get('label').classes()).not.toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', readonly: true, modelValue: 'du texte'}})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ'}})
|
||||||
|
const msg = wrapper.find('[data-test="message-line"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', reserveMessageSpace: false}})
|
||||||
|
expect(wrapper.find('[data-test="message-line"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mount(InputTextAreaForTest, {props: {label: 'Champ', reserveMessageSpace: false, error: 'Erreur'}})
|
||||||
|
const msg = wrapper.find('[data-test="message-line"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,89 +1,101 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div :class="mergedGroupClass">
|
||||||
class="relative w-full"
|
<div class="relative w-full flex-1">
|
||||||
>
|
<textarea
|
||||||
<textarea
|
:id="inputId"
|
||||||
:id="inputId"
|
:name="name"
|
||||||
:name="name"
|
|
||||||
|
|
||||||
:autocomplete="autocomplete"
|
:autocomplete="autocomplete"
|
||||||
class="floating-input peer w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent focus:border-2 overflow-auto"
|
class="floating-input peer w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent overflow-auto"
|
||||||
:class="[
|
:class="[
|
||||||
isFilled ? 'border-black' : 'border-m-muted',
|
isReadonly ? 'border-black' : (isFilled ? 'border-black' : 'border-m-muted'),
|
||||||
disabled ? 'cursor-not-allowed text-black/60 border-m-muted' : 'cursor-text',
|
disabled ? 'cursor-not-allowed text-black/60 border-m-muted' : (isReadonly ? 'cursor-default' : 'cursor-text'),
|
||||||
hasError
|
|
||||||
? 'border-m-danger focus:border-m-danger focus:pl-[11px]'
|
|
||||||
: hasSuccess
|
|
||||||
? 'border-m-success focus:border-m-success focus:pl-[11px]'
|
|
||||||
: 'focus:border-m-primary focus:pl-[11px]',
|
|
||||||
textInput,
|
|
||||||
showCounterComputed ? 'pb-6' : '',
|
|
||||||
rounded,
|
|
||||||
]"
|
|
||||||
:required="required"
|
|
||||||
:maxlength="maxLength"
|
|
||||||
:rows="rowsCount"
|
|
||||||
:disabled="disabled"
|
|
||||||
:value="currentValue"
|
|
||||||
:readonly="readonly"
|
|
||||||
:aria-invalid="hasError"
|
|
||||||
:aria-describedby="describedBy"
|
|
||||||
:style="textareaStyle"
|
|
||||||
v-bind="attrs"
|
|
||||||
placeholder="_"
|
|
||||||
@input="onInput"
|
|
||||||
@focus="isFocused = true"
|
|
||||||
@blur="isFocused = false"
|
|
||||||
/>
|
|
||||||
<label
|
|
||||||
v-if="label"
|
|
||||||
:for="inputId"
|
|
||||||
class="floating-label absolute left-3 top-2 mt-1 inline-block origin-left transition-transform duration-150 font-medium"
|
|
||||||
:class="[
|
|
||||||
shouldFloatLabel ? '-translate-y-[1.30rem] scale-90' : '',
|
|
||||||
disabled ? 'text-black/60' : '',
|
|
||||||
hasError
|
|
||||||
? 'text-m-danger'
|
|
||||||
: hasSuccess
|
|
||||||
? 'text-m-success'
|
|
||||||
: isFocused ? 'text-m-primary' : shouldFloatLabel ? 'text-black' : 'text-m-muted',
|
|
||||||
textLabel,
|
|
||||||
]"
|
|
||||||
>
|
|
||||||
{{ label }}
|
|
||||||
</label>
|
|
||||||
<span
|
|
||||||
v-if="showCounterComputed"
|
|
||||||
class="pointer-events-none absolute bottom-2 left-3 text-xs text-m-muted"
|
|
||||||
>
|
|
||||||
{{ currentLength }}/{{ maxLength }}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
v-if="hasError || hasSuccess || hint"
|
|
||||||
class="mt-1 flex items-center justify-between gap-2 text-xs"
|
|
||||||
>
|
|
||||||
<p
|
|
||||||
:id="`${inputId}-describedby`"
|
|
||||||
:class="[
|
|
||||||
hasError
|
hasError
|
||||||
? 'text-m-danger'
|
? 'border-m-danger focus:border-m-danger'
|
||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'border-m-success focus:border-m-success'
|
||||||
: 'text-m-muted',
|
: isReadonly ? '' : 'focus:border-m-primary',
|
||||||
'ml-[2px]',
|
isReadonly ? '' : (isFocused ? 'textarea-scrollbar-primary' : ''),
|
||||||
|
textInput,
|
||||||
|
showCounterComputed ? 'pb-6' : '',
|
||||||
|
rounded,
|
||||||
|
keyboardFocused ? 'm-focus-ring-kbd' : '',
|
||||||
]"
|
]"
|
||||||
|
:required="required"
|
||||||
|
:maxlength="maxLength"
|
||||||
|
:rows="rowsCount"
|
||||||
|
:disabled="disabled"
|
||||||
|
:value="currentValue"
|
||||||
|
:readonly="readonly"
|
||||||
|
:aria-invalid="hasError"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
:style="textareaStyle"
|
||||||
|
v-bind="attrs"
|
||||||
|
placeholder="_"
|
||||||
|
@input="onInput"
|
||||||
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="inputId"
|
||||||
|
class="floating-label absolute left-3 top-2 mt-1 inline-block origin-left transition-transform duration-150 font-medium"
|
||||||
|
:class="[
|
||||||
|
shouldFloatLabel ? '-translate-y-[1.30rem] scale-90' : '',
|
||||||
|
hasError
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess
|
||||||
|
? 'text-m-success'
|
||||||
|
: disabled
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly
|
||||||
|
? (isFilled ? 'text-black' : 'text-m-muted')
|
||||||
|
: (isFocused ? 'text-m-primary' : shouldFloatLabel ? 'text-black' : 'text-m-muted'),
|
||||||
|
textLabel,
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
|
</label>
|
||||||
|
<span
|
||||||
|
v-if="showCounterComputed"
|
||||||
|
class="pointer-events-none absolute bottom-2 left-3 text-xs text-m-muted"
|
||||||
|
>
|
||||||
|
{{ currentLength }}/{{ maxLength }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
|
data-test="message-line"
|
||||||
|
class="mt-1 flex items-center justify-between gap-2 text-xs"
|
||||||
|
:class="reserveMessageSpace ? 'min-h-[1rem]' : ''"
|
||||||
>
|
>
|
||||||
{{ error || success || hint }}
|
<p
|
||||||
</p>
|
:id="`${inputId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess
|
||||||
|
? 'text-m-success'
|
||||||
|
: 'text-m-muted',
|
||||||
|
'ml-[2px]',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ error || success || hint }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputTextArea', inheritAttrs: false})
|
defineOptions({name: 'MalioInputTextArea', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -108,6 +120,8 @@ const props = withDefaults(
|
|||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
rounded?: string
|
rounded?: string
|
||||||
|
groupClass?: string
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
|
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
@@ -133,9 +147,17 @@ const props = withDefaults(
|
|||||||
maxResizeWidth: 640,
|
maxResizeWidth: 640,
|
||||||
minResizeHeight: 40,
|
minResizeHeight: 40,
|
||||||
maxResizeHeight: 320,
|
maxResizeHeight: 320,
|
||||||
|
groupClass: '',
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
// pt-1 (4px) aligne le haut de la textarea avec les inputs floating-label,
|
||||||
|
// qui centrent un champ de 40px dans un groupe h-12 (≈ 4px de décalage en haut).
|
||||||
|
twMerge('flex flex-col w-full pt-1', props.groupClass),
|
||||||
|
)
|
||||||
|
|
||||||
const attrs = useAttrs()
|
const attrs = useAttrs()
|
||||||
const generatedId = useId()
|
const generatedId = useId()
|
||||||
const localValue = ref('')
|
const localValue = ref('')
|
||||||
@@ -144,9 +166,15 @@ const isFocused = ref(false)
|
|||||||
const inputId = computed(() => props.id?.toString() || `malio-input-textarea-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-input-textarea-${generatedId}`)
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
const currentValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
const shouldFloatLabel = computed(() => isFocused.value || currentValue.value.length > 0)
|
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
|
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentValue.value.length > 0,
|
||||||
|
)
|
||||||
const rowsCount = computed(() => Math.max(1, Number(props.size || 3)))
|
const rowsCount = computed(() => Math.max(1, Number(props.size || 3)))
|
||||||
const currentLength = computed(() => (currentValue.value ?? '').length)
|
const currentLength = computed(() => (currentValue.value ?? '').length)
|
||||||
const showCounterComputed = computed(() =>
|
const showCounterComputed = computed(() =>
|
||||||
@@ -160,7 +188,6 @@ const textareaStyle = computed(() => ({
|
|||||||
minHeight: toCssSize(props.minResizeHeight),
|
minHeight: toCssSize(props.minResizeHeight),
|
||||||
maxHeight: toCssSize(props.maxResizeHeight),
|
maxHeight: toCssSize(props.maxResizeHeight),
|
||||||
}))
|
}))
|
||||||
const isFilled = computed(() => currentValue.value.trim().length > 0)
|
|
||||||
const describedBy = computed(() =>
|
const describedBy = computed(() =>
|
||||||
(hasError.value || hasSuccess.value || !!props.hint) ? `${inputId.value}-describedby` : undefined,
|
(hasError.value || hasSuccess.value || !!props.hint) ? `${inputId.value}-describedby` : undefined,
|
||||||
)
|
)
|
||||||
@@ -183,4 +210,8 @@ const onInput = (event: Event) => {
|
|||||||
background: white;
|
background: white;
|
||||||
padding: 0 0.25rem;
|
padding: 0 0.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.textarea-scrollbar-primary {
|
||||||
|
scrollbar-color: rgb(var(--m-primary)) transparent;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import {describe, expect, it} from 'vitest'
|
import {describe, expect, it, vi} from 'vitest'
|
||||||
import {mount} from '@vue/test-utils'
|
import {mount} from '@vue/test-utils'
|
||||||
import type {DefineComponent} from 'vue'
|
import type {DefineComponent} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
@@ -12,11 +12,14 @@ type InputUploadProps = {
|
|||||||
labelClass?: string
|
labelClass?: string
|
||||||
groupClass?: string
|
groupClass?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
displayIcon?: boolean
|
displayIcon?: boolean
|
||||||
accept?: string
|
accept?: string
|
||||||
|
required?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const InputUploadForTest = InputUpload as DefineComponent<InputUploadProps>
|
const InputUploadForTest = InputUpload as DefineComponent<InputUploadProps>
|
||||||
@@ -167,9 +170,94 @@ describe('MalioInputUpload', () => {
|
|||||||
expect(wrapper.get('input[type="text"]').attributes('aria-invalid')).toBe('false')
|
expect(wrapper.get('input[type="text"]').attributes('aria-invalid')).toBe('false')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('expose aria-required sur le champ visible quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.get('input[type="text"]').attributes('aria-required')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
it('passes accept attribute to file input', () => {
|
it('passes accept attribute to file input', () => {
|
||||||
const wrapper = mountComponent({accept: '.pdf,.doc'})
|
const wrapper = mountComponent({accept: '.pdf,.doc'})
|
||||||
|
|
||||||
expect(wrapper.get('input[type="file"]').attributes('accept')).toBe('.pdf,.doc')
|
expect(wrapper.get('input[type="file"]').attributes('accept')).toBe('.pdf,.doc')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('shows primary icon color on focus', async () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
|
||||||
|
await wrapper.get('input[type="text"]').trigger('focus')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black icon color when filled and unfocused', () => {
|
||||||
|
const wrapper = mountComponent({modelValue: 'document.pdf'})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', required: true})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même vide, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const field = wrapper.get('input[type="text"]')
|
||||||
|
expect(field.classes()).toContain('border-black')
|
||||||
|
expect(field.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(field.classes()).not.toContain('grow-height')
|
||||||
|
expect(field.classes()).not.toContain('focus:border-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const label = wrapper.get('label')
|
||||||
|
expect(label.classes()).not.toContain('peer-focus:text-m-primary')
|
||||||
|
expect(label.classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : icône en text-m-muted', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly rempli : label noir + icône noire', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true, modelValue: 'fichier.pdf'})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="icon"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly empêche l\'ouverture du sélecteur de fichier', async () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', readonly: true})
|
||||||
|
const fileInput = wrapper.get('input[type="file"]').element as HTMLInputElement
|
||||||
|
const clickSpy = vi.spyOn(fileInput, 'click')
|
||||||
|
await wrapper.get('input[type="text"]').trigger('click')
|
||||||
|
expect(clickSpy).not.toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mountComponent({label: 'Champ', reserveMessageSpace: false, error: 'Erreur'})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
:accept="accept"
|
:accept="accept"
|
||||||
class="hidden"
|
class="hidden"
|
||||||
:disabled="disabled"
|
:disabled="disabled"
|
||||||
|
:required="required"
|
||||||
@change="onFileChange"
|
@change="onFileChange"
|
||||||
>
|
>
|
||||||
|
|
||||||
@@ -19,13 +20,16 @@
|
|||||||
:value="currentDisplayValue"
|
:value="currentDisplayValue"
|
||||||
:readonly="true"
|
:readonly="true"
|
||||||
:aria-invalid="!!error"
|
:aria-invalid="!!error"
|
||||||
|
:aria-required="required || undefined"
|
||||||
:aria-describedby="describedBy"
|
:aria-describedby="describedBy"
|
||||||
v-bind="attrs"
|
v-bind="attrs"
|
||||||
placeholder="_"
|
placeholder="_"
|
||||||
type="text"
|
type="text"
|
||||||
@click="openFilePicker"
|
@click="openFilePicker"
|
||||||
@focus="isFocused = true"
|
@keydown.enter.prevent="openFilePicker"
|
||||||
@blur="isFocused = false"
|
@keydown.space.prevent="openFilePicker"
|
||||||
|
@focus="isFocused = true; onKbdFocus()"
|
||||||
|
@blur="isFocused = false; onKbdBlur()"
|
||||||
>
|
>
|
||||||
|
|
||||||
<label
|
<label
|
||||||
@@ -33,27 +37,40 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<IconifyIcon
|
<div
|
||||||
v-if="displayIcon"
|
v-if="displayIcon || showClear"
|
||||||
icon="mdi:cloud-arrow-up-outline"
|
class="absolute right-[10px] top-1/2 flex -translate-y-1/2 items-center gap-1"
|
||||||
:width="24"
|
>
|
||||||
:height="24"
|
<button
|
||||||
data-test="icon"
|
v-if="showClear"
|
||||||
:class="[
|
type="button"
|
||||||
hasError
|
data-test="clear"
|
||||||
? 'text-m-danger'
|
class="m-focus-ring rounded-malio text-m-muted hover:text-m-primary"
|
||||||
: hasSuccess
|
aria-label="Retirer le fichier"
|
||||||
? 'text-m-success' : 'text-m-muted',
|
@click.stop="onClear"
|
||||||
'pointer-events-none absolute right-[10px] top-1/2 -translate-y-1/2',
|
>
|
||||||
]"
|
<IconifyIcon
|
||||||
/>
|
icon="mdi:close"
|
||||||
|
:width="16"
|
||||||
|
:height="16"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<IconifyIcon
|
||||||
|
v-if="displayIcon"
|
||||||
|
icon="mdi:cloud-arrow-up-outline"
|
||||||
|
:width="24"
|
||||||
|
:height="24"
|
||||||
|
data-test="icon"
|
||||||
|
:class="[iconStateClass, 'pointer-events-none']"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<p
|
<p
|
||||||
v-if="hint || hasError || hasSuccess"
|
v-if="reserveMessageSpace || hint || error || success"
|
||||||
:id="`${inputId}-describedby`"
|
:id="`${inputId}-describedby`"
|
||||||
:class="[
|
:class="[
|
||||||
hasError
|
hasError
|
||||||
@@ -61,7 +78,8 @@
|
|||||||
: hasSuccess
|
: hasSuccess
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'text-m-muted',
|
: 'text-m-muted',
|
||||||
'mt-1 text-xs ml-[2px] ',
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
reserveMessageSpace ? 'min-h-[1rem]' : '',
|
||||||
]"
|
]"
|
||||||
>
|
>
|
||||||
{{ hint || error || success }}
|
{{ hint || error || success }}
|
||||||
@@ -74,9 +92,13 @@
|
|||||||
import {computed, ref, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import { Icon as IconifyIcon } from '@iconify/vue'
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
import {useKbdFocusRing} from '../shared/useKbdFocusRing'
|
||||||
|
|
||||||
defineOptions({name: 'MalioInputUpload', inheritAttrs: false})
|
defineOptions({name: 'MalioInputUpload', inheritAttrs: false})
|
||||||
|
|
||||||
|
const {keyboardFocused, onFocus: onKbdFocus, onBlur: onKbdBlur} = useKbdFocusRing()
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
id?: string
|
id?: string
|
||||||
@@ -86,11 +108,15 @@ const props = withDefaults(
|
|||||||
labelClass?: string
|
labelClass?: string
|
||||||
groupClass?: string
|
groupClass?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
displayIcon?: boolean
|
displayIcon?: boolean
|
||||||
accept?: string
|
accept?: string
|
||||||
|
required?: boolean
|
||||||
|
clearable?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
id: '',
|
id: '',
|
||||||
@@ -100,11 +126,15 @@ const props = withDefaults(
|
|||||||
labelClass: '',
|
labelClass: '',
|
||||||
groupClass: '',
|
groupClass: '',
|
||||||
disabled: false,
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
hint: '',
|
hint: '',
|
||||||
error: '',
|
error: '',
|
||||||
success: '',
|
success: '',
|
||||||
displayIcon: true,
|
displayIcon: true,
|
||||||
accept: '',
|
accept: '',
|
||||||
|
required: false,
|
||||||
|
clearable: false,
|
||||||
|
reserveMessageSpace: true,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -117,10 +147,16 @@ const fileInputRef = ref<HTMLInputElement | null>(null)
|
|||||||
const inputId = computed(() => props.id?.toString() || `malio-input-upload-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-input-upload-${generatedId}`)
|
||||||
const isControlled = computed(() => props.modelValue !== undefined)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
const currentDisplayValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
const currentDisplayValue = computed(() => (isControlled.value ? (props.modelValue ?? '') : localValue.value))
|
||||||
const shouldFloatLabel = computed(() => isFocused.value || currentDisplayValue.value.length > 0)
|
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success)
|
const hasSuccess = computed(() => !!props.success)
|
||||||
const isFilled = computed(() => currentDisplayValue.value.trim().length > 0)
|
const isFilled = computed(() => currentDisplayValue.value.trim().length > 0)
|
||||||
|
const disabled = computed(() => props.disabled)
|
||||||
|
const isReadonly = computed(() => props.readonly && !props.disabled)
|
||||||
|
const shouldFloatLabel = computed(() =>
|
||||||
|
isReadonly.value
|
||||||
|
? isFilled.value
|
||||||
|
: isFocused.value || currentDisplayValue.value.length > 0,
|
||||||
|
)
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'relative flex h-12 w-full items-center',
|
'relative flex h-12 w-full items-center',
|
||||||
@@ -129,16 +165,24 @@ const mergedGroupClass = computed(() =>
|
|||||||
)
|
)
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'floating-input grow-height peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent focus:border-2 text-lg rounded-md',
|
'floating-input peer min-h-[40px] w-full border bg-white pl-3 pr-3 py-1 outline-none placeholder:text-transparent text-lg rounded-md cursor-pointer',
|
||||||
isFilled.value ? 'border-black' : 'border-m-muted',
|
keyboardFocused.value ? 'm-focus-ring-kbd' : '',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : 'cursor-pointer',
|
isReadonly.value ? '' : 'grow-height',
|
||||||
|
isReadonly.value
|
||||||
|
? 'border-black'
|
||||||
|
: isFilled.value ? 'border-black' : 'border-m-muted',
|
||||||
|
disabled.value ? 'text-black/60 [&:not(:placeholder-shown)]:border-m-muted border-m-muted' : '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
? 'border-m-danger focus:border-m-danger [&:not(:placeholder-shown)]:border-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
? 'border-m-success focus:border-m-success [&:not(:placeholder-shown)]:border-m-success'
|
||||||
: 'focus:border-m-primary',
|
: isReadonly.value ? '' : 'focus:border-m-primary',
|
||||||
props.displayIcon ? '!pr-10' : '',
|
showClear.value
|
||||||
'focus:pl-[11px]',
|
? (props.displayIcon ? '!pr-16' : '!pr-10')
|
||||||
|
: (props.displayIcon ? '!pr-10' : ''),
|
||||||
|
isReadonly.value ? '' : 'focus:pl-[11px]',
|
||||||
|
isReadonly.value ? 'cursor-default' : '',
|
||||||
|
disabled.value ? 'cursor-not-allowed' : '',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -146,13 +190,18 @@ const mergedLabelClass = computed(() =>
|
|||||||
twMerge(
|
twMerge(
|
||||||
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
'floating-label absolute top-2 mt-[5px] inline-block origin-left transition-transform duration-150 font-medium text-sm',
|
||||||
'left-3',
|
'left-3',
|
||||||
shouldFloatLabel.value ? '-translate-y-[1.25rem] peer-focus:-translate-y-[1.55rem] scale-90' : '',
|
shouldFloatLabel.value
|
||||||
disabled.value ? 'peer-[&:not(:placeholder-shown):not(:focus)]:text-black/60' : '',
|
? `-translate-y-[1.25rem] scale-90${isReadonly.value ? '' : ' peer-focus:-translate-y-[1.55rem]'}`
|
||||||
|
: '',
|
||||||
hasError.value
|
hasError.value
|
||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess.value
|
: hasSuccess.value
|
||||||
? 'text-m-success'
|
? 'text-m-success'
|
||||||
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
: disabled.value
|
||||||
|
? 'text-m-muted'
|
||||||
|
: isReadonly.value
|
||||||
|
? isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
: 'peer-placeholder-shown:text-m-muted peer-[&:not(:placeholder-shown):not(:focus)]:text-black peer-focus:text-m-primary',
|
||||||
props.labelClass,
|
props.labelClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -168,10 +217,23 @@ const describedBy = computed(() => {
|
|||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(event: 'update:modelValue', value: string): void
|
(event: 'update:modelValue', value: string): void
|
||||||
(event: 'file-selected', file: File): void
|
(event: 'file-selected', file: File): void
|
||||||
|
(event: 'clear'): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
|
const showClear = computed(() =>
|
||||||
|
props.clearable && isFilled.value && !props.disabled && !isReadonly.value,
|
||||||
|
)
|
||||||
|
|
||||||
|
const onClear = () => {
|
||||||
|
if (props.disabled || isReadonly.value) return
|
||||||
|
if (!isControlled.value) localValue.value = ''
|
||||||
|
if (fileInputRef.value) fileInputRef.value.value = ''
|
||||||
|
emit('update:modelValue', '')
|
||||||
|
emit('clear')
|
||||||
|
}
|
||||||
|
|
||||||
const openFilePicker = () => {
|
const openFilePicker = () => {
|
||||||
if (props.disabled) return
|
if (props.disabled || props.readonly) return
|
||||||
fileInputRef.value?.click()
|
fileInputRef.value?.click()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,7 +250,15 @@ const onFileChange = (event: Event) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const disabled = computed(() => props.disabled)
|
const iconStateClass = computed(() => {
|
||||||
|
if (hasError.value) return 'text-m-danger'
|
||||||
|
if (hasSuccess.value) return 'text-m-success'
|
||||||
|
if (disabled.value) return 'text-m-muted'
|
||||||
|
if (isReadonly.value) return isFilled.value ? 'text-black' : 'text-m-muted'
|
||||||
|
if (isFocused.value) return 'text-m-primary'
|
||||||
|
if (isFilled.value) return 'text-black'
|
||||||
|
return 'text-m-muted'
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
|||||||
@@ -0,0 +1,74 @@
|
|||||||
|
import {describe, expect, it} from 'vitest'
|
||||||
|
import {normalizeAmount, formatGroupedAmount, countSignificant, caretFromSignificant} from './amountFormat'
|
||||||
|
|
||||||
|
describe('normalizeAmount', () => {
|
||||||
|
it('garde le point décimal', () => {
|
||||||
|
expect(normalizeAmount('12.5')).toBe('12.5')
|
||||||
|
})
|
||||||
|
it('convertit la virgule en point et nettoie', () => {
|
||||||
|
expect(normalizeAmount('0012,345abc')).toBe('12.34')
|
||||||
|
})
|
||||||
|
it('normalise une décimale en tête', () => {
|
||||||
|
expect(normalizeAmount(',5')).toBe('0.5')
|
||||||
|
})
|
||||||
|
it('retire les espaces', () => {
|
||||||
|
expect(normalizeAmount('1 234 567')).toBe('1234567')
|
||||||
|
})
|
||||||
|
it('limite à 2 décimales', () => {
|
||||||
|
expect(normalizeAmount('1234.567')).toBe('1234.56')
|
||||||
|
})
|
||||||
|
it('garde une décimale en cours de saisie', () => {
|
||||||
|
expect(normalizeAmount('12.')).toBe('12.')
|
||||||
|
})
|
||||||
|
it('renvoie une chaîne vide pour une saisie non numérique', () => {
|
||||||
|
expect(normalizeAmount('abc')).toBe('')
|
||||||
|
})
|
||||||
|
it('garde un zéro seul', () => {
|
||||||
|
expect(normalizeAmount('0')).toBe('0')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('formatGroupedAmount', () => {
|
||||||
|
it('groupe la partie entière par 3 avec des espaces', () => {
|
||||||
|
expect(formatGroupedAmount('1234567')).toBe('1 234 567')
|
||||||
|
})
|
||||||
|
it('utilise la virgule comme séparateur décimal', () => {
|
||||||
|
expect(formatGroupedAmount('1234.56')).toBe('1 234,56')
|
||||||
|
})
|
||||||
|
it('affiche une virgule pour une décimale en cours', () => {
|
||||||
|
expect(formatGroupedAmount('12.')).toBe('12,')
|
||||||
|
})
|
||||||
|
it('gère les valeurs sous 1000 sans séparateur', () => {
|
||||||
|
expect(formatGroupedAmount('12')).toBe('12')
|
||||||
|
})
|
||||||
|
it('groupe avec une décimale en tête', () => {
|
||||||
|
expect(formatGroupedAmount('0.5')).toBe('0,5')
|
||||||
|
})
|
||||||
|
it('renvoie une chaîne vide pour une chaîne vide', () => {
|
||||||
|
expect(formatGroupedAmount('')).toBe('')
|
||||||
|
})
|
||||||
|
it('garde un zéro seul', () => {
|
||||||
|
expect(formatGroupedAmount('0')).toBe('0')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('countSignificant', () => {
|
||||||
|
it('compte les caractères hors espaces à gauche du curseur', () => {
|
||||||
|
expect(countSignificant('1 234', 5)).toBe(4)
|
||||||
|
})
|
||||||
|
it('ignore un espace juste avant le curseur', () => {
|
||||||
|
expect(countSignificant('1 234', 2)).toBe(1)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('caretFromSignificant', () => {
|
||||||
|
it('place le curseur après le n-ième caractère significatif', () => {
|
||||||
|
expect(caretFromSignificant('1 234 567', 4)).toBe(5)
|
||||||
|
})
|
||||||
|
it('place le curseur en fin si on dépasse', () => {
|
||||||
|
expect(caretFromSignificant('1 234', 10)).toBe(5)
|
||||||
|
})
|
||||||
|
it('place le curseur au début pour 0 caractère significatif', () => {
|
||||||
|
expect(caretFromSignificant('1 234', 0)).toBe(0)
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
// Parse : texte saisi (espaces, virgule, caractères parasites) → chaîne numérique propre.
|
||||||
|
export const normalizeAmount = (value: string): string => {
|
||||||
|
const sanitizedValue = value
|
||||||
|
.replace(/\s+/g, '')
|
||||||
|
.replace(/,/g, '.')
|
||||||
|
.replace(/[^\d.]/g, '')
|
||||||
|
const [integerPartRaw = '', ...decimalParts] = sanitizedValue.split('.')
|
||||||
|
const integerPart = integerPartRaw.replace(/^0+(?=\d)/, '')
|
||||||
|
const decimalPart = decimalParts.join('').slice(0, 2)
|
||||||
|
|
||||||
|
if (sanitizedValue.includes('.')) {
|
||||||
|
return `${integerPart || '0'}.${decimalPart}`
|
||||||
|
}
|
||||||
|
|
||||||
|
return integerPart
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format : modèle propre (point décimal) → affichage groupé FR (espaces + virgule).
|
||||||
|
export const formatGroupedAmount = (model: string): string => {
|
||||||
|
if (model === '') return ''
|
||||||
|
const hasDot = model.includes('.')
|
||||||
|
const [integerPart = '', decimalPart = ''] = model.split('.')
|
||||||
|
const groupedInteger = integerPart.replace(/\B(?=(\d{3})+(?!\d))/g, ' ')
|
||||||
|
return hasDot ? `${groupedInteger},${decimalPart}` : groupedInteger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nombre de caractères significatifs (hors espaces de groupement) à gauche d'une position.
|
||||||
|
export const countSignificant = (str: string, upTo: number): number =>
|
||||||
|
str.slice(0, upTo).replace(/ /g, '').length
|
||||||
|
|
||||||
|
// Position de curseur après le n-ième caractère significatif dans la chaîne affichée.
|
||||||
|
export const caretFromSignificant = (display: string, sig: number): number => {
|
||||||
|
if (sig <= 0) return 0
|
||||||
|
let seen = 0
|
||||||
|
for (let i = 0; i < display.length; i++) {
|
||||||
|
if (display[i] !== ' ') seen++
|
||||||
|
if (seen >= sig) return i + 1
|
||||||
|
}
|
||||||
|
return display.length
|
||||||
|
}
|
||||||
@@ -0,0 +1,320 @@
|
|||||||
|
import { afterEach, describe, expect, it } from 'vitest'
|
||||||
|
import { enableAutoUnmount, mount } from '@vue/test-utils'
|
||||||
|
import type { DefineComponent } from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import Modal from './Modal.vue'
|
||||||
|
|
||||||
|
type ModalProps = {
|
||||||
|
id?: string
|
||||||
|
modelValue?: boolean
|
||||||
|
showClose?: boolean
|
||||||
|
dismissable?: boolean
|
||||||
|
closeOnEscape?: boolean
|
||||||
|
ariaLabel?: string
|
||||||
|
modalClass?: string
|
||||||
|
overlayClass?: string
|
||||||
|
headerClass?: string
|
||||||
|
bodyClass?: string
|
||||||
|
footerClass?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const ModalForTest = Modal as DefineComponent<ModalProps>
|
||||||
|
|
||||||
|
function mountComponent(props: ModalProps = {}, slots?: Record<string, string>) {
|
||||||
|
return mount(ModalForTest, {
|
||||||
|
props,
|
||||||
|
slots,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('MalioModal', () => {
|
||||||
|
enableAutoUnmount(afterEach)
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render when modelValue is false', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: false })
|
||||||
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the panel when modelValue is true', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('centers the modal (items-center justify-center)', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
const root = wrapper.find('.fixed')
|
||||||
|
expect(root.classes()).toContain('items-center')
|
||||||
|
expect(root.classes()).toContain('justify-center')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders default slot in the body', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true },
|
||||||
|
{ default: '<p data-test="content">Contenu</p>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="body"] [data-test="content"]').text()).toBe('Contenu')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('works in uncontrolled mode (defaults closed)', () => {
|
||||||
|
const wrapper = mountComponent()
|
||||||
|
expect(wrapper.find('[data-test="panel"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses custom id when provided', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, id: 'my-modal' })
|
||||||
|
expect(wrapper.find('.fixed').attributes('id')).toBe('my-modal')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an id when not provided', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('.fixed').attributes('id')).toMatch(/^malio-modal-/)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has role="dialog" and aria-modal on the panel', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
|
expect(panel.attributes('role')).toBe('dialog')
|
||||||
|
expect(panel.attributes('aria-modal')).toBe('true')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies modalClass to the panel', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, modalClass: 'max-w-2xl' })
|
||||||
|
expect(wrapper.find('[data-test="panel"]').classes()).toContain('max-w-2xl')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the #header slot inside the header bar', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true },
|
||||||
|
{ header: '<h2 data-test="title">Titre</h2>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="header"] [data-test="title"]').text()).toBe('Titre')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the header bar when showClose is true even without #header', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="header"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render the header bar when no #header and showClose is false', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, showClose: false })
|
||||||
|
expect(wrapper.find('[data-test="header"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows the close button by default', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the close button when showClose is false', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, showClose: false },
|
||||||
|
{ header: '<h2>Titre</h2>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="close-button"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('close button renders mdi:cancel-bold icon', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
const icon = wrapper.findComponent(IconifyIcon)
|
||||||
|
expect(icon.props('icon')).toBe('mdi:cancel-bold')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('close button has aria-label "Fermer"', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="close-button"]').attributes('aria-label')).toBe('Fermer')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue false and close on close button click', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="close-button"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-labelledby to the header id when #header is provided', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, id: 'test-modal' },
|
||||||
|
{ header: '<h2>Titre</h2>' },
|
||||||
|
)
|
||||||
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
|
expect(panel.attributes('aria-labelledby')).toBe('test-modal-header')
|
||||||
|
expect(wrapper.find('[data-test="header-content"]').attributes('id')).toBe('test-modal-header')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-label from ariaLabel when no #header is provided', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, ariaLabel: 'Boîte de dialogue' })
|
||||||
|
const panel = wrapper.find('[data-test="panel"]')
|
||||||
|
expect(panel.attributes('aria-label')).toBe('Boîte de dialogue')
|
||||||
|
expect(panel.attributes('aria-labelledby')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies headerClass to the header bar', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, headerClass: 'bg-m-primary' })
|
||||||
|
expect(wrapper.find('[data-test="header"]').classes()).toContain('bg-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the #footer slot in a footer pinned below the body', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true },
|
||||||
|
{ footer: '<button data-test="save">Enregistrer</button>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="body"] [data-test="footer"]').exists()).toBe(false)
|
||||||
|
expect(wrapper.find('[data-test="footer"] [data-test="save"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render the footer when no #footer slot', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
expect(wrapper.find('[data-test="footer"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies bodyClass to the body', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, bodyClass: 'px-10' })
|
||||||
|
expect(wrapper.find('[data-test="body"]').classes()).toContain('px-10')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies footerClass to the footer', () => {
|
||||||
|
const wrapper = mountComponent(
|
||||||
|
{ modelValue: true, footerClass: 'justify-end' },
|
||||||
|
{ footer: '<span>pied</span>' },
|
||||||
|
)
|
||||||
|
expect(wrapper.find('[data-test="footer"]').classes()).toContain('justify-end')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('emits update:modelValue false and close on backdrop click (dismissable)', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="backdrop"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not close on backdrop click when dismissable is false', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, dismissable: false })
|
||||||
|
await wrapper.find('[data-test="backdrop"]').trigger('click')
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies overlayClass to the backdrop', () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, overlayClass: 'bg-black/70' })
|
||||||
|
expect(wrapper.find('[data-test="backdrop"]').classes()).toContain('bg-black/70')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('closes on Escape key when closeOnEscape is true', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true })
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Escape' })
|
||||||
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual([false])
|
||||||
|
expect(wrapper.emitted('close')).toHaveLength(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not close on Escape when closeOnEscape is false', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: true, closeOnEscape: false })
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Escape' })
|
||||||
|
expect(wrapper.emitted('update:modelValue')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('locks body scroll when opened and restores it when closed', async () => {
|
||||||
|
const wrapper = mountComponent({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
await wrapper.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('moves focus into the panel when opened', async () => {
|
||||||
|
const wrapper = mount(ModalForTest, {
|
||||||
|
props: { modelValue: false, showClose: false },
|
||||||
|
slots: { default: '<button data-test="first">OK</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const first = wrapper.find('[data-test="first"]').element
|
||||||
|
expect(document.activeElement).toBe(first)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('restores focus to the trigger when closed', async () => {
|
||||||
|
const trigger = document.createElement('button')
|
||||||
|
document.body.appendChild(trigger)
|
||||||
|
trigger.focus()
|
||||||
|
expect(document.activeElement).toBe(trigger)
|
||||||
|
|
||||||
|
const wrapper = mount(ModalForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
slots: { default: '<button>OK</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.setProps({ modelValue: true })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
await wrapper.setProps({ modelValue: false })
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
expect(document.activeElement).toBe(trigger)
|
||||||
|
|
||||||
|
wrapper.unmount()
|
||||||
|
trigger.remove()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wraps focus to the first element when Tab is pressed on the last element', async () => {
|
||||||
|
const wrapper = mount(ModalForTest, {
|
||||||
|
props: { modelValue: true, showClose: false },
|
||||||
|
slots: { default: '<button data-test="btn1">First</button><button data-test="btn2">Last</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const last = wrapper.find('[data-test="btn2"]').element as HTMLElement
|
||||||
|
last.focus()
|
||||||
|
expect(document.activeElement).toBe(last)
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Tab' })
|
||||||
|
expect(document.activeElement).toBe(wrapper.find('[data-test="btn1"]').element)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('wraps focus to the last element when Shift+Tab is pressed on the first element', async () => {
|
||||||
|
const wrapper = mount(ModalForTest, {
|
||||||
|
props: { modelValue: true, showClose: false },
|
||||||
|
slots: { default: '<button data-test="btn1">First</button><button data-test="btn2">Last</button>' },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
const first = wrapper.find('[data-test="btn1"]').element as HTMLElement
|
||||||
|
first.focus()
|
||||||
|
expect(document.activeElement).toBe(first)
|
||||||
|
await wrapper.find('[data-test="panel"]').trigger('keydown', { key: 'Tab', shiftKey: true })
|
||||||
|
expect(document.activeElement).toBe(wrapper.find('[data-test="btn2"]').element)
|
||||||
|
wrapper.unmount()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not release body scroll-lock when one stacked modal closes while another is still open', async () => {
|
||||||
|
const wrapperA = mount(ModalForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
const wrapperB = mount(ModalForTest, {
|
||||||
|
props: { modelValue: false },
|
||||||
|
attachTo: document.body,
|
||||||
|
global: { stubs: { Teleport: true } },
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapperA.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
await wrapperB.setProps({ modelValue: true })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
await wrapperB.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('hidden')
|
||||||
|
|
||||||
|
await wrapperA.setProps({ modelValue: false })
|
||||||
|
expect(document.body.style.overflow).toBe('')
|
||||||
|
})
|
||||||
|
})
|
||||||
@@ -0,0 +1,279 @@
|
|||||||
|
<template>
|
||||||
|
<Teleport to="body">
|
||||||
|
<Transition
|
||||||
|
name="modal"
|
||||||
|
appear
|
||||||
|
@after-leave="isRendered = false"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-if="isRendered && isOpen"
|
||||||
|
:id="componentId"
|
||||||
|
class="fixed inset-0 z-50 flex items-center justify-center p-4"
|
||||||
|
v-bind="attrs"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
:class="twMerge('absolute inset-0 bg-black/40', overlayClass)"
|
||||||
|
data-test="backdrop"
|
||||||
|
@click="onBackdropClick"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div
|
||||||
|
ref="panelRef"
|
||||||
|
:class="twMerge(
|
||||||
|
'relative z-50 flex max-h-[85vh] w-full max-w-md flex-col rounded-malio bg-white shadow-[0_4px_4px_0_rgba(0,0,0,0.25)]',
|
||||||
|
modalClass,
|
||||||
|
)"
|
||||||
|
role="dialog"
|
||||||
|
aria-modal="true"
|
||||||
|
:aria-labelledby="hasHeader ? headerId : undefined"
|
||||||
|
:aria-label="hasHeader ? undefined : (ariaLabel || undefined)"
|
||||||
|
tabindex="-1"
|
||||||
|
data-test="panel"
|
||||||
|
@keydown="onKeydown"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-if="hasHeader || showClose"
|
||||||
|
:class="twMerge('flex shrink-0 items-center justify-between gap-4 px-5 py-[25px]', headerClass)"
|
||||||
|
data-test="header"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
:id="headerId"
|
||||||
|
class="min-w-0 flex-1"
|
||||||
|
data-test="header-content"
|
||||||
|
>
|
||||||
|
<slot name="header" />
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
v-if="showClose"
|
||||||
|
type="button"
|
||||||
|
aria-label="Fermer"
|
||||||
|
class="flex h-8 w-8 shrink-0 cursor-pointer items-center justify-center rounded-full transition-colors hover:bg-m-surface"
|
||||||
|
data-test="close-button"
|
||||||
|
@click="close"
|
||||||
|
>
|
||||||
|
<IconifyIcon
|
||||||
|
icon="mdi:cancel-bold"
|
||||||
|
:width="16"
|
||||||
|
:height="16"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
:class="twMerge('flex-1 overflow-y-auto px-5', bodyClass)"
|
||||||
|
data-test="body"
|
||||||
|
>
|
||||||
|
<slot />
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="$slots.footer"
|
||||||
|
:class="twMerge('flex shrink-0 items-center gap-3 px-5 py-4', footerClass)"
|
||||||
|
data-test="footer"
|
||||||
|
>
|
||||||
|
<slot name="footer" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Transition>
|
||||||
|
</Teleport>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {
|
||||||
|
computed,
|
||||||
|
nextTick,
|
||||||
|
onBeforeUnmount,
|
||||||
|
onMounted,
|
||||||
|
ref,
|
||||||
|
useAttrs,
|
||||||
|
useId,
|
||||||
|
useSlots,
|
||||||
|
watch,
|
||||||
|
} from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import { twMerge } from 'tailwind-merge'
|
||||||
|
|
||||||
|
defineOptions({ name: 'MalioModal', inheritAttrs: false })
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
modelValue?: boolean
|
||||||
|
showClose?: boolean
|
||||||
|
dismissable?: boolean
|
||||||
|
closeOnEscape?: boolean
|
||||||
|
ariaLabel?: string
|
||||||
|
modalClass?: string
|
||||||
|
overlayClass?: string
|
||||||
|
headerClass?: string
|
||||||
|
bodyClass?: string
|
||||||
|
footerClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
showClose: true,
|
||||||
|
dismissable: true,
|
||||||
|
closeOnEscape: true,
|
||||||
|
ariaLabel: '',
|
||||||
|
modalClass: '',
|
||||||
|
overlayClass: '',
|
||||||
|
headerClass: '',
|
||||||
|
bodyClass: '',
|
||||||
|
footerClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'update:modelValue', value: boolean): void
|
||||||
|
(e: 'close'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const attrs = useAttrs()
|
||||||
|
const generatedId = useId()
|
||||||
|
|
||||||
|
const componentId = computed(() => props.id || `malio-modal-${generatedId}`)
|
||||||
|
|
||||||
|
const slots = useSlots()
|
||||||
|
const headerId = computed(() => `${componentId.value}-header`)
|
||||||
|
const hasHeader = computed(() => !!slots.header)
|
||||||
|
|
||||||
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const localValue = ref(false)
|
||||||
|
const isOpen = computed(() =>
|
||||||
|
isControlled.value ? props.modelValue! : localValue.value,
|
||||||
|
)
|
||||||
|
const isRendered = ref(isOpen.value)
|
||||||
|
|
||||||
|
const panelRef = ref<HTMLElement | null>(null)
|
||||||
|
|
||||||
|
let previouslyFocused: HTMLElement | null = null
|
||||||
|
// Per-instance flag: true while this modal holds a scroll-lock count slot.
|
||||||
|
let lockedByThisInstance = false
|
||||||
|
|
||||||
|
function getFocusable(container: HTMLElement): HTMLElement[] {
|
||||||
|
return Array.from(
|
||||||
|
container.querySelectorAll<HTMLElement>(
|
||||||
|
'a[href], button:not([disabled]), textarea:not([disabled]), input:not([disabled]), select:not([disabled]), [tabindex]:not([tabindex="-1"]), [contenteditable]:not([contenteditable="false"])',
|
||||||
|
),
|
||||||
|
).filter((el) => el.tabIndex !== -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onOpen() {
|
||||||
|
previouslyFocused = (document.activeElement as HTMLElement | null) ?? null
|
||||||
|
if (!lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = true
|
||||||
|
openModalCount++
|
||||||
|
if (openModalCount === 1) {
|
||||||
|
document.body.style.overflow = 'hidden'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nextTick(() => {
|
||||||
|
const panel = panelRef.value
|
||||||
|
if (!panel) return
|
||||||
|
const focusable = getFocusable(panel)
|
||||||
|
;(focusable[0] ?? panel).focus()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function onClose() {
|
||||||
|
if (lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = false
|
||||||
|
openModalCount = Math.max(0, openModalCount - 1)
|
||||||
|
if (openModalCount === 0) {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
previouslyFocused?.focus?.()
|
||||||
|
previouslyFocused = null
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(isOpen, (val) => {
|
||||||
|
if (val) {
|
||||||
|
isRendered.value = true
|
||||||
|
onOpen()
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
onClose()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
if (isOpen.value) onOpen()
|
||||||
|
})
|
||||||
|
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
// If this instance is still holding a scroll-lock slot, release it.
|
||||||
|
if (lockedByThisInstance) {
|
||||||
|
lockedByThisInstance = false
|
||||||
|
openModalCount = Math.max(0, openModalCount - 1)
|
||||||
|
if (openModalCount === 0) {
|
||||||
|
document.body.style.overflow = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
function onBackdropClick() {
|
||||||
|
if (props.dismissable) close()
|
||||||
|
}
|
||||||
|
|
||||||
|
function onKeydown(e: KeyboardEvent) {
|
||||||
|
if (e.key === 'Escape' && props.closeOnEscape) {
|
||||||
|
e.stopPropagation()
|
||||||
|
close()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (e.key !== 'Tab') return
|
||||||
|
|
||||||
|
const panel = panelRef.value
|
||||||
|
if (!panel) return
|
||||||
|
const focusable = getFocusable(panel)
|
||||||
|
if (focusable.length === 0) {
|
||||||
|
e.preventDefault()
|
||||||
|
panel.focus()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const first = focusable[0]!
|
||||||
|
const last = focusable[focusable.length - 1]!
|
||||||
|
if (e.shiftKey && document.activeElement === first) {
|
||||||
|
e.preventDefault()
|
||||||
|
last.focus()
|
||||||
|
}
|
||||||
|
else if (!e.shiftKey && document.activeElement === last) {
|
||||||
|
e.preventDefault()
|
||||||
|
first.focus()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function close() {
|
||||||
|
if (!isControlled.value) localValue.value = false
|
||||||
|
emit('update:modelValue', false)
|
||||||
|
emit('close')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
// Shared across all MalioModal instances: only the last open modal releases the body scroll-lock.
|
||||||
|
let openModalCount = 0
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.modal-enter-active,
|
||||||
|
.modal-leave-active {
|
||||||
|
transition: opacity 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal-enter-active > div:last-child,
|
||||||
|
.modal-leave-active > div:last-child {
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal-enter-from,
|
||||||
|
.modal-leave-to {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal-enter-from > div:last-child,
|
||||||
|
.modal-leave-to > div:last-child {
|
||||||
|
transform: scale(0.95);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -153,4 +153,43 @@ describe('MalioRadioButton', () => {
|
|||||||
expect(wrapper.get('input').classes()).toContain('border-red-500')
|
expect(wrapper.get('input').classes()).toContain('border-red-500')
|
||||||
expect(wrapper.get('.radio-text').classes()).toContain('font-bold')
|
expect(wrapper.get('.radio-text').classes()).toContain('font-bold')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('uses muted label color and muted border when unchecked', () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Option 1', value: 'a', modelValue: 'b'})
|
||||||
|
|
||||||
|
expect(wrapper.get('.radio-text').classes()).toContain('text-m-muted')
|
||||||
|
expect(wrapper.get('input').classes()).toContain('border-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses black label color when checked', () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Option 1', value: 'a', modelValue: 'a'})
|
||||||
|
|
||||||
|
expect(wrapper.get('.radio-text').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('has checked:border-black on input', () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Option 1', value: 'a', modelValue: 'a'})
|
||||||
|
|
||||||
|
expect(wrapper.get('input').classes()).toContain('checked:border-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Champ', required: true})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Champ'})
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('updates label color when toggled without v-model (uncontrolled)', async () => {
|
||||||
|
const wrapper = mountRadioButton({label: 'Option 1', value: 'a'})
|
||||||
|
|
||||||
|
expect(wrapper.get('.radio-text').classes()).toContain('text-m-muted')
|
||||||
|
|
||||||
|
await wrapper.get('input').trigger('change')
|
||||||
|
|
||||||
|
expect(wrapper.get('.radio-text').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -29,7 +29,7 @@
|
|||||||
:for="inputId"
|
:for="inputId"
|
||||||
:class="mergedLabelClass"
|
:class="mergedLabelClass"
|
||||||
>
|
>
|
||||||
{{ label }}
|
{{ label }}<MalioRequiredMark v-if="required" />
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -44,8 +44,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, useAttrs, useId} from 'vue'
|
import {computed, ref, useAttrs, useId} from 'vue'
|
||||||
import {twMerge} from 'tailwind-merge'
|
import {twMerge} from 'tailwind-merge'
|
||||||
|
import MalioRequiredMark from '../shared/RequiredMark.vue'
|
||||||
|
|
||||||
defineOptions({name: 'MalioRadioButton', inheritAttrs: false})
|
defineOptions({name: 'MalioRadioButton', inheritAttrs: false})
|
||||||
|
|
||||||
@@ -86,9 +87,13 @@ const props = withDefaults(
|
|||||||
|
|
||||||
const attrs = useAttrs()
|
const attrs = useAttrs()
|
||||||
const generatedId = useId()
|
const generatedId = useId()
|
||||||
|
const localValue = ref<string | number | boolean | null | undefined>(undefined)
|
||||||
|
|
||||||
const inputId = computed(() => props.id?.toString() || `malio-radio-${generatedId}`)
|
const inputId = computed(() => props.id?.toString() || `malio-radio-${generatedId}`)
|
||||||
const isChecked = computed(() => props.modelValue === props.value)
|
const isControlled = computed(() => props.modelValue !== undefined)
|
||||||
|
const isChecked = computed(() =>
|
||||||
|
isControlled.value ? props.modelValue === props.value : localValue.value === props.value,
|
||||||
|
)
|
||||||
const hasError = computed(() => !!props.error)
|
const hasError = computed(() => !!props.error)
|
||||||
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
const disabled = computed(() => props.disabled)
|
const disabled = computed(() => props.disabled)
|
||||||
@@ -117,14 +122,15 @@ const mergedControlClass = computed(() =>
|
|||||||
|
|
||||||
const mergedInputClass = computed(() =>
|
const mergedInputClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'h-5 w-5 cursor-pointer appearance-none rounded-full border-2 border-black',
|
'h-5 w-5 cursor-pointer appearance-none rounded-full border-2 border-m-muted checked:border-black',
|
||||||
props.inputClass,
|
props.inputClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
const mergedLabelClass = computed(() =>
|
const mergedLabelClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'radio-text mt-px cursor-pointer text-black',
|
'radio-text mt-px cursor-pointer',
|
||||||
|
isChecked.value ? 'text-black' : 'text-m-muted',
|
||||||
hasError.value ? 'text-m-danger' : '',
|
hasError.value ? 'text-m-danger' : '',
|
||||||
hasSuccess.value ? 'text-m-success' : '',
|
hasSuccess.value ? 'text-m-success' : '',
|
||||||
disabled.value ? 'cursor-not-allowed text-black/60' : '',
|
disabled.value ? 'cursor-not-allowed text-black/60' : '',
|
||||||
@@ -160,6 +166,10 @@ const onChange = (event: Event) => {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!isControlled.value) {
|
||||||
|
localValue.value = props.value
|
||||||
|
}
|
||||||
|
|
||||||
emit('update:modelValue', props.value)
|
emit('update:modelValue', props.value)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -169,6 +179,11 @@ const onChange = (event: Event) => {
|
|||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.radio-control input[type='radio']:focus-visible {
|
||||||
|
outline: 2px solid rgb(var(--m-primary) / 1);
|
||||||
|
outline-offset: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
.radio-control.is-error input[type='radio'] {
|
.radio-control.is-error input[type='radio'] {
|
||||||
border-color: rgb(var(--m-danger) / 1);
|
border-color: rgb(var(--m-danger) / 1);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,13 +16,14 @@ type SelectProps = {
|
|||||||
hint?: string
|
hint?: string
|
||||||
error?: string
|
error?: string
|
||||||
success?: string
|
success?: string
|
||||||
minWidth?: string
|
|
||||||
maxWidth?: string
|
|
||||||
textField?: string
|
textField?: string
|
||||||
textValue?: string
|
textValue?: string
|
||||||
textLabel?: string
|
textLabel?: string
|
||||||
rounded?: string
|
rounded?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
required?: boolean
|
||||||
|
reserveMessageSpace?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const SelectForTest = Select as DefineComponent<SelectProps>
|
const SelectForTest = Select as DefineComponent<SelectProps>
|
||||||
@@ -209,4 +210,182 @@ describe('MalioSelect', () => {
|
|||||||
expect(wrapper.find('p.text-m-success').exists()).toBe(false)
|
expect(wrapper.find('p.text-m-success').exists()).toBe(false)
|
||||||
expect(wrapper.get('p.text-m-danger').text()).toBe('Selection error')
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Selection error')
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('shows muted chevron color when empty and closed', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows primary chevron color when open', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options},
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows black chevron color when an option is selected and closed', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: 'fr', options},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('hides the chevron when disabled', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: 'fr', options, disabled: true},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="chevron"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('greys the label and the selected value when disabled', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: 'fr', label: 'Pays', options, disabled: true},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-muted')
|
||||||
|
expect(wrapper.get('button span.block').classes()).toContain('text-black/60')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows danger chevron color on error even when open', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options, error: 'Selection error'},
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success chevron color on success', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options, success: 'OK'},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('affiche l\'astérisque quand required est vrai', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ', required: true},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'affiche pas l\'astérisque par défaut', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ'},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.find('[data-test="required-mark"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('expose aria-required quand required est vrai', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options, required: true},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.find('[aria-required="true"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('n\'expose pas aria-required par défaut', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options},
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(wrapper.find('[aria-required="true"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('keeps the bottom border allocation when open downward (transparent, not zero)', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, options},
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
|
||||||
|
const buttonClasses = wrapper.get('button').classes()
|
||||||
|
// !border-b-0 would shrink the bottom border to 0px and grow content area by 1px;
|
||||||
|
// !border-b-transparent keeps the 1px allocation but hides the line
|
||||||
|
expect(buttonClasses).not.toContain('!border-b-0')
|
||||||
|
expect(buttonClasses).toContain('!border-b-transparent')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly : bordure noire même sans sélection, pas de grow/bleu', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ', readonly: true, options: [{label: 'A', value: 'a'}]},
|
||||||
|
})
|
||||||
|
const trigger = wrapper.get('button')
|
||||||
|
expect(trigger.classes()).toContain('border-black')
|
||||||
|
expect(trigger.classes()).not.toContain('border-m-muted')
|
||||||
|
expect(trigger.classes()).not.toContain('grow-height')
|
||||||
|
expect(trigger.classes()).not.toContain('focus-visible:border-m-primary')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly vide : label gris, pas de bleu', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ', readonly: true, options: [{label: 'A', value: 'a'}]},
|
||||||
|
})
|
||||||
|
const label = wrapper.get('label')
|
||||||
|
expect(label.classes()).not.toContain('text-m-primary')
|
||||||
|
expect(label.classes()).toContain('text-m-muted')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly sélectionné : label noir + chevron noir', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {label: 'Champ', readonly: true, modelValue: 'a', options: [{label: 'A', value: 'a'}]},
|
||||||
|
})
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-black')
|
||||||
|
expect(wrapper.get('[data-test="chevron"]').classes()).toContain('text-black')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly empêche l’ouverture du dropdown', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ', readonly: true, options: [{label: 'A', value: 'a'}]},
|
||||||
|
})
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
expect(wrapper.find('[role="listbox"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('readonly expose aria-readonly et reste focusable (pas disabled)', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {modelValue: null, label: 'Champ', readonly: true, options},
|
||||||
|
})
|
||||||
|
const trigger = wrapper.get('button')
|
||||||
|
expect(trigger.attributes('aria-readonly')).toBe('true')
|
||||||
|
expect(trigger.attributes('disabled')).toBeUndefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disabled + readonly : pas d’aria-readonly (disabled prime)', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {props: {modelValue: null, label: 'Champ', disabled: true, readonly: true, options: [{label: 'A', value: 'a'}]}})
|
||||||
|
const trigger = wrapper.get('button')
|
||||||
|
expect(trigger.attributes('aria-readonly')).toBeUndefined()
|
||||||
|
expect(trigger.attributes('disabled')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('réserve l’espace message par défaut même sans message', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {props: {modelValue: null, label: 'Champ', options}})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false sans message : pas de ligne réservée', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {props: {modelValue: null, label: 'Champ', options, reserveMessageSpace: false}})
|
||||||
|
expect(wrapper.find('[id$="-describedby"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('reserveMessageSpace=false avec message : ligne rendue sans min-h', () => {
|
||||||
|
const wrapper = mount(SelectForTest, {props: {modelValue: null, label: 'Champ', options, reserveMessageSpace: false, error: 'Erreur'}})
|
||||||
|
const msg = wrapper.find('[id$="-describedby"]')
|
||||||
|
expect(msg.exists()).toBe(true)
|
||||||
|
expect(msg.classes()).not.toContain('min-h-[1rem]')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user