valtech-components 2.0.426 → 2.0.428

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