@ktortu/aaa 0.9.0 → 0.9.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,275 @@
1
+ import * as i0 from '@angular/core';
2
+ import { InjectionToken, Provider, OnDestroy } from '@angular/core';
3
+ import { Observable } from 'rxjs';
4
+
5
+ /** Politesse de l'annonce au lecteur d'écran (relayée à `LiveAnnouncer`). */
6
+ type KtSnackbarPoliteness = 'polite' | 'assertive';
7
+ /** Bord d'ancrage de la snackbar dans le viewport. */
8
+ type KtSnackbarPosition = 'top' | 'bottom';
9
+ /**
10
+ * Régime temporel de la snackbar :
11
+ * - `'auto'` : disparition automatique après `duration`, **mise en pause au survol et au focus**
12
+ * clavier (WCAG 1.4.13 / 2.2.1). Conforme **AA**. C'est le défaut.
13
+ * - `'manual'` : **aucune** minuterie, la snackbar reste jusqu'à fermeture explicite
14
+ * (bouton de fermeture ou `ref.dismiss()`). Conforme **2.2.3 (AAA)** au sens strict.
15
+ *
16
+ * Conformément aux recommandations d'accessibilité (Roselli/Soueidan), la snackbar ne porte **pas
17
+ * d'action interactive** (ex. « Annuler ») : une action auto-disparaissante n'est pas atteignable et
18
+ * une live region n'expose pas ses boutons. Pour un undo, préférez un mécanisme atteignable côté app
19
+ * (Ctrl+Z, bannière persistante).
20
+ */
21
+ type KtSnackbarTiming = 'auto' | 'manual';
22
+ /**
23
+ * Variante sémantique de la snackbar. Pilote uniquement l'**apparence** (couleur d'accent + icône),
24
+ * via l'attribut `data-variant` et les tokens CSS `--snackbar-*` — donc entièrement gérée par le
25
+ * thème, sans logique TypeScript. Découplée de la `politeness` (une « erreur » n'est pas forcément
26
+ * une urgence assertive). La couleur n'est jamais le seul indice : chaque variante porte une **icône**
27
+ * de forme distincte (WCAG 1.4.1).
28
+ * - `'neutral'` : pastille neutre, sans icône (défaut) ;
29
+ * - `'info' | 'success' | 'warning' | 'error'` : accent + icône dédiés.
30
+ */
31
+ type KtSnackbarVariant = 'neutral' | 'info' | 'success' | 'warning' | 'error';
32
+ /**
33
+ * Défauts de la snackbar, injectables via `provideKtSnackbar` / `KT_SNACKBAR_CONFIG`.
34
+ * Tous les champs sont surchargeables **par appel** via les options de `KtSnackbar.open()`.
35
+ *
36
+ * Résolution en cascade (convention de la lib) : `option d'open ?? KT_SNACKBAR_CONFIG ?? défaut`.
37
+ */
38
+ interface KtSnackbarConfig {
39
+ /**
40
+ * Durée d'affichage en régime `'auto'`. Soit un **nombre fixe** (ms), soit le sentinel
41
+ * **`'reading-time'`** (DÉFAUT) qui **calcule** la durée d'après la longueur du message :
42
+ * `clamp(longueur × readingTimePerChar, readingTimeMin, readingTimeMax)`. Passer un nombre à
43
+ * `open()` force donc une durée fixe pour cet appel.
44
+ *
45
+ * ⚠️ Gardez le message **court** : une snackbar est un message transitoire (pas un paragraphe).
46
+ * Un message long fait grimper la durée jusqu'au plafond `readingTimeMax`.
47
+ * @default 'reading-time'
48
+ */
49
+ duration: number | 'reading-time';
50
+ /** Plancher (ms) de la durée calculée (`'reading-time'`) — laisse le temps de lire un message court. @default 4000 */
51
+ readingTimeMin: number;
52
+ /** Plafond (ms) de la durée calculée (`'reading-time'`) — borne un message long. @default 10000 */
53
+ readingTimeMax: number;
54
+ /** Coefficient de lecture : millisecondes ajoutées par caractère du message. @default 60 (~200 mots/min) */
55
+ readingTimePerChar: number;
56
+ /** Régime temporel (cf. {@link KtSnackbarTiming}). @default 'auto' (AA + pause) */
57
+ timing: KtSnackbarTiming;
58
+ /** Bord d'ancrage dans le viewport. @default 'bottom' */
59
+ position: KtSnackbarPosition;
60
+ /** Politesse de l'annonce lecteur d'écran. @default 'polite' */
61
+ politeness: KtSnackbarPoliteness;
62
+ /** Variante sémantique (apparence seule, cf. {@link KtSnackbarVariant}). @default 'neutral' */
63
+ variant: KtSnackbarVariant;
64
+ /** Affiche un bouton de fermeture (cible 44px, AAA). @default true */
65
+ closable: boolean;
66
+ /** Nom accessible du bouton de fermeture. @default 'Close' (FR fourni en lot L3) */
67
+ closeLabel: string;
68
+ /**
69
+ * Taille maximale de la file FIFO (snackbar affichée + en attente). Au-delà, les **plus anciennes
70
+ * en attente** sont retirées silencieusement. Une seule snackbar est visible à la fois. @default 3
71
+ */
72
+ max: number;
73
+ }
74
+ /** Défauts AAA-orientés de la snackbar (anglais neutre). */
75
+ declare const KT_SNACKBAR_DEFAULTS: KtSnackbarConfig;
76
+ declare const KT_SNACKBAR_CONFIG: InjectionToken<Partial<KtSnackbarConfig>>;
77
+ /**
78
+ * Options ponctuelles d'ouverture d'une snackbar : un sous-ensemble (toutes facultatives) de la
79
+ * config, prioritaire sur `KT_SNACKBAR_CONFIG` et sur les défauts.
80
+ */
81
+ type KtSnackbarOptions = Partial<KtSnackbarConfig>;
82
+ /**
83
+ * Fournit des défauts de snackbar pour un sous-arbre ou l'application entière.
84
+ *
85
+ * @example
86
+ * ```ts
87
+ * // app.config.ts — bascule TOUTE l'app en AAA strict (aucune disparition automatique)
88
+ * providers: [provideKtSnackbar({ timing: 'manual' })]
89
+ * ```
90
+ * @example
91
+ * ```ts
92
+ * providers: [provideKtSnackbar({ duration: 8000, position: 'top' })]
93
+ * ```
94
+ */
95
+ declare function provideKtSnackbar(config: Partial<KtSnackbarConfig>): Provider;
96
+
97
+ /**
98
+ * Raison de fermeture d'une snackbar, transmise par {@link KtSnackbarRef.afterDismissed}.
99
+ * - `'timeout'` : la minuterie (régime `'auto'`) est arrivée à échéance ;
100
+ * - `'dismiss'` : fermeture explicite (bouton de fermeture, `Échap`, ou `ref.dismiss()`) ;
101
+ * - `'replaced'` : la snackbar a été retirée de la file sans être affichée (file pleine — cf. `max`).
102
+ */
103
+ type KtSnackbarDismissReason = 'timeout' | 'dismiss' | 'replaced';
104
+ /**
105
+ * Référence d'une snackbar ouverte, renvoyée par `KtSnackbar.open()`. Permet de fermer la
106
+ * snackbar par programmation et d'observer sa fermeture.
107
+ *
108
+ * Le débutant peut l'ignorer (`snackbar.open('…')`) ; le code avancé s'en sert pour piloter la
109
+ * fermeture et réagir à la raison de fin.
110
+ *
111
+ * @example
112
+ * ```ts
113
+ * const ref = snackbar.open('Brouillon enregistré');
114
+ * ref.afterDismissed().subscribe((reason) => console.log(reason)); // 'timeout' | 'dismiss' | 'replaced'
115
+ * // plus tard : ref.dismiss();
116
+ * ```
117
+ */
118
+ declare class KtSnackbarRef {
119
+ private readonly onDismiss;
120
+ private readonly _afterDismissed;
121
+ private settled;
122
+ /**
123
+ * @param onDismiss Rappel exécuté une fois à la fermeture (animation de sortie + démontage de
124
+ * l'overlay + avance de la file), fourni par le service.
125
+ * @internal Construit par `KtSnackbar` — n'instanciez pas `KtSnackbarRef` directement.
126
+ */
127
+ constructor(onDismiss: (reason: KtSnackbarDismissReason) => void);
128
+ /**
129
+ * Ferme la snackbar (idempotent). Déclenche la sortie/démontage puis émet la raison sur
130
+ * `afterDismissed()`.
131
+ * @param reason Raison de fermeture. @default 'dismiss'
132
+ */
133
+ dismiss(reason?: KtSnackbarDismissReason): void;
134
+ /** Émet une fois (puis complète) à la fermeture, avec la {@link KtSnackbarDismissReason}. */
135
+ afterDismissed(): Observable<KtSnackbarDismissReason>;
136
+ }
137
+
138
+ /**
139
+ * Service d'ouverture des snackbars `@ktortu/aaa` — feedback **non bloquant** qui confirme une
140
+ * action ou signale un événement transitoire sans interrompre la tâche (n'utilisez PAS de snackbar
141
+ * pour une information critique à acquitter : préférez le dialog).
142
+ *
143
+ * Architecture a11y :
144
+ * - **CDK Overlay** pour le panneau visuel, **CDK LiveAnnouncer** pour l'annonce — **un seul canal**
145
+ * d'annonce (le panneau n'est pas une live region) ;
146
+ * - **le focus n'est jamais déplacé** vers la snackbar (RGAA « message de statut » / WCAG 4.1.3) ;
147
+ * - disparition automatique **en pause au survol et au focus** (régime `'auto'`, défaut AA), ou
148
+ * persistante (`timing: 'manual'`, AAA) ;
149
+ * - **file FIFO** : une seule snackbar visible, les suivantes patientent (live region non saturée).
150
+ * La file est bornée par `max` (défaut 3) ; les messages **identiques** sont fusionnés (coalescing) ;
151
+ * - **Échap** ferme la snackbar affichée (la plus récente).
152
+ *
153
+ * Requiert côté hôte les styles `@angular/cdk/overlay-prebuilt.css` **et**
154
+ * `@angular/cdk/a11y-prebuilt.css` (ce dernier masque l'élément du LiveAnnouncer).
155
+ *
156
+ * @example
157
+ * ```ts
158
+ * private readonly snackbar = inject(KtSnackbar);
159
+ * this.snackbar.open('Brouillon enregistré'); // disparaît, annonce polie
160
+ * this.snackbar.open('Fichier supprimé', { variant: 'success' }); // variante (couleur + icône)
161
+ * this.snackbar.open('Hors ligne', { timing: 'manual' }); // reste jusqu'à fermeture (AAA)
162
+ * ```
163
+ */
164
+ declare class KtSnackbar implements OnDestroy {
165
+ private readonly overlay;
166
+ private readonly injector;
167
+ private readonly liveAnnouncer;
168
+ private readonly config;
169
+ private readonly platformId;
170
+ private readonly doc;
171
+ /** File FIFO : `queue[0]` est la snackbar affichée dès qu'elle est attachée. */
172
+ private readonly queue;
173
+ private active;
174
+ private escapeRegistered;
175
+ /**
176
+ * Ouvre une snackbar affichant `message`. Les `options` priment sur `KT_SNACKBAR_CONFIG`, qui
177
+ * prime sur les défauts. Renvoie une {@link KtSnackbarRef} (ignorable dans le cas simple).
178
+ *
179
+ * Une seule snackbar est visible : si une autre est affichée, celle-ci patiente en file (FIFO).
180
+ * Un `message` identique à une snackbar déjà affichée ou en attente est **fusionné** : on renvoie
181
+ * alors la référence existante sans rien ré-empiler.
182
+ *
183
+ * @param message Texte affiché et annoncé au lecteur d'écran.
184
+ * @param options Surcharges ponctuelles (durée, régime, position, politesse, variante, fermeture).
185
+ * @returns La référence de la snackbar (existante en cas de fusion).
186
+ */
187
+ open(message: string, options?: KtSnackbarOptions): KtSnackbarRef;
188
+ ngOnDestroy(): void;
189
+ /** Borne la file (affichée + en attente) : retire les plus ANCIENNES en attente au-delà de `max`. */
190
+ private enforceMax;
191
+ /** Affiche la tête de file si rien n'est actuellement visible. */
192
+ private showHead;
193
+ /** Retire l'élément de la file ; s'il est affiché, joue la sortie puis démonte et avance la file. */
194
+ private handleDismiss;
195
+ private readonly onDocumentKeydown;
196
+ private registerEscape;
197
+ private unregisterEscape;
198
+ /**
199
+ * Durée effective (ms) en régime `'auto'`. Un nombre est pris tel quel (durée fixe) ; le sentinel
200
+ * `'reading-time'` calcule `clamp(longueur × perChar, min, max)`.
201
+ */
202
+ private computeDuration;
203
+ /** Résolution en cascade `option ?? KT_SNACKBAR_CONFIG ?? défaut`, champ par champ. */
204
+ private resolve;
205
+ static ɵfac: i0.ɵɵFactoryDeclaration<KtSnackbar, never>;
206
+ static ɵprov: i0.ɵɵInjectableDeclaration<KtSnackbar>;
207
+ }
208
+
209
+ /**
210
+ * Données résolues passées au conteneur visuel à l'ouverture (injectées via {@link KT_SNACKBAR_CONTEXT}).
211
+ * Produites par `KtSnackbar` après résolution `option ?? config ?? défaut`.
212
+ */
213
+ interface KtSnackbarContext {
214
+ /** Message affiché et annoncé. */
215
+ readonly message: string;
216
+ /** Variante sémantique (apparence : accent + icône). */
217
+ readonly variant: KtSnackbarVariant;
218
+ /** Affiche le bouton de fermeture. */
219
+ readonly closable: boolean;
220
+ /** Nom accessible du bouton de fermeture. */
221
+ readonly closeLabel: string;
222
+ /** Régime temporel (la minuterie n'existe qu'en `'auto'`). */
223
+ readonly timing: KtSnackbarTiming;
224
+ /** Durée (ms) avant disparition automatique en régime `'auto'`. */
225
+ readonly duration: number;
226
+ }
227
+ declare const KT_SNACKBAR_CONTEXT: InjectionToken<KtSnackbarContext>;
228
+ /**
229
+ * Conteneur visuel d'une snackbar. **Volontairement pas une live region** (aucun `role`/`aria-live`
230
+ * sur l'hôte) : l'annonce passe par un canal UNIQUE, le `LiveAnnouncer` du service. On évite ainsi
231
+ * la double-annonce et l'écrasement de politesse observés sur d'autres libs.
232
+ *
233
+ * En régime `'auto'`, la minuterie de disparition se met **en pause au survol et au focus** clavier
234
+ * (WCAG 1.4.13 / 2.2.1) et reprend quand ni le pointeur ni le focus ne sont sur la snackbar.
235
+ * L'icône de variante est **décorative** (`aria-hidden`) : la forme distincte sert d'indice non
236
+ * coloré (WCAG 1.4.1), le sens reste porté par le texte.
237
+ *
238
+ * @internal Monté par `KtSnackbar` via un `ComponentPortal`.
239
+ */
240
+ declare class KtSnackbarContainer {
241
+ protected readonly context: KtSnackbarContext;
242
+ private readonly ref;
243
+ private readonly platformId;
244
+ private readonly host;
245
+ /** Vrai pendant l'animation de sortie (déclenche la classe `kt-snackbar--leaving`). */
246
+ protected readonly leaving: i0.WritableSignal<boolean>;
247
+ private timer;
248
+ private remaining;
249
+ private deadline;
250
+ private hovered;
251
+ private focused;
252
+ constructor();
253
+ protected close(): void;
254
+ /**
255
+ * Joue l'animation de sortie puis invoque `done` (démontage de l'overlay côté service). Appelé par
256
+ * le service à la fermeture. Sans animation (durée nulle, `prefers-reduced-motion`, ou SSR), `done`
257
+ * est invoqué immédiatement. Filet de sécurité par `setTimeout` si `transitionend` ne se déclenche pas.
258
+ */
259
+ playExit(done: () => void): void;
260
+ protected onPointerEnter(): void;
261
+ protected onPointerLeave(): void;
262
+ protected onFocusEnter(): void;
263
+ protected onFocusLeave(): void;
264
+ private maybeResume;
265
+ private startTimer;
266
+ private pauseTimer;
267
+ private clearTimer;
268
+ /** Durée (ms) de la transition de sortie, lue sur l'hôte (0 si aucune — ex. reduced-motion). */
269
+ private exitDurationMs;
270
+ static ɵfac: i0.ɵɵFactoryDeclaration<KtSnackbarContainer, never>;
271
+ static ɵcmp: i0.ɵɵComponentDeclaration<KtSnackbarContainer, "kt-snackbar-container", never, {}, {}, never, never, true, never>;
272
+ }
273
+
274
+ export { KT_SNACKBAR_CONFIG, KT_SNACKBAR_CONTEXT, KT_SNACKBAR_DEFAULTS, KtSnackbar, KtSnackbarContainer, KtSnackbarRef, provideKtSnackbar };
275
+ export type { KtSnackbarConfig, KtSnackbarContext, KtSnackbarDismissReason, KtSnackbarOptions, KtSnackbarPoliteness, KtSnackbarPosition, KtSnackbarTiming, KtSnackbarVariant };
@@ -5,5 +5,6 @@ export * from '@ktortu/aaa/dialog';
5
5
  export * from '@ktortu/aaa/forms';
6
6
  export * from '@ktortu/aaa/i18n';
7
7
  export * from '@ktortu/aaa/menu';
8
+ export * from '@ktortu/aaa/snackbar';
8
9
  export * from '@ktortu/aaa/tabs';
9
10
  export * from '@ktortu/aaa/tooltip';