valtech-components 2.0.846 → 2.0.847

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,53 @@
1
+ import * as i0 from "@angular/core";
2
+ /**
3
+ * `val-modal-shell` — wrapper canónico para modales del factory.
4
+ *
5
+ * Implementa el patrón obligatorio de la Regla #5 (`frontend/CLAUDE.md`):
6
+ * - Header sin `<ion-title>`, solo botón de cierre con texto `Cerrar` en
7
+ * `slot="end"`.
8
+ * - Título principal del modal en el BODY como `val-display size="small"`
9
+ * (32/24 px).
10
+ * - Subtítulo opcional como `val-title size="large"` (24/18 px).
11
+ *
12
+ * El contenido del modal se proyecta vía `<ng-content>`.
13
+ *
14
+ * @example
15
+ * ```html
16
+ * <val-modal-shell [title]="title" [subtitle]="subtitle" (close)="dismiss()">
17
+ * <!-- body del modal -->
18
+ * <p>Contenido...</p>
19
+ * </val-modal-shell>
20
+ * ```
21
+ *
22
+ * @example Sin subtitle, con label de cierre custom:
23
+ * ```html
24
+ * <val-modal-shell [title]="title" [closeLabel]="'Fechar'" (close)="dismiss()">
25
+ * ...
26
+ * </val-modal-shell>
27
+ * ```
28
+ */
29
+ export declare class ModalShellComponent {
30
+ private i18n;
31
+ /** Título principal del modal — se renderiza como `val-display size="small"`. */
32
+ readonly title: import("@angular/core").InputSignal<string>;
33
+ /** Subtítulo opcional — se renderiza como `val-title size="large"`. */
34
+ readonly subtitle: import("@angular/core").InputSignal<string>;
35
+ /**
36
+ * Label del botón de cierre. Si no se provee, se resuelve vía i18n con la
37
+ * clave `close` del namespace `_global` ("Cerrar" / "Close" / "Fechar").
38
+ */
39
+ readonly closeLabel: import("@angular/core").InputSignal<string>;
40
+ /**
41
+ * Muestra el botón de cierre del header. `true` por defecto (estándar
42
+ * canónico). Solo se desactiva en modales no-dismissable (ej. confirmaciones
43
+ * bloqueantes que requieren respuesta explícita).
44
+ */
45
+ readonly showClose: import("@angular/core").InputSignal<boolean>;
46
+ /** Emite cuando el user toca el botón de cierre del header. */
47
+ readonly close: import("@angular/core").OutputEmitterRef<void>;
48
+ /** Label efectivo — usa `closeLabel` si viene, sino cae al i18n `_global.close`. */
49
+ readonly resolvedCloseLabel: import("@angular/core").Signal<string>;
50
+ onClose(): void;
51
+ static ɵfac: i0.ɵɵFactoryDeclaration<ModalShellComponent, never>;
52
+ static ɵcmp: i0.ɵɵComponentDeclaration<ModalShellComponent, "val-modal-shell", never, { "title": { "alias": "title"; "required": false; "isSignal": true; }; "subtitle": { "alias": "subtitle"; "required": false; "isSignal": true; }; "closeLabel": { "alias": "closeLabel"; "required": false; "isSignal": true; }; "showClose": { "alias": "showClose"; "required": false; "isSignal": true; }; }, { "close": "close"; }, never, ["*"], true, never>;
53
+ }
@@ -3,6 +3,10 @@ import * as i0 from "@angular/core";
3
3
  /**
4
4
  * Internal component for simple content modals.
5
5
  * Used by ModalService.openSimple()
6
+ *
7
+ * Header + título + close button delegados a `val-modal-shell` (estándar
8
+ * canónico — ver Regla #5 en `frontend/CLAUDE.md`). El cuerpo (innerHTML) +
9
+ * footer con botones quedan locales.
6
10
  */
7
11
  export declare class SimpleModalContentComponent {
8
12
  title: string;
package/lib/version.d.ts CHANGED
@@ -2,4 +2,4 @@
2
2
  * Current version of valtech-components.
3
3
  * This is automatically updated during the publish process.
4
4
  */
5
- export declare const VERSION = "2.0.846";
5
+ export declare const VERSION = "2.0.847";
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "valtech-components",
3
- "version": "2.0.846",
3
+ "version": "2.0.847",
4
4
  "private": false,
5
5
  "bin": {
6
6
  "valtech-firebase-config": "./src/lib/services/firebase/scripts/generate-sw-config.js"
package/public-api.d.ts CHANGED
@@ -175,6 +175,7 @@ export * from './lib/components/molecules/username-input/types';
175
175
  export * from './lib/components/molecules/linked-providers/linked-providers.component';
176
176
  export * from './lib/components/molecules/linked-providers/types';
177
177
  export * from './lib/components/molecules/image-crop/image-crop.component';
178
+ export * from './lib/components/molecules/modal-shell/modal-shell.component';
178
179
  export * from './lib/components/organisms/article/article.component';
179
180
  export * from './lib/components/organisms/article/types';
180
181
  export * from './lib/components/organisms/banner/banner.component';
@@ -1,145 +0,0 @@
1
- /**
2
- * Firebase Messaging Service Worker
3
- *
4
- * Service Worker estático para Firebase Cloud Messaging.
5
- * Carga la configuración dinámicamente desde /firebase-config.js.
6
- *
7
- * CONFIGURACIÓN:
8
- * 1. Crea firebase.config.json con tu configuración de Firebase
9
- * 2. Ejecuta: npm run generate:firebase-config
10
- * Esto genera /firebase-config.js con: self.FIREBASE_CONFIG = {...}
11
- * 3. Agrega este SW y firebase-config.js a los assets de angular.json
12
- *
13
- * Ver README.md para documentación completa.
14
- */
15
-
16
- // Importar Firebase scripts
17
- importScripts('https://www.gstatic.com/firebasejs/10.7.0/firebase-app-compat.js');
18
- importScripts('https://www.gstatic.com/firebasejs/10.7.0/firebase-messaging-compat.js');
19
-
20
- // Importar configuración desde archivo externo (generado en build)
21
- // Este archivo define: self.FIREBASE_CONFIG = { ... }
22
- try {
23
- importScripts('/firebase-config.js');
24
- } catch (e) {
25
- console.error('[SW] No se pudo cargar firebase-config.js:', e);
26
- }
27
-
28
- // Verificar que la configuración existe
29
- if (!self.FIREBASE_CONFIG) {
30
- console.error('[SW] FIREBASE_CONFIG no está definido.');
31
- console.error('[SW] Ejecuta: npm run generate:firebase-config');
32
- } else {
33
- // Inicializar Firebase
34
- firebase.initializeApp(self.FIREBASE_CONFIG);
35
-
36
- // Obtener instancia de messaging
37
- const messaging = firebase.messaging();
38
-
39
- /**
40
- * Handler para mensajes en background.
41
- */
42
- messaging.onBackgroundMessage((payload) => {
43
- console.log('[SW] Mensaje recibido en background:', payload);
44
-
45
- // Web push usa mensajes data-only (sin bloque `notification`) para que el
46
- // navegador NO auto-muestre una notificación duplicada — el SW es el único
47
- // que la pinta. Title/body/icon llegan dentro de `data`. Se mantiene el
48
- // fallback a `payload.notification` por compatibilidad con mensajes
49
- // legacy o nativos que sí traen el bloque.
50
- const data = payload.data || {};
51
- const notificationTitle =
52
- payload.notification?.title || data.title || 'Nueva notificación';
53
- const notificationBody = payload.notification?.body || data.body || '';
54
- const notificationIcon =
55
- payload.notification?.icon || data.icon || '/assets/icon/favicon.ico';
56
-
57
- const notificationOptions = {
58
- body: notificationBody,
59
- icon: notificationIcon,
60
- image: payload.notification?.image,
61
- badge: '/assets/icon/badge.png',
62
- tag: payload.messageId || data.messageId || 'default',
63
- data: {
64
- ...data,
65
- messageId: payload.messageId || data.messageId,
66
- title: notificationTitle,
67
- body: notificationBody,
68
- },
69
- vibrate: [200, 100, 200],
70
- requireInteraction: data.require_interaction === 'true',
71
- };
72
-
73
- return self.registration.showNotification(notificationTitle, notificationOptions);
74
- });
75
-
76
- /**
77
- * Handler para clicks en notificaciones.
78
- */
79
- self.addEventListener('notificationclick', (event) => {
80
- console.log('[SW] Click en notificación:', event);
81
- event.notification.close();
82
-
83
- const data = event.notification.data || {};
84
- let targetUrl = '/';
85
-
86
- if (data.route) {
87
- targetUrl = data.route;
88
- } else if (data.url) {
89
- targetUrl = data.url;
90
- }
91
-
92
- if (data.query_params) {
93
- const separator = targetUrl.includes('?') ? '&' : '?';
94
- targetUrl += separator + data.query_params;
95
- }
96
-
97
- const notificationPayload = {
98
- type: 'NOTIFICATION_CLICK',
99
- notification: {
100
- title: data.title,
101
- body: data.body,
102
- data: data,
103
- messageId: data.messageId,
104
- },
105
- };
106
-
107
- event.waitUntil(
108
- clients
109
- .matchAll({ type: 'window', includeUncontrolled: true })
110
- .then((clientList) => {
111
- // Siempre enviar postMessage para que la app pueda reaccionar
112
- for (const client of clientList) {
113
- client.postMessage(notificationPayload);
114
- }
115
-
116
- // Navegar si hay route o url
117
- if (targetUrl !== '/') {
118
- for (const client of clientList) {
119
- if ('navigate' in client) {
120
- return client.navigate(targetUrl).then((c) => c?.focus());
121
- }
122
- }
123
- // Si no hay cliente abierto, abrir nueva ventana
124
- if (clients.openWindow) {
125
- return clients.openWindow(targetUrl);
126
- }
127
- }
128
-
129
- // Solo hacer focus si no hay navegación
130
- for (const client of clientList) {
131
- if ('focus' in client) {
132
- return client.focus();
133
- }
134
- }
135
- if (clients.openWindow) {
136
- return clients.openWindow('/');
137
- }
138
- })
139
- );
140
- });
141
-
142
- self.addEventListener('notificationclose', (event) => {
143
- console.log('[SW] Notificación cerrada:', event.notification.tag);
144
- });
145
- }