valtech-components 2.0.695 → 2.0.711
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/version.mjs +2 -2
- package/fesm2022/valtech-components.mjs +107 -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":"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","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"]}
|
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.711';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC43MTEnO1xuIl19
|
|
@@ -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.711';
|
|
54
54
|
|
|
55
55
|
/**
|
|
56
56
|
* Servicio para gestionar presets de componentes.
|
|
@@ -9058,18 +9058,28 @@ class SelectSearchComponent {
|
|
|
9058
9058
|
return;
|
|
9059
9059
|
}
|
|
9060
9060
|
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] : [];
|
|
9061
|
+
if (this.props.mode === 'legacy') {
|
|
9062
|
+
// Permisivo: muestra el valor aunque no esté en las opciones
|
|
9063
|
+
if (controlValue !== null && controlValue !== undefined) {
|
|
9064
|
+
this.displayValue = controlValue.toString();
|
|
9065
|
+
this.selectedItems = [];
|
|
9066
|
+
return;
|
|
9067
|
+
}
|
|
9070
9068
|
}
|
|
9071
9069
|
else {
|
|
9072
|
-
|
|
9070
|
+
// Estricto: solo muestra si coincide con una opción
|
|
9071
|
+
if (controlValue === null || controlValue === undefined) {
|
|
9072
|
+
this.selectedItems = [];
|
|
9073
|
+
return;
|
|
9074
|
+
}
|
|
9075
|
+
if (this.props.options && this.props.options.length > 0) {
|
|
9076
|
+
const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));
|
|
9077
|
+
const selectedOption = map.get(controlValue);
|
|
9078
|
+
this.selectedItems = selectedOption ? [selectedOption] : [];
|
|
9079
|
+
}
|
|
9080
|
+
else {
|
|
9081
|
+
this.selectedItems = [];
|
|
9082
|
+
}
|
|
9073
9083
|
}
|
|
9074
9084
|
}
|
|
9075
9085
|
applyDefaultValue() {
|
|
@@ -9161,6 +9171,10 @@ class SelectSearchComponent {
|
|
|
9161
9171
|
return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);
|
|
9162
9172
|
}
|
|
9163
9173
|
updateDisplayValue() {
|
|
9174
|
+
if (this.props?.mode === 'legacy' && this.selectedItems.length === 0 && this.props?.control?.value) {
|
|
9175
|
+
this.displayValue = this.props.control.value.toString();
|
|
9176
|
+
return;
|
|
9177
|
+
}
|
|
9164
9178
|
if (this.selectedItems.length === 0) {
|
|
9165
9179
|
this.displayValue = '';
|
|
9166
9180
|
return;
|
|
@@ -22682,8 +22696,6 @@ class FormComponent {
|
|
|
22682
22696
|
this.onSelectChange = new EventEmitter();
|
|
22683
22697
|
this.types = InputType;
|
|
22684
22698
|
this.subscriptions = [];
|
|
22685
|
-
// Cache para evitar crear nuevos objetos en cada ciclo de change detection
|
|
22686
|
-
this.fieldPropsCache = new Map();
|
|
22687
22699
|
this.actionsCache = [];
|
|
22688
22700
|
}
|
|
22689
22701
|
ngOnInit() {
|
|
@@ -22711,15 +22723,14 @@ class FormComponent {
|
|
|
22711
22723
|
this.trackSelectChanges(field.name);
|
|
22712
22724
|
});
|
|
22713
22725
|
});
|
|
22714
|
-
|
|
22715
|
-
this.rebuildFieldPropsCache();
|
|
22726
|
+
this.syncFieldControlStates();
|
|
22716
22727
|
this.updateActionsCache();
|
|
22717
22728
|
this.previousState = this.props.state;
|
|
22718
22729
|
}
|
|
22719
22730
|
ngOnChanges(changes) {
|
|
22720
22731
|
// Cuando props cambia, reconstruir el cache
|
|
22721
22732
|
if (changes['props'] && this.form) {
|
|
22722
|
-
this.
|
|
22733
|
+
this.syncFieldControlStates();
|
|
22723
22734
|
this.updateActionsCache();
|
|
22724
22735
|
this.previousState = this.props.state;
|
|
22725
22736
|
}
|
|
@@ -22732,14 +22743,11 @@ class FormComponent {
|
|
|
22732
22743
|
}
|
|
22733
22744
|
}
|
|
22734
22745
|
/**
|
|
22735
|
-
*
|
|
22736
|
-
* Esto evita crear nuevos objetos en cada ciclo de change detection.
|
|
22746
|
+
* Synchronizes form control disabled/enabled state with field metadata.
|
|
22737
22747
|
*/
|
|
22738
|
-
|
|
22739
|
-
this.fieldPropsCache.clear();
|
|
22748
|
+
syncFieldControlStates() {
|
|
22740
22749
|
this.props.sections.forEach(section => {
|
|
22741
22750
|
section.fields.forEach(field => {
|
|
22742
|
-
const token = field.token || `input-${field.type}-${field.name}`;
|
|
22743
22751
|
if (field.type === this.types.NUMBER_FROM_TO) {
|
|
22744
22752
|
const fromControl = this.getControl(`${field.name}_from`);
|
|
22745
22753
|
const toControl = this.getControl(`${field.name}_to`);
|
|
@@ -22751,13 +22759,6 @@ class FormComponent {
|
|
|
22751
22759
|
fromControl.enable({ emitEvent: false });
|
|
22752
22760
|
toControl.enable({ emitEvent: false });
|
|
22753
22761
|
}
|
|
22754
|
-
this.fieldPropsCache.set(field.name, {
|
|
22755
|
-
...field,
|
|
22756
|
-
token,
|
|
22757
|
-
fromControl,
|
|
22758
|
-
toControl,
|
|
22759
|
-
control: undefined,
|
|
22760
|
-
});
|
|
22761
22762
|
}
|
|
22762
22763
|
else {
|
|
22763
22764
|
const control = this.getControl(field.name);
|
|
@@ -22767,11 +22768,6 @@ class FormComponent {
|
|
|
22767
22768
|
else {
|
|
22768
22769
|
control.enable({ emitEvent: false });
|
|
22769
22770
|
}
|
|
22770
|
-
this.fieldPropsCache.set(field.name, {
|
|
22771
|
-
...field,
|
|
22772
|
-
token,
|
|
22773
|
-
control,
|
|
22774
|
-
});
|
|
22775
22771
|
}
|
|
22776
22772
|
});
|
|
22777
22773
|
});
|
|
@@ -22810,50 +22806,16 @@ class FormComponent {
|
|
|
22810
22806
|
return this.Form.get(field);
|
|
22811
22807
|
}
|
|
22812
22808
|
getFieldProp(field) {
|
|
22813
|
-
|
|
22814
|
-
|
|
22815
|
-
if (cached) {
|
|
22816
|
-
return cached;
|
|
22809
|
+
if (!field.token) {
|
|
22810
|
+
field.token = `input-${field.type}-${field.name}`;
|
|
22817
22811
|
}
|
|
22818
|
-
// Fallback: generar y cachear si no existe (no debería ocurrir normalmente)
|
|
22819
|
-
const token = field.token || `input-${field.type}-${field.name}`;
|
|
22820
22812
|
if (field.type === this.types.NUMBER_FROM_TO) {
|
|
22821
22813
|
const fromControl = this.getControl(`${field.name}_from`);
|
|
22822
22814
|
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;
|
|
22815
|
+
return { ...field, fromControl, toControl, control: undefined };
|
|
22856
22816
|
}
|
|
22817
|
+
const control = this.getControl(field.name);
|
|
22818
|
+
return { ...field, control };
|
|
22857
22819
|
}
|
|
22858
22820
|
get isAtEndOfForm() {
|
|
22859
22821
|
return isAtEnd(this.elementRef);
|
|
@@ -28467,6 +28429,38 @@ class TypedCollection {
|
|
|
28467
28429
|
}
|
|
28468
28430
|
return this.firestore.deleteDoc(this.collectionPath, id);
|
|
28469
28431
|
}
|
|
28432
|
+
/**
|
|
28433
|
+
* Actualiza múltiples documentos en una sola operación atómica.
|
|
28434
|
+
*/
|
|
28435
|
+
async batchUpdate(ids, data) {
|
|
28436
|
+
if (ids.length === 0)
|
|
28437
|
+
return;
|
|
28438
|
+
const BATCH_LIMIT = 500; // Firestore batch limit
|
|
28439
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
28440
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
28441
|
+
await this.firestore.batch(batch => {
|
|
28442
|
+
for (const id of chunk) {
|
|
28443
|
+
batch.update(`${this.collectionPath}/${id}`, data);
|
|
28444
|
+
}
|
|
28445
|
+
});
|
|
28446
|
+
}
|
|
28447
|
+
}
|
|
28448
|
+
/**
|
|
28449
|
+
* Elimina múltiples documentos en una sola operación atómica.
|
|
28450
|
+
*/
|
|
28451
|
+
async batchDelete(ids) {
|
|
28452
|
+
if (ids.length === 0)
|
|
28453
|
+
return;
|
|
28454
|
+
const BATCH_LIMIT = 500;
|
|
28455
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
28456
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
28457
|
+
await this.firestore.batch(batch => {
|
|
28458
|
+
for (const id of chunk) {
|
|
28459
|
+
batch.delete(`${this.collectionPath}/${id}`);
|
|
28460
|
+
}
|
|
28461
|
+
});
|
|
28462
|
+
}
|
|
28463
|
+
}
|
|
28470
28464
|
/**
|
|
28471
28465
|
* Restaura un documento soft-deleted.
|
|
28472
28466
|
*/
|
|
@@ -29909,6 +29903,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
29909
29903
|
* Se auto-inicializa cuando AuthService tiene un usuario autenticado.
|
|
29910
29904
|
* También puede inicializarse manualmente con `initialize(userId)`.
|
|
29911
29905
|
*/
|
|
29906
|
+
/** Límite por defecto de notificaciones a cargar */
|
|
29907
|
+
const DEFAULT_LIMIT = 50;
|
|
29912
29908
|
/**
|
|
29913
29909
|
* Servicio para leer notificaciones desde Firestore.
|
|
29914
29910
|
*
|
|
@@ -29986,7 +29982,9 @@ class NotificationsService {
|
|
|
29986
29982
|
// Path relativo - FirestoreService agrega automáticamente apps/{appId}/
|
|
29987
29983
|
// NO agregar apps/ aquí para evitar doble prefijo
|
|
29988
29984
|
const basePath = `users/${userId}/notifications`;
|
|
29989
|
-
this.collection = this.collectionFactory.create(basePath, {
|
|
29985
|
+
this.collection = this.collectionFactory.create(basePath, {
|
|
29986
|
+
timestamps: true,
|
|
29987
|
+
});
|
|
29990
29988
|
console.log('[Notifications] Initialized with path:', basePath);
|
|
29991
29989
|
}
|
|
29992
29990
|
/**
|
|
@@ -30005,39 +30003,59 @@ class NotificationsService {
|
|
|
30005
30003
|
// LECTURA
|
|
30006
30004
|
// ===========================================================================
|
|
30007
30005
|
/**
|
|
30008
|
-
* Obtiene
|
|
30009
|
-
*
|
|
30006
|
+
* Obtiene notificaciones ordenadas por fecha descendente (real-time).
|
|
30007
|
+
* Se actualiza automáticamente cuando cambian los datos.
|
|
30008
|
+
*
|
|
30009
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
30010
30010
|
*/
|
|
30011
|
-
getAll() {
|
|
30011
|
+
getAll(limit = DEFAULT_LIMIT) {
|
|
30012
30012
|
if (!this.collection)
|
|
30013
30013
|
return of([]);
|
|
30014
30014
|
return this.collection.watchAll({
|
|
30015
30015
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30016
|
+
limit,
|
|
30016
30017
|
});
|
|
30017
30018
|
}
|
|
30018
30019
|
/**
|
|
30019
|
-
* Obtiene
|
|
30020
|
+
* Obtiene notificaciones (one-time fetch sin listener).
|
|
30020
30021
|
* Útil para cargas iniciales sin necesidad de updates en tiempo real.
|
|
30022
|
+
*
|
|
30023
|
+
* @param limit - Máximo de notificaciones a cargar (default: 50)
|
|
30021
30024
|
*/
|
|
30022
|
-
async getAllOnce() {
|
|
30025
|
+
async getAllOnce(limit = DEFAULT_LIMIT) {
|
|
30023
30026
|
if (!this.collection)
|
|
30024
30027
|
return [];
|
|
30025
30028
|
return this.collection.getAllOnce({
|
|
30026
30029
|
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30030
|
+
limit,
|
|
30027
30031
|
});
|
|
30028
30032
|
}
|
|
30029
30033
|
/**
|
|
30030
|
-
* Obtiene solo notificaciones no leídas.
|
|
30034
|
+
* Obtiene solo notificaciones no leídas (real-time, filtrado server-side).
|
|
30035
|
+
*
|
|
30036
|
+
* @param limit - Máximo de notificaciones (default: 50)
|
|
30031
30037
|
*/
|
|
30032
|
-
getUnread() {
|
|
30033
|
-
|
|
30038
|
+
getUnread(limit = DEFAULT_LIMIT) {
|
|
30039
|
+
if (!this.collection)
|
|
30040
|
+
return of([]);
|
|
30041
|
+
return this.collection.watchAll({
|
|
30042
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30043
|
+
orderBy: [{ field: 'createdAt', direction: 'desc' }],
|
|
30044
|
+
limit,
|
|
30045
|
+
});
|
|
30034
30046
|
}
|
|
30035
30047
|
/**
|
|
30036
|
-
* Cuenta notificaciones no leídas.
|
|
30048
|
+
* Cuenta notificaciones no leídas (real-time, filtrado server-side).
|
|
30037
30049
|
* Útil para badges en UI.
|
|
30038
30050
|
*/
|
|
30039
30051
|
getUnreadCount() {
|
|
30040
|
-
|
|
30052
|
+
if (!this.collection)
|
|
30053
|
+
return of(0);
|
|
30054
|
+
return this.collection
|
|
30055
|
+
.watchAll({
|
|
30056
|
+
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30057
|
+
})
|
|
30058
|
+
.pipe(map$1(n => n.length));
|
|
30041
30059
|
}
|
|
30042
30060
|
/**
|
|
30043
30061
|
* Obtiene una notificación por ID.
|
|
@@ -30059,7 +30077,7 @@ class NotificationsService {
|
|
|
30059
30077
|
await this.collection.update(notificationId, { isRead: true });
|
|
30060
30078
|
}
|
|
30061
30079
|
/**
|
|
30062
|
-
* Marca todas las notificaciones como leídas.
|
|
30080
|
+
* Marca todas las notificaciones no leídas como leídas (batch write).
|
|
30063
30081
|
*/
|
|
30064
30082
|
async markAllAsRead() {
|
|
30065
30083
|
if (!this.collection)
|
|
@@ -30067,7 +30085,8 @@ class NotificationsService {
|
|
|
30067
30085
|
const unread = await this.collection.query({
|
|
30068
30086
|
where: [{ field: 'isRead', operator: '==', value: false }],
|
|
30069
30087
|
});
|
|
30070
|
-
|
|
30088
|
+
const ids = unread.map(n => n.id).filter(Boolean);
|
|
30089
|
+
await this.collection.batchUpdate(ids, { isRead: true });
|
|
30071
30090
|
}
|
|
30072
30091
|
/**
|
|
30073
30092
|
* Elimina una notificación.
|
|
@@ -30078,13 +30097,14 @@ class NotificationsService {
|
|
|
30078
30097
|
await this.collection.delete(notificationId);
|
|
30079
30098
|
}
|
|
30080
30099
|
/**
|
|
30081
|
-
* Elimina todas las notificaciones del usuario.
|
|
30100
|
+
* Elimina todas las notificaciones del usuario (batch delete).
|
|
30082
30101
|
*/
|
|
30083
30102
|
async deleteAll() {
|
|
30084
30103
|
if (!this.collection)
|
|
30085
30104
|
return;
|
|
30086
30105
|
const all = await this.collection.getAll();
|
|
30087
|
-
|
|
30106
|
+
const ids = all.map(n => n.id).filter(Boolean);
|
|
30107
|
+
await this.collection.batchDelete(ids);
|
|
30088
30108
|
}
|
|
30089
30109
|
/**
|
|
30090
30110
|
* Limpia el estado del servicio.
|