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.
@@ -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, { timestamps: true });
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 todas las notificaciones ordenadas por fecha descendente.
113
- * Real-time: se actualiza automáticamente cuando cambian los datos.
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 todas las notificaciones (one-time fetch sin listener).
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
- return this.getAll().pipe(map(notifications => notifications.filter(n => !n.isRead)));
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
- return this.getUnread().pipe(map(n => n.length));
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
- await Promise.all(unread.map(n => this.collection.update(n.id, { isRead: true })));
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
- await Promise.all(all.map(n => this.collection.delete(n.id)));
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":"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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/** 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"]}
@@ -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.695';
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC42OTUnO1xuIl19
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.695';
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 (controlValue === null || controlValue === undefined) {
9062
- this.selectedItems = [];
9063
- return;
9064
- }
9065
- // PERF: Use a Map for faster lookup if options are large
9066
- if (this.props.options && this.props.options.length > 0) {
9067
- const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));
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
- this.selectedItems = [];
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
- // Construir cache inicial
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.rebuildFieldPropsCache();
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
- * Reconstruye el cache de props para cada field.
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
- rebuildFieldPropsCache() {
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
- // Retornar del cache para evitar crear nuevos objetos en cada ciclo de change detection
22814
- const cached = this.fieldPropsCache.get(field.name);
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
- if (field.state === ComponentStates.DISABLED) {
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, { timestamps: true });
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 todas las notificaciones ordenadas por fecha descendente.
30009
- * Real-time: se actualiza automáticamente cuando cambian los datos.
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 todas las notificaciones (one-time fetch sin listener).
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
- return this.getAll().pipe(map$1(notifications => notifications.filter(n => !n.isRead)));
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
- return this.getUnread().pipe(map$1(n => n.length));
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
- await Promise.all(unread.map(n => this.collection.update(n.id, { isRead: true })));
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
- await Promise.all(all.map(n => this.collection.delete(n.id)));
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.