valtech-components 2.0.695 → 2.0.712
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/esm2022/lib/components/molecules/select-search/select-search.component.mjs +25 -11
- package/esm2022/lib/components/organisms/form/form.component.mjs +10 -62
- package/esm2022/lib/services/firebase/firestore-collection.mjs +33 -1
- package/esm2022/lib/services/firebase/notifications.service.mjs +42 -16
- package/esm2022/lib/services/i18n/config.mjs +19 -1
- package/esm2022/lib/version.mjs +2 -2
- package/fesm2022/valtech-components.mjs +125 -87
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/select-search/select-search.component.d.ts +3 -1
- package/lib/components/organisms/form/form.component.d.ts +2 -4
- package/lib/services/firebase/firestore-collection.d.ts +8 -0
- package/lib/services/firebase/notifications.service.d.ts +16 -10
- package/lib/version.d.ts +1 -1
- package/package.json +1 -1
|
@@ -13,6 +13,8 @@ import { map } from 'rxjs/operators';
|
|
|
13
13
|
import { AuthService } from '../auth/auth.service';
|
|
14
14
|
import * as i0 from "@angular/core";
|
|
15
15
|
import * as i1 from "./firestore-collection";
|
|
16
|
+
/** Límite por defecto de notificaciones a cargar */
|
|
17
|
+
const DEFAULT_LIMIT = 50;
|
|
16
18
|
/**
|
|
17
19
|
* Servicio para leer notificaciones desde Firestore.
|
|
18
20
|
*
|
|
@@ -90,7 +92,9 @@ export class NotificationsService {
|
|
|
90
92
|
// Path relativo - FirestoreService agrega automáticamente apps/{appId}/
|
|
91
93
|
// NO agregar apps/ aquí para evitar doble prefijo
|
|
92
94
|
const basePath = `users/${userId}/notifications`;
|
|
93
|
-
this.collection = this.collectionFactory.create(basePath, {
|
|
95
|
+
this.collection = this.collectionFactory.create(basePath, {
|
|
96
|
+
timestamps: true,
|
|
97
|
+
});
|
|
94
98
|
console.log('[Notifications] Initialized with path:', basePath);
|
|
95
99
|
}
|
|
96
100
|
/**
|
|
@@ -109,39 +113,59 @@ export class NotificationsService {
|
|
|
109
113
|
// LECTURA
|
|
110
114
|
// ===========================================================================
|
|
111
115
|
/**
|
|
112
|
-
* Obtiene
|
|
113
|
-
*
|
|
116
|
+
* Obtiene notificaciones ordenadas por fecha descendente (real-time).
|
|
117
|
+
* Se actualiza automáticamente cuando cambian los datos.
|
|
118
|
+
*
|
|
119
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
114
120
|
*/
|
|
115
|
-
getAll() {
|
|
121
|
+
getAll(limit = DEFAULT_LIMIT) {
|
|
116
122
|
if (!this.collection)
|
|
117
123
|
return of([]);
|
|
118
124
|
return this.collection.watchAll({
|
|
119
125
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
126
|
+
limit,
|
|
120
127
|
});
|
|
121
128
|
}
|
|
122
129
|
/**
|
|
123
|
-
* Obtiene
|
|
130
|
+
* Obtiene notificaciones (one-time fetch sin listener).
|
|
124
131
|
* Útil para cargas iniciales sin necesidad de updates en tiempo real.
|
|
132
|
+
*
|
|
133
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
125
134
|
*/
|
|
126
|
-
async getAllOnce() {
|
|
135
|
+
async getAllOnce(limit = DEFAULT_LIMIT) {
|
|
127
136
|
if (!this.collection)
|
|
128
137
|
return [];
|
|
129
138
|
return this.collection.getAllOnce({
|
|
130
139
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
140
|
+
limit,
|
|
131
141
|
});
|
|
132
142
|
}
|
|
133
143
|
/**
|
|
134
|
-
* Obtiene solo notificaciones no leídas.
|
|
144
|
+
* Obtiene solo notificaciones no leídas (real-time, filtrado server-side).
|
|
145
|
+
*
|
|
146
|
+
* @param limit - Máximo de notificaciones (default: 50)
|
|
135
147
|
*/
|
|
136
|
-
getUnread() {
|
|
137
|
-
|
|
148
|
+
getUnread(limit = DEFAULT_LIMIT) {
|
|
149
|
+
if (!this.collection)
|
|
150
|
+
return of([]);
|
|
151
|
+
return this.collection.watchAll({
|
|
152
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
153
|
+
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
154
|
+
limit,
|
|
155
|
+
});
|
|
138
156
|
}
|
|
139
157
|
/**
|
|
140
|
-
* Cuenta notificaciones no leídas.
|
|
158
|
+
* Cuenta notificaciones no leídas (real-time, filtrado server-side).
|
|
141
159
|
* Útil para badges en UI.
|
|
142
160
|
*/
|
|
143
161
|
getUnreadCount() {
|
|
144
|
-
|
|
162
|
+
if (!this.collection)
|
|
163
|
+
return of(0);
|
|
164
|
+
return this.collection
|
|
165
|
+
.watchAll({
|
|
166
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
167
|
+
})
|
|
168
|
+
.pipe(map(n => n.length));
|
|
145
169
|
}
|
|
146
170
|
/**
|
|
147
171
|
* Obtiene una notificación por ID.
|
|
@@ -163,7 +187,7 @@ export class NotificationsService {
|
|
|
163
187
|
await this.collection.update(notificationId, { isRead: true });
|
|
164
188
|
}
|
|
165
189
|
/**
|
|
166
|
-
* Marca todas las notificaciones como leídas.
|
|
190
|
+
* Marca todas las notificaciones no leídas como leídas (batch write).
|
|
167
191
|
*/
|
|
168
192
|
async markAllAsRead() {
|
|
169
193
|
if (!this.collection)
|
|
@@ -171,7 +195,8 @@ export class NotificationsService {
|
|
|
171
195
|
const unread = await this.collection.query({
|
|
172
196
|
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
173
197
|
});
|
|
174
|
-
|
|
198
|
+
const ids = unread.map(n => n.id).filter(Boolean);
|
|
199
|
+
await this.collection.batchUpdate(ids, { isRead: true });
|
|
175
200
|
}
|
|
176
201
|
/**
|
|
177
202
|
* Elimina una notificación.
|
|
@@ -182,13 +207,14 @@ export class NotificationsService {
|
|
|
182
207
|
await this.collection.delete(notificationId);
|
|
183
208
|
}
|
|
184
209
|
/**
|
|
185
|
-
* Elimina todas las notificaciones del usuario.
|
|
210
|
+
* Elimina todas las notificaciones del usuario (batch delete).
|
|
186
211
|
*/
|
|
187
212
|
async deleteAll() {
|
|
188
213
|
if (!this.collection)
|
|
189
214
|
return;
|
|
190
215
|
const all = await this.collection.getAll();
|
|
191
|
-
|
|
216
|
+
const ids = all.map(n => n.id).filter(Boolean);
|
|
217
|
+
await this.collection.batchDelete(ids);
|
|
192
218
|
}
|
|
193
219
|
/**
|
|
194
220
|
* Limpia el estado del servicio.
|
|
@@ -207,4 +233,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
207
233
|
}], ctorParameters: () => [{ type: i0.Injector }, { type: i1.FirestoreCollectionFactory, decorators: [{
|
|
208
234
|
type: Optional
|
|
209
235
|
}] }] });
|
|
210
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notifications.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/notifications.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAY,QAAQ,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC9F,OAAO,EAAc,EAAE,EAAE,MAAM,MAAM,CAAC;AACtC,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;;;AAuBnD;;;;;;;;;;;;;;;;;;GAkBG;AAEH,MAAM,OAAO,oBAAoB;IAO/B,YACU,QAAkB,EACN,iBAA6C;QADzD,aAAQ,GAAR,QAAQ,CAAU;QACN,sBAAiB,GAAjB,iBAAiB,CAA4B;QAR3D,eAAU,GAAiD,IAAI,CAAC;QAChE,kBAAa,GAAkB,IAAI,CAAC;QAE5C,8DAA8D;QACtD,gBAAW,GAAuB,IAAI,CAAC;QAM7C,0EAA0E;QAC1E,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACjC,CAAC;IAED;;;OAGG;IACK,uBAAuB;QAC7B,oCAAoC;QACpC,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;QAC1D,CAAC;QAAC,MAAM,CAAC;YACP,8DAA8D;YAC9D,OAAO;QACT,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,OAAO;QACT,CAAC;QAED,wDAAwD;QACxD,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YACxC,MAAM,CAAC,GAAG,EAAE;gBACV,MAAM,IAAI,GAAG,IAAI,CAAC,WAAY,CAAC,IAAI,EAAE,CAAC;gBAEtC,IAAI,IAAI,EAAE,MAAM,IAAI,IAAI,CAAC,aAAa,KAAK,IAAI,CAAC,MAAM,EAAE,CAAC;oBACvD,6CAA6C;oBAC7C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;qBAAM,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;oBACvC,yBAAyB;oBACzB,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,UAAU,CAAC,MAAc;QACvB,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC5B,OAAO,CAAC,IAAI,CAAC,0GAA0G,CAAC,CAAC;YACzH,OAAO;QACT,CAAC;QAED,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAE5B,wEAAwE;QACxE,kDAAkD;QAClD,MAAM,QAAQ,GAAG,SAAS,MAAM,gBAAgB,CAAC;QAEjD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAC7C,QAAQ,EACR,EAAE,UAAU,EAAE,IAAI,EAAE,CACrB,CAAC;QAEF,OAAO,CAAC,GAAG,CAAC,wCAAwC,EAAE,QAAQ,CAAC,CAAC;IAClE,CAAC;IAED;;OAEG;IACH,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,UAAU,KAAK,IAAI,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,aAAa,CAAC;IAC5B,CAAC;IAED,8EAA8E;IAC9E,UAAU;IACV,8EAA8E;IAE9E;;;OAGG;IACH,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;QAEpC,OAAO,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;YAC9B,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC;SACrD,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,UAAU;QACd,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC;QAEhC,OAAO,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;YAChC,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC;SACrD,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC,IAAI,CACvB,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAC3D,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,OAAO,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IACnD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,cAAsB;QAClC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,IAAI,CAAC;QAClC,OAAO,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;IACjD,CAAC;IAED,8EAA8E;IAC9E,WAAW;IACX,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,UAAU,CAAC,cAAsB;QACrC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAC7B,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa;QACjB,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;YACzC,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;SAC3D,CAAC,CAAC;QAEH,MAAM,OAAO,CAAC,GAAG,CACf,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAG,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAClE,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,cAAsB;QACjC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAC7B,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;IAC/C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,SAAS;QACb,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC;QAC3C,MAAM,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAG,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC5B,CAAC;+GAlMU,oBAAoB;mHAApB,oBAAoB,cADP,MAAM;;4FACnB,oBAAoB;kBADhC,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAU7B,QAAQ","sourcesContent":["/**\n * Notifications Service\n *\n * Servicio para leer notificaciones desde Firestore.\n * El backend escribe las notificaciones, el frontend solo las lee y actualiza estado.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * También puede inicializarse manualmente con `initialize(userId)`.\n */\n\nimport { effect, Injectable, Injector, Optional, runInInjectionContext } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { map } from 'rxjs/operators';\n\nimport { AuthService } from '../auth/auth.service';\nimport { FirestoreCollectionFactory, TypedCollection } from './firestore-collection';\nimport { FirestoreDocument } from './types';\n\n/**\n * Documento de notificación en Firestore.\n * Estructura escrita por el backend.\n */\nexport interface NotificationDocument extends FirestoreDocument {\n  /** Título de la notificación */\n  title?: string;\n  /** Cuerpo del mensaje */\n  body?: string;\n  /** URL de imagen */\n  image?: string;\n  /** Datos personalizados (ej: route, actionType) */\n  data?: Record<string, string>;\n  /** Tipo de notificación */\n  type?: 'fcm' | 'system' | 'reminder' | string;\n  /** Si la notificación fue leída */\n  isRead: boolean;\n}\n\n/**\n * Servicio para leer notificaciones desde Firestore.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * No requiere llamar a `initialize()` manualmente si AuthService está configurado.\n *\n * @example\n * ```typescript\n * // Con AuthService configurado: auto-inicialización\n * // Solo inyectar y usar directamente\n * private notifications = inject(NotificationsService);\n *\n * notifications$ = this.notifications.getAll();\n * unreadCount$ = this.notifications.getUnreadCount();\n *\n * // Sin AuthService: inicialización manual\n * this.notifications.initialize(userId);\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class NotificationsService {\n  private collection: TypedCollection<NotificationDocument> | null = null;\n  private currentUserId: string | null = null;\n\n  // Inyección opcional - AuthService puede no estar configurado\n  private authService: AuthService | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Optional() private collectionFactory: FirestoreCollectionFactory\n  ) {\n    // Intentar obtener AuthService de forma lazy (puede no estar configurado)\n    this.setupAutoInitialization();\n  }\n\n  /**\n   * Configura auto-inicialización observando el estado de AuthService.\n   * Se ejecuta en el contexto del injector para poder usar effect().\n   */\n  private setupAutoInitialization(): void {\n    // Obtener AuthService de forma lazy\n    try {\n      this.authService = this.injector.get(AuthService, null);\n    } catch {\n      // AuthService no está configurado, no hay auto-inicialización\n      return;\n    }\n\n    if (!this.authService) {\n      return;\n    }\n\n    // Usar runInInjectionContext para poder crear el effect\n    runInInjectionContext(this.injector, () => {\n      effect(() => {\n        const user = this.authService!.user();\n\n        if (user?.userId && this.currentUserId !== user.userId) {\n          // Usuario autenticado: inicializar con su ID\n          this.initialize(user.userId);\n        } else if (!user && this.currentUserId) {\n          // Logout: limpiar estado\n          this.reset();\n        }\n      });\n    });\n  }\n\n  /**\n   * Inicializa el servicio para un usuario específico.\n   *\n   * NOTA: Se llama automáticamente si AuthService está configurado.\n   * Solo usar manualmente si AuthService no está disponible o se necesita\n   * un userId diferente al del usuario autenticado.\n   */\n  initialize(userId: string): void {\n    if (!this.collectionFactory) {\n      console.warn('[Notifications] FirestoreCollectionFactory not available. Ensure provideValtechFirebase() is configured.');\n      return;\n    }\n\n    this.currentUserId = userId;\n\n    // Path relativo - FirestoreService agrega automáticamente apps/{appId}/\n    // NO agregar apps/ aquí para evitar doble prefijo\n    const basePath = `users/${userId}/notifications`;\n\n    this.collection = this.collectionFactory.create<NotificationDocument>(\n      basePath,\n      { timestamps: true }\n    );\n\n    console.log('[Notifications] Initialized with path:', basePath);\n  }\n\n  /**\n   * Verifica si el servicio está inicializado.\n   */\n  get isReady(): boolean {\n    return this.collection !== null;\n  }\n\n  /**\n   * Obtiene el ID del usuario actual.\n   */\n  get userId(): string | null {\n    return this.currentUserId;\n  }\n\n  // ===========================================================================\n  // LECTURA\n  // ===========================================================================\n\n  /**\n   * Obtiene todas las notificaciones ordenadas por fecha descendente.\n   * Real-time: se actualiza automáticamente cuando cambian los datos.\n   */\n  getAll(): Observable<NotificationDocument[]> {\n    if (!this.collection) return of([]);\n\n    return this.collection.watchAll({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n    });\n  }\n\n  /**\n   * Obtiene todas las notificaciones (one-time fetch sin listener).\n   * Útil para cargas iniciales sin necesidad de updates en tiempo real.\n   */\n  async getAllOnce(): Promise<NotificationDocument[]> {\n    if (!this.collection) return [];\n\n    return this.collection.getAllOnce({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n    });\n  }\n\n  /**\n   * Obtiene solo notificaciones no leídas.\n   */\n  getUnread(): Observable<NotificationDocument[]> {\n    return this.getAll().pipe(\n      map(notifications => notifications.filter(n => !n.isRead))\n    );\n  }\n\n  /**\n   * Cuenta notificaciones no leídas.\n   * Útil para badges en UI.\n   */\n  getUnreadCount(): Observable<number> {\n    return this.getUnread().pipe(map(n => n.length));\n  }\n\n  /**\n   * Obtiene una notificación por ID.\n   */\n  async getById(notificationId: string): Promise<NotificationDocument | null> {\n    if (!this.collection) return null;\n    return this.collection.getById(notificationId);\n  }\n\n  // ===========================================================================\n  // ACCIONES\n  // ===========================================================================\n\n  /**\n   * Marca una notificación como leída.\n   */\n  async markAsRead(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.update(notificationId, { isRead: true });\n  }\n\n  /**\n   * Marca todas las notificaciones como leídas.\n   */\n  async markAllAsRead(): Promise<void> {\n    if (!this.collection) return;\n\n    const unread = await this.collection.query({\n      where: [{ field: 'isRead', operator: '==', value: false }],\n    });\n\n    await Promise.all(\n      unread.map(n => this.collection!.update(n.id!, { isRead: true }))\n    );\n  }\n\n  /**\n   * Elimina una notificación.\n   */\n  async delete(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.delete(notificationId);\n  }\n\n  /**\n   * Elimina todas las notificaciones del usuario.\n   */\n  async deleteAll(): Promise<void> {\n    if (!this.collection) return;\n\n    const all = await this.collection.getAll();\n    await Promise.all(all.map(n => this.collection!.delete(n.id!)));\n  }\n\n  /**\n   * Limpia el estado del servicio.\n   * Útil para logout.\n   */\n  reset(): void {\n    this.collection = null;\n    this.currentUserId = null;\n  }\n}\n"]}
|
|
236
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notifications.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/notifications.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAY,QAAQ,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC9F,OAAO,EAAc,EAAE,EAAE,MAAM,MAAM,CAAC;AACtC,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;;;AAInD,oDAAoD;AACpD,MAAM,aAAa,GAAG,EAAE,CAAC;AAqBzB;;;;;;;;;;;;;;;;;;GAkBG;AAEH,MAAM,OAAO,oBAAoB;IAO/B,YACU,QAAkB,EACN,iBAA6C;QADzD,aAAQ,GAAR,QAAQ,CAAU;QACN,sBAAiB,GAAjB,iBAAiB,CAA4B;QAR3D,eAAU,GAAiD,IAAI,CAAC;QAChE,kBAAa,GAAkB,IAAI,CAAC;QAE5C,8DAA8D;QACtD,gBAAW,GAAuB,IAAI,CAAC;QAM7C,0EAA0E;QAC1E,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACjC,CAAC;IAED;;;OAGG;IACK,uBAAuB;QAC7B,oCAAoC;QACpC,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;QAC1D,CAAC;QAAC,MAAM,CAAC;YACP,8DAA8D;YAC9D,OAAO;QACT,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,OAAO;QACT,CAAC;QAED,wDAAwD;QACxD,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YACxC,MAAM,CAAC,GAAG,EAAE;gBACV,MAAM,IAAI,GAAG,IAAI,CAAC,WAAY,CAAC,IAAI,EAAE,CAAC;gBAEtC,IAAI,IAAI,EAAE,MAAM,IAAI,IAAI,CAAC,aAAa,KAAK,IAAI,CAAC,MAAM,EAAE,CAAC;oBACvD,6CAA6C;oBAC7C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;qBAAM,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;oBACvC,yBAAyB;oBACzB,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,UAAU,CAAC,MAAc;QACvB,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC5B,OAAO,CAAC,IAAI,CACV,0GAA0G,CAC3G,CAAC;YACF,OAAO;QACT,CAAC;QAED,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAE5B,wEAAwE;QACxE,kDAAkD;QAClD,MAAM,QAAQ,GAAG,SAAS,MAAM,gBAAgB,CAAC;QAEjD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAuB,QAAQ,EAAE;YAC9E,UAAU,EAAE,IAAI;SACjB,CAAC,CAAC;QAEH,OAAO,CAAC,GAAG,CAAC,wCAAwC,EAAE,QAAQ,CAAC,CAAC;IAClE,CAAC;IAED;;OAEG;IACH,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,UAAU,KAAK,IAAI,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,aAAa,CAAC;IAC5B,CAAC;IAED,8EAA8E;IAC9E,UAAU;IACV,8EAA8E;IAE9E;;;;;OAKG;IACH,MAAM,CAAC,KAAK,GAAG,aAAa;QAC1B,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;QAEpC,OAAO,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;YAC9B,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC;YACpD,KAAK;SACN,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,UAAU,CAAC,KAAK,GAAG,aAAa;QACpC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC;QAEhC,OAAO,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;YAChC,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC;YACpD,KAAK;SACN,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,SAAS,CAAC,KAAK,GAAG,aAAa;QAC7B,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;QAEpC,OAAO,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;YAC9B,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;YAC1D,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC;YACpD,KAAK;SACN,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC;QAEnC,OAAO,IAAI,CAAC,UAAU;aACnB,QAAQ,CAAC;YACR,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;SAC3D,CAAC;aACD,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,cAAsB;QAClC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,IAAI,CAAC;QAClC,OAAO,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;IACjD,CAAC;IAED,8EAA8E;IAC9E,WAAW;IACX,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,UAAU,CAAC,cAAsB;QACrC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAC7B,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa;QACjB,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;YACzC,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;SAC3D,CAAC,CAAC;QAEH,MAAM,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAG,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QACnD,MAAM,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,cAAsB;QACjC,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAC7B,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;IAC/C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,SAAS;QACb,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC;QAC3C,MAAM,GAAG,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAG,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAChD,MAAM,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;IACzC,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC5B,CAAC;+GArNU,oBAAoB;mHAApB,oBAAoB,cADP,MAAM;;4FACnB,oBAAoB;kBADhC,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAU7B,QAAQ","sourcesContent":["/**\n * Notifications Service\n *\n * Servicio para leer notificaciones desde Firestore.\n * El backend escribe las notificaciones, el frontend solo las lee y actualiza estado.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * También puede inicializarse manualmente con `initialize(userId)`.\n */\n\nimport { effect, Injectable, Injector, Optional, runInInjectionContext } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { map } from 'rxjs/operators';\n\nimport { AuthService } from '../auth/auth.service';\nimport { FirestoreCollectionFactory, TypedCollection } from './firestore-collection';\nimport { FirestoreDocument } from './types';\n\n/** Límite por defecto de notificaciones a cargar */\nconst DEFAULT_LIMIT = 50;\n\n/**\n * Documento de notificación en Firestore.\n * Estructura escrita por el backend.\n */\nexport interface NotificationDocument extends FirestoreDocument {\n  /** Título de la notificación */\n  title?: string;\n  /** Cuerpo del mensaje */\n  body?: string;\n  /** URL de imagen */\n  image?: string;\n  /** Datos personalizados (ej: route, actionType) */\n  data?: Record<string, string>;\n  /** Tipo de notificación */\n  type?: 'fcm' | 'system' | 'reminder' | string;\n  /** Si la notificación fue leída */\n  isRead: boolean;\n}\n\n/**\n * Servicio para leer notificaciones desde Firestore.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * No requiere llamar a `initialize()` manualmente si AuthService está configurado.\n *\n * @example\n * ```typescript\n * // Con AuthService configurado: auto-inicialización\n * // Solo inyectar y usar directamente\n * private notifications = inject(NotificationsService);\n *\n * notifications$ = this.notifications.getAll();\n * unreadCount$ = this.notifications.getUnreadCount();\n *\n * // Sin AuthService: inicialización manual\n * this.notifications.initialize(userId);\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class NotificationsService {\n  private collection: TypedCollection<NotificationDocument> | null = null;\n  private currentUserId: string | null = null;\n\n  // Inyección opcional - AuthService puede no estar configurado\n  private authService: AuthService | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Optional() private collectionFactory: FirestoreCollectionFactory\n  ) {\n    // Intentar obtener AuthService de forma lazy (puede no estar configurado)\n    this.setupAutoInitialization();\n  }\n\n  /**\n   * Configura auto-inicialización observando el estado de AuthService.\n   * Se ejecuta en el contexto del injector para poder usar effect().\n   */\n  private setupAutoInitialization(): void {\n    // Obtener AuthService de forma lazy\n    try {\n      this.authService = this.injector.get(AuthService, null);\n    } catch {\n      // AuthService no está configurado, no hay auto-inicialización\n      return;\n    }\n\n    if (!this.authService) {\n      return;\n    }\n\n    // Usar runInInjectionContext para poder crear el effect\n    runInInjectionContext(this.injector, () => {\n      effect(() => {\n        const user = this.authService!.user();\n\n        if (user?.userId && this.currentUserId !== user.userId) {\n          // Usuario autenticado: inicializar con su ID\n          this.initialize(user.userId);\n        } else if (!user && this.currentUserId) {\n          // Logout: limpiar estado\n          this.reset();\n        }\n      });\n    });\n  }\n\n  /**\n   * Inicializa el servicio para un usuario específico.\n   *\n   * NOTA: Se llama automáticamente si AuthService está configurado.\n   * Solo usar manualmente si AuthService no está disponible o se necesita\n   * un userId diferente al del usuario autenticado.\n   */\n  initialize(userId: string): void {\n    if (!this.collectionFactory) {\n      console.warn(\n        '[Notifications] FirestoreCollectionFactory not available. Ensure provideValtechFirebase() is configured.'\n      );\n      return;\n    }\n\n    this.currentUserId = userId;\n\n    // Path relativo - FirestoreService agrega automáticamente apps/{appId}/\n    // NO agregar apps/ aquí para evitar doble prefijo\n    const basePath = `users/${userId}/notifications`;\n\n    this.collection = this.collectionFactory.create<NotificationDocument>(basePath, {\n      timestamps: true,\n    });\n\n    console.log('[Notifications] Initialized with path:', basePath);\n  }\n\n  /**\n   * Verifica si el servicio está inicializado.\n   */\n  get isReady(): boolean {\n    return this.collection !== null;\n  }\n\n  /**\n   * Obtiene el ID del usuario actual.\n   */\n  get userId(): string | null {\n    return this.currentUserId;\n  }\n\n  // ===========================================================================\n  // LECTURA\n  // ===========================================================================\n\n  /**\n   * Obtiene notificaciones ordenadas por fecha descendente (real-time).\n   * Se actualiza automáticamente cuando cambian los datos.\n   *\n   * @param limit - Máximo de notificaciones a cargar (default: 50)\n   */\n  getAll(limit = DEFAULT_LIMIT): Observable<NotificationDocument[]> {\n    if (!this.collection) return of([]);\n\n    return this.collection.watchAll({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n      limit,\n    });\n  }\n\n  /**\n   * Obtiene notificaciones (one-time fetch sin listener).\n   * Útil para cargas iniciales sin necesidad de updates en tiempo real.\n   *\n   * @param limit - Máximo de notificaciones a cargar (default: 50)\n   */\n  async getAllOnce(limit = DEFAULT_LIMIT): Promise<NotificationDocument[]> {\n    if (!this.collection) return [];\n\n    return this.collection.getAllOnce({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n      limit,\n    });\n  }\n\n  /**\n   * Obtiene solo notificaciones no leídas (real-time, filtrado server-side).\n   *\n   * @param limit - Máximo de notificaciones (default: 50)\n   */\n  getUnread(limit = DEFAULT_LIMIT): Observable<NotificationDocument[]> {\n    if (!this.collection) return of([]);\n\n    return this.collection.watchAll({\n      where: [{ field: 'isRead', operator: '==', value: false }],\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n      limit,\n    });\n  }\n\n  /**\n   * Cuenta notificaciones no leídas (real-time, filtrado server-side).\n   * Útil para badges en UI.\n   */\n  getUnreadCount(): Observable<number> {\n    if (!this.collection) return of(0);\n\n    return this.collection\n      .watchAll({\n        where: [{ field: 'isRead', operator: '==', value: false }],\n      })\n      .pipe(map(n => n.length));\n  }\n\n  /**\n   * Obtiene una notificación por ID.\n   */\n  async getById(notificationId: string): Promise<NotificationDocument | null> {\n    if (!this.collection) return null;\n    return this.collection.getById(notificationId);\n  }\n\n  // ===========================================================================\n  // ACCIONES\n  // ===========================================================================\n\n  /**\n   * Marca una notificación como leída.\n   */\n  async markAsRead(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.update(notificationId, { isRead: true });\n  }\n\n  /**\n   * Marca todas las notificaciones no leídas como leídas (batch write).\n   */\n  async markAllAsRead(): Promise<void> {\n    if (!this.collection) return;\n\n    const unread = await this.collection.query({\n      where: [{ field: 'isRead', operator: '==', value: false }],\n    });\n\n    const ids = unread.map(n => n.id!).filter(Boolean);\n    await this.collection.batchUpdate(ids, { isRead: true });\n  }\n\n  /**\n   * Elimina una notificación.\n   */\n  async delete(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.delete(notificationId);\n  }\n\n  /**\n   * Elimina todas las notificaciones del usuario (batch delete).\n   */\n  async deleteAll(): Promise<void> {\n    if (!this.collection) return;\n\n    const all = await this.collection.getAll();\n    const ids = all.map(n => n.id!).filter(Boolean);\n    await this.collection.batchDelete(ids);\n  }\n\n  /**\n   * Limpia el estado del servicio.\n   * Útil para logout.\n   */\n  reset(): void {\n    this.collection = null;\n    this.currentUserId = null;\n  }\n}\n"]}
|
|
@@ -51,6 +51,24 @@ export function provideValtechI18n(config = {}) {
|
|
|
51
51
|
if (Object.keys(contentToRegister).length > 0) {
|
|
52
52
|
i18n.registerContentBulk(contentToRegister);
|
|
53
53
|
}
|
|
54
|
+
// Aplicar idioma según configuración
|
|
55
|
+
if (!mergedConfig.detectBrowserLanguage) {
|
|
56
|
+
// Sin detección de navegador: forzar idioma por defecto
|
|
57
|
+
i18n.setLanguage(mergedConfig.defaultLanguage);
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
// Con detección: re-evaluar ahora que supportedLanguages está configurado
|
|
61
|
+
const stored = localStorage.getItem('app_lang');
|
|
62
|
+
if (!stored || !mergedConfig.supportedLanguages.includes(stored)) {
|
|
63
|
+
const browserLang = navigator.language.split('-')[0];
|
|
64
|
+
if (mergedConfig.supportedLanguages.includes(browserLang)) {
|
|
65
|
+
i18n.setLanguage(browserLang);
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
i18n.setLanguage(mergedConfig.defaultLanguage);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
54
72
|
};
|
|
55
73
|
},
|
|
56
74
|
deps: [I18nService],
|
|
@@ -107,4 +125,4 @@ function deepMergeLanguagesContent(base, override) {
|
|
|
107
125
|
}
|
|
108
126
|
return result;
|
|
109
127
|
}
|
|
110
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/i18n/config.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EnvironmentProviders,\n  makeEnvironmentProviders,\n  APP_INITIALIZER,\n} from '@angular/core';\nimport { I18nConfig, DEFAULT_I18N_CONFIG, ContentStore, LanguagesContent, I18nLang } from './types';\nimport { I18nService } from './i18n.service';\nimport { VALTECH_DEFAULT_CONTENT } from './default-content';\n\n/**\n * Configura el sistema de internacionalización de Valtech Components.\n *\n * @param config Configuración de i18n\n * @returns Providers para agregar en app.config.ts\n *\n * @example\n * // app.config.ts\n * import { provideValtechI18n } from 'valtech-components';\n * import { GLOBAL_CONTENT } from './i18n/_global';\n * import { LOGIN_CONTENT } from './i18n/login.i18n';\n *\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     provideValtechI18n({\n *       defaultLanguage: 'es',\n *       supportedLanguages: ['es', 'en'],\n *       detectBrowserLanguage: true,\n *       content: {\n *         '_global': GLOBAL_CONTENT,\n *         'Login': LOGIN_CONTENT,\n *       }\n *     }),\n *   ]\n * };\n */\nexport function provideValtechI18n(config: I18nConfig = {}): EnvironmentProviders {\n  const mergedConfig = { ...DEFAULT_I18N_CONFIG, ...config };\n\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      useFactory: (i18n: I18nService) => {\n        return () => {\n          // Configurar idiomas soportados\n          i18n.setI18nLanguages(mergedConfig.supportedLanguages);\n\n          // Determinar contenido a registrar\n          let contentToRegister: ContentStore = {};\n\n          if (mergedConfig.includeDefaultContent !== false) {\n            // Merge: defaults + user config (user gana)\n            contentToRegister = deepMergeContent(\n              VALTECH_DEFAULT_CONTENT,\n              mergedConfig.content || {}\n            );\n          } else if (mergedConfig.content) {\n            // Solo contenido del usuario\n            contentToRegister = mergedConfig.content;\n          }\n\n          // Registrar contenido\n          if (Object.keys(contentToRegister).length > 0) {\n            i18n.registerContentBulk(contentToRegister);\n          }\n        };\n      },\n      deps: [I18nService],\n      multi: true,\n    },\n  ]);\n}\n\n/**\n * Deep merge de ContentStore.\n * El contenido de `override` sobrescribe keys específicas de `base`.\n *\n * @param base - Contenido base (defaults de valtech-components)\n * @param override - Contenido del usuario que sobrescribe\n * @returns ContentStore mergeado\n *\n * @example\n * const base = { _global: { es: { ok: 'Aceptar' } } };\n * const override = { _global: { es: { ok: 'Dale' } } };\n * // Resultado: { _global: { es: { ok: 'Dale' } } }\n */\nfunction deepMergeContent(base: ContentStore, override: ContentStore): ContentStore {\n  const result: ContentStore = {};\n\n  // Copiar todos los namespaces del base\n  for (const namespace of Object.keys(base)) {\n    result[namespace] = deepMergeLanguagesContent(base[namespace], {});\n  }\n\n  // Mergear namespaces del override\n  for (const namespace of Object.keys(override)) {\n    if (result[namespace]) {\n      // Namespace existe en base, hacer deep merge\n      result[namespace] = deepMergeLanguagesContent(result[namespace], override[namespace]);\n    } else {\n      // Namespace nuevo, copiar completo\n      result[namespace] = deepMergeLanguagesContent({}, override[namespace]);\n    }\n  }\n\n  return result;\n}\n\n/**\n * Deep merge de LanguagesContent (un namespace).\n */\nfunction deepMergeLanguagesContent(\n  base: LanguagesContent,\n  override: LanguagesContent\n): LanguagesContent {\n  const result: LanguagesContent = {};\n  const allLangs = new Set([\n    ...Object.keys(base),\n    ...Object.keys(override),\n  ]) as Set<I18nLang>;\n\n  for (const lang of allLangs) {\n    result[lang] = {\n      ...(base[lang] || {}),\n      ...(override[lang] || {}),\n    };\n  }\n\n  return result;\n}\n"]}
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/i18n/config.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EnvironmentProviders,\n  makeEnvironmentProviders,\n  APP_INITIALIZER,\n} from '@angular/core';\nimport { I18nConfig, DEFAULT_I18N_CONFIG, ContentStore, LanguagesContent, I18nLang } from './types';\nimport { I18nService } from './i18n.service';\nimport { VALTECH_DEFAULT_CONTENT } from './default-content';\n\n/**\n * Configura el sistema de internacionalización de Valtech Components.\n *\n * @param config Configuración de i18n\n * @returns Providers para agregar en app.config.ts\n *\n * @example\n * // app.config.ts\n * import { provideValtechI18n } from 'valtech-components';\n * import { GLOBAL_CONTENT } from './i18n/_global';\n * import { LOGIN_CONTENT } from './i18n/login.i18n';\n *\n * export const appConfig: ApplicationConfig = {\n *   providers: [\n *     provideValtechI18n({\n *       defaultLanguage: 'es',\n *       supportedLanguages: ['es', 'en'],\n *       detectBrowserLanguage: true,\n *       content: {\n *         '_global': GLOBAL_CONTENT,\n *         'Login': LOGIN_CONTENT,\n *       }\n *     }),\n *   ]\n * };\n */\nexport function provideValtechI18n(config: I18nConfig = {}): EnvironmentProviders {\n  const mergedConfig = { ...DEFAULT_I18N_CONFIG, ...config };\n\n  return makeEnvironmentProviders([\n    {\n      provide: APP_INITIALIZER,\n      useFactory: (i18n: I18nService) => {\n        return () => {\n          // Configurar idiomas soportados\n          i18n.setI18nLanguages(mergedConfig.supportedLanguages);\n\n          // Determinar contenido a registrar\n          let contentToRegister: ContentStore = {};\n\n          if (mergedConfig.includeDefaultContent !== false) {\n            // Merge: defaults + user config (user gana)\n            contentToRegister = deepMergeContent(\n              VALTECH_DEFAULT_CONTENT,\n              mergedConfig.content || {}\n            );\n          } else if (mergedConfig.content) {\n            // Solo contenido del usuario\n            contentToRegister = mergedConfig.content;\n          }\n\n          // Registrar contenido\n          if (Object.keys(contentToRegister).length > 0) {\n            i18n.registerContentBulk(contentToRegister);\n          }\n\n          // Aplicar idioma según configuración\n          if (!mergedConfig.detectBrowserLanguage) {\n            // Sin detección de navegador: forzar idioma por defecto\n            i18n.setLanguage(mergedConfig.defaultLanguage);\n          } else {\n            // Con detección: re-evaluar ahora que supportedLanguages está configurado\n            const stored = localStorage.getItem('app_lang');\n            if (!stored || !mergedConfig.supportedLanguages.includes(stored as I18nLang)) {\n              const browserLang = navigator.language.split('-')[0] as I18nLang;\n              if (mergedConfig.supportedLanguages.includes(browserLang)) {\n                i18n.setLanguage(browserLang);\n              } else {\n                i18n.setLanguage(mergedConfig.defaultLanguage);\n              }\n            }\n          }\n        };\n      },\n      deps: [I18nService],\n      multi: true,\n    },\n  ]);\n}\n\n/**\n * Deep merge de ContentStore.\n * El contenido de `override` sobrescribe keys específicas de `base`.\n *\n * @param base - Contenido base (defaults de valtech-components)\n * @param override - Contenido del usuario que sobrescribe\n * @returns ContentStore mergeado\n *\n * @example\n * const base = { _global: { es: { ok: 'Aceptar' } } };\n * const override = { _global: { es: { ok: 'Dale' } } };\n * // Resultado: { _global: { es: { ok: 'Dale' } } }\n */\nfunction deepMergeContent(base: ContentStore, override: ContentStore): ContentStore {\n  const result: ContentStore = {};\n\n  // Copiar todos los namespaces del base\n  for (const namespace of Object.keys(base)) {\n    result[namespace] = deepMergeLanguagesContent(base[namespace], {});\n  }\n\n  // Mergear namespaces del override\n  for (const namespace of Object.keys(override)) {\n    if (result[namespace]) {\n      // Namespace existe en base, hacer deep merge\n      result[namespace] = deepMergeLanguagesContent(result[namespace], override[namespace]);\n    } else {\n      // Namespace nuevo, copiar completo\n      result[namespace] = deepMergeLanguagesContent({}, override[namespace]);\n    }\n  }\n\n  return result;\n}\n\n/**\n * Deep merge de LanguagesContent (un namespace).\n */\nfunction deepMergeLanguagesContent(\n  base: LanguagesContent,\n  override: LanguagesContent\n): LanguagesContent {\n  const result: LanguagesContent = {};\n  const allLangs = new Set([\n    ...Object.keys(base),\n    ...Object.keys(override),\n  ]) as Set<I18nLang>;\n\n  for (const lang of allLangs) {\n    result[lang] = {\n      ...(base[lang] || {}),\n      ...(override[lang] || {}),\n    };\n  }\n\n  return result;\n}\n"]}
|
package/esm2022/lib/version.mjs
CHANGED
|
@@ -2,5 +2,5 @@
|
|
|
2
2
|
* Current version of valtech-components.
|
|
3
3
|
* This is automatically updated during the publish process.
|
|
4
4
|
*/
|
|
5
|
-
export const VERSION = '2.0.
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
export const VERSION = '2.0.712';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC43MTInO1xuIl19
|
|
@@ -50,7 +50,7 @@ import 'prismjs/components/prism-json';
|
|
|
50
50
|
* Current version of valtech-components.
|
|
51
51
|
* This is automatically updated during the publish process.
|
|
52
52
|
*/
|
|
53
|
-
const VERSION = '2.0.
|
|
53
|
+
const VERSION = '2.0.712';
|
|
54
54
|
|
|
55
55
|
/**
|
|
56
56
|
* Servicio para gestionar presets de componentes.
|
|
@@ -4364,6 +4364,24 @@ function provideValtechI18n(config = {}) {
|
|
|
4364
4364
|
if (Object.keys(contentToRegister).length > 0) {
|
|
4365
4365
|
i18n.registerContentBulk(contentToRegister);
|
|
4366
4366
|
}
|
|
4367
|
+
// Aplicar idioma según configuración
|
|
4368
|
+
if (!mergedConfig.detectBrowserLanguage) {
|
|
4369
|
+
// Sin detección de navegador: forzar idioma por defecto
|
|
4370
|
+
i18n.setLanguage(mergedConfig.defaultLanguage);
|
|
4371
|
+
}
|
|
4372
|
+
else {
|
|
4373
|
+
// Con detección: re-evaluar ahora que supportedLanguages está configurado
|
|
4374
|
+
const stored = localStorage.getItem('app_lang');
|
|
4375
|
+
if (!stored || !mergedConfig.supportedLanguages.includes(stored)) {
|
|
4376
|
+
const browserLang = navigator.language.split('-')[0];
|
|
4377
|
+
if (mergedConfig.supportedLanguages.includes(browserLang)) {
|
|
4378
|
+
i18n.setLanguage(browserLang);
|
|
4379
|
+
}
|
|
4380
|
+
else {
|
|
4381
|
+
i18n.setLanguage(mergedConfig.defaultLanguage);
|
|
4382
|
+
}
|
|
4383
|
+
}
|
|
4384
|
+
}
|
|
4367
4385
|
};
|
|
4368
4386
|
},
|
|
4369
4387
|
deps: [I18nService],
|
|
@@ -9058,18 +9076,28 @@ class SelectSearchComponent {
|
|
|
9058
9076
|
return;
|
|
9059
9077
|
}
|
|
9060
9078
|
const controlValue = this.props.control.value;
|
|
9061
|
-
if (
|
|
9062
|
-
|
|
9063
|
-
|
|
9064
|
-
|
|
9065
|
-
|
|
9066
|
-
|
|
9067
|
-
|
|
9068
|
-
const selectedOption = map.get(controlValue);
|
|
9069
|
-
this.selectedItems = selectedOption ? [selectedOption] : [];
|
|
9079
|
+
if (this.props.mode === 'legacy') {
|
|
9080
|
+
// Permisivo: muestra el valor aunque no esté en las opciones
|
|
9081
|
+
if (controlValue !== null && controlValue !== undefined) {
|
|
9082
|
+
this.displayValue = controlValue.toString();
|
|
9083
|
+
this.selectedItems = [];
|
|
9084
|
+
return;
|
|
9085
|
+
}
|
|
9070
9086
|
}
|
|
9071
9087
|
else {
|
|
9072
|
-
|
|
9088
|
+
// Estricto: solo muestra si coincide con una opción
|
|
9089
|
+
if (controlValue === null || controlValue === undefined) {
|
|
9090
|
+
this.selectedItems = [];
|
|
9091
|
+
return;
|
|
9092
|
+
}
|
|
9093
|
+
if (this.props.options && this.props.options.length > 0) {
|
|
9094
|
+
const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));
|
|
9095
|
+
const selectedOption = map.get(controlValue);
|
|
9096
|
+
this.selectedItems = selectedOption ? [selectedOption] : [];
|
|
9097
|
+
}
|
|
9098
|
+
else {
|
|
9099
|
+
this.selectedItems = [];
|
|
9100
|
+
}
|
|
9073
9101
|
}
|
|
9074
9102
|
}
|
|
9075
9103
|
applyDefaultValue() {
|
|
@@ -9161,6 +9189,10 @@ class SelectSearchComponent {
|
|
|
9161
9189
|
return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);
|
|
9162
9190
|
}
|
|
9163
9191
|
updateDisplayValue() {
|
|
9192
|
+
if (this.props?.mode === 'legacy' && this.selectedItems.length === 0 && this.props?.control?.value) {
|
|
9193
|
+
this.displayValue = this.props.control.value.toString();
|
|
9194
|
+
return;
|
|
9195
|
+
}
|
|
9164
9196
|
if (this.selectedItems.length === 0) {
|
|
9165
9197
|
this.displayValue = '';
|
|
9166
9198
|
return;
|
|
@@ -22682,8 +22714,6 @@ class FormComponent {
|
|
|
22682
22714
|
this.onSelectChange = new EventEmitter();
|
|
22683
22715
|
this.types = InputType;
|
|
22684
22716
|
this.subscriptions = [];
|
|
22685
|
-
// Cache para evitar crear nuevos objetos en cada ciclo de change detection
|
|
22686
|
-
this.fieldPropsCache = new Map();
|
|
22687
22717
|
this.actionsCache = [];
|
|
22688
22718
|
}
|
|
22689
22719
|
ngOnInit() {
|
|
@@ -22711,15 +22741,14 @@ class FormComponent {
|
|
|
22711
22741
|
this.trackSelectChanges(field.name);
|
|
22712
22742
|
});
|
|
22713
22743
|
});
|
|
22714
|
-
|
|
22715
|
-
this.rebuildFieldPropsCache();
|
|
22744
|
+
this.syncFieldControlStates();
|
|
22716
22745
|
this.updateActionsCache();
|
|
22717
22746
|
this.previousState = this.props.state;
|
|
22718
22747
|
}
|
|
22719
22748
|
ngOnChanges(changes) {
|
|
22720
22749
|
// Cuando props cambia, reconstruir el cache
|
|
22721
22750
|
if (changes['props'] && this.form) {
|
|
22722
|
-
this.
|
|
22751
|
+
this.syncFieldControlStates();
|
|
22723
22752
|
this.updateActionsCache();
|
|
22724
22753
|
this.previousState = this.props.state;
|
|
22725
22754
|
}
|
|
@@ -22732,14 +22761,11 @@ class FormComponent {
|
|
|
22732
22761
|
}
|
|
22733
22762
|
}
|
|
22734
22763
|
/**
|
|
22735
|
-
*
|
|
22736
|
-
* Esto evita crear nuevos objetos en cada ciclo de change detection.
|
|
22764
|
+
* Synchronizes form control disabled/enabled state with field metadata.
|
|
22737
22765
|
*/
|
|
22738
|
-
|
|
22739
|
-
this.fieldPropsCache.clear();
|
|
22766
|
+
syncFieldControlStates() {
|
|
22740
22767
|
this.props.sections.forEach(section => {
|
|
22741
22768
|
section.fields.forEach(field => {
|
|
22742
|
-
const token = field.token || `input-${field.type}-${field.name}`;
|
|
22743
22769
|
if (field.type === this.types.NUMBER_FROM_TO) {
|
|
22744
22770
|
const fromControl = this.getControl(`${field.name}_from`);
|
|
22745
22771
|
const toControl = this.getControl(`${field.name}_to`);
|
|
@@ -22751,13 +22777,6 @@ class FormComponent {
|
|
|
22751
22777
|
fromControl.enable({ emitEvent: false });
|
|
22752
22778
|
toControl.enable({ emitEvent: false });
|
|
22753
22779
|
}
|
|
22754
|
-
this.fieldPropsCache.set(field.name, {
|
|
22755
|
-
...field,
|
|
22756
|
-
token,
|
|
22757
|
-
fromControl,
|
|
22758
|
-
toControl,
|
|
22759
|
-
control: undefined,
|
|
22760
|
-
});
|
|
22761
22780
|
}
|
|
22762
22781
|
else {
|
|
22763
22782
|
const control = this.getControl(field.name);
|
|
@@ -22767,11 +22786,6 @@ class FormComponent {
|
|
|
22767
22786
|
else {
|
|
22768
22787
|
control.enable({ emitEvent: false });
|
|
22769
22788
|
}
|
|
22770
|
-
this.fieldPropsCache.set(field.name, {
|
|
22771
|
-
...field,
|
|
22772
|
-
token,
|
|
22773
|
-
control,
|
|
22774
|
-
});
|
|
22775
22789
|
}
|
|
22776
22790
|
});
|
|
22777
22791
|
});
|
|
@@ -22810,50 +22824,16 @@ class FormComponent {
|
|
|
22810
22824
|
return this.Form.get(field);
|
|
22811
22825
|
}
|
|
22812
22826
|
getFieldProp(field) {
|
|
22813
|
-
|
|
22814
|
-
|
|
22815
|
-
if (cached) {
|
|
22816
|
-
return cached;
|
|
22827
|
+
if (!field.token) {
|
|
22828
|
+
field.token = `input-${field.type}-${field.name}`;
|
|
22817
22829
|
}
|
|
22818
|
-
// Fallback: generar y cachear si no existe (no debería ocurrir normalmente)
|
|
22819
|
-
const token = field.token || `input-${field.type}-${field.name}`;
|
|
22820
22830
|
if (field.type === this.types.NUMBER_FROM_TO) {
|
|
22821
22831
|
const fromControl = this.getControl(`${field.name}_from`);
|
|
22822
22832
|
const toControl = this.getControl(`${field.name}_to`);
|
|
22823
|
-
|
|
22824
|
-
fromControl.disable({ emitEvent: false });
|
|
22825
|
-
toControl.disable({ emitEvent: false });
|
|
22826
|
-
}
|
|
22827
|
-
else {
|
|
22828
|
-
fromControl.enable({ emitEvent: false });
|
|
22829
|
-
toControl.enable({ emitEvent: false });
|
|
22830
|
-
}
|
|
22831
|
-
const props = {
|
|
22832
|
-
...field,
|
|
22833
|
-
token,
|
|
22834
|
-
fromControl,
|
|
22835
|
-
toControl,
|
|
22836
|
-
control: undefined,
|
|
22837
|
-
};
|
|
22838
|
-
this.fieldPropsCache.set(field.name, props);
|
|
22839
|
-
return props;
|
|
22840
|
-
}
|
|
22841
|
-
else {
|
|
22842
|
-
const control = this.getControl(field.name);
|
|
22843
|
-
if (field.state === ComponentStates.DISABLED) {
|
|
22844
|
-
control.disable({ emitEvent: false });
|
|
22845
|
-
}
|
|
22846
|
-
else {
|
|
22847
|
-
control.enable({ emitEvent: false });
|
|
22848
|
-
}
|
|
22849
|
-
const props = {
|
|
22850
|
-
...field,
|
|
22851
|
-
token,
|
|
22852
|
-
control,
|
|
22853
|
-
};
|
|
22854
|
-
this.fieldPropsCache.set(field.name, props);
|
|
22855
|
-
return props;
|
|
22833
|
+
return { ...field, fromControl, toControl, control: undefined };
|
|
22856
22834
|
}
|
|
22835
|
+
const control = this.getControl(field.name);
|
|
22836
|
+
return { ...field, control };
|
|
22857
22837
|
}
|
|
22858
22838
|
get isAtEndOfForm() {
|
|
22859
22839
|
return isAtEnd(this.elementRef);
|
|
@@ -28467,6 +28447,38 @@ class TypedCollection {
|
|
|
28467
28447
|
}
|
|
28468
28448
|
return this.firestore.deleteDoc(this.collectionPath, id);
|
|
28469
28449
|
}
|
|
28450
|
+
/**
|
|
28451
|
+
* Actualiza múltiples documentos en una sola operación atómica.
|
|
28452
|
+
*/
|
|
28453
|
+
async batchUpdate(ids, data) {
|
|
28454
|
+
if (ids.length === 0)
|
|
28455
|
+
return;
|
|
28456
|
+
const BATCH_LIMIT = 500; // Firestore batch limit
|
|
28457
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
28458
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
28459
|
+
await this.firestore.batch(batch => {
|
|
28460
|
+
for (const id of chunk) {
|
|
28461
|
+
batch.update(`${this.collectionPath}/${id}`, data);
|
|
28462
|
+
}
|
|
28463
|
+
});
|
|
28464
|
+
}
|
|
28465
|
+
}
|
|
28466
|
+
/**
|
|
28467
|
+
* Elimina múltiples documentos en una sola operación atómica.
|
|
28468
|
+
*/
|
|
28469
|
+
async batchDelete(ids) {
|
|
28470
|
+
if (ids.length === 0)
|
|
28471
|
+
return;
|
|
28472
|
+
const BATCH_LIMIT = 500;
|
|
28473
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
28474
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
28475
|
+
await this.firestore.batch(batch => {
|
|
28476
|
+
for (const id of chunk) {
|
|
28477
|
+
batch.delete(`${this.collectionPath}/${id}`);
|
|
28478
|
+
}
|
|
28479
|
+
});
|
|
28480
|
+
}
|
|
28481
|
+
}
|
|
28470
28482
|
/**
|
|
28471
28483
|
* Restaura un documento soft-deleted.
|
|
28472
28484
|
*/
|
|
@@ -29909,6 +29921,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
29909
29921
|
* Se auto-inicializa cuando AuthService tiene un usuario autenticado.
|
|
29910
29922
|
* También puede inicializarse manualmente con `initialize(userId)`.
|
|
29911
29923
|
*/
|
|
29924
|
+
/** Límite por defecto de notificaciones a cargar */
|
|
29925
|
+
const DEFAULT_LIMIT = 50;
|
|
29912
29926
|
/**
|
|
29913
29927
|
* Servicio para leer notificaciones desde Firestore.
|
|
29914
29928
|
*
|
|
@@ -29986,7 +30000,9 @@ class NotificationsService {
|
|
|
29986
30000
|
// Path relativo - FirestoreService agrega automáticamente apps/{appId}/
|
|
29987
30001
|
// NO agregar apps/ aquí para evitar doble prefijo
|
|
29988
30002
|
const basePath = `users/${userId}/notifications`;
|
|
29989
|
-
this.collection = this.collectionFactory.create(basePath, {
|
|
30003
|
+
this.collection = this.collectionFactory.create(basePath, {
|
|
30004
|
+
timestamps: true,
|
|
30005
|
+
});
|
|
29990
30006
|
console.log('[Notifications] Initialized with path:', basePath);
|
|
29991
30007
|
}
|
|
29992
30008
|
/**
|
|
@@ -30005,39 +30021,59 @@ class NotificationsService {
|
|
|
30005
30021
|
// LECTURA
|
|
30006
30022
|
// ===========================================================================
|
|
30007
30023
|
/**
|
|
30008
|
-
* Obtiene
|
|
30009
|
-
*
|
|
30024
|
+
* Obtiene notificaciones ordenadas por fecha descendente (real-time).
|
|
30025
|
+
* Se actualiza automáticamente cuando cambian los datos.
|
|
30026
|
+
*
|
|
30027
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
30010
30028
|
*/
|
|
30011
|
-
getAll() {
|
|
30029
|
+
getAll(limit = DEFAULT_LIMIT) {
|
|
30012
30030
|
if (!this.collection)
|
|
30013
30031
|
return of([]);
|
|
30014
30032
|
return this.collection.watchAll({
|
|
30015
30033
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30034
|
+
limit,
|
|
30016
30035
|
});
|
|
30017
30036
|
}
|
|
30018
30037
|
/**
|
|
30019
|
-
* Obtiene
|
|
30038
|
+
* Obtiene notificaciones (one-time fetch sin listener).
|
|
30020
30039
|
* Útil para cargas iniciales sin necesidad de updates en tiempo real.
|
|
30040
|
+
*
|
|
30041
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
30021
30042
|
*/
|
|
30022
|
-
async getAllOnce() {
|
|
30043
|
+
async getAllOnce(limit = DEFAULT_LIMIT) {
|
|
30023
30044
|
if (!this.collection)
|
|
30024
30045
|
return [];
|
|
30025
30046
|
return this.collection.getAllOnce({
|
|
30026
30047
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30048
|
+
limit,
|
|
30027
30049
|
});
|
|
30028
30050
|
}
|
|
30029
30051
|
/**
|
|
30030
|
-
* Obtiene solo notificaciones no leídas.
|
|
30052
|
+
* Obtiene solo notificaciones no leídas (real-time, filtrado server-side).
|
|
30053
|
+
*
|
|
30054
|
+
* @param limit - Máximo de notificaciones (default: 50)
|
|
30031
30055
|
*/
|
|
30032
|
-
getUnread() {
|
|
30033
|
-
|
|
30056
|
+
getUnread(limit = DEFAULT_LIMIT) {
|
|
30057
|
+
if (!this.collection)
|
|
30058
|
+
return of([]);
|
|
30059
|
+
return this.collection.watchAll({
|
|
30060
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30061
|
+
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30062
|
+
limit,
|
|
30063
|
+
});
|
|
30034
30064
|
}
|
|
30035
30065
|
/**
|
|
30036
|
-
* Cuenta notificaciones no leídas.
|
|
30066
|
+
* Cuenta notificaciones no leídas (real-time, filtrado server-side).
|
|
30037
30067
|
* Útil para badges en UI.
|
|
30038
30068
|
*/
|
|
30039
30069
|
getUnreadCount() {
|
|
30040
|
-
|
|
30070
|
+
if (!this.collection)
|
|
30071
|
+
return of(0);
|
|
30072
|
+
return this.collection
|
|
30073
|
+
.watchAll({
|
|
30074
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30075
|
+
})
|
|
30076
|
+
.pipe(map$1(n => n.length));
|
|
30041
30077
|
}
|
|
30042
30078
|
/**
|
|
30043
30079
|
* Obtiene una notificación por ID.
|
|
@@ -30059,7 +30095,7 @@ class NotificationsService {
|
|
|
30059
30095
|
await this.collection.update(notificationId, { isRead: true });
|
|
30060
30096
|
}
|
|
30061
30097
|
/**
|
|
30062
|
-
* Marca todas las notificaciones como leídas.
|
|
30098
|
+
* Marca todas las notificaciones no leídas como leídas (batch write).
|
|
30063
30099
|
*/
|
|
30064
30100
|
async markAllAsRead() {
|
|
30065
30101
|
if (!this.collection)
|
|
@@ -30067,7 +30103,8 @@ class NotificationsService {
|
|
|
30067
30103
|
const unread = await this.collection.query({
|
|
30068
30104
|
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30069
30105
|
});
|
|
30070
|
-
|
|
30106
|
+
const ids = unread.map(n => n.id).filter(Boolean);
|
|
30107
|
+
await this.collection.batchUpdate(ids, { isRead: true });
|
|
30071
30108
|
}
|
|
30072
30109
|
/**
|
|
30073
30110
|
* Elimina una notificación.
|
|
@@ -30078,13 +30115,14 @@ class NotificationsService {
|
|
|
30078
30115
|
await this.collection.delete(notificationId);
|
|
30079
30116
|
}
|
|
30080
30117
|
/**
|
|
30081
|
-
* Elimina todas las notificaciones del usuario.
|
|
30118
|
+
* Elimina todas las notificaciones del usuario (batch delete).
|
|
30082
30119
|
*/
|
|
30083
30120
|
async deleteAll() {
|
|
30084
30121
|
if (!this.collection)
|
|
30085
30122
|
return;
|
|
30086
30123
|
const all = await this.collection.getAll();
|
|
30087
|
-
|
|
30124
|
+
const ids = all.map(n => n.id).filter(Boolean);
|
|
30125
|
+
await this.collection.batchDelete(ids);
|
|
30088
30126
|
}
|
|
30089
30127
|
/**
|
|
30090
30128
|
* Limpia el estado del servicio.
|