valtech-components 2.0.694 → 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/checkbox-radio-input/checkbox-radio-input.component.mjs +3 -3
- 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 +109 -89
- 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
|
@@ -186,6 +186,38 @@ export class TypedCollection {
|
|
|
186
186
|
}
|
|
187
187
|
return this.firestore.deleteDoc(this.collectionPath, id);
|
|
188
188
|
}
|
|
189
|
+
/**
|
|
190
|
+
* Actualiza múltiples documentos en una sola operación atómica.
|
|
191
|
+
*/
|
|
192
|
+
async batchUpdate(ids, data) {
|
|
193
|
+
if (ids.length === 0)
|
|
194
|
+
return;
|
|
195
|
+
const BATCH_LIMIT = 500; // Firestore batch limit
|
|
196
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
197
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
198
|
+
await this.firestore.batch(batch => {
|
|
199
|
+
for (const id of chunk) {
|
|
200
|
+
batch.update(`${this.collectionPath}/${id}`, data);
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Elimina múltiples documentos en una sola operación atómica.
|
|
207
|
+
*/
|
|
208
|
+
async batchDelete(ids) {
|
|
209
|
+
if (ids.length === 0)
|
|
210
|
+
return;
|
|
211
|
+
const BATCH_LIMIT = 500;
|
|
212
|
+
for (let i = 0; i < ids.length; i += BATCH_LIMIT) {
|
|
213
|
+
const chunk = ids.slice(i, i + BATCH_LIMIT);
|
|
214
|
+
await this.firestore.batch(batch => {
|
|
215
|
+
for (const id of chunk) {
|
|
216
|
+
batch.delete(`${this.collectionPath}/${id}`);
|
|
217
|
+
}
|
|
218
|
+
});
|
|
219
|
+
}
|
|
220
|
+
}
|
|
189
221
|
/**
|
|
190
222
|
* Restaura un documento soft-deleted.
|
|
191
223
|
*/
|
|
@@ -259,4 +291,4 @@ export class TypedCollection {
|
|
|
259
291
|
return this.collectionPath;
|
|
260
292
|
}
|
|
261
293
|
}
|
|
262
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore-collection.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore-collection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firestore Collection Factory\n *\n * Patrón factory para crear instancias de colección tipadas.\n * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.\n */\n\nimport { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\n\nimport { FirestoreService } from './firestore.service';\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\n\n/**\n * Opciones de configuración para una colección.\n */\nexport interface CollectionOptions {\n  /**\n   * Si true, usa soft delete (marca deletedAt en lugar de eliminar).\n   * Default: false\n   */\n  softDelete?: boolean;\n\n  /**\n   * Si true, maneja automáticamente createdAt/updatedAt.\n   * Default: true\n   */\n  timestamps?: boolean;\n}\n\n/**\n * Referencia a una sub-colección tipada.\n */\nexport interface SubCollectionRef<T extends FirestoreDocument> {\n  getById(id: string): Promise<T | null>;\n  getAll(options?: QueryOptions): Promise<T[]>;\n  watch(id: string): Observable<T | null>;\n  watchAll(options?: QueryOptions): Observable<T[]>;\n  create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T>;\n  update(id: string, data: Partial<T>): Promise<void>;\n  delete(id: string): Promise<void>;\n}\n\n/**\n * Factory para crear instancias de colección tipadas.\n *\n * @example\n * ```typescript\n * @Injectable({ providedIn: 'root' })\n * export class UsersService {\n *   private users = inject(FirestoreCollectionFactory).create<User>('users');\n *\n *   getAll = () => this.users.getAll();\n *   getById = (id: string) => this.users.getById(id);\n *   create = (data: Omit<User, 'id'>) => this.users.create(data);\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.users.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreCollectionFactory {\n  constructor(private firestore: FirestoreService) {}\n\n  /**\n   * Crea una instancia de colección tipada.\n   *\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   * @returns Instancia de TypedCollection\n   */\n  create<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: CollectionOptions\n  ): TypedCollection<T> {\n    return new TypedCollection<T>(this.firestore, collectionPath, options);\n  }\n}\n\n/**\n * Colección tipada con métodos CRUD.\n *\n * NO usa inject() - recibe FirestoreService por constructor.\n * Esto evita el error NG0203.\n */\nexport class TypedCollection<T extends FirestoreDocument> {\n  private readonly options: CollectionOptions;\n\n  constructor(\n    private firestore: FirestoreService,\n    private collectionPath: string,\n    options: CollectionOptions = {}\n  ) {\n    this.options = {\n      softDelete: false,\n      timestamps: true,\n      ...options,\n    };\n  }\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID.\n   */\n  async getById(id: string): Promise<T | null> {\n    return this.firestore.getDoc<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Obtiene todos los documentos de la colección.\n   */\n  async getAll(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Ejecuta una query personalizada.\n   */\n  async query(options: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene documentos con paginación.\n   */\n  async paginate(options: QueryOptions & { limit: number }): Promise<PaginatedResult<T>> {\n    const queryOptions = this.applyDefaultFilters(options) as QueryOptions & { limit: number };\n    return this.firestore.getPaginated<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene el primer documento que coincida con la query.\n   */\n  async getFirst(options?: QueryOptions): Promise<T | null> {\n    const queryOptions = this.applyDefaultFilters({\n      ...options,\n      limit: 1,\n    });\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results[0] ?? null;\n  }\n\n  /**\n   * Cuenta los documentos que coinciden con la query.\n   * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.\n   */\n  async count(options?: QueryOptions): Promise<number> {\n    const queryOptions = this.applyDefaultFilters(options);\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results.length;\n  }\n\n  /**\n   * Obtiene todos los documentos (one-time fetch sin listener).\n   * A diferencia de watchAll(), no mantiene subscripciones activas.\n   */\n  async getAllOnce(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Verifica si un documento existe.\n   */\n  async exists(id: string): Promise<boolean> {\n    return this.firestore.exists(this.collectionPath, id);\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento.\n   */\n  watch(id: string): Observable<T | null> {\n    return this.firestore.docChanges<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Suscribe a cambios de la colección.\n   */\n  watchAll(options?: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Suscribe a una query personalizada.\n   */\n  watchQuery(options: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Crea un nuevo documento con ID auto-generado.\n   */\n  async create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T> {\n    return this.firestore.addDoc<T>(this.collectionPath, data);\n  }\n\n  /**\n   * Crea un documento con ID específico.\n   */\n  async createWithId(id: string, data: Omit<T, 'id'>): Promise<void> {\n    return this.firestore.setDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Actualiza campos de un documento.\n   */\n  async update(id: string, data: Partial<Omit<T, 'id' | 'createdAt'>>): Promise<void> {\n    return this.firestore.updateDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Elimina un documento.\n   * Si softDelete está habilitado, marca como eliminado en lugar de borrar.\n   */\n  async delete(id: string): Promise<void> {\n    if (this.options.softDelete) {\n      return this.firestore.updateDoc<T>(this.collectionPath, id, {\n        deletedAt: new Date(),\n      } as unknown as Partial<T>);\n    }\n    return this.firestore.deleteDoc(this.collectionPath, id);\n  }\n\n  /**\n   * Restaura un documento soft-deleted.\n   */\n  async restore(id: string): Promise<void> {\n    if (!this.options.softDelete) {\n      throw new Error('Soft delete no está habilitado para esta colección');\n    }\n    return this.firestore.updateDoc<T>(this.collectionPath, id, {\n      deletedAt: null,\n    } as unknown as Partial<T>);\n  }\n\n  // ===========================================================================\n  // SUB-COLECCIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene una referencia a una sub-colección.\n   *\n   * @example\n   * ```typescript\n   * // En UsersService\n   * getUserDocuments(userId: string) {\n   *   return this.users.subcollection<Document>(userId, 'documents');\n   * }\n   *\n   * // Uso\n   * const docs = await users.getUserDocuments('user123').getAll();\n   * ```\n   */\n  subcollection<S extends FirestoreDocument>(\n    parentId: string,\n    subcollectionName: string\n  ): SubCollectionRef<S> {\n    const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;\n\n    return {\n      getById: (id: string) => this.firestore.getDoc<S>(subPath, id),\n      getAll: (options?: QueryOptions) => this.firestore.getDocs<S>(subPath, options),\n      watch: (id: string) => this.firestore.docChanges<S>(subPath, id),\n      watchAll: (options?: QueryOptions) => this.firestore.collectionChanges<S>(subPath, options),\n      create: (data: Omit<S, 'id' | 'createdAt' | 'updatedAt'>) =>\n        this.firestore.addDoc<S>(subPath, data),\n      update: (id: string, data: Partial<S>) => this.firestore.updateDoc<S>(subPath, id, data),\n      delete: (id: string) => this.firestore.deleteDoc(subPath, id),\n    };\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   */\n  private applyDefaultFilters(options?: QueryOptions): QueryOptions {\n    if (!this.options.softDelete) {\n      return options ?? {};\n    }\n\n    // Excluir documentos soft-deleted por defecto\n    const whereClause = { field: 'deletedAt', operator: '==' as const, value: null };\n\n    return {\n      ...options,\n      where: [...(options?.where ?? []), whereClause],\n    };\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Genera un nuevo ID sin crear el documento.\n   */\n  generateId(): string {\n    return this.firestore.generateId(this.collectionPath);\n  }\n\n  /**\n   * Obtiene la ruta de la colección.\n   */\n  getPath(): string {\n    return this.collectionPath;\n  }\n}\n\n/**\n * @deprecated Use FirestoreCollectionFactory.create() instead.\n * Type alias for backwards compatibility.\n */\nexport type FirestoreCollection<T extends FirestoreDocument> = TypedCollection<T>;\n"]}
|
|
294
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore-collection.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore-collection.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;;AAoC3C;;;;;;;;;;;;;;;;;;;;;GAqBG;AAEH,MAAM,OAAO,0BAA0B;IACrC,YAAoB,SAA2B;QAA3B,cAAS,GAAT,SAAS,CAAkB;IAAG,CAAC;IAEnD;;;;;;OAMG;IACH,MAAM,CACJ,cAAsB,EACtB,OAA2B;QAE3B,OAAO,IAAI,eAAe,CAAI,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,OAAO,CAAC,CAAC;IACzE,CAAC;+GAfU,0BAA0B;mHAA1B,0BAA0B,cADb,MAAM;;4FACnB,0BAA0B;kBADtC,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;AAmBlC;;;;;GAKG;AACH,MAAM,OAAO,eAAe;IAG1B,YACU,SAA2B,EAC3B,cAAsB,EAC9B,UAA6B,EAAE;QAFvB,cAAS,GAAT,SAAS,CAAkB;QAC3B,mBAAc,GAAd,cAAc,CAAQ;QAG9B,IAAI,CAAC,OAAO,GAAG;YACb,UAAU,EAAE,KAAK;YACjB,UAAU,EAAE,IAAI;YAChB,GAAG,OAAO;SACX,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,oBAAoB;IACpB,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,EAAU;QACtB,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAAsB;QACjC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAK,CAAC,OAAqB;QAC/B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,OAAyC;QACtD,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAqC,CAAC;QAC3F,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAC3E,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,OAAsB;QACnC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC;YAC5C,GAAG,OAAO;YACV,KAAK,EAAE,CAAC;SACT,CAAC,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;QACnF,OAAO,OAAO,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,KAAK,CAAC,OAAsB;QAChC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;QACnF,OAAO,OAAO,CAAC,MAAM,CAAC;IACxB,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,UAAU,CAAC,OAAsB;QACrC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU;QACrB,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IACxD,CAAC;IAED,8EAA8E;IAC9E,2BAA2B;IAC3B,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,EAAU;QACd,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,OAAsB;QAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAChF,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,OAAqB;QAC9B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAChF,CAAC;IAED,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,IAA+C;QAC1D,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,CAAC;IAC7D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,EAAU,EAAE,IAAmB;QAChD,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU,EAAE,IAA0C;QACjE,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;IACpE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU;QACrB,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC5B,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE;gBAC1D,SAAS,EAAE,IAAI,IAAI,EAAE;aACG,CAAC,CAAC;QAC9B,CAAC;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,GAAa,EAAE,IAA0C;QACzE,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO;QAE7B,MAAM,WAAW,GAAG,GAAG,CAAC,CAAC,wBAAwB;QACjD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,IAAI,WAAW,EAAE,CAAC;YACjD,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,WAAW,CAAC,CAAC;YAC5C,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE;gBACjC,KAAK,MAAM,EAAE,IAAI,KAAK,EAAE,CAAC;oBACvB,KAAK,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,cAAc,IAAI,EAAE,EAAE,EAAE,IAA+B,CAAC,CAAC;gBAChF,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,GAAa;QAC7B,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO;QAE7B,MAAM,WAAW,GAAG,GAAG,CAAC;QACxB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,IAAI,WAAW,EAAE,CAAC;YACjD,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,WAAW,CAAC,CAAC;YAC5C,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE;gBACjC,KAAK,MAAM,EAAE,IAAI,KAAK,EAAE,CAAC;oBACvB,KAAK,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,cAAc,IAAI,EAAE,EAAE,CAAC,CAAC;gBAC/C,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,EAAU;QACtB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;QACxE,CAAC;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE;YAC1D,SAAS,EAAE,IAAI;SACS,CAAC,CAAC;IAC9B,CAAC;IAED,8EAA8E;IAC9E,kBAAkB;IAClB,8EAA8E;IAE9E;;;;;;;;;;;;;OAaG;IACH,aAAa,CACX,QAAgB,EAChB,iBAAyB;QAEzB,MAAM,OAAO,GAAG,GAAG,IAAI,CAAC,cAAc,IAAI,QAAQ,IAAI,iBAAiB,EAAE,CAAC;QAE1E,OAAO;YACL,OAAO,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,OAAO,EAAE,EAAE,CAAC;YAC9D,MAAM,EAAE,CAAC,OAAsB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,OAAO,EAAE,OAAO,CAAC;YAC/E,KAAK,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAI,OAAO,EAAE,EAAE,CAAC;YAChE,QAAQ,EAAE,CAAC,OAAsB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,OAAO,EAAE,OAAO,CAAC;YAC3F,MAAM,EAAE,CAAC,IAA+C,EAAE,EAAE,CAC1D,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,OAAO,EAAE,IAAI,CAAC;YACzC,MAAM,EAAE,CAAC,EAAU,EAAE,IAAgB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,OAAO,EAAE,EAAE,EAAE,IAAI,CAAC;YACxF,MAAM,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,OAAO,EAAE,EAAE,CAAC;SAC9D,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,mBAAmB;IACnB,8EAA8E;IAE9E;;OAEG;IACK,mBAAmB,CAAC,OAAsB;QAChD,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC7B,OAAO,OAAO,IAAI,EAAE,CAAC;QACvB,CAAC;QAED,8CAA8C;QAC9C,MAAM,WAAW,GAAG,EAAE,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,IAAa,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QAEjF,OAAO;YACL,GAAG,OAAO;YACV,KAAK,EAAE,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE,CAAC,EAAE,WAAW,CAAC;SAChD,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,aAAa;IACb,8EAA8E;IAE9E;;OAEG;IACH,UAAU;QACR,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACxD,CAAC;IAED;;OAEG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;CACF","sourcesContent":["/**\n * Firestore Collection Factory\n *\n * Patrón factory para crear instancias de colección tipadas.\n * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.\n */\n\nimport { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\n\nimport { FirestoreService } from './firestore.service';\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\n\n/**\n * Opciones de configuración para una colección.\n */\nexport interface CollectionOptions {\n  /**\n   * Si true, usa soft delete (marca deletedAt en lugar de eliminar).\n   * Default: false\n   */\n  softDelete?: boolean;\n\n  /**\n   * Si true, maneja automáticamente createdAt/updatedAt.\n   * Default: true\n   */\n  timestamps?: boolean;\n}\n\n/**\n * Referencia a una sub-colección tipada.\n */\nexport interface SubCollectionRef<T extends FirestoreDocument> {\n  getById(id: string): Promise<T | null>;\n  getAll(options?: QueryOptions): Promise<T[]>;\n  watch(id: string): Observable<T | null>;\n  watchAll(options?: QueryOptions): Observable<T[]>;\n  create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T>;\n  update(id: string, data: Partial<T>): Promise<void>;\n  delete(id: string): Promise<void>;\n}\n\n/**\n * Factory para crear instancias de colección tipadas.\n *\n * @example\n * ```typescript\n * @Injectable({ providedIn: 'root' })\n * export class UsersService {\n *   private users = inject(FirestoreCollectionFactory).create<User>('users');\n *\n *   getAll = () => this.users.getAll();\n *   getById = (id: string) => this.users.getById(id);\n *   create = (data: Omit<User, 'id'>) => this.users.create(data);\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.users.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreCollectionFactory {\n  constructor(private firestore: FirestoreService) {}\n\n  /**\n   * Crea una instancia de colección tipada.\n   *\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   * @returns Instancia de TypedCollection\n   */\n  create<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: CollectionOptions\n  ): TypedCollection<T> {\n    return new TypedCollection<T>(this.firestore, collectionPath, options);\n  }\n}\n\n/**\n * Colección tipada con métodos CRUD.\n *\n * NO usa inject() - recibe FirestoreService por constructor.\n * Esto evita el error NG0203.\n */\nexport class TypedCollection<T extends FirestoreDocument> {\n  private readonly options: CollectionOptions;\n\n  constructor(\n    private firestore: FirestoreService,\n    private collectionPath: string,\n    options: CollectionOptions = {}\n  ) {\n    this.options = {\n      softDelete: false,\n      timestamps: true,\n      ...options,\n    };\n  }\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID.\n   */\n  async getById(id: string): Promise<T | null> {\n    return this.firestore.getDoc<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Obtiene todos los documentos de la colección.\n   */\n  async getAll(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Ejecuta una query personalizada.\n   */\n  async query(options: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene documentos con paginación.\n   */\n  async paginate(options: QueryOptions & { limit: number }): Promise<PaginatedResult<T>> {\n    const queryOptions = this.applyDefaultFilters(options) as QueryOptions & { limit: number };\n    return this.firestore.getPaginated<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene el primer documento que coincida con la query.\n   */\n  async getFirst(options?: QueryOptions): Promise<T | null> {\n    const queryOptions = this.applyDefaultFilters({\n      ...options,\n      limit: 1,\n    });\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results[0] ?? null;\n  }\n\n  /**\n   * Cuenta los documentos que coinciden con la query.\n   * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.\n   */\n  async count(options?: QueryOptions): Promise<number> {\n    const queryOptions = this.applyDefaultFilters(options);\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results.length;\n  }\n\n  /**\n   * Obtiene todos los documentos (one-time fetch sin listener).\n   * A diferencia de watchAll(), no mantiene subscripciones activas.\n   */\n  async getAllOnce(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Verifica si un documento existe.\n   */\n  async exists(id: string): Promise<boolean> {\n    return this.firestore.exists(this.collectionPath, id);\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento.\n   */\n  watch(id: string): Observable<T | null> {\n    return this.firestore.docChanges<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Suscribe a cambios de la colección.\n   */\n  watchAll(options?: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Suscribe a una query personalizada.\n   */\n  watchQuery(options: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Crea un nuevo documento con ID auto-generado.\n   */\n  async create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T> {\n    return this.firestore.addDoc<T>(this.collectionPath, data);\n  }\n\n  /**\n   * Crea un documento con ID específico.\n   */\n  async createWithId(id: string, data: Omit<T, 'id'>): Promise<void> {\n    return this.firestore.setDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Actualiza campos de un documento.\n   */\n  async update(id: string, data: Partial<Omit<T, 'id' | 'createdAt'>>): Promise<void> {\n    return this.firestore.updateDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Elimina un documento.\n   * Si softDelete está habilitado, marca como eliminado en lugar de borrar.\n   */\n  async delete(id: string): Promise<void> {\n    if (this.options.softDelete) {\n      return this.firestore.updateDoc<T>(this.collectionPath, id, {\n        deletedAt: new Date(),\n      } as unknown as Partial<T>);\n    }\n    return this.firestore.deleteDoc(this.collectionPath, id);\n  }\n\n  /**\n   * Actualiza múltiples documentos en una sola operación atómica.\n   */\n  async batchUpdate(ids: string[], data: Partial<Omit<T, 'id' | 'createdAt'>>): Promise<void> {\n    if (ids.length === 0) return;\n\n    const BATCH_LIMIT = 500; // Firestore batch limit\n    for (let i = 0; i < ids.length; i += BATCH_LIMIT) {\n      const chunk = ids.slice(i, i + BATCH_LIMIT);\n      await this.firestore.batch(batch => {\n        for (const id of chunk) {\n          batch.update(`${this.collectionPath}/${id}`, data as Record<string, unknown>);\n        }\n      });\n    }\n  }\n\n  /**\n   * Elimina múltiples documentos en una sola operación atómica.\n   */\n  async batchDelete(ids: string[]): Promise<void> {\n    if (ids.length === 0) return;\n\n    const BATCH_LIMIT = 500;\n    for (let i = 0; i < ids.length; i += BATCH_LIMIT) {\n      const chunk = ids.slice(i, i + BATCH_LIMIT);\n      await this.firestore.batch(batch => {\n        for (const id of chunk) {\n          batch.delete(`${this.collectionPath}/${id}`);\n        }\n      });\n    }\n  }\n\n  /**\n   * Restaura un documento soft-deleted.\n   */\n  async restore(id: string): Promise<void> {\n    if (!this.options.softDelete) {\n      throw new Error('Soft delete no está habilitado para esta colección');\n    }\n    return this.firestore.updateDoc<T>(this.collectionPath, id, {\n      deletedAt: null,\n    } as unknown as Partial<T>);\n  }\n\n  // ===========================================================================\n  // SUB-COLECCIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene una referencia a una sub-colección.\n   *\n   * @example\n   * ```typescript\n   * // En UsersService\n   * getUserDocuments(userId: string) {\n   *   return this.users.subcollection<Document>(userId, 'documents');\n   * }\n   *\n   * // Uso\n   * const docs = await users.getUserDocuments('user123').getAll();\n   * ```\n   */\n  subcollection<S extends FirestoreDocument>(\n    parentId: string,\n    subcollectionName: string\n  ): SubCollectionRef<S> {\n    const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;\n\n    return {\n      getById: (id: string) => this.firestore.getDoc<S>(subPath, id),\n      getAll: (options?: QueryOptions) => this.firestore.getDocs<S>(subPath, options),\n      watch: (id: string) => this.firestore.docChanges<S>(subPath, id),\n      watchAll: (options?: QueryOptions) => this.firestore.collectionChanges<S>(subPath, options),\n      create: (data: Omit<S, 'id' | 'createdAt' | 'updatedAt'>) =>\n        this.firestore.addDoc<S>(subPath, data),\n      update: (id: string, data: Partial<S>) => this.firestore.updateDoc<S>(subPath, id, data),\n      delete: (id: string) => this.firestore.deleteDoc(subPath, id),\n    };\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   */\n  private applyDefaultFilters(options?: QueryOptions): QueryOptions {\n    if (!this.options.softDelete) {\n      return options ?? {};\n    }\n\n    // Excluir documentos soft-deleted por defecto\n    const whereClause = { field: 'deletedAt', operator: '==' as const, value: null };\n\n    return {\n      ...options,\n      where: [...(options?.where ?? []), whereClause],\n    };\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Genera un nuevo ID sin crear el documento.\n   */\n  generateId(): string {\n    return this.firestore.generateId(this.collectionPath);\n  }\n\n  /**\n   * Obtiene la ruta de la colección.\n   */\n  getPath(): string {\n    return this.collectionPath;\n  }\n}\n\n/**\n * @deprecated Use FirestoreCollectionFactory.create() instead.\n * Type alias for backwards compatibility.\n */\nexport type FirestoreCollection<T extends FirestoreDocument> = TypedCollection<T>;\n"]}
|
|
@@ -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":"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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"]}
|
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
|