@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.
- package/README.md +5 -3
- package/fesm2022/ktortu-aaa-cdk.mjs +1 -0
- package/fesm2022/ktortu-aaa-cdk.mjs.map +1 -1
- package/fesm2022/ktortu-aaa-dialog.mjs +69 -16
- package/fesm2022/ktortu-aaa-dialog.mjs.map +1 -1
- package/fesm2022/ktortu-aaa-forms.mjs +328 -69
- package/fesm2022/ktortu-aaa-forms.mjs.map +1 -1
- package/fesm2022/ktortu-aaa-i18n.mjs +27 -2
- package/fesm2022/ktortu-aaa-i18n.mjs.map +1 -1
- package/fesm2022/ktortu-aaa-menu.mjs.map +1 -1
- package/fesm2022/ktortu-aaa-snackbar.mjs +465 -0
- package/fesm2022/ktortu-aaa-snackbar.mjs.map +1 -0
- package/fesm2022/ktortu-aaa-tabs.mjs.map +1 -1
- package/fesm2022/ktortu-aaa.mjs +1 -0
- package/fesm2022/ktortu-aaa.mjs.map +1 -1
- package/forms/checkbox/checkbox-group.css +0 -8
- package/forms/chips/chip-list.css +5 -0
- package/forms/radio/radio-group.css +1 -9
- package/forms/styles/field-box.css +3 -1
- package/forms/styles/field-pending.css +46 -0
- package/forms/styles/select-panel.css +4 -0
- package/package.json +5 -1
- package/snackbar/snackbar-tokens.css +53 -0
- package/snackbar/snackbar.css +175 -0
- package/styles/forms.css +1 -0
- package/styles/snackbar.css +9 -0
- package/styles/styles.css +1 -0
- package/types/ktortu-aaa-dialog.d.ts +89 -27
- package/types/ktortu-aaa-forms.d.ts +180 -24
- package/types/ktortu-aaa-i18n.d.ts +3 -0
- package/types/ktortu-aaa-snackbar.d.ts +275 -0
- package/types/ktortu-aaa.d.ts +1 -0
|
@@ -0,0 +1,465 @@
|
|
|
1
|
+
import { LiveAnnouncer } from '@angular/cdk/a11y';
|
|
2
|
+
import { Overlay } from '@angular/cdk/overlay';
|
|
3
|
+
import { ComponentPortal } from '@angular/cdk/portal';
|
|
4
|
+
import { isPlatformBrowser, DOCUMENT } from '@angular/common';
|
|
5
|
+
import * as i0 from '@angular/core';
|
|
6
|
+
import { InjectionToken, inject, PLATFORM_ID, ElementRef, signal, afterNextRender, DestroyRef, ChangeDetectionStrategy, Component, Injector, Injectable } from '@angular/core';
|
|
7
|
+
import { Subject } from 'rxjs';
|
|
8
|
+
|
|
9
|
+
/** Défauts AAA-orientés de la snackbar (anglais neutre). */
|
|
10
|
+
const KT_SNACKBAR_DEFAULTS = {
|
|
11
|
+
// Stryker disable next-line StringLiteral: équivalent — computeDuration() traite toute valeur non numérique comme reading-time
|
|
12
|
+
duration: 'reading-time',
|
|
13
|
+
readingTimeMin: 4000,
|
|
14
|
+
readingTimeMax: 10000,
|
|
15
|
+
readingTimePerChar: 60,
|
|
16
|
+
timing: 'auto',
|
|
17
|
+
position: 'bottom',
|
|
18
|
+
politeness: 'polite',
|
|
19
|
+
variant: 'neutral',
|
|
20
|
+
closable: true,
|
|
21
|
+
closeLabel: 'Close',
|
|
22
|
+
max: 3,
|
|
23
|
+
};
|
|
24
|
+
// Stryker disable next-line StringLiteral: libellé de debug du token d'injection (sans effet runtime)
|
|
25
|
+
const KT_SNACKBAR_CONFIG = new InjectionToken('KT_SNACKBAR_CONFIG');
|
|
26
|
+
/**
|
|
27
|
+
* Fournit des défauts de snackbar pour un sous-arbre ou l'application entière.
|
|
28
|
+
*
|
|
29
|
+
* @example
|
|
30
|
+
* ```ts
|
|
31
|
+
* // app.config.ts — bascule TOUTE l'app en AAA strict (aucune disparition automatique)
|
|
32
|
+
* providers: [provideKtSnackbar({ timing: 'manual' })]
|
|
33
|
+
* ```
|
|
34
|
+
* @example
|
|
35
|
+
* ```ts
|
|
36
|
+
* providers: [provideKtSnackbar({ duration: 8000, position: 'top' })]
|
|
37
|
+
* ```
|
|
38
|
+
*/
|
|
39
|
+
function provideKtSnackbar(config) {
|
|
40
|
+
return { provide: KT_SNACKBAR_CONFIG, useValue: config };
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Référence d'une snackbar ouverte, renvoyée par `KtSnackbar.open()`. Permet de fermer la
|
|
45
|
+
* snackbar par programmation et d'observer sa fermeture.
|
|
46
|
+
*
|
|
47
|
+
* Le débutant peut l'ignorer (`snackbar.open('…')`) ; le code avancé s'en sert pour piloter la
|
|
48
|
+
* fermeture et réagir à la raison de fin.
|
|
49
|
+
*
|
|
50
|
+
* @example
|
|
51
|
+
* ```ts
|
|
52
|
+
* const ref = snackbar.open('Brouillon enregistré');
|
|
53
|
+
* ref.afterDismissed().subscribe((reason) => console.log(reason)); // 'timeout' | 'dismiss' | 'replaced'
|
|
54
|
+
* // plus tard : ref.dismiss();
|
|
55
|
+
* ```
|
|
56
|
+
*/
|
|
57
|
+
class KtSnackbarRef {
|
|
58
|
+
onDismiss;
|
|
59
|
+
_afterDismissed = new Subject();
|
|
60
|
+
settled = false;
|
|
61
|
+
/**
|
|
62
|
+
* @param onDismiss Rappel exécuté une fois à la fermeture (animation de sortie + démontage de
|
|
63
|
+
* l'overlay + avance de la file), fourni par le service.
|
|
64
|
+
* @internal Construit par `KtSnackbar` — n'instanciez pas `KtSnackbarRef` directement.
|
|
65
|
+
*/
|
|
66
|
+
constructor(onDismiss) {
|
|
67
|
+
this.onDismiss = onDismiss;
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Ferme la snackbar (idempotent). Déclenche la sortie/démontage puis émet la raison sur
|
|
71
|
+
* `afterDismissed()`.
|
|
72
|
+
* @param reason Raison de fermeture. @default 'dismiss'
|
|
73
|
+
*/
|
|
74
|
+
dismiss(reason = 'dismiss') {
|
|
75
|
+
if (this.settled)
|
|
76
|
+
return;
|
|
77
|
+
this.settled = true;
|
|
78
|
+
this.onDismiss(reason);
|
|
79
|
+
this._afterDismissed.next(reason);
|
|
80
|
+
this._afterDismissed.complete();
|
|
81
|
+
}
|
|
82
|
+
/** Émet une fois (puis complète) à la fermeture, avec la {@link KtSnackbarDismissReason}. */
|
|
83
|
+
afterDismissed() {
|
|
84
|
+
return this._afterDismissed.asObservable();
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// Stryker disable next-line StringLiteral: libellé de debug du token d'injection (sans effet runtime)
|
|
89
|
+
const KT_SNACKBAR_CONTEXT = new InjectionToken('KT_SNACKBAR_CONTEXT');
|
|
90
|
+
/**
|
|
91
|
+
* Conteneur visuel d'une snackbar. **Volontairement pas une live region** (aucun `role`/`aria-live`
|
|
92
|
+
* sur l'hôte) : l'annonce passe par un canal UNIQUE, le `LiveAnnouncer` du service. On évite ainsi
|
|
93
|
+
* la double-annonce et l'écrasement de politesse observés sur d'autres libs.
|
|
94
|
+
*
|
|
95
|
+
* En régime `'auto'`, la minuterie de disparition se met **en pause au survol et au focus** clavier
|
|
96
|
+
* (WCAG 1.4.13 / 2.2.1) et reprend quand ni le pointeur ni le focus ne sont sur la snackbar.
|
|
97
|
+
* L'icône de variante est **décorative** (`aria-hidden`) : la forme distincte sert d'indice non
|
|
98
|
+
* coloré (WCAG 1.4.1), le sens reste porté par le texte.
|
|
99
|
+
*
|
|
100
|
+
* @internal Monté par `KtSnackbar` via un `ComponentPortal`.
|
|
101
|
+
*/
|
|
102
|
+
class KtSnackbarContainer {
|
|
103
|
+
context = inject(KT_SNACKBAR_CONTEXT);
|
|
104
|
+
ref = inject(KtSnackbarRef);
|
|
105
|
+
platformId = inject(PLATFORM_ID);
|
|
106
|
+
host = inject(ElementRef).nativeElement;
|
|
107
|
+
/** Vrai pendant l'animation de sortie (déclenche la classe `kt-snackbar--leaving`). */
|
|
108
|
+
leaving = signal(false, /* @ts-ignore */
|
|
109
|
+
...(ngDevMode ? [{ debugName: "leaving" }] : /* istanbul ignore next */ []));
|
|
110
|
+
timer;
|
|
111
|
+
remaining = this.context.duration;
|
|
112
|
+
deadline = 0;
|
|
113
|
+
hovered = false;
|
|
114
|
+
focused = false;
|
|
115
|
+
constructor() {
|
|
116
|
+
if (this.context.timing === 'auto' && isPlatformBrowser(this.platformId)) {
|
|
117
|
+
afterNextRender(() => this.startTimer(this.context.duration));
|
|
118
|
+
}
|
|
119
|
+
inject(DestroyRef).onDestroy(() => this.clearTimer());
|
|
120
|
+
}
|
|
121
|
+
close() {
|
|
122
|
+
this.ref.dismiss('dismiss');
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Joue l'animation de sortie puis invoque `done` (démontage de l'overlay côté service). Appelé par
|
|
126
|
+
* le service à la fermeture. Sans animation (durée nulle, `prefers-reduced-motion`, ou SSR), `done`
|
|
127
|
+
* est invoqué immédiatement. Filet de sécurité par `setTimeout` si `transitionend` ne se déclenche pas.
|
|
128
|
+
*/
|
|
129
|
+
playExit(done) {
|
|
130
|
+
this.clearTimer();
|
|
131
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
132
|
+
done();
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
135
|
+
const durationMs = this.exitDurationMs();
|
|
136
|
+
// Stryker disable all: animation de sortie pilotée par une transition CSS réelle. En jsdom
|
|
137
|
+
// `transitionDuration` est vide → `exitDurationMs()` vaut 0 → ce code n'est jamais atteint.
|
|
138
|
+
// Comportement (jeu de l'animation puis démontage) vérifié en e2e (sortie + reduced-motion).
|
|
139
|
+
if (durationMs <= 0) {
|
|
140
|
+
done();
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
this.leaving.set(true);
|
|
144
|
+
let finished = false;
|
|
145
|
+
const finish = () => {
|
|
146
|
+
if (finished)
|
|
147
|
+
return;
|
|
148
|
+
finished = true;
|
|
149
|
+
this.host.removeEventListener('transitionend', onTransitionEnd);
|
|
150
|
+
done();
|
|
151
|
+
};
|
|
152
|
+
const onTransitionEnd = (event) => {
|
|
153
|
+
if (event.target === this.host)
|
|
154
|
+
finish();
|
|
155
|
+
};
|
|
156
|
+
this.host.addEventListener('transitionend', onTransitionEnd);
|
|
157
|
+
setTimeout(finish, durationMs + 50);
|
|
158
|
+
// Stryker restore all
|
|
159
|
+
}
|
|
160
|
+
onPointerEnter() {
|
|
161
|
+
this.hovered = true;
|
|
162
|
+
this.pauseTimer();
|
|
163
|
+
}
|
|
164
|
+
onPointerLeave() {
|
|
165
|
+
this.hovered = false;
|
|
166
|
+
this.maybeResume();
|
|
167
|
+
}
|
|
168
|
+
onFocusEnter() {
|
|
169
|
+
this.focused = true;
|
|
170
|
+
this.pauseTimer();
|
|
171
|
+
}
|
|
172
|
+
onFocusLeave() {
|
|
173
|
+
this.focused = false;
|
|
174
|
+
this.maybeResume();
|
|
175
|
+
}
|
|
176
|
+
maybeResume() {
|
|
177
|
+
if (this.hovered || this.focused)
|
|
178
|
+
return;
|
|
179
|
+
if (this.context.timing !== 'auto' || this.timer !== undefined || this.leaving())
|
|
180
|
+
return;
|
|
181
|
+
if (this.remaining <= 0) {
|
|
182
|
+
this.ref.dismiss('timeout');
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
this.startTimer(this.remaining);
|
|
186
|
+
}
|
|
187
|
+
startTimer(ms) {
|
|
188
|
+
this.deadline = Date.now() + ms;
|
|
189
|
+
this.timer = setTimeout(() => this.ref.dismiss('timeout'), ms);
|
|
190
|
+
}
|
|
191
|
+
pauseTimer() {
|
|
192
|
+
if (this.timer === undefined)
|
|
193
|
+
return;
|
|
194
|
+
this.remaining = Math.max(0, this.deadline - Date.now());
|
|
195
|
+
this.clearTimer();
|
|
196
|
+
}
|
|
197
|
+
clearTimer() {
|
|
198
|
+
if (this.timer === undefined)
|
|
199
|
+
return;
|
|
200
|
+
clearTimeout(this.timer);
|
|
201
|
+
this.timer = undefined;
|
|
202
|
+
}
|
|
203
|
+
/** Durée (ms) de la transition de sortie, lue sur l'hôte (0 si aucune — ex. reduced-motion). */
|
|
204
|
+
exitDurationMs() {
|
|
205
|
+
// Stryker disable all: lecture/parsing de `transition-duration` CSS — en jsdom
|
|
206
|
+
// `getComputedStyle().transitionDuration` est toujours vide (renvoie 0). Le parsing réel
|
|
207
|
+
// (ms vs s, valeurs multiples, NaN) est exercé en e2e ; intestable en environnement jsdom.
|
|
208
|
+
const raw = getComputedStyle(this.host).transitionDuration || '';
|
|
209
|
+
const first = raw.split(',')[0].trim();
|
|
210
|
+
if (!first)
|
|
211
|
+
return 0;
|
|
212
|
+
const value = parseFloat(first);
|
|
213
|
+
if (Number.isNaN(value))
|
|
214
|
+
return 0;
|
|
215
|
+
return first.endsWith('ms') ? value : value * 1000;
|
|
216
|
+
// Stryker restore all
|
|
217
|
+
}
|
|
218
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "22.0.1", ngImport: i0, type: KtSnackbarContainer, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
219
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "22.0.1", type: KtSnackbarContainer, isStandalone: true, selector: "kt-snackbar-container", host: { listeners: { "mouseenter": "onPointerEnter()", "mouseleave": "onPointerLeave()", "focusin": "onFocusEnter()", "focusout": "onFocusLeave()" }, properties: { "attr.data-variant": "context.variant", "class.kt-snackbar--leaving": "leaving()" }, classAttribute: "kt-snackbar" }, ngImport: i0, template: `
|
|
220
|
+
@if (context.variant !== 'neutral') {
|
|
221
|
+
<span class="kt-snackbar__icon" aria-hidden="true"></span>
|
|
222
|
+
}
|
|
223
|
+
<span class="kt-snackbar__message">{{ context.message }}</span>
|
|
224
|
+
@if (context.closable) {
|
|
225
|
+
<button
|
|
226
|
+
type="button"
|
|
227
|
+
class="kt-snackbar__close"
|
|
228
|
+
[attr.aria-label]="context.closeLabel"
|
|
229
|
+
(click)="close()"
|
|
230
|
+
></button>
|
|
231
|
+
}
|
|
232
|
+
`, isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
233
|
+
}
|
|
234
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "22.0.1", ngImport: i0, type: KtSnackbarContainer, decorators: [{
|
|
235
|
+
type: Component,
|
|
236
|
+
args: [{
|
|
237
|
+
selector: 'kt-snackbar-container',
|
|
238
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
239
|
+
host: {
|
|
240
|
+
class: 'kt-snackbar',
|
|
241
|
+
'[attr.data-variant]': 'context.variant',
|
|
242
|
+
'[class.kt-snackbar--leaving]': 'leaving()',
|
|
243
|
+
'(mouseenter)': 'onPointerEnter()',
|
|
244
|
+
'(mouseleave)': 'onPointerLeave()',
|
|
245
|
+
'(focusin)': 'onFocusEnter()',
|
|
246
|
+
'(focusout)': 'onFocusLeave()',
|
|
247
|
+
},
|
|
248
|
+
template: `
|
|
249
|
+
@if (context.variant !== 'neutral') {
|
|
250
|
+
<span class="kt-snackbar__icon" aria-hidden="true"></span>
|
|
251
|
+
}
|
|
252
|
+
<span class="kt-snackbar__message">{{ context.message }}</span>
|
|
253
|
+
@if (context.closable) {
|
|
254
|
+
<button
|
|
255
|
+
type="button"
|
|
256
|
+
class="kt-snackbar__close"
|
|
257
|
+
[attr.aria-label]="context.closeLabel"
|
|
258
|
+
(click)="close()"
|
|
259
|
+
></button>
|
|
260
|
+
}
|
|
261
|
+
`,
|
|
262
|
+
}]
|
|
263
|
+
}], ctorParameters: () => [] });
|
|
264
|
+
|
|
265
|
+
/**
|
|
266
|
+
* Service d'ouverture des snackbars `@ktortu/aaa` — feedback **non bloquant** qui confirme une
|
|
267
|
+
* action ou signale un événement transitoire sans interrompre la tâche (n'utilisez PAS de snackbar
|
|
268
|
+
* pour une information critique à acquitter : préférez le dialog).
|
|
269
|
+
*
|
|
270
|
+
* Architecture a11y :
|
|
271
|
+
* - **CDK Overlay** pour le panneau visuel, **CDK LiveAnnouncer** pour l'annonce — **un seul canal**
|
|
272
|
+
* d'annonce (le panneau n'est pas une live region) ;
|
|
273
|
+
* - **le focus n'est jamais déplacé** vers la snackbar (RGAA « message de statut » / WCAG 4.1.3) ;
|
|
274
|
+
* - disparition automatique **en pause au survol et au focus** (régime `'auto'`, défaut AA), ou
|
|
275
|
+
* persistante (`timing: 'manual'`, AAA) ;
|
|
276
|
+
* - **file FIFO** : une seule snackbar visible, les suivantes patientent (live region non saturée).
|
|
277
|
+
* La file est bornée par `max` (défaut 3) ; les messages **identiques** sont fusionnés (coalescing) ;
|
|
278
|
+
* - **Échap** ferme la snackbar affichée (la plus récente).
|
|
279
|
+
*
|
|
280
|
+
* Requiert côté hôte les styles `@angular/cdk/overlay-prebuilt.css` **et**
|
|
281
|
+
* `@angular/cdk/a11y-prebuilt.css` (ce dernier masque l'élément du LiveAnnouncer).
|
|
282
|
+
*
|
|
283
|
+
* @example
|
|
284
|
+
* ```ts
|
|
285
|
+
* private readonly snackbar = inject(KtSnackbar);
|
|
286
|
+
* this.snackbar.open('Brouillon enregistré'); // disparaît, annonce polie
|
|
287
|
+
* this.snackbar.open('Fichier supprimé', { variant: 'success' }); // variante (couleur + icône)
|
|
288
|
+
* this.snackbar.open('Hors ligne', { timing: 'manual' }); // reste jusqu'à fermeture (AAA)
|
|
289
|
+
* ```
|
|
290
|
+
*/
|
|
291
|
+
class KtSnackbar {
|
|
292
|
+
overlay = inject(Overlay);
|
|
293
|
+
injector = inject(Injector);
|
|
294
|
+
liveAnnouncer = inject(LiveAnnouncer);
|
|
295
|
+
config = inject(KT_SNACKBAR_CONFIG, { optional: true });
|
|
296
|
+
platformId = inject(PLATFORM_ID);
|
|
297
|
+
doc = inject(DOCUMENT);
|
|
298
|
+
/** File FIFO : `queue[0]` est la snackbar affichée dès qu'elle est attachée. */
|
|
299
|
+
queue = [];
|
|
300
|
+
active = null;
|
|
301
|
+
escapeRegistered = false;
|
|
302
|
+
/**
|
|
303
|
+
* Ouvre une snackbar affichant `message`. Les `options` priment sur `KT_SNACKBAR_CONFIG`, qui
|
|
304
|
+
* prime sur les défauts. Renvoie une {@link KtSnackbarRef} (ignorable dans le cas simple).
|
|
305
|
+
*
|
|
306
|
+
* Une seule snackbar est visible : si une autre est affichée, celle-ci patiente en file (FIFO).
|
|
307
|
+
* Un `message` identique à une snackbar déjà affichée ou en attente est **fusionné** : on renvoie
|
|
308
|
+
* alors la référence existante sans rien ré-empiler.
|
|
309
|
+
*
|
|
310
|
+
* @param message Texte affiché et annoncé au lecteur d'écran.
|
|
311
|
+
* @param options Surcharges ponctuelles (durée, régime, position, politesse, variante, fermeture).
|
|
312
|
+
* @returns La référence de la snackbar (existante en cas de fusion).
|
|
313
|
+
*/
|
|
314
|
+
open(message, options) {
|
|
315
|
+
const resolved = this.resolve(options);
|
|
316
|
+
// SSR : aucun overlay côté serveur — on renvoie une référence inerte déjà fermée.
|
|
317
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
318
|
+
const inert = new KtSnackbarRef(() => {
|
|
319
|
+
/* SSR : aucun overlay à fermer. */
|
|
320
|
+
});
|
|
321
|
+
inert.dismiss('dismiss');
|
|
322
|
+
return inert;
|
|
323
|
+
}
|
|
324
|
+
// Coalescing : message identique déjà affiché ou en file → on réutilise la référence existante.
|
|
325
|
+
const existing = this.queue.find((entry) => entry.context.message === message);
|
|
326
|
+
if (existing)
|
|
327
|
+
return existing.ref;
|
|
328
|
+
const context = {
|
|
329
|
+
message,
|
|
330
|
+
variant: resolved.variant,
|
|
331
|
+
closable: resolved.closable,
|
|
332
|
+
closeLabel: resolved.closeLabel,
|
|
333
|
+
timing: resolved.timing,
|
|
334
|
+
duration: this.computeDuration(message, resolved),
|
|
335
|
+
};
|
|
336
|
+
const item = { context, position: resolved.position, politeness: resolved.politeness };
|
|
337
|
+
item.ref = new KtSnackbarRef(() => this.handleDismiss(item));
|
|
338
|
+
this.queue.push(item);
|
|
339
|
+
this.enforceMax(resolved.max);
|
|
340
|
+
this.showHead();
|
|
341
|
+
return item.ref;
|
|
342
|
+
}
|
|
343
|
+
ngOnDestroy() {
|
|
344
|
+
this.unregisterEscape();
|
|
345
|
+
this.active?.overlayRef.dispose();
|
|
346
|
+
this.active = null;
|
|
347
|
+
this.queue.length = 0;
|
|
348
|
+
}
|
|
349
|
+
/** Borne la file (affichée + en attente) : retire les plus ANCIENNES en attente au-delà de `max`. */
|
|
350
|
+
enforceMax(max) {
|
|
351
|
+
const limit = Math.max(1, max);
|
|
352
|
+
while (this.queue.length > limit) {
|
|
353
|
+
const oldestWaiting = this.queue.find((entry) => entry !== this.active?.item);
|
|
354
|
+
if (!oldestWaiting)
|
|
355
|
+
break;
|
|
356
|
+
// Jamais affichée : la fermeture est routée via handleDismiss (qui la retire de la file).
|
|
357
|
+
oldestWaiting.ref.dismiss('replaced');
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
/** Affiche la tête de file si rien n'est actuellement visible. */
|
|
361
|
+
showHead() {
|
|
362
|
+
if (this.active || this.queue.length === 0)
|
|
363
|
+
return;
|
|
364
|
+
const item = this.queue[0];
|
|
365
|
+
const positionStrategy = this.overlay.position().global().centerHorizontally();
|
|
366
|
+
if (item.position === 'top')
|
|
367
|
+
positionStrategy.top('0');
|
|
368
|
+
else
|
|
369
|
+
positionStrategy.bottom('0');
|
|
370
|
+
const overlayRef = this.overlay.create({
|
|
371
|
+
positionStrategy,
|
|
372
|
+
panelClass: ['kt-snackbar-pane', `kt-snackbar-pane--${item.position}`],
|
|
373
|
+
});
|
|
374
|
+
const injector = Injector.create({
|
|
375
|
+
parent: this.injector,
|
|
376
|
+
providers: [
|
|
377
|
+
{ provide: KtSnackbarRef, useValue: item.ref },
|
|
378
|
+
{ provide: KT_SNACKBAR_CONTEXT, useValue: item.context },
|
|
379
|
+
],
|
|
380
|
+
});
|
|
381
|
+
const cmp = overlayRef.attach(new ComponentPortal(KtSnackbarContainer, null, injector));
|
|
382
|
+
// Canal d'annonce UNIQUE : le panneau visuel n'est pas une live region.
|
|
383
|
+
void this.liveAnnouncer.announce(item.context.message, item.politeness);
|
|
384
|
+
this.active = { item, overlayRef, cmp };
|
|
385
|
+
this.registerEscape();
|
|
386
|
+
}
|
|
387
|
+
/** Retire l'élément de la file ; s'il est affiché, joue la sortie puis démonte et avance la file. */
|
|
388
|
+
handleDismiss(item) {
|
|
389
|
+
const index = this.queue.indexOf(item);
|
|
390
|
+
if (index === -1)
|
|
391
|
+
return;
|
|
392
|
+
if (this.active && this.active.item === item) {
|
|
393
|
+
const { overlayRef, cmp } = this.active;
|
|
394
|
+
this.active = null;
|
|
395
|
+
this.queue.splice(index, 1);
|
|
396
|
+
cmp.instance.playExit(() => {
|
|
397
|
+
overlayRef.dispose();
|
|
398
|
+
this.showHead();
|
|
399
|
+
if (this.queue.length === 0)
|
|
400
|
+
this.unregisterEscape();
|
|
401
|
+
});
|
|
402
|
+
return;
|
|
403
|
+
}
|
|
404
|
+
// En attente : aucun visuel à démonter.
|
|
405
|
+
this.queue.splice(index, 1);
|
|
406
|
+
}
|
|
407
|
+
onDocumentKeydown = (event) => {
|
|
408
|
+
// Échap ferme la plus récente (= la snackbar affichée).
|
|
409
|
+
if (event.key === 'Escape')
|
|
410
|
+
this.active?.item.ref.dismiss('dismiss');
|
|
411
|
+
};
|
|
412
|
+
registerEscape() {
|
|
413
|
+
if (this.escapeRegistered)
|
|
414
|
+
return;
|
|
415
|
+
this.doc.addEventListener('keydown', this.onDocumentKeydown);
|
|
416
|
+
this.escapeRegistered = true;
|
|
417
|
+
}
|
|
418
|
+
unregisterEscape() {
|
|
419
|
+
if (!this.escapeRegistered)
|
|
420
|
+
return;
|
|
421
|
+
this.doc.removeEventListener('keydown', this.onDocumentKeydown);
|
|
422
|
+
this.escapeRegistered = false;
|
|
423
|
+
}
|
|
424
|
+
/**
|
|
425
|
+
* Durée effective (ms) en régime `'auto'`. Un nombre est pris tel quel (durée fixe) ; le sentinel
|
|
426
|
+
* `'reading-time'` calcule `clamp(longueur × perChar, min, max)`.
|
|
427
|
+
*/
|
|
428
|
+
computeDuration(message, resolved) {
|
|
429
|
+
const setting = resolved.duration;
|
|
430
|
+
if (typeof setting === 'number')
|
|
431
|
+
return setting;
|
|
432
|
+
const computed = message.length * resolved.readingTimePerChar;
|
|
433
|
+
return Math.min(resolved.readingTimeMax, Math.max(resolved.readingTimeMin, computed));
|
|
434
|
+
}
|
|
435
|
+
/** Résolution en cascade `option ?? KT_SNACKBAR_CONFIG ?? défaut`, champ par champ. */
|
|
436
|
+
resolve(options) {
|
|
437
|
+
const config = this.config;
|
|
438
|
+
return {
|
|
439
|
+
duration: options?.duration ?? config?.duration ?? KT_SNACKBAR_DEFAULTS.duration,
|
|
440
|
+
readingTimeMin: options?.readingTimeMin ?? config?.readingTimeMin ?? KT_SNACKBAR_DEFAULTS.readingTimeMin,
|
|
441
|
+
readingTimeMax: options?.readingTimeMax ?? config?.readingTimeMax ?? KT_SNACKBAR_DEFAULTS.readingTimeMax,
|
|
442
|
+
readingTimePerChar: options?.readingTimePerChar ?? config?.readingTimePerChar ?? KT_SNACKBAR_DEFAULTS.readingTimePerChar,
|
|
443
|
+
timing: options?.timing ?? config?.timing ?? KT_SNACKBAR_DEFAULTS.timing,
|
|
444
|
+
position: options?.position ?? config?.position ?? KT_SNACKBAR_DEFAULTS.position,
|
|
445
|
+
politeness: options?.politeness ?? config?.politeness ?? KT_SNACKBAR_DEFAULTS.politeness,
|
|
446
|
+
variant: options?.variant ?? config?.variant ?? KT_SNACKBAR_DEFAULTS.variant,
|
|
447
|
+
closable: options?.closable ?? config?.closable ?? KT_SNACKBAR_DEFAULTS.closable,
|
|
448
|
+
closeLabel: options?.closeLabel ?? config?.closeLabel ?? KT_SNACKBAR_DEFAULTS.closeLabel,
|
|
449
|
+
max: options?.max ?? config?.max ?? KT_SNACKBAR_DEFAULTS.max,
|
|
450
|
+
};
|
|
451
|
+
}
|
|
452
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "22.0.1", ngImport: i0, type: KtSnackbar, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
453
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "22.0.1", ngImport: i0, type: KtSnackbar, providedIn: 'root' });
|
|
454
|
+
}
|
|
455
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "22.0.1", ngImport: i0, type: KtSnackbar, decorators: [{
|
|
456
|
+
type: Injectable,
|
|
457
|
+
args: [{ providedIn: 'root' }]
|
|
458
|
+
}] });
|
|
459
|
+
|
|
460
|
+
/**
|
|
461
|
+
* Generated bundle index. Do not edit.
|
|
462
|
+
*/
|
|
463
|
+
|
|
464
|
+
export { KT_SNACKBAR_CONFIG, KT_SNACKBAR_CONTEXT, KT_SNACKBAR_DEFAULTS, KtSnackbar, KtSnackbarContainer, KtSnackbarRef, provideKtSnackbar };
|
|
465
|
+
//# sourceMappingURL=ktortu-aaa-snackbar.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ktortu-aaa-snackbar.mjs","sources":["../../../../projects/ktortu/aaa/snackbar/snackbar-config.ts","../../../../projects/ktortu/aaa/snackbar/snackbar-ref.ts","../../../../projects/ktortu/aaa/snackbar/snackbar-container.ts","../../../../projects/ktortu/aaa/snackbar/snackbar.service.ts","../../../../projects/ktortu/aaa/snackbar/ktortu-aaa-snackbar.ts"],"sourcesContent":["import { InjectionToken, Provider } from '@angular/core';\n\n/** Politesse de l'annonce au lecteur d'écran (relayée à `LiveAnnouncer`). */\nexport type KtSnackbarPoliteness = 'polite' | 'assertive';\n\n/** Bord d'ancrage de la snackbar dans le viewport. */\nexport type KtSnackbarPosition = 'top' | 'bottom';\n\n/**\n * Régime temporel de la snackbar :\n * - `'auto'` : disparition automatique après `duration`, **mise en pause au survol et au focus**\n * clavier (WCAG 1.4.13 / 2.2.1). Conforme **AA**. C'est le défaut.\n * - `'manual'` : **aucune** minuterie, la snackbar reste jusqu'à fermeture explicite\n * (bouton de fermeture ou `ref.dismiss()`). Conforme **2.2.3 (AAA)** au sens strict.\n *\n * Conformément aux recommandations d'accessibilité (Roselli/Soueidan), la snackbar ne porte **pas\n * d'action interactive** (ex. « Annuler ») : une action auto-disparaissante n'est pas atteignable et\n * une live region n'expose pas ses boutons. Pour un undo, préférez un mécanisme atteignable côté app\n * (Ctrl+Z, bannière persistante).\n */\nexport type KtSnackbarTiming = 'auto' | 'manual';\n\n/**\n * Variante sémantique de la snackbar. Pilote uniquement l'**apparence** (couleur d'accent + icône),\n * via l'attribut `data-variant` et les tokens CSS `--snackbar-*` — donc entièrement gérée par le\n * thème, sans logique TypeScript. Découplée de la `politeness` (une « erreur » n'est pas forcément\n * une urgence assertive). La couleur n'est jamais le seul indice : chaque variante porte une **icône**\n * de forme distincte (WCAG 1.4.1).\n * - `'neutral'` : pastille neutre, sans icône (défaut) ;\n * - `'info' | 'success' | 'warning' | 'error'` : accent + icône dédiés.\n */\nexport type KtSnackbarVariant = 'neutral' | 'info' | 'success' | 'warning' | 'error';\n\n/**\n * Défauts de la snackbar, injectables via `provideKtSnackbar` / `KT_SNACKBAR_CONFIG`.\n * Tous les champs sont surchargeables **par appel** via les options de `KtSnackbar.open()`.\n *\n * Résolution en cascade (convention de la lib) : `option d'open ?? KT_SNACKBAR_CONFIG ?? défaut`.\n */\nexport interface KtSnackbarConfig {\n /**\n * Durée d'affichage en régime `'auto'`. Soit un **nombre fixe** (ms), soit le sentinel\n * **`'reading-time'`** (DÉFAUT) qui **calcule** la durée d'après la longueur du message :\n * `clamp(longueur × readingTimePerChar, readingTimeMin, readingTimeMax)`. Passer un nombre à\n * `open()` force donc une durée fixe pour cet appel.\n *\n * ⚠️ Gardez le message **court** : une snackbar est un message transitoire (pas un paragraphe).\n * Un message long fait grimper la durée jusqu'au plafond `readingTimeMax`.\n * @default 'reading-time'\n */\n duration: number | 'reading-time';\n /** Plancher (ms) de la durée calculée (`'reading-time'`) — laisse le temps de lire un message court. @default 4000 */\n readingTimeMin: number;\n /** Plafond (ms) de la durée calculée (`'reading-time'`) — borne un message long. @default 10000 */\n readingTimeMax: number;\n /** Coefficient de lecture : millisecondes ajoutées par caractère du message. @default 60 (~200 mots/min) */\n readingTimePerChar: number;\n /** Régime temporel (cf. {@link KtSnackbarTiming}). @default 'auto' (AA + pause) */\n timing: KtSnackbarTiming;\n /** Bord d'ancrage dans le viewport. @default 'bottom' */\n position: KtSnackbarPosition;\n /** Politesse de l'annonce lecteur d'écran. @default 'polite' */\n politeness: KtSnackbarPoliteness;\n /** Variante sémantique (apparence seule, cf. {@link KtSnackbarVariant}). @default 'neutral' */\n variant: KtSnackbarVariant;\n /** Affiche un bouton de fermeture (cible 44px, AAA). @default true */\n closable: boolean;\n /** Nom accessible du bouton de fermeture. @default 'Close' (FR fourni en lot L3) */\n closeLabel: string;\n /**\n * Taille maximale de la file FIFO (snackbar affichée + en attente). Au-delà, les **plus anciennes\n * en attente** sont retirées silencieusement. Une seule snackbar est visible à la fois. @default 3\n */\n max: number;\n}\n\n/** Défauts AAA-orientés de la snackbar (anglais neutre). */\nexport const KT_SNACKBAR_DEFAULTS: KtSnackbarConfig = {\n // Stryker disable next-line StringLiteral: équivalent — computeDuration() traite toute valeur non numérique comme reading-time\n duration: 'reading-time',\n readingTimeMin: 4000,\n readingTimeMax: 10000,\n readingTimePerChar: 60,\n timing: 'auto',\n position: 'bottom',\n politeness: 'polite',\n variant: 'neutral',\n closable: true,\n closeLabel: 'Close',\n max: 3,\n};\n\n// Stryker disable next-line StringLiteral: libellé de debug du token d'injection (sans effet runtime)\nexport const KT_SNACKBAR_CONFIG = new InjectionToken<Partial<KtSnackbarConfig>>('KT_SNACKBAR_CONFIG');\n\n/**\n * Options ponctuelles d'ouverture d'une snackbar : un sous-ensemble (toutes facultatives) de la\n * config, prioritaire sur `KT_SNACKBAR_CONFIG` et sur les défauts.\n */\nexport type KtSnackbarOptions = Partial<KtSnackbarConfig>;\n\n/**\n * Fournit des défauts de snackbar pour un sous-arbre ou l'application entière.\n *\n * @example\n * ```ts\n * // app.config.ts — bascule TOUTE l'app en AAA strict (aucune disparition automatique)\n * providers: [provideKtSnackbar({ timing: 'manual' })]\n * ```\n * @example\n * ```ts\n * providers: [provideKtSnackbar({ duration: 8000, position: 'top' })]\n * ```\n */\nexport function provideKtSnackbar(config: Partial<KtSnackbarConfig>): Provider {\n return { provide: KT_SNACKBAR_CONFIG, useValue: config };\n}\n","import { Observable, Subject } from 'rxjs';\n\n/**\n * Raison de fermeture d'une snackbar, transmise par {@link KtSnackbarRef.afterDismissed}.\n * - `'timeout'` : la minuterie (régime `'auto'`) est arrivée à échéance ;\n * - `'dismiss'` : fermeture explicite (bouton de fermeture, `Échap`, ou `ref.dismiss()`) ;\n * - `'replaced'` : la snackbar a été retirée de la file sans être affichée (file pleine — cf. `max`).\n */\nexport type KtSnackbarDismissReason = 'timeout' | 'dismiss' | 'replaced';\n\n/**\n * Référence d'une snackbar ouverte, renvoyée par `KtSnackbar.open()`. Permet de fermer la\n * snackbar par programmation et d'observer sa fermeture.\n *\n * Le débutant peut l'ignorer (`snackbar.open('…')`) ; le code avancé s'en sert pour piloter la\n * fermeture et réagir à la raison de fin.\n *\n * @example\n * ```ts\n * const ref = snackbar.open('Brouillon enregistré');\n * ref.afterDismissed().subscribe((reason) => console.log(reason)); // 'timeout' | 'dismiss' | 'replaced'\n * // plus tard : ref.dismiss();\n * ```\n */\nexport class KtSnackbarRef {\n private readonly _afterDismissed = new Subject<KtSnackbarDismissReason>();\n private settled = false;\n\n /**\n * @param onDismiss Rappel exécuté une fois à la fermeture (animation de sortie + démontage de\n * l'overlay + avance de la file), fourni par le service.\n * @internal Construit par `KtSnackbar` — n'instanciez pas `KtSnackbarRef` directement.\n */\n constructor(private readonly onDismiss: (reason: KtSnackbarDismissReason) => void) {}\n\n /**\n * Ferme la snackbar (idempotent). Déclenche la sortie/démontage puis émet la raison sur\n * `afterDismissed()`.\n * @param reason Raison de fermeture. @default 'dismiss'\n */\n dismiss(reason: KtSnackbarDismissReason = 'dismiss'): void {\n if (this.settled) return;\n this.settled = true;\n this.onDismiss(reason);\n this._afterDismissed.next(reason);\n this._afterDismissed.complete();\n }\n\n /** Émet une fois (puis complète) à la fermeture, avec la {@link KtSnackbarDismissReason}. */\n afterDismissed(): Observable<KtSnackbarDismissReason> {\n return this._afterDismissed.asObservable();\n }\n}\n","import { isPlatformBrowser } from '@angular/common';\nimport {\n ChangeDetectionStrategy,\n Component,\n DestroyRef,\n ElementRef,\n InjectionToken,\n PLATFORM_ID,\n afterNextRender,\n inject,\n signal,\n} from '@angular/core';\n\nimport { KtSnackbarTiming, KtSnackbarVariant } from './snackbar-config';\nimport { KtSnackbarRef } from './snackbar-ref';\n\n/**\n * Données résolues passées au conteneur visuel à l'ouverture (injectées via {@link KT_SNACKBAR_CONTEXT}).\n * Produites par `KtSnackbar` après résolution `option ?? config ?? défaut`.\n */\nexport interface KtSnackbarContext {\n /** Message affiché et annoncé. */\n readonly message: string;\n /** Variante sémantique (apparence : accent + icône). */\n readonly variant: KtSnackbarVariant;\n /** Affiche le bouton de fermeture. */\n readonly closable: boolean;\n /** Nom accessible du bouton de fermeture. */\n readonly closeLabel: string;\n /** Régime temporel (la minuterie n'existe qu'en `'auto'`). */\n readonly timing: KtSnackbarTiming;\n /** Durée (ms) avant disparition automatique en régime `'auto'`. */\n readonly duration: number;\n}\n\n// Stryker disable next-line StringLiteral: libellé de debug du token d'injection (sans effet runtime)\nexport const KT_SNACKBAR_CONTEXT = new InjectionToken<KtSnackbarContext>('KT_SNACKBAR_CONTEXT');\n\n/**\n * Conteneur visuel d'une snackbar. **Volontairement pas une live region** (aucun `role`/`aria-live`\n * sur l'hôte) : l'annonce passe par un canal UNIQUE, le `LiveAnnouncer` du service. On évite ainsi\n * la double-annonce et l'écrasement de politesse observés sur d'autres libs.\n *\n * En régime `'auto'`, la minuterie de disparition se met **en pause au survol et au focus** clavier\n * (WCAG 1.4.13 / 2.2.1) et reprend quand ni le pointeur ni le focus ne sont sur la snackbar.\n * L'icône de variante est **décorative** (`aria-hidden`) : la forme distincte sert d'indice non\n * coloré (WCAG 1.4.1), le sens reste porté par le texte.\n *\n * @internal Monté par `KtSnackbar` via un `ComponentPortal`.\n */\n@Component({\n selector: 'kt-snackbar-container',\n changeDetection: ChangeDetectionStrategy.OnPush,\n host: {\n class: 'kt-snackbar',\n '[attr.data-variant]': 'context.variant',\n '[class.kt-snackbar--leaving]': 'leaving()',\n '(mouseenter)': 'onPointerEnter()',\n '(mouseleave)': 'onPointerLeave()',\n '(focusin)': 'onFocusEnter()',\n '(focusout)': 'onFocusLeave()',\n },\n template: `\n @if (context.variant !== 'neutral') {\n <span class=\"kt-snackbar__icon\" aria-hidden=\"true\"></span>\n }\n <span class=\"kt-snackbar__message\">{{ context.message }}</span>\n @if (context.closable) {\n <button\n type=\"button\"\n class=\"kt-snackbar__close\"\n [attr.aria-label]=\"context.closeLabel\"\n (click)=\"close()\"\n ></button>\n }\n `,\n})\nexport class KtSnackbarContainer {\n protected readonly context = inject(KT_SNACKBAR_CONTEXT);\n private readonly ref = inject(KtSnackbarRef);\n private readonly platformId = inject(PLATFORM_ID);\n private readonly host = inject<ElementRef<HTMLElement>>(ElementRef).nativeElement;\n\n /** Vrai pendant l'animation de sortie (déclenche la classe `kt-snackbar--leaving`). */\n protected readonly leaving = signal(false);\n\n private timer: ReturnType<typeof setTimeout> | undefined;\n private remaining = this.context.duration;\n private deadline = 0;\n private hovered = false;\n private focused = false;\n\n constructor() {\n if (this.context.timing === 'auto' && isPlatformBrowser(this.platformId)) {\n afterNextRender(() => this.startTimer(this.context.duration));\n }\n inject(DestroyRef).onDestroy(() => this.clearTimer());\n }\n\n protected close(): void {\n this.ref.dismiss('dismiss');\n }\n\n /**\n * Joue l'animation de sortie puis invoque `done` (démontage de l'overlay côté service). Appelé par\n * le service à la fermeture. Sans animation (durée nulle, `prefers-reduced-motion`, ou SSR), `done`\n * est invoqué immédiatement. Filet de sécurité par `setTimeout` si `transitionend` ne se déclenche pas.\n */\n playExit(done: () => void): void {\n this.clearTimer();\n if (!isPlatformBrowser(this.platformId)) {\n done();\n return;\n }\n const durationMs = this.exitDurationMs();\n // Stryker disable all: animation de sortie pilotée par une transition CSS réelle. En jsdom\n // `transitionDuration` est vide → `exitDurationMs()` vaut 0 → ce code n'est jamais atteint.\n // Comportement (jeu de l'animation puis démontage) vérifié en e2e (sortie + reduced-motion).\n if (durationMs <= 0) {\n done();\n return;\n }\n\n this.leaving.set(true);\n let finished = false;\n const finish = (): void => {\n if (finished) return;\n finished = true;\n this.host.removeEventListener('transitionend', onTransitionEnd);\n done();\n };\n const onTransitionEnd = (event: TransitionEvent): void => {\n if (event.target === this.host) finish();\n };\n this.host.addEventListener('transitionend', onTransitionEnd);\n setTimeout(finish, durationMs + 50);\n // Stryker restore all\n }\n\n protected onPointerEnter(): void {\n this.hovered = true;\n this.pauseTimer();\n }\n\n protected onPointerLeave(): void {\n this.hovered = false;\n this.maybeResume();\n }\n\n protected onFocusEnter(): void {\n this.focused = true;\n this.pauseTimer();\n }\n\n protected onFocusLeave(): void {\n this.focused = false;\n this.maybeResume();\n }\n\n private maybeResume(): void {\n if (this.hovered || this.focused) return;\n if (this.context.timing !== 'auto' || this.timer !== undefined || this.leaving()) return;\n if (this.remaining <= 0) {\n this.ref.dismiss('timeout');\n return;\n }\n this.startTimer(this.remaining);\n }\n\n private startTimer(ms: number): void {\n this.deadline = Date.now() + ms;\n this.timer = setTimeout(() => this.ref.dismiss('timeout'), ms);\n }\n\n private pauseTimer(): void {\n if (this.timer === undefined) return;\n this.remaining = Math.max(0, this.deadline - Date.now());\n this.clearTimer();\n }\n\n private clearTimer(): void {\n if (this.timer === undefined) return;\n clearTimeout(this.timer);\n this.timer = undefined;\n }\n\n /** Durée (ms) de la transition de sortie, lue sur l'hôte (0 si aucune — ex. reduced-motion). */\n private exitDurationMs(): number {\n // Stryker disable all: lecture/parsing de `transition-duration` CSS — en jsdom\n // `getComputedStyle().transitionDuration` est toujours vide (renvoie 0). Le parsing réel\n // (ms vs s, valeurs multiples, NaN) est exercé en e2e ; intestable en environnement jsdom.\n const raw = getComputedStyle(this.host).transitionDuration || '';\n const first = raw.split(',')[0].trim();\n if (!first) return 0;\n const value = parseFloat(first);\n if (Number.isNaN(value)) return 0;\n return first.endsWith('ms') ? value : value * 1000;\n // Stryker restore all\n }\n}\n","import { LiveAnnouncer } from '@angular/cdk/a11y';\nimport { Overlay, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { DOCUMENT, isPlatformBrowser } from '@angular/common';\nimport { ComponentRef, Injectable, Injector, OnDestroy, PLATFORM_ID, inject } from '@angular/core';\n\nimport {\n KT_SNACKBAR_CONFIG,\n KT_SNACKBAR_DEFAULTS,\n KtSnackbarConfig,\n KtSnackbarOptions,\n KtSnackbarPoliteness,\n KtSnackbarPosition,\n} from './snackbar-config';\nimport { KT_SNACKBAR_CONTEXT, KtSnackbarContainer, KtSnackbarContext } from './snackbar-container';\nimport { KtSnackbarRef } from './snackbar-ref';\n\n/** Élément de la file FIFO interne (une seule snackbar visible à la fois). */\ninterface KtSnackbarQueueItem {\n ref: KtSnackbarRef;\n readonly context: KtSnackbarContext;\n readonly position: KtSnackbarPosition;\n readonly politeness: KtSnackbarPoliteness;\n}\n\n/**\n * Service d'ouverture des snackbars `@ktortu/aaa` — feedback **non bloquant** qui confirme une\n * action ou signale un événement transitoire sans interrompre la tâche (n'utilisez PAS de snackbar\n * pour une information critique à acquitter : préférez le dialog).\n *\n * Architecture a11y :\n * - **CDK Overlay** pour le panneau visuel, **CDK LiveAnnouncer** pour l'annonce — **un seul canal**\n * d'annonce (le panneau n'est pas une live region) ;\n * - **le focus n'est jamais déplacé** vers la snackbar (RGAA « message de statut » / WCAG 4.1.3) ;\n * - disparition automatique **en pause au survol et au focus** (régime `'auto'`, défaut AA), ou\n * persistante (`timing: 'manual'`, AAA) ;\n * - **file FIFO** : une seule snackbar visible, les suivantes patientent (live region non saturée).\n * La file est bornée par `max` (défaut 3) ; les messages **identiques** sont fusionnés (coalescing) ;\n * - **Échap** ferme la snackbar affichée (la plus récente).\n *\n * Requiert côté hôte les styles `@angular/cdk/overlay-prebuilt.css` **et**\n * `@angular/cdk/a11y-prebuilt.css` (ce dernier masque l'élément du LiveAnnouncer).\n *\n * @example\n * ```ts\n * private readonly snackbar = inject(KtSnackbar);\n * this.snackbar.open('Brouillon enregistré'); // disparaît, annonce polie\n * this.snackbar.open('Fichier supprimé', { variant: 'success' }); // variante (couleur + icône)\n * this.snackbar.open('Hors ligne', { timing: 'manual' }); // reste jusqu'à fermeture (AAA)\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class KtSnackbar implements OnDestroy {\n private readonly overlay = inject(Overlay);\n private readonly injector = inject(Injector);\n private readonly liveAnnouncer = inject(LiveAnnouncer);\n private readonly config = inject(KT_SNACKBAR_CONFIG, { optional: true });\n private readonly platformId = inject(PLATFORM_ID);\n private readonly doc = inject(DOCUMENT);\n\n /** File FIFO : `queue[0]` est la snackbar affichée dès qu'elle est attachée. */\n private readonly queue: KtSnackbarQueueItem[] = [];\n private active: { item: KtSnackbarQueueItem; overlayRef: OverlayRef; cmp: ComponentRef<KtSnackbarContainer> } | null =\n null;\n private escapeRegistered = false;\n\n /**\n * Ouvre une snackbar affichant `message`. Les `options` priment sur `KT_SNACKBAR_CONFIG`, qui\n * prime sur les défauts. Renvoie une {@link KtSnackbarRef} (ignorable dans le cas simple).\n *\n * Une seule snackbar est visible : si une autre est affichée, celle-ci patiente en file (FIFO).\n * Un `message` identique à une snackbar déjà affichée ou en attente est **fusionné** : on renvoie\n * alors la référence existante sans rien ré-empiler.\n *\n * @param message Texte affiché et annoncé au lecteur d'écran.\n * @param options Surcharges ponctuelles (durée, régime, position, politesse, variante, fermeture).\n * @returns La référence de la snackbar (existante en cas de fusion).\n */\n open(message: string, options?: KtSnackbarOptions): KtSnackbarRef {\n const resolved = this.resolve(options);\n\n // SSR : aucun overlay côté serveur — on renvoie une référence inerte déjà fermée.\n if (!isPlatformBrowser(this.platformId)) {\n const inert = new KtSnackbarRef(() => {\n /* SSR : aucun overlay à fermer. */\n });\n inert.dismiss('dismiss');\n return inert;\n }\n\n // Coalescing : message identique déjà affiché ou en file → on réutilise la référence existante.\n const existing = this.queue.find((entry) => entry.context.message === message);\n if (existing) return existing.ref;\n\n const context: KtSnackbarContext = {\n message,\n variant: resolved.variant,\n closable: resolved.closable,\n closeLabel: resolved.closeLabel,\n timing: resolved.timing,\n duration: this.computeDuration(message, resolved),\n };\n const item = { context, position: resolved.position, politeness: resolved.politeness } as KtSnackbarQueueItem;\n item.ref = new KtSnackbarRef(() => this.handleDismiss(item));\n\n this.queue.push(item);\n this.enforceMax(resolved.max);\n this.showHead();\n return item.ref;\n }\n\n ngOnDestroy(): void {\n this.unregisterEscape();\n this.active?.overlayRef.dispose();\n this.active = null;\n this.queue.length = 0;\n }\n\n /** Borne la file (affichée + en attente) : retire les plus ANCIENNES en attente au-delà de `max`. */\n private enforceMax(max: number): void {\n const limit = Math.max(1, max);\n while (this.queue.length > limit) {\n const oldestWaiting = this.queue.find((entry) => entry !== this.active?.item);\n if (!oldestWaiting) break;\n // Jamais affichée : la fermeture est routée via handleDismiss (qui la retire de la file).\n oldestWaiting.ref.dismiss('replaced');\n }\n }\n\n /** Affiche la tête de file si rien n'est actuellement visible. */\n private showHead(): void {\n if (this.active || this.queue.length === 0) return;\n const item = this.queue[0];\n\n const positionStrategy = this.overlay.position().global().centerHorizontally();\n if (item.position === 'top') positionStrategy.top('0');\n else positionStrategy.bottom('0');\n\n const overlayRef = this.overlay.create({\n positionStrategy,\n panelClass: ['kt-snackbar-pane', `kt-snackbar-pane--${item.position}`],\n });\n\n const injector = Injector.create({\n parent: this.injector,\n providers: [\n { provide: KtSnackbarRef, useValue: item.ref },\n { provide: KT_SNACKBAR_CONTEXT, useValue: item.context },\n ],\n });\n\n const cmp = overlayRef.attach(new ComponentPortal(KtSnackbarContainer, null, injector));\n\n // Canal d'annonce UNIQUE : le panneau visuel n'est pas une live region.\n void this.liveAnnouncer.announce(item.context.message, item.politeness);\n\n this.active = { item, overlayRef, cmp };\n this.registerEscape();\n }\n\n /** Retire l'élément de la file ; s'il est affiché, joue la sortie puis démonte et avance la file. */\n private handleDismiss(item: KtSnackbarQueueItem): void {\n const index = this.queue.indexOf(item);\n if (index === -1) return;\n\n if (this.active && this.active.item === item) {\n const { overlayRef, cmp } = this.active;\n this.active = null;\n this.queue.splice(index, 1);\n cmp.instance.playExit(() => {\n overlayRef.dispose();\n this.showHead();\n if (this.queue.length === 0) this.unregisterEscape();\n });\n return;\n }\n\n // En attente : aucun visuel à démonter.\n this.queue.splice(index, 1);\n }\n\n private readonly onDocumentKeydown = (event: KeyboardEvent): void => {\n // Échap ferme la plus récente (= la snackbar affichée).\n if (event.key === 'Escape') this.active?.item.ref.dismiss('dismiss');\n };\n\n private registerEscape(): void {\n if (this.escapeRegistered) return;\n this.doc.addEventListener('keydown', this.onDocumentKeydown);\n this.escapeRegistered = true;\n }\n\n private unregisterEscape(): void {\n if (!this.escapeRegistered) return;\n this.doc.removeEventListener('keydown', this.onDocumentKeydown);\n this.escapeRegistered = false;\n }\n\n /**\n * Durée effective (ms) en régime `'auto'`. Un nombre est pris tel quel (durée fixe) ; le sentinel\n * `'reading-time'` calcule `clamp(longueur × perChar, min, max)`.\n */\n private computeDuration(message: string, resolved: KtSnackbarConfig): number {\n const setting = resolved.duration;\n if (typeof setting === 'number') return setting;\n const computed = message.length * resolved.readingTimePerChar;\n return Math.min(resolved.readingTimeMax, Math.max(resolved.readingTimeMin, computed));\n }\n\n /** Résolution en cascade `option ?? KT_SNACKBAR_CONFIG ?? défaut`, champ par champ. */\n private resolve(options?: KtSnackbarOptions): KtSnackbarConfig {\n const config = this.config;\n return {\n duration: options?.duration ?? config?.duration ?? KT_SNACKBAR_DEFAULTS.duration,\n readingTimeMin: options?.readingTimeMin ?? config?.readingTimeMin ?? KT_SNACKBAR_DEFAULTS.readingTimeMin,\n readingTimeMax: options?.readingTimeMax ?? config?.readingTimeMax ?? KT_SNACKBAR_DEFAULTS.readingTimeMax,\n readingTimePerChar:\n options?.readingTimePerChar ?? config?.readingTimePerChar ?? KT_SNACKBAR_DEFAULTS.readingTimePerChar,\n timing: options?.timing ?? config?.timing ?? KT_SNACKBAR_DEFAULTS.timing,\n position: options?.position ?? config?.position ?? KT_SNACKBAR_DEFAULTS.position,\n politeness: options?.politeness ?? config?.politeness ?? KT_SNACKBAR_DEFAULTS.politeness,\n variant: options?.variant ?? config?.variant ?? KT_SNACKBAR_DEFAULTS.variant,\n closable: options?.closable ?? config?.closable ?? KT_SNACKBAR_DEFAULTS.closable,\n closeLabel: options?.closeLabel ?? config?.closeLabel ?? KT_SNACKBAR_DEFAULTS.closeLabel,\n max: options?.max ?? config?.max ?? KT_SNACKBAR_DEFAULTS.max,\n };\n }\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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"}
|