valtech-components 2.0.449 → 2.0.450

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.
Files changed (27) hide show
  1. package/esm2022/public-api.mjs +3 -2
  2. package/fesm2022/valtech-components.mjs +4 -2732
  3. package/fesm2022/valtech-components.mjs.map +1 -1
  4. package/package.json +1 -1
  5. package/public-api.d.ts +0 -1
  6. package/esm2022/lib/services/firebase/config.mjs +0 -108
  7. package/esm2022/lib/services/firebase/firebase.service.mjs +0 -285
  8. package/esm2022/lib/services/firebase/firestore-collection.mjs +0 -254
  9. package/esm2022/lib/services/firebase/firestore.service.mjs +0 -508
  10. package/esm2022/lib/services/firebase/index.mjs +0 -49
  11. package/esm2022/lib/services/firebase/messaging.service.mjs +0 -503
  12. package/esm2022/lib/services/firebase/shared-config.mjs +0 -138
  13. package/esm2022/lib/services/firebase/storage.service.mjs +0 -421
  14. package/esm2022/lib/services/firebase/types.mjs +0 -8
  15. package/esm2022/lib/services/firebase/utils/path-builder.mjs +0 -195
  16. package/esm2022/lib/services/firebase/utils/query-builder.mjs +0 -302
  17. package/lib/services/firebase/config.d.ts +0 -49
  18. package/lib/services/firebase/firebase.service.d.ts +0 -140
  19. package/lib/services/firebase/firestore-collection.d.ts +0 -174
  20. package/lib/services/firebase/firestore.service.d.ts +0 -303
  21. package/lib/services/firebase/index.d.ts +0 -39
  22. package/lib/services/firebase/messaging.service.d.ts +0 -254
  23. package/lib/services/firebase/shared-config.d.ts +0 -126
  24. package/lib/services/firebase/storage.service.d.ts +0 -204
  25. package/lib/services/firebase/types.d.ts +0 -281
  26. package/lib/services/firebase/utils/path-builder.d.ts +0 -132
  27. package/lib/services/firebase/utils/query-builder.d.ts +0 -210
@@ -1,254 +0,0 @@
1
- /**
2
- * Firestore Collection Factory
3
- *
4
- * Patrón factory para crear instancias de colección tipadas.
5
- * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.
6
- */
7
- import { Injectable, inject } from '@angular/core';
8
- import { FirestoreService } from './firestore.service';
9
- import * as i0 from "@angular/core";
10
- /**
11
- * Factory para crear instancias de colección tipadas.
12
- *
13
- * @example
14
- * ```typescript
15
- * @Injectable({ providedIn: 'root' })
16
- * export class UsersService {
17
- * private users = inject(FirestoreCollectionFactory).create<User>('users');
18
- *
19
- * getAll = () => this.users.getAll();
20
- * getById = (id: string) => this.users.getById(id);
21
- * create = (data: Omit<User, 'id'>) => this.users.create(data);
22
- *
23
- * // Métodos personalizados
24
- * async getActiveUsers(): Promise<User[]> {
25
- * return this.users.query({
26
- * where: [{ field: 'active', operator: '==', value: true }]
27
- * });
28
- * }
29
- * }
30
- * ```
31
- */
32
- export class FirestoreCollectionFactory {
33
- constructor() {
34
- this.firestore = inject(FirestoreService);
35
- }
36
- /**
37
- * Crea una instancia de colección tipada.
38
- *
39
- * @param collectionPath - Ruta de la colección en Firestore
40
- * @param options - Opciones de configuración
41
- * @returns Instancia de TypedCollection
42
- */
43
- create(collectionPath, options) {
44
- return new TypedCollection(this.firestore, collectionPath, options);
45
- }
46
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
47
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, providedIn: 'root' }); }
48
- }
49
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, decorators: [{
50
- type: Injectable,
51
- args: [{ providedIn: 'root' }]
52
- }] });
53
- /**
54
- * Colección tipada con métodos CRUD.
55
- *
56
- * NO usa inject() - recibe FirestoreService por constructor.
57
- * Esto evita el error NG0203.
58
- */
59
- export class TypedCollection {
60
- constructor(firestore, collectionPath, options = {}) {
61
- this.firestore = firestore;
62
- this.collectionPath = collectionPath;
63
- this.options = {
64
- softDelete: false,
65
- timestamps: true,
66
- ...options,
67
- };
68
- }
69
- // ===========================================================================
70
- // LECTURAS ONE-TIME
71
- // ===========================================================================
72
- /**
73
- * Obtiene un documento por ID.
74
- */
75
- async getById(id) {
76
- return this.firestore.getDoc(this.collectionPath, id);
77
- }
78
- /**
79
- * Obtiene todos los documentos de la colección.
80
- */
81
- async getAll(options) {
82
- const queryOptions = this.applyDefaultFilters(options);
83
- return this.firestore.getDocs(this.collectionPath, queryOptions);
84
- }
85
- /**
86
- * Ejecuta una query personalizada.
87
- */
88
- async query(options) {
89
- const queryOptions = this.applyDefaultFilters(options);
90
- return this.firestore.getDocs(this.collectionPath, queryOptions);
91
- }
92
- /**
93
- * Obtiene documentos con paginación.
94
- */
95
- async paginate(options) {
96
- const queryOptions = this.applyDefaultFilters(options);
97
- return this.firestore.getPaginated(this.collectionPath, queryOptions);
98
- }
99
- /**
100
- * Obtiene el primer documento que coincida con la query.
101
- */
102
- async getFirst(options) {
103
- const queryOptions = this.applyDefaultFilters({
104
- ...options,
105
- limit: 1,
106
- });
107
- const results = await this.firestore.getDocs(this.collectionPath, queryOptions);
108
- return results[0] ?? null;
109
- }
110
- /**
111
- * Cuenta los documentos que coinciden con la query.
112
- * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.
113
- */
114
- async count(options) {
115
- const queryOptions = this.applyDefaultFilters(options);
116
- const results = await this.firestore.getDocs(this.collectionPath, queryOptions);
117
- return results.length;
118
- }
119
- /**
120
- * Verifica si un documento existe.
121
- */
122
- async exists(id) {
123
- return this.firestore.exists(this.collectionPath, id);
124
- }
125
- // ===========================================================================
126
- // SUBSCRIPCIONES REAL-TIME
127
- // ===========================================================================
128
- /**
129
- * Suscribe a cambios de un documento.
130
- */
131
- watch(id) {
132
- return this.firestore.docChanges(this.collectionPath, id);
133
- }
134
- /**
135
- * Suscribe a cambios de la colección.
136
- */
137
- watchAll(options) {
138
- const queryOptions = this.applyDefaultFilters(options);
139
- return this.firestore.collectionChanges(this.collectionPath, queryOptions);
140
- }
141
- /**
142
- * Suscribe a una query personalizada.
143
- */
144
- watchQuery(options) {
145
- const queryOptions = this.applyDefaultFilters(options);
146
- return this.firestore.collectionChanges(this.collectionPath, queryOptions);
147
- }
148
- // ===========================================================================
149
- // ESCRITURA
150
- // ===========================================================================
151
- /**
152
- * Crea un nuevo documento con ID auto-generado.
153
- */
154
- async create(data) {
155
- return this.firestore.addDoc(this.collectionPath, data);
156
- }
157
- /**
158
- * Crea un documento con ID específico.
159
- */
160
- async createWithId(id, data) {
161
- return this.firestore.setDoc(this.collectionPath, id, data);
162
- }
163
- /**
164
- * Actualiza campos de un documento.
165
- */
166
- async update(id, data) {
167
- return this.firestore.updateDoc(this.collectionPath, id, data);
168
- }
169
- /**
170
- * Elimina un documento.
171
- * Si softDelete está habilitado, marca como eliminado en lugar de borrar.
172
- */
173
- async delete(id) {
174
- if (this.options.softDelete) {
175
- return this.firestore.updateDoc(this.collectionPath, id, {
176
- deletedAt: new Date(),
177
- });
178
- }
179
- return this.firestore.deleteDoc(this.collectionPath, id);
180
- }
181
- /**
182
- * Restaura un documento soft-deleted.
183
- */
184
- async restore(id) {
185
- if (!this.options.softDelete) {
186
- throw new Error('Soft delete no está habilitado para esta colección');
187
- }
188
- return this.firestore.updateDoc(this.collectionPath, id, {
189
- deletedAt: null,
190
- });
191
- }
192
- // ===========================================================================
193
- // SUB-COLECCIONES
194
- // ===========================================================================
195
- /**
196
- * Obtiene una referencia a una sub-colección.
197
- *
198
- * @example
199
- * ```typescript
200
- * // En UsersService
201
- * getUserDocuments(userId: string) {
202
- * return this.users.subcollection<Document>(userId, 'documents');
203
- * }
204
- *
205
- * // Uso
206
- * const docs = await users.getUserDocuments('user123').getAll();
207
- * ```
208
- */
209
- subcollection(parentId, subcollectionName) {
210
- const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;
211
- return {
212
- getById: (id) => this.firestore.getDoc(subPath, id),
213
- getAll: (options) => this.firestore.getDocs(subPath, options),
214
- watch: (id) => this.firestore.docChanges(subPath, id),
215
- watchAll: (options) => this.firestore.collectionChanges(subPath, options),
216
- create: (data) => this.firestore.addDoc(subPath, data),
217
- update: (id, data) => this.firestore.updateDoc(subPath, id, data),
218
- delete: (id) => this.firestore.deleteDoc(subPath, id),
219
- };
220
- }
221
- // ===========================================================================
222
- // MÉTODOS PRIVADOS
223
- // ===========================================================================
224
- /**
225
- * Aplica filtros por defecto a las queries.
226
- */
227
- applyDefaultFilters(options) {
228
- if (!this.options.softDelete) {
229
- return options ?? {};
230
- }
231
- // Excluir documentos soft-deleted por defecto
232
- const whereClause = { field: 'deletedAt', operator: '==', value: null };
233
- return {
234
- ...options,
235
- where: [...(options?.where ?? []), whereClause],
236
- };
237
- }
238
- // ===========================================================================
239
- // UTILIDADES
240
- // ===========================================================================
241
- /**
242
- * Genera un nuevo ID sin crear el documento.
243
- */
244
- generateId() {
245
- return this.firestore.generateId(this.collectionPath);
246
- }
247
- /**
248
- * Obtiene la ruta de la colección.
249
- */
250
- getPath() {
251
- return this.collectionPath;
252
- }
253
- }
254
- //# 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, inject } 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  private firestore = inject(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   * 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"]}