Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c646df9fe3 | |||
| 7fc072ad08 | |||
| f30619a497 | |||
| d7bf038fdd | |||
| 2059556ffe | |||
| a95cf8cdfb | |||
| ba2ecb5768 | |||
| 87940481d6 |
91
.playground/pages/composant/input/inputRichText.vue
Normal file
91
.playground/pages/composant/input/inputRichText.vue
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
<template>
|
||||||
|
<div class="grid grid-cols-1 items-start gap-6 p-4 lg:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="simpleValue"
|
||||||
|
label="Note"
|
||||||
|
placeholder="Écrire ici…"
|
||||||
|
/>
|
||||||
|
<pre class="mt-3 overflow-auto rounded bg-m-bg p-2 text-xs">{{ simpleValue }}</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec contenu initial + hint</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="hintValue"
|
||||||
|
label="Description"
|
||||||
|
hint="Tu peux mettre en forme avec la barre d'outils"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="errorValue"
|
||||||
|
label="Compte-rendu"
|
||||||
|
error="Le compte-rendu doit faire au moins 20 caractères"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="successValue"
|
||||||
|
label="Compte-rendu"
|
||||||
|
success="Compte-rendu validé"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="readonlyValue"
|
||||||
|
label="Note (lecture seule)"
|
||||||
|
readonly
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Disabled</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="disabledValue"
|
||||||
|
label="Note (désactivée)"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Affichage seul (editable=false)</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
:model-value="readonlyValue"
|
||||||
|
:editable="false"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Sortie HTML</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="htmlValue"
|
||||||
|
label="Article"
|
||||||
|
output-format="html"
|
||||||
|
min-height="200px"
|
||||||
|
placeholder="Tape ici, la sortie sera en HTML…"
|
||||||
|
/>
|
||||||
|
<pre class="mt-3 overflow-auto rounded bg-m-bg p-2 text-xs">{{ htmlValue }}</pre>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
import MalioInputRichText from '../../../../app/components/malio/input/InputRichText.vue'
|
||||||
|
|
||||||
|
const simpleValue = ref('')
|
||||||
|
const hintValue = ref('## Titre\n\nUn paragraphe avec du **gras**, de l\'*italique* et un [lien](https://malio.fr).')
|
||||||
|
const errorValue = ref('Trop court')
|
||||||
|
const successValue = ref('Tout est bon de mon côté.')
|
||||||
|
const readonlyValue = ref('## Compte-rendu\n\n- Point 1\n- Point 2\n\n> Citation importante')
|
||||||
|
const disabledValue = ref('Contenu indisponible.')
|
||||||
|
const htmlValue = ref('<p>Contenu <strong>riche</strong>.</p>')
|
||||||
|
</script>
|
||||||
@@ -2,8 +2,26 @@
|
|||||||
"branches": ["main", "master"],
|
"branches": ["main", "master"],
|
||||||
"repositoryUrl": "https://gitea.malio.fr/MALIO-DEV/malio-layer-ui.git",
|
"repositoryUrl": "https://gitea.malio.fr/MALIO-DEV/malio-layer-ui.git",
|
||||||
"plugins": [
|
"plugins": [
|
||||||
"@semantic-release/commit-analyzer",
|
[
|
||||||
"@semantic-release/release-notes-generator",
|
"@semantic-release/commit-analyzer",
|
||||||
|
{
|
||||||
|
"preset": "angular",
|
||||||
|
"parserOpts": {
|
||||||
|
"headerPattern": "^(\\w+)(?:\\(([\\w$.\\-* ]+)\\))?\\s*:\\s+(.+)$",
|
||||||
|
"headerCorrespondence": ["type", "scope", "subject"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"@semantic-release/release-notes-generator",
|
||||||
|
{
|
||||||
|
"preset": "angular",
|
||||||
|
"parserOpts": {
|
||||||
|
"headerPattern": "^(\\w+)(?:\\(([\\w$.\\-* ]+)\\))?\\s*:\\s+(.+)$",
|
||||||
|
"headerCorrespondence": ["type", "scope", "subject"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
"@semantic-release/npm"
|
"@semantic-release/npm"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ Liste des évolutions de la librairie Malio layer UI
|
|||||||
* [#MUI-15] Création d'un composant drawer
|
* [#MUI-15] Création d'un composant drawer
|
||||||
* [#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
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
|
|||||||
@@ -132,6 +132,41 @@ Zone de texte multiligne avec compteur et redimensionnement.
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## MalioInputRichText
|
||||||
|
|
||||||
|
Éditeur de texte riche basé sur **TipTap v3** + **StarterKit** + **tiptap-markdown** + **TextStyle/Color/Highlight**. Toolbar avec gras, italique, barré, titres H2/H3, listes, citation, code, code-block, lien, **couleur du texte**, **surlignage**, undo/redo. Sortie en HTML (par défaut) ou markdown.
|
||||||
|
|
||||||
|
> Couleurs et surlignages ne sont **pas persistés en markdown**. Pour les conserver au save/reload, utiliser `output-format="html"`.
|
||||||
|
|
||||||
|
| Prop | Type | Défaut | Description |
|
||||||
|
|------|------|--------|-------------|
|
||||||
|
| `id` | `string` | auto | Identifiant HTML |
|
||||||
|
| `label` | `string` | `''` | Label affiché au-dessus de l'éditeur |
|
||||||
|
| `modelValue` | `string \| null` | `undefined` | Contenu (v-model) |
|
||||||
|
| `placeholder` | `string` | `''` | Texte affiché quand vide |
|
||||||
|
| `minHeight` | `string` | `'160px'` | Hauteur min de la zone d'édition |
|
||||||
|
| `editable` | `boolean` | `true` | `false` → mode affichage seul (toolbar masquée) |
|
||||||
|
| `disabled` | `boolean` | `false` | Désactive l'édition et la toolbar |
|
||||||
|
| `readonly` | `boolean` | `false` | Lecture seule (toolbar visible mais désactivée) |
|
||||||
|
| `hint` | `string` | `''` | Message d'aide |
|
||||||
|
| `error` | `string` | `''` | Message d'erreur |
|
||||||
|
| `success` | `string` | `''` | Message de succès |
|
||||||
|
| `outputFormat` | `'markdown' \| 'html'` | `'html'` | Format émis dans `update:modelValue` |
|
||||||
|
| `groupClass` | `string` | `''` | Classes CSS conteneur (twMerge) |
|
||||||
|
| `labelClass` | `string` | `''` | Classes CSS label (twMerge) |
|
||||||
|
| `editorClass` | `string` | `''` | Classes CSS wrapper éditeur (twMerge) |
|
||||||
|
|
||||||
|
**Events :** `update:modelValue(value: string)`
|
||||||
|
|
||||||
|
```vue
|
||||||
|
<MalioInputRichText v-model="note" label="Note" placeholder="Écrire ici…" />
|
||||||
|
<MalioInputRichText v-model="cr" label="Compte-rendu" error="Trop court" />
|
||||||
|
<MalioInputRichText v-model="article" label="Article" min-height="240px" />
|
||||||
|
<MalioInputRichText :model-value="content" :editable="false" />
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## MalioInputUpload
|
## MalioInputUpload
|
||||||
|
|
||||||
Champ d'upload de fichier.
|
Champ d'upload de fichier.
|
||||||
|
|||||||
@@ -94,7 +94,7 @@ const describedBy = computed(() => {
|
|||||||
|
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge(
|
twMerge(
|
||||||
'checkbox-wrapper-4 mt-4 w-full',
|
'checkbox-wrapper-4 w-full',
|
||||||
props.groupClass,
|
props.groupClass,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
165
app/components/malio/input/InputRichText.test.ts
Normal file
165
app/components/malio/input/InputRichText.test.ts
Normal file
@@ -0,0 +1,165 @@
|
|||||||
|
import {afterEach, describe, expect, it} from 'vitest'
|
||||||
|
import {flushPromises, mount} from '@vue/test-utils'
|
||||||
|
import type {DefineComponent} from 'vue'
|
||||||
|
import InputRichText from './InputRichText.vue'
|
||||||
|
|
||||||
|
type InputRichTextProps = {
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null
|
||||||
|
placeholder?: string
|
||||||
|
minHeight?: string
|
||||||
|
editable?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
outputFormat?: 'markdown' | 'html'
|
||||||
|
groupClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
editorClass?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const InputRichTextForTest = InputRichText as DefineComponent<InputRichTextProps>
|
||||||
|
|
||||||
|
const mountComponent = async (props: InputRichTextProps = {}) => {
|
||||||
|
const wrapper = mount(InputRichTextForTest, {
|
||||||
|
props,
|
||||||
|
attachTo: document.body,
|
||||||
|
})
|
||||||
|
await flushPromises()
|
||||||
|
return wrapper
|
||||||
|
}
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
document.body.replaceChildren()
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('MalioInputRichText', () => {
|
||||||
|
it('renders the label and reuses a provided id', async () => {
|
||||||
|
const wrapper = await mountComponent({id: 'custom-rt-id', label: 'Description'})
|
||||||
|
|
||||||
|
const label = wrapper.get('label')
|
||||||
|
expect(label.text()).toBe('Description')
|
||||||
|
expect(label.attributes('for')).toBe('custom-rt-id')
|
||||||
|
expect(wrapper.get('#custom-rt-id').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('generates an id when missing', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Description'})
|
||||||
|
|
||||||
|
const labelFor = wrapper.get('label').attributes('for')
|
||||||
|
expect(labelFor?.startsWith('malio-input-rich-text-')).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders the toolbar buttons in editable mode', async () => {
|
||||||
|
const wrapper = await mountComponent({modelValue: ''})
|
||||||
|
|
||||||
|
const buttons = wrapper.findAll('button[type="button"]')
|
||||||
|
expect(buttons.length).toBeGreaterThanOrEqual(13)
|
||||||
|
expect(wrapper.find('button[title="Gras"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Italique"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Lien"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Couleur du texte"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Surlignage"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Annuler"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('button[title="Rétablir"]').exists()).toBe(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens and closes the text color palette', async () => {
|
||||||
|
const wrapper = await mountComponent({modelValue: ''})
|
||||||
|
|
||||||
|
expect(wrapper.find('[aria-label="Palette couleur du texte"]').exists()).toBe(false)
|
||||||
|
|
||||||
|
await wrapper.get('button[title="Couleur du texte"]').trigger('click')
|
||||||
|
expect(wrapper.find('[aria-label="Palette couleur du texte"]').exists()).toBe(true)
|
||||||
|
|
||||||
|
await wrapper.get('button[title="Couleur du texte"]').trigger('click')
|
||||||
|
expect(wrapper.find('[aria-label="Palette couleur du texte"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('opens the highlight palette and closes the color palette', async () => {
|
||||||
|
const wrapper = await mountComponent({modelValue: ''})
|
||||||
|
|
||||||
|
await wrapper.get('button[title="Couleur du texte"]').trigger('click')
|
||||||
|
expect(wrapper.find('[aria-label="Palette couleur du texte"]').exists()).toBe(true)
|
||||||
|
|
||||||
|
await wrapper.get('button[title="Surlignage"]').trigger('click')
|
||||||
|
expect(wrapper.find('[aria-label="Palette de surlignage"]').exists()).toBe(true)
|
||||||
|
expect(wrapper.find('[aria-label="Palette couleur du texte"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables color and highlight buttons when readonly', async () => {
|
||||||
|
const wrapper = await mountComponent({readonly: true, modelValue: ''})
|
||||||
|
|
||||||
|
expect(wrapper.get('button[title="Couleur du texte"]').attributes('disabled')).toBeDefined()
|
||||||
|
expect(wrapper.get('button[title="Surlignage"]').attributes('disabled')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('does not render the toolbar in readonly display mode (editable=false)', async () => {
|
||||||
|
const wrapper = await mountComponent({editable: false, modelValue: '**hi**'})
|
||||||
|
|
||||||
|
expect(wrapper.find('button[title="Gras"]').exists()).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables toolbar buttons when disabled', async () => {
|
||||||
|
const wrapper = await mountComponent({disabled: true, modelValue: ''})
|
||||||
|
|
||||||
|
const boldBtn = wrapper.get('button[title="Gras"]')
|
||||||
|
expect(boldBtn.attributes('disabled')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('disables toolbar buttons when readonly', async () => {
|
||||||
|
const wrapper = await mountComponent({readonly: true, modelValue: ''})
|
||||||
|
|
||||||
|
const boldBtn = wrapper.get('button[title="Gras"]')
|
||||||
|
expect(boldBtn.attributes('disabled')).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows hint message in muted color', async () => {
|
||||||
|
const wrapper = await mountComponent({hint: 'Helpful hint'})
|
||||||
|
|
||||||
|
expect(wrapper.get('p.text-m-muted').text()).toBe('Helpful hint')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows error state on wrapper, label and message', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Description', error: 'Editor error'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-danger')
|
||||||
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Editor error')
|
||||||
|
expect(wrapper.get('.rich-text-wrapper').classes()).toContain('border-m-danger')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows success state on wrapper, label and message', async () => {
|
||||||
|
const wrapper = await mountComponent({label: 'Description', success: 'Editor success'})
|
||||||
|
|
||||||
|
expect(wrapper.get('label').classes()).toContain('text-m-success')
|
||||||
|
expect(wrapper.get('p.text-m-success').text()).toBe('Editor success')
|
||||||
|
expect(wrapper.get('.rich-text-wrapper').classes()).toContain('border-m-success')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('prioritizes error over success', async () => {
|
||||||
|
const wrapper = await mountComponent({error: 'Editor error', success: 'Editor success'})
|
||||||
|
|
||||||
|
expect(wrapper.get('.rich-text-wrapper').classes()).toContain('border-m-danger')
|
||||||
|
expect(wrapper.find('p.text-m-success').exists()).toBe(false)
|
||||||
|
expect(wrapper.get('p.text-m-danger').text()).toBe('Editor error')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('sets aria-invalid and aria-describedby on the editor content when error', async () => {
|
||||||
|
const wrapper = await mountComponent({id: 'rt-aria', error: 'Boom'})
|
||||||
|
|
||||||
|
const editorContent = wrapper.find('[aria-invalid="true"]')
|
||||||
|
expect(editorContent.exists()).toBe(true)
|
||||||
|
expect(editorContent.attributes('aria-describedby')).toBe('rt-aria-describedby')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders initial markdown content visually', async () => {
|
||||||
|
const wrapper = await mountComponent({modelValue: '## Mon titre\n\nUn paragraphe.'})
|
||||||
|
|
||||||
|
const html = wrapper.html()
|
||||||
|
expect(html).toContain('Mon titre')
|
||||||
|
expect(html).toContain('Un paragraphe.')
|
||||||
|
})
|
||||||
|
})
|
||||||
574
app/components/malio/input/InputRichText.vue
Normal file
574
app/components/malio/input/InputRichText.vue
Normal file
@@ -0,0 +1,574 @@
|
|||||||
|
<template>
|
||||||
|
<div :class="mergedGroupClass">
|
||||||
|
<label
|
||||||
|
v-if="label"
|
||||||
|
:for="editorId"
|
||||||
|
:class="mergedLabelClass"
|
||||||
|
>
|
||||||
|
{{ label }}
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<!-- Mode lecture seule (rendu uniquement) -->
|
||||||
|
<div
|
||||||
|
v-if="!editable"
|
||||||
|
:id="editorId"
|
||||||
|
:class="mergedReadonlyClass"
|
||||||
|
>
|
||||||
|
<EditorContent :editor="editor" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Mode éditable -->
|
||||||
|
<div
|
||||||
|
v-else
|
||||||
|
:id="editorId"
|
||||||
|
:class="mergedEditorWrapperClass"
|
||||||
|
@click="focusEditor"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="flex flex-wrap items-center gap-0.5 border-b border-m-border bg-m-bg p-1"
|
||||||
|
@click.stop
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
v-for="btn in toolbarButtons"
|
||||||
|
:key="btn.key"
|
||||||
|
type="button"
|
||||||
|
class="flex h-8 w-8 items-center justify-center rounded text-m-text transition-colors hover:bg-m-primary/10 disabled:cursor-not-allowed disabled:opacity-40"
|
||||||
|
:class="btn.isActive() ? 'bg-m-primary/15 text-m-primary' : ''"
|
||||||
|
:title="btn.title"
|
||||||
|
:disabled="disabled || readonly"
|
||||||
|
:aria-label="btn.title"
|
||||||
|
:aria-pressed="btn.isActive()"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="btn.action()"
|
||||||
|
>
|
||||||
|
<IconifyIcon :icon="btn.icon" :width="18" :height="18" />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<span class="mx-1 h-5 w-px bg-m-border" aria-hidden="true" />
|
||||||
|
|
||||||
|
<div class="relative">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex h-8 w-8 flex-col items-center justify-center rounded text-m-text transition-colors hover:bg-m-primary/10 disabled:cursor-not-allowed disabled:opacity-40"
|
||||||
|
:class="colorPickerOpen ? 'bg-m-primary/15 text-m-primary' : ''"
|
||||||
|
title="Couleur du texte"
|
||||||
|
aria-label="Couleur du texte"
|
||||||
|
:aria-expanded="colorPickerOpen"
|
||||||
|
:disabled="disabled || readonly"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="toggleColorPicker"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:format-color-text" :width="18" :height="18" />
|
||||||
|
<span
|
||||||
|
class="-mt-0.5 block h-1 w-4 rounded-sm"
|
||||||
|
:style="{ backgroundColor: currentTextColor ?? 'transparent' }"
|
||||||
|
aria-hidden="true"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
v-if="colorPickerOpen"
|
||||||
|
class="absolute left-0 top-full z-10 mt-1 flex w-44 flex-col gap-2 rounded-md border border-m-border bg-white p-2 shadow-lg"
|
||||||
|
role="dialog"
|
||||||
|
aria-label="Palette couleur du texte"
|
||||||
|
>
|
||||||
|
<div class="grid grid-cols-4 gap-1">
|
||||||
|
<button
|
||||||
|
v-for="swatch in textColorSwatches"
|
||||||
|
:key="swatch.value"
|
||||||
|
type="button"
|
||||||
|
class="h-7 w-7 rounded border border-m-border transition-transform hover:scale-110"
|
||||||
|
:class="currentTextColor === swatch.value ? 'ring-2 ring-m-primary ring-offset-1' : ''"
|
||||||
|
:style="{ backgroundColor: swatch.value }"
|
||||||
|
:title="swatch.label"
|
||||||
|
:aria-label="swatch.label"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="applyTextColor(swatch.value)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex items-center justify-center gap-1 rounded border border-m-border px-2 py-1 text-xs text-m-text transition-colors hover:bg-m-bg"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="applyTextColor(null)"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:format-color-marker-cancel" :width="14" :height="14" />
|
||||||
|
Aucune couleur
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="relative">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex h-8 w-8 flex-col items-center justify-center rounded text-m-text transition-colors hover:bg-m-primary/10 disabled:cursor-not-allowed disabled:opacity-40"
|
||||||
|
:class="highlightPickerOpen ? 'bg-m-primary/15 text-m-primary' : ''"
|
||||||
|
title="Surlignage"
|
||||||
|
aria-label="Surlignage"
|
||||||
|
:aria-expanded="highlightPickerOpen"
|
||||||
|
:disabled="disabled || readonly"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="toggleHighlightPicker"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:marker" :width="18" :height="18" />
|
||||||
|
<span
|
||||||
|
class="-mt-0.5 block h-1 w-4 rounded-sm"
|
||||||
|
:style="{ backgroundColor: currentHighlightColor ?? 'transparent' }"
|
||||||
|
aria-hidden="true"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
v-if="highlightPickerOpen"
|
||||||
|
class="absolute left-0 top-full z-10 mt-1 flex w-44 flex-col gap-2 rounded-md border border-m-border bg-white p-2 shadow-lg"
|
||||||
|
role="dialog"
|
||||||
|
aria-label="Palette de surlignage"
|
||||||
|
>
|
||||||
|
<div class="grid grid-cols-4 gap-1">
|
||||||
|
<button
|
||||||
|
v-for="swatch in highlightSwatches"
|
||||||
|
:key="swatch.value"
|
||||||
|
type="button"
|
||||||
|
class="h-7 w-7 rounded border border-m-border transition-transform hover:scale-110"
|
||||||
|
:class="currentHighlightColor === swatch.value ? 'ring-2 ring-m-primary ring-offset-1' : ''"
|
||||||
|
:style="{ backgroundColor: swatch.value }"
|
||||||
|
:title="swatch.label"
|
||||||
|
:aria-label="swatch.label"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="applyHighlight(swatch.value)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex items-center justify-center gap-1 rounded border border-m-border px-2 py-1 text-xs text-m-text transition-colors hover:bg-m-bg"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="applyHighlight(null)"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:format-color-marker-cancel" :width="14" :height="14" />
|
||||||
|
Aucun surlignage
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span class="mx-1 h-5 w-px bg-m-border" aria-hidden="true" />
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex h-8 w-8 items-center justify-center rounded text-m-text transition-colors hover:bg-m-primary/10 disabled:cursor-not-allowed disabled:opacity-40"
|
||||||
|
title="Annuler"
|
||||||
|
aria-label="Annuler"
|
||||||
|
:disabled="disabled || readonly || !editor?.can().undo()"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="editor?.chain().focus().undo().run()"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:undo" :width="18" :height="18" />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="flex h-8 w-8 items-center justify-center rounded text-m-text transition-colors hover:bg-m-primary/10 disabled:cursor-not-allowed disabled:opacity-40"
|
||||||
|
title="Rétablir"
|
||||||
|
aria-label="Rétablir"
|
||||||
|
:disabled="disabled || readonly || !editor?.can().redo()"
|
||||||
|
@mousedown.prevent
|
||||||
|
@click="editor?.chain().focus().redo().run()"
|
||||||
|
>
|
||||||
|
<IconifyIcon icon="mdi:redo" :width="18" :height="18" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<EditorContent
|
||||||
|
:editor="editor"
|
||||||
|
class="malio-rich-text flex flex-1 cursor-text"
|
||||||
|
:style="{ minHeight }"
|
||||||
|
:aria-invalid="hasError || undefined"
|
||||||
|
:aria-describedby="describedBy"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p
|
||||||
|
v-if="hint || hasError || hasSuccess"
|
||||||
|
:id="`${editorId}-describedby`"
|
||||||
|
:class="[
|
||||||
|
hasError
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess
|
||||||
|
? 'text-m-success'
|
||||||
|
: 'text-m-muted',
|
||||||
|
'mt-1 text-xs ml-[2px]',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ error || success || hint }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, onBeforeUnmount, onMounted, ref, shallowRef, useId, watch } from 'vue'
|
||||||
|
import { Icon as IconifyIcon } from '@iconify/vue'
|
||||||
|
import { Editor, EditorContent } from '@tiptap/vue-3'
|
||||||
|
import StarterKit from '@tiptap/starter-kit'
|
||||||
|
import Placeholder from '@tiptap/extension-placeholder'
|
||||||
|
import { TextStyle } from '@tiptap/extension-text-style'
|
||||||
|
import Color from '@tiptap/extension-color'
|
||||||
|
import Highlight from '@tiptap/extension-highlight'
|
||||||
|
import { Markdown } from 'tiptap-markdown'
|
||||||
|
import { twMerge } from 'tailwind-merge'
|
||||||
|
|
||||||
|
defineOptions({ name: 'MalioInputRichText', inheritAttrs: false })
|
||||||
|
|
||||||
|
type OutputFormat = 'markdown' | 'html'
|
||||||
|
|
||||||
|
const props = withDefaults(
|
||||||
|
defineProps<{
|
||||||
|
id?: string
|
||||||
|
label?: string
|
||||||
|
modelValue?: string | null | undefined
|
||||||
|
placeholder?: string
|
||||||
|
minHeight?: string
|
||||||
|
editable?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
readonly?: boolean
|
||||||
|
hint?: string
|
||||||
|
error?: string
|
||||||
|
success?: string
|
||||||
|
outputFormat?: OutputFormat
|
||||||
|
groupClass?: string
|
||||||
|
labelClass?: string
|
||||||
|
editorClass?: string
|
||||||
|
}>(),
|
||||||
|
{
|
||||||
|
id: '',
|
||||||
|
label: '',
|
||||||
|
modelValue: undefined,
|
||||||
|
placeholder: '',
|
||||||
|
minHeight: '160px',
|
||||||
|
editable: true,
|
||||||
|
disabled: false,
|
||||||
|
readonly: false,
|
||||||
|
hint: '',
|
||||||
|
error: '',
|
||||||
|
success: '',
|
||||||
|
outputFormat: 'html',
|
||||||
|
groupClass: '',
|
||||||
|
labelClass: '',
|
||||||
|
editorClass: '',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(event: 'update:modelValue', value: string): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const generatedId = useId()
|
||||||
|
const editor = shallowRef<Editor>()
|
||||||
|
const isFocused = shallowRef(false)
|
||||||
|
|
||||||
|
const editorId = computed(() => props.id?.toString() || `malio-input-rich-text-${generatedId}`)
|
||||||
|
const hasError = computed(() => !!props.error)
|
||||||
|
const hasSuccess = computed(() => !!props.success && !hasError.value)
|
||||||
|
const isInteractionLocked = computed(() => props.disabled || props.readonly)
|
||||||
|
|
||||||
|
const describedBy = computed(() =>
|
||||||
|
hasError.value || hasSuccess.value || props.hint ? `${editorId.value}-describedby` : undefined,
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedGroupClass = computed(() => twMerge('w-full', props.groupClass))
|
||||||
|
|
||||||
|
const mergedLabelClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'mb-1 block text-sm font-medium',
|
||||||
|
hasError.value
|
||||||
|
? 'text-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'text-m-success'
|
||||||
|
: isFocused.value
|
||||||
|
? 'text-m-primary'
|
||||||
|
: 'text-m-text',
|
||||||
|
props.disabled ? 'text-black/60' : '',
|
||||||
|
props.labelClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedEditorWrapperClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'rich-text-wrapper flex flex-col overflow-hidden rounded-md border bg-white transition-colors',
|
||||||
|
hasError.value
|
||||||
|
? 'border-m-danger focus-within:border-m-danger'
|
||||||
|
: hasSuccess.value
|
||||||
|
? 'border-m-success focus-within:border-m-success'
|
||||||
|
: isFocused.value
|
||||||
|
? 'border-m-primary'
|
||||||
|
: 'border-m-muted hover:border-m-text/60',
|
||||||
|
props.disabled ? 'cursor-not-allowed bg-m-bg/50 opacity-70' : '',
|
||||||
|
props.editorClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const mergedReadonlyClass = computed(() =>
|
||||||
|
twMerge(
|
||||||
|
'malio-rich-text prose prose-sm max-w-none rounded-md border border-m-border bg-white p-3',
|
||||||
|
'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',
|
||||||
|
props.editorClass,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
const focusEditor = () => {
|
||||||
|
if (isInteractionLocked.value) return
|
||||||
|
closePickers()
|
||||||
|
editor.value?.commands.focus()
|
||||||
|
}
|
||||||
|
|
||||||
|
const htmlPattern = /<\/?[a-z][\s\S]*>/i
|
||||||
|
|
||||||
|
const normalizeEditorInput = (value: string | null | undefined): string => {
|
||||||
|
const content = (value ?? '').replace(/\r\n?/g, '\n')
|
||||||
|
if (htmlPattern.test(content)) return content
|
||||||
|
return content.split('\n').join('\n\n').replace(/\n{3,}/g, '\n\n')
|
||||||
|
}
|
||||||
|
|
||||||
|
const promptForLink = () => {
|
||||||
|
if (!editor.value) return
|
||||||
|
const previous = editor.value.getAttributes('link').href as string | undefined
|
||||||
|
const url = window.prompt('URL du lien (vide pour retirer)', previous ?? '')
|
||||||
|
if (url === null) return
|
||||||
|
if (url === '') {
|
||||||
|
editor.value.chain().focus().extendMarkRange('link').unsetLink().run()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
editor.value.chain().focus().extendMarkRange('link').setLink({ href: url }).run()
|
||||||
|
}
|
||||||
|
|
||||||
|
type ColorSwatch = { label: string; value: string }
|
||||||
|
|
||||||
|
const textColorSwatches: ColorSwatch[] = [
|
||||||
|
{ label: 'Rouge', value: '#bf2600' },
|
||||||
|
{ label: 'Orange', value: '#ff8b00' },
|
||||||
|
{ label: 'Jaune', value: '#ffc400' },
|
||||||
|
{ label: 'Vert', value: '#00875a' },
|
||||||
|
{ label: 'Turquoise', value: '#00a3bf' },
|
||||||
|
{ label: 'Bleu', value: '#0747a6' },
|
||||||
|
{ label: 'Violet', value: '#5243aa' },
|
||||||
|
{ label: 'Gris', value: '#42526e' },
|
||||||
|
]
|
||||||
|
|
||||||
|
const highlightSwatches: ColorSwatch[] = [
|
||||||
|
{ label: 'Rouge', value: '#fdd0c8' },
|
||||||
|
{ label: 'Orange', value: '#ffe2c2' },
|
||||||
|
{ label: 'Jaune', value: '#fff0b3' },
|
||||||
|
{ label: 'Vert', value: '#c6edd0' },
|
||||||
|
{ label: 'Turquoise', value: '#c1ecf0' },
|
||||||
|
{ label: 'Bleu', value: '#cce0ff' },
|
||||||
|
{ label: 'Violet', value: '#dfd8fa' },
|
||||||
|
{ label: 'Gris', value: '#dfe1e6' },
|
||||||
|
]
|
||||||
|
|
||||||
|
const colorPickerOpen = ref(false)
|
||||||
|
const highlightPickerOpen = ref(false)
|
||||||
|
|
||||||
|
const closePickers = () => {
|
||||||
|
colorPickerOpen.value = false
|
||||||
|
highlightPickerOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleColorPicker = () => {
|
||||||
|
highlightPickerOpen.value = false
|
||||||
|
colorPickerOpen.value = !colorPickerOpen.value
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleHighlightPicker = () => {
|
||||||
|
colorPickerOpen.value = false
|
||||||
|
highlightPickerOpen.value = !highlightPickerOpen.value
|
||||||
|
}
|
||||||
|
|
||||||
|
const applyTextColor = (value: string | null) => {
|
||||||
|
if (!editor.value) return
|
||||||
|
if (value === null) {
|
||||||
|
editor.value.chain().focus().unsetColor().run()
|
||||||
|
} else {
|
||||||
|
editor.value.chain().focus().setColor(value).run()
|
||||||
|
}
|
||||||
|
colorPickerOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const applyHighlight = (value: string | null) => {
|
||||||
|
if (!editor.value) return
|
||||||
|
if (value === null) {
|
||||||
|
editor.value.chain().focus().unsetHighlight().run()
|
||||||
|
} else {
|
||||||
|
editor.value.chain().focus().setHighlight({ color: value }).run()
|
||||||
|
}
|
||||||
|
highlightPickerOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const currentTextColor = computed(() => {
|
||||||
|
const attrs = editor.value?.getAttributes('textStyle') as { color?: string } | undefined
|
||||||
|
return attrs?.color ?? null
|
||||||
|
})
|
||||||
|
|
||||||
|
const currentHighlightColor = computed(() => {
|
||||||
|
const attrs = editor.value?.getAttributes('highlight') as { color?: string } | undefined
|
||||||
|
return attrs?.color ?? null
|
||||||
|
})
|
||||||
|
|
||||||
|
const toolbarButtons = computed(() => {
|
||||||
|
const e = editor.value
|
||||||
|
return [
|
||||||
|
{ key: 'bold', icon: 'mdi:format-bold', title: 'Gras', isActive: () => !!e?.isActive('bold'), action: () => e?.chain().focus().toggleBold().run() },
|
||||||
|
{ key: 'italic', icon: 'mdi:format-italic', title: 'Italique', isActive: () => !!e?.isActive('italic'), action: () => e?.chain().focus().toggleItalic().run() },
|
||||||
|
{ key: 'strike', icon: 'mdi:format-strikethrough', title: 'Barré', isActive: () => !!e?.isActive('strike'), action: () => e?.chain().focus().toggleStrike().run() },
|
||||||
|
{ key: 'h2', icon: 'mdi:format-header-2', title: 'Titre H2', isActive: () => !!e?.isActive('heading', { level: 2 }), action: () => e?.chain().focus().toggleHeading({ level: 2 }).run() },
|
||||||
|
{ key: 'h3', icon: 'mdi:format-header-3', title: 'Titre H3', isActive: () => !!e?.isActive('heading', { level: 3 }), action: () => e?.chain().focus().toggleHeading({ level: 3 }).run() },
|
||||||
|
{ key: 'bulletList', icon: 'mdi:format-list-bulleted', title: 'Liste à puces', isActive: () => !!e?.isActive('bulletList'), action: () => e?.chain().focus().toggleBulletList().run() },
|
||||||
|
{ key: 'orderedList', icon: 'mdi:format-list-numbered', title: 'Liste numérotée', isActive: () => !!e?.isActive('orderedList'), action: () => e?.chain().focus().toggleOrderedList().run() },
|
||||||
|
{ key: 'blockquote', icon: 'mdi:format-quote-close', title: 'Citation', isActive: () => !!e?.isActive('blockquote'), action: () => e?.chain().focus().toggleBlockquote().run() },
|
||||||
|
{ key: 'code', icon: 'mdi:code-tags', title: 'Code inline', isActive: () => !!e?.isActive('code'), action: () => e?.chain().focus().toggleCode().run() },
|
||||||
|
{ key: 'codeBlock', icon: 'mdi:code-braces-box', title: 'Bloc de code', isActive: () => !!e?.isActive('codeBlock'), action: () => e?.chain().focus().toggleCodeBlock().run() },
|
||||||
|
{ key: 'link', icon: 'mdi:link-variant', title: 'Lien', isActive: () => !!e?.isActive('link'), action: promptForLink },
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
const getCurrentValue = (): string => {
|
||||||
|
if (!editor.value) return ''
|
||||||
|
if (props.outputFormat === 'html') return editor.value.getHTML()
|
||||||
|
const storage = (editor.value.storage as unknown as Record<string, { getMarkdown?: () => string } | undefined>).markdown
|
||||||
|
return storage?.getMarkdown ? storage.getMarkdown() : editor.value.getHTML()
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDocumentMousedown = (event: MouseEvent) => {
|
||||||
|
if (!colorPickerOpen.value && !highlightPickerOpen.value) return
|
||||||
|
const target = event.target as Node | null
|
||||||
|
if (!target) return
|
||||||
|
const popovers = document.querySelectorAll(`#${editorId.value} [role="dialog"]`)
|
||||||
|
const triggers = document.querySelectorAll(`#${editorId.value} [aria-expanded]`)
|
||||||
|
for (const node of [...popovers, ...triggers]) {
|
||||||
|
if (node.contains(target)) return
|
||||||
|
}
|
||||||
|
closePickers()
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDocumentKeydown = (event: KeyboardEvent) => {
|
||||||
|
if (event.key === 'Escape' && (colorPickerOpen.value || highlightPickerOpen.value)) {
|
||||||
|
closePickers()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
document.addEventListener('mousedown', handleDocumentMousedown)
|
||||||
|
document.addEventListener('keydown', handleDocumentKeydown)
|
||||||
|
|
||||||
|
editor.value = new Editor({
|
||||||
|
content: normalizeEditorInput(props.modelValue),
|
||||||
|
editable: props.editable && !props.disabled && !props.readonly,
|
||||||
|
extensions: [
|
||||||
|
StarterKit.configure({
|
||||||
|
heading: { levels: [2, 3] },
|
||||||
|
link: {
|
||||||
|
openOnClick: false,
|
||||||
|
autolink: true,
|
||||||
|
HTMLAttributes: { rel: 'noopener noreferrer nofollow', target: '_blank' },
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
TextStyle,
|
||||||
|
Color.configure({ types: ['textStyle'] }),
|
||||||
|
Highlight.configure({ multicolor: true }),
|
||||||
|
Placeholder.configure({
|
||||||
|
placeholder: props.placeholder,
|
||||||
|
}),
|
||||||
|
Markdown.configure({
|
||||||
|
html: true,
|
||||||
|
tightLists: true,
|
||||||
|
bulletListMarker: '-',
|
||||||
|
linkify: true,
|
||||||
|
breaks: false,
|
||||||
|
transformPastedText: true,
|
||||||
|
transformCopiedText: true,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
editorProps: {
|
||||||
|
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',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
onUpdate: () => {
|
||||||
|
emit('update:modelValue', getCurrentValue())
|
||||||
|
},
|
||||||
|
onFocus: () => {
|
||||||
|
isFocused.value = true
|
||||||
|
},
|
||||||
|
onBlur: () => {
|
||||||
|
isFocused.value = false
|
||||||
|
},
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
document.removeEventListener('mousedown', handleDocumentMousedown)
|
||||||
|
document.removeEventListener('keydown', handleDocumentKeydown)
|
||||||
|
editor.value?.destroy()
|
||||||
|
})
|
||||||
|
|
||||||
|
watch(() => props.modelValue, (incoming) => {
|
||||||
|
if (!editor.value) return
|
||||||
|
if ((incoming ?? '') === getCurrentValue()) return
|
||||||
|
editor.value.commands.setContent(normalizeEditorInput(incoming), { emitUpdate: false })
|
||||||
|
})
|
||||||
|
|
||||||
|
watch(() => [props.editable, props.disabled, props.readonly], () => {
|
||||||
|
editor.value?.setEditable(props.editable && !props.disabled && !props.readonly)
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.malio-rich-text :deep(.ProseMirror) {
|
||||||
|
outline: none;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(.ProseMirror > *:first-child) {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(.ProseMirror > *:last-child) {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(.ProseMirror p.is-editor-empty:first-child::before) {
|
||||||
|
content: attr(data-placeholder);
|
||||||
|
float: left;
|
||||||
|
color: rgb(var(--m-muted));
|
||||||
|
pointer-events: none;
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(h2) {
|
||||||
|
margin: 0.75rem 0 0.5rem;
|
||||||
|
font-size: 1.5rem;
|
||||||
|
line-height: 2rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: rgb(var(--m-text));
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(h3) {
|
||||||
|
margin: 0.65rem 0 0.4rem;
|
||||||
|
font-size: 1.25rem;
|
||||||
|
line-height: 1.75rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: rgb(var(--m-text));
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(p) {
|
||||||
|
margin: 0.45rem 0;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(ul),
|
||||||
|
.malio-rich-text :deep(ol) {
|
||||||
|
margin: 0.5rem 0;
|
||||||
|
padding-left: 1.5rem;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(ul) {
|
||||||
|
list-style: disc;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(ol) {
|
||||||
|
list-style: decimal;
|
||||||
|
}
|
||||||
|
.malio-rich-text :deep(blockquote) {
|
||||||
|
margin: 0.75rem 0;
|
||||||
|
border-left: 3px solid rgb(var(--m-border));
|
||||||
|
padding-left: 0.75rem;
|
||||||
|
color: rgb(var(--m-muted));
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -88,11 +88,46 @@ describe('MalioSelect', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
await wrapper.get('button').trigger('click')
|
await wrapper.get('button').trigger('click')
|
||||||
await wrapper.findAll('li')[2].trigger('click')
|
await wrapper.findAll('li')[1].trigger('click')
|
||||||
|
|
||||||
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['be'])
|
expect(wrapper.emitted('update:modelValue')?.[0]).toEqual(['be'])
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('does not render empty option when emptyOptionLabel is empty', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {
|
||||||
|
modelValue: null,
|
||||||
|
options: [
|
||||||
|
{label: 'AM', value: 'am'},
|
||||||
|
{label: 'PM', value: 'pm'},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('li[role="option"]')
|
||||||
|
expect(items).toHaveLength(2)
|
||||||
|
expect(items[0].text()).toBe('AM')
|
||||||
|
expect(items[1].text()).toBe('PM')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('renders empty option when emptyOptionLabel is provided', async () => {
|
||||||
|
const wrapper = mount(SelectForTest, {
|
||||||
|
props: {
|
||||||
|
modelValue: null,
|
||||||
|
options: [{label: 'AM', value: 'am'}],
|
||||||
|
emptyOptionLabel: 'Choisir...',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
await wrapper.get('button').trigger('click')
|
||||||
|
|
||||||
|
const items = wrapper.findAll('li[role="option"]')
|
||||||
|
expect(items).toHaveLength(2)
|
||||||
|
expect(items[0].text()).toBe('Choisir...')
|
||||||
|
})
|
||||||
|
|
||||||
it('renders the empty option with muted text style', async () => {
|
it('renders the empty option with muted text style', async () => {
|
||||||
const wrapper = mount(SelectForTest, {
|
const wrapper = mount(SelectForTest, {
|
||||||
props: {
|
props: {
|
||||||
|
|||||||
@@ -208,10 +208,10 @@ const buttonId = `custom-select-btn-${uid}`
|
|||||||
const listboxId = `custom-select-listbox-${uid}`
|
const listboxId = `custom-select-listbox-${uid}`
|
||||||
const listRef = ref<HTMLElement | null>(null)
|
const listRef = ref<HTMLElement | null>(null)
|
||||||
const listHeight = ref(0)
|
const listHeight = ref(0)
|
||||||
const normalizedOptions = computed<Option[]>(() => [
|
const normalizedOptions = computed<Option[]>(() => {
|
||||||
{label: props.emptyOptionLabel, value: null},
|
if (!props.emptyOptionLabel) return props.options
|
||||||
...props.options,
|
return [{label: props.emptyOptionLabel, value: null}, ...props.options]
|
||||||
])
|
})
|
||||||
const mergedGroupClass = computed(() =>
|
const mergedGroupClass = computed(() =>
|
||||||
twMerge('relative w-full', props.minWidth, props.maxWidth, props.groupClass),
|
twMerge('relative w-full', props.minWidth, props.maxWidth, props.groupClass),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ type SelectCheckboxProps = {
|
|||||||
displaySelectAll?: boolean
|
displaySelectAll?: boolean
|
||||||
selectAllLabel?: string
|
selectAllLabel?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
|
groupClass?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const SelectCheckboxForTest = SelectCheckbox as DefineComponent<SelectCheckboxProps>
|
const SelectCheckboxForTest = SelectCheckbox as DefineComponent<SelectCheckboxProps>
|
||||||
@@ -175,4 +176,21 @@ describe('MalioSelectCheckbox', () => {
|
|||||||
const checkboxes = wrapper.findAll('input[type="checkbox"]')
|
const checkboxes = wrapper.findAll('input[type="checkbox"]')
|
||||||
expect((checkboxes[0].element as HTMLInputElement).checked).toBe(false)
|
expect((checkboxes[0].element as HTMLInputElement).checked).toBe(false)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('applies minWidth via twMerge so it overrides w-full (parity with MalioSelect)', () => {
|
||||||
|
const wrapper = mount(SelectCheckboxForTest, {
|
||||||
|
props: {modelValue: [], options: [], minWidth: 'w-80'},
|
||||||
|
})
|
||||||
|
const root = wrapper.find('button').element.parentElement
|
||||||
|
expect(root?.className).toContain('w-80')
|
||||||
|
expect(root?.className).not.toContain('w-full')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('applies groupClass via twMerge', () => {
|
||||||
|
const wrapper = mount(SelectCheckboxForTest, {
|
||||||
|
props: {modelValue: [], options: [], groupClass: 'mt-4'},
|
||||||
|
})
|
||||||
|
const root = wrapper.find('button').element.parentElement
|
||||||
|
expect(root?.className).toContain('mt-4')
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -2,8 +2,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<div
|
<div
|
||||||
ref="root"
|
ref="root"
|
||||||
class="relative w-full"
|
:class="mergedGroupClass"
|
||||||
:class="[minWidth, maxWidth]"
|
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
:id="buttonId"
|
:id="buttonId"
|
||||||
@@ -26,7 +25,7 @@
|
|||||||
? openDirection === 'down'
|
? openDirection === 'down'
|
||||||
? 'rounded-b-none !border-2 !border-m-primary !border-b-0'
|
? 'rounded-b-none !border-2 !border-m-primary !border-b-0'
|
||||||
: 'rounded-t-none !border-2 !border-m-primary !border-t-0'
|
: 'rounded-t-none !border-2 !border-m-primary !border-t-0'
|
||||||
: isOptionSelected
|
: isOptionSelected
|
||||||
? 'border-black'
|
? 'border-black'
|
||||||
: 'border-m-muted',
|
: 'border-m-muted',
|
||||||
disabled ? 'cursor-not-allowed border-m-muted text-black/60' : 'cursor-pointer',
|
disabled ? 'cursor-not-allowed border-m-muted text-black/60' : 'cursor-pointer',
|
||||||
@@ -45,7 +44,7 @@
|
|||||||
v-if="label"
|
v-if="label"
|
||||||
class="floating-label pointer-events-none absolute left-3 inline-block origin-left transition-transform duration-150 font-medium"
|
class="floating-label pointer-events-none absolute left-3 inline-block origin-left transition-transform duration-150 font-medium"
|
||||||
:class="[
|
:class="[
|
||||||
shouldFloatLabel ? 'top-2 z-30' : 'top-1/2 -translate-y-1/2',
|
isOpen ? 'top-2 z-30' : 'top-2',
|
||||||
hasError
|
hasError
|
||||||
? 'text-m-danger'
|
? 'text-m-danger'
|
||||||
: hasSuccess
|
: hasSuccess
|
||||||
@@ -206,6 +205,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, onBeforeUnmount, onMounted, ref, useId, nextTick} from 'vue'
|
import {computed, onBeforeUnmount, onMounted, ref, useId, nextTick} from 'vue'
|
||||||
import {Icon as IconifyIcon} from '@iconify/vue'
|
import {Icon as IconifyIcon} from '@iconify/vue'
|
||||||
|
import {twMerge} from 'tailwind-merge'
|
||||||
import Checkbox from '../checkbox/Checkbox.vue'
|
import Checkbox from '../checkbox/Checkbox.vue'
|
||||||
|
|
||||||
defineOptions({name: 'MalioSelectCheckbox', inheritAttrs: false})
|
defineOptions({name: 'MalioSelectCheckbox', inheritAttrs: false})
|
||||||
@@ -232,6 +232,7 @@ const props = withDefaults(defineProps<{
|
|||||||
displaySelectAll?: boolean
|
displaySelectAll?: boolean
|
||||||
selectAllLabel?: string
|
selectAllLabel?: string
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
|
groupClass?: string
|
||||||
}>(), {
|
}>(), {
|
||||||
options: () => [],
|
options: () => [],
|
||||||
emptyOptionLabel: '',
|
emptyOptionLabel: '',
|
||||||
@@ -249,6 +250,7 @@ const props = withDefaults(defineProps<{
|
|||||||
displaySelectAll: false,
|
displaySelectAll: false,
|
||||||
selectAllLabel: 'Tout sélectionner',
|
selectAllLabel: 'Tout sélectionner',
|
||||||
disabled: false,
|
disabled: false,
|
||||||
|
groupClass: '',
|
||||||
})
|
})
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
@@ -264,6 +266,9 @@ const listboxId = `custom-select-listbox-${uid}`
|
|||||||
const listRef = ref<HTMLElement | null>(null)
|
const listRef = ref<HTMLElement | null>(null)
|
||||||
const listHeight = ref(0)
|
const listHeight = ref(0)
|
||||||
const normalizedOptions = computed<Option[]>(() => props.options)
|
const normalizedOptions = computed<Option[]>(() => props.options)
|
||||||
|
const mergedGroupClass = computed(() =>
|
||||||
|
twMerge('relative w-full', props.minWidth, props.maxWidth, props.groupClass),
|
||||||
|
)
|
||||||
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 isOptionSelected = computed(() =>
|
const isOptionSelected = computed(() =>
|
||||||
@@ -281,6 +286,10 @@ const shouldFloatLabel = computed(() =>
|
|||||||
const selectionSummary = computed(() =>
|
const selectionSummary = computed(() =>
|
||||||
`${props.modelValue.length}/${normalizedOptions.value.length}`
|
`${props.modelValue.length}/${normalizedOptions.value.length}`
|
||||||
)
|
)
|
||||||
|
const allSelected = computed(() =>
|
||||||
|
normalizedOptions.value.length > 0
|
||||||
|
&& normalizedOptions.value.every(opt => props.modelValue.includes(opt.value)),
|
||||||
|
)
|
||||||
const describedBy = computed(() =>
|
const describedBy = computed(() =>
|
||||||
(hasError.value || hasSuccess.value || !!props.hint) ? `${buttonId}-describedby` : undefined,
|
(hasError.value || hasSuccess.value || !!props.hint) ? `${buttonId}-describedby` : undefined,
|
||||||
)
|
)
|
||||||
@@ -320,18 +329,22 @@ function open() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const labelTransformStyle = computed(() => {
|
const labelTransformStyle = computed(() => {
|
||||||
|
// label non flottant
|
||||||
if (!shouldFloatLabel.value) {
|
if (!shouldFloatLabel.value) {
|
||||||
return undefined
|
return {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// fermé ou ouverture vers le bas : comportement classique
|
||||||
if (!isOpen.value || openDirection.value === 'down') {
|
if (!isOpen.value || openDirection.value === 'down') {
|
||||||
return {
|
return {
|
||||||
transform: 'translateY(-1.15rem) scale(0.9)',
|
transform: 'translateY(-1.15rem) scale(0.9)',
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ouverture vers le haut : on remonte en fonction de la hauteur de la liste
|
||||||
const extraOffset = 8 // marge visuelle au-dessus de la liste en px
|
const extraOffset = 8 // marge visuelle au-dessus de la liste en px
|
||||||
const total = 4 + listHeight.value + extraOffset
|
const total = 4 +listHeight.value + extraOffset
|
||||||
|
// 18 ≈ 1.15rem pour garder la même base que votre flottant actuel
|
||||||
|
|
||||||
return {
|
return {
|
||||||
transform: `translateY(-${total}px) scale(0.9)`,
|
transform: `translateY(-${total}px) scale(0.9)`,
|
||||||
@@ -351,19 +364,6 @@ function toggle() {
|
|||||||
open()
|
open()
|
||||||
}
|
}
|
||||||
|
|
||||||
const allSelected = computed(() =>
|
|
||||||
normalizedOptions.value.length > 0
|
|
||||||
&& normalizedOptions.value.every(opt => props.modelValue.includes(opt.value)),
|
|
||||||
)
|
|
||||||
|
|
||||||
function toggleAll() {
|
|
||||||
if (allSelected.value) {
|
|
||||||
emit('update:modelValue', [])
|
|
||||||
} else {
|
|
||||||
emit('update:modelValue', normalizedOptions.value.map(opt => opt.value))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function isChecked(value: string | number) {
|
function isChecked(value: string | number) {
|
||||||
return props.modelValue.includes(value)
|
return props.modelValue.includes(value)
|
||||||
}
|
}
|
||||||
@@ -373,10 +373,17 @@ function toggleOption(value: string | number) {
|
|||||||
emit('update:modelValue', props.modelValue.filter(item => item !== value))
|
emit('update:modelValue', props.modelValue.filter(item => item !== value))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
emit('update:modelValue', [...props.modelValue, value])
|
emit('update:modelValue', [...props.modelValue, value])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function toggleAll() {
|
||||||
|
if (allSelected.value) {
|
||||||
|
emit('update:modelValue', [])
|
||||||
|
} else {
|
||||||
|
emit('update:modelValue', normalizedOptions.value.map(opt => opt.value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function onClickOutside(e: MouseEvent) {
|
function onClickOutside(e: MouseEvent) {
|
||||||
if (!root.value) return
|
if (!root.value) return
|
||||||
if (!root.value.contains(e.target as Node)) close()
|
if (!root.value.contains(e.target as Node)) close()
|
||||||
|
|||||||
221
app/story/input/inputRichText.story.vue
Normal file
221
app/story/input/inputRichText.story.vue
Normal file
@@ -0,0 +1,221 @@
|
|||||||
|
<template>
|
||||||
|
<Story title="Input/RichText">
|
||||||
|
<div class="grid grid-cols-1 gap-6 lg:grid-cols-2">
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Simple</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="simpleValue"
|
||||||
|
label="Note"
|
||||||
|
placeholder="Écrire ici…"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Avec contenu initial + hint</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="hintValue"
|
||||||
|
label="Description"
|
||||||
|
hint="Mise en forme via la barre d'outils"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Erreur</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="errorValue"
|
||||||
|
label="Compte-rendu"
|
||||||
|
error="Le compte-rendu doit faire au moins 20 caractères"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Succès</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="successValue"
|
||||||
|
label="Compte-rendu"
|
||||||
|
success="Compte-rendu validé"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Désactivé</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="disabledValue"
|
||||||
|
label="Note"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Readonly</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="readonlyValue"
|
||||||
|
label="Note"
|
||||||
|
readonly
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4 lg:col-span-2">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Affichage seul (editable=false)</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
:model-value="readonlyValue"
|
||||||
|
:editable="false"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4 lg:col-span-2">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Sortie HTML</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="htmlValue"
|
||||||
|
label="Article"
|
||||||
|
output-format="html"
|
||||||
|
min-height="200px"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="rounded-lg border p-4 lg:col-span-2">
|
||||||
|
<h2 class="mb-4 text-xl font-bold">Couleurs & surlignage</h2>
|
||||||
|
<MalioInputRichText
|
||||||
|
v-model="colorValue"
|
||||||
|
label="Note colorée"
|
||||||
|
output-format="html"
|
||||||
|
min-height="180px"
|
||||||
|
hint="Tester les boutons couleur du texte et surlignage (palettes Jira-like)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Story>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<docs lang="md">
|
||||||
|
# MalioInputRichText
|
||||||
|
|
||||||
|
Éditeur de texte riche basé sur **TipTap v3** + **StarterKit** + **tiptap-markdown**.
|
||||||
|
Sortie en **HTML** (par défaut) ou en **markdown**. Aligné sur le thème Malio
|
||||||
|
(couleurs `m-*`, icônes `mdi:*`, états error / success / hint).
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
## Props détaillées
|
||||||
|
|
||||||
|
### id
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Description: Identifiant HTML.
|
||||||
|
- Comportement: Généré automatiquement si non fourni (`malio-input-rich-text-…`).
|
||||||
|
|
||||||
|
### label
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Description: Label affiché au-dessus de l'éditeur.
|
||||||
|
- Comportement: Change de couleur selon l'état (focus → `m-primary`, error → `m-danger`, success → `m-success`).
|
||||||
|
|
||||||
|
### modelValue
|
||||||
|
|
||||||
|
- Type: `string | null | undefined`
|
||||||
|
- Description: Contenu de l'éditeur (markdown ou HTML selon `outputFormat`).
|
||||||
|
- Comportement: `v-model` ; sync bidirectionnelle.
|
||||||
|
|
||||||
|
### placeholder
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Défaut: `''`
|
||||||
|
- Description: Texte affiché quand l'éditeur est vide.
|
||||||
|
|
||||||
|
### minHeight
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Défaut: `160px`
|
||||||
|
- Description: Hauteur minimale de la zone d'édition (CSS valid value).
|
||||||
|
|
||||||
|
### editable
|
||||||
|
|
||||||
|
- Type: `boolean`
|
||||||
|
- Défaut: `true`
|
||||||
|
- Description: `false` → mode affichage seul, **toolbar masquée**, contenu rendu en `prose`.
|
||||||
|
|
||||||
|
### disabled
|
||||||
|
|
||||||
|
- Type: `boolean`
|
||||||
|
- Défaut: `false`
|
||||||
|
- Description: Désactive l'édition et la toolbar (opacité réduite).
|
||||||
|
|
||||||
|
### readonly
|
||||||
|
|
||||||
|
- Type: `boolean`
|
||||||
|
- Défaut: `false`
|
||||||
|
- Description: Lecture seule (toolbar visible mais désactivée, pas de saisie).
|
||||||
|
|
||||||
|
### hint / error / success
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Description: Messages contextuels affichés sous l'éditeur.
|
||||||
|
- Priorité: `error` > `success` > `hint`.
|
||||||
|
|
||||||
|
### outputFormat
|
||||||
|
|
||||||
|
- Type: `'markdown' | 'html'`
|
||||||
|
- Défaut: `'html'`
|
||||||
|
- Description: Format émis dans `update:modelValue`.
|
||||||
|
- `html` : utilise `editor.getHTML()`.
|
||||||
|
- `markdown` : utilise `tiptap-markdown` (`getMarkdown()`).
|
||||||
|
|
||||||
|
### groupClass / labelClass / editorClass
|
||||||
|
|
||||||
|
- Type: `string`
|
||||||
|
- Description: Classes Tailwind additionnelles fusionnées via `twMerge` pour override.
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
## Toolbar
|
||||||
|
|
||||||
|
Boutons (icônes `mdi:*`) :
|
||||||
|
|
||||||
|
- Gras, Italique, Barré
|
||||||
|
- Titre H2, Titre H3
|
||||||
|
- Liste à puces, Liste numérotée
|
||||||
|
- Citation
|
||||||
|
- Code inline, Bloc de code
|
||||||
|
- Lien (prompt URL ; vide pour retirer)
|
||||||
|
- Couleur du texte (palette de 8 swatches + reset)
|
||||||
|
- Surlignage (palette de 8 swatches + reset)
|
||||||
|
- Annuler / Rétablir
|
||||||
|
|
||||||
|
Les palettes couleur/surlignage s'ouvrent en popover sous leur bouton.
|
||||||
|
Fermeture : clic sur un swatch, clic en dehors, ou touche **Échap**.
|
||||||
|
|
||||||
|
> Les couleurs et surlignages ne sont **pas persistés en markdown** (spec Markdown ne couvre pas la couleur). Pour préserver les couleurs au save/reload, utiliser `output-format="html"`.
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
## Accessibilité
|
||||||
|
|
||||||
|
- Le label est lié à la zone d'édition via `for` / `id`.
|
||||||
|
- `aria-invalid="true"` sur la zone d'édition en cas d'erreur.
|
||||||
|
- `aria-describedby` référence le message d'erreur / succès / hint.
|
||||||
|
- Boutons toolbar : `aria-pressed` reflète l'état actif, `aria-label` pour l'usage screen-reader.
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
## Events
|
||||||
|
|
||||||
|
### update:modelValue
|
||||||
|
|
||||||
|
- Émis à chaque modification du contenu.
|
||||||
|
- Payload : `string` (markdown ou HTML selon `outputFormat`).
|
||||||
|
|
||||||
|
</docs>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import {ref} from 'vue'
|
||||||
|
import MalioInputRichText from '../../components/malio/input/InputRichText.vue'
|
||||||
|
|
||||||
|
const simpleValue = ref('')
|
||||||
|
const hintValue = ref('## Titre\n\nUn paragraphe avec du **gras**, de l\'*italique* et un [lien](https://malio.fr).')
|
||||||
|
const errorValue = ref('Trop court')
|
||||||
|
const successValue = ref('Tout est bon de mon côté.')
|
||||||
|
const disabledValue = ref('Contenu indisponible.')
|
||||||
|
const readonlyValue = ref('## Compte-rendu\n\n- Point 1\n- Point 2\n\n> Citation importante')
|
||||||
|
const htmlValue = ref('<p>Contenu <strong>riche</strong>.</p>')
|
||||||
|
const colorValue = ref('<p>Sélectionner du texte puis utiliser les boutons <span style="color: #bf2600">couleur</span> ou <mark data-color="#fff0b3" style="background-color: #fff0b3">surlignage</mark>.</p>')
|
||||||
|
</script>
|
||||||
@@ -12,6 +12,7 @@ export default defineConfig({
|
|||||||
resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
'@': path.resolve(__dirname, './'),
|
'@': path.resolve(__dirname, './'),
|
||||||
|
'tiptap-markdown': path.resolve(__dirname, 'node_modules/tiptap-markdown/dist/tiptap-markdown.es.js'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
css: {
|
css: {
|
||||||
@@ -19,6 +20,17 @@ export default defineConfig({
|
|||||||
plugins: [tailwindcss(), autoprefixer()],
|
plugins: [tailwindcss(), autoprefixer()],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
ssr: {
|
||||||
|
noExternal: ['tiptap-markdown', /^@tiptap\//],
|
||||||
|
},
|
||||||
|
optimizeDeps: {
|
||||||
|
include: [
|
||||||
|
'tiptap-markdown',
|
||||||
|
'@tiptap/vue-3',
|
||||||
|
'@tiptap/starter-kit',
|
||||||
|
'@tiptap/extension-placeholder',
|
||||||
|
],
|
||||||
|
},
|
||||||
},
|
},
|
||||||
plugins: [HstVue()],
|
plugins: [HstVue()],
|
||||||
})
|
})
|
||||||
|
|||||||
760
package-lock.json
generated
760
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
10
package.json
10
package.json
@@ -42,7 +42,15 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@nuxt/icon": "^2.2.1",
|
"@nuxt/icon": "^2.2.1",
|
||||||
"@nuxtjs/tailwindcss": "^6.14.0",
|
"@nuxtjs/tailwindcss": "^6.14.0",
|
||||||
|
"@tiptap/extension-color": "^3.22.5",
|
||||||
|
"@tiptap/extension-highlight": "^3.22.5",
|
||||||
|
"@tiptap/extension-placeholder": "^3.22.5",
|
||||||
|
"@tiptap/extension-text-style": "^3.22.5",
|
||||||
|
"@tiptap/pm": "^3.22.5",
|
||||||
|
"@tiptap/starter-kit": "^3.22.5",
|
||||||
|
"@tiptap/vue-3": "^3.22.5",
|
||||||
"maska": "^3.2.0",
|
"maska": "^3.2.0",
|
||||||
"tailwind-merge": "^3.3.1"
|
"tailwind-merge": "^3.3.1",
|
||||||
|
"tiptap-markdown": "^0.9.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ export default {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
fontFamily: {
|
fontFamily: {
|
||||||
sans: ['"Helvetica Neue"', 'Helvetica', 'Arial', 'sans-serif'],
|
sans: ['"Inter"', 'Helvetica Neue', 'Helvetica', 'Arial', 'sans-serif'],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
Reference in New Issue
Block a user