valtech-components 2.0.417 → 2.0.418

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.
@@ -1,508 +0,0 @@
1
- /**
2
- * Firestore Service
3
- *
4
- * Servicio genérico para operaciones CRUD en Firestore.
5
- * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.
6
- */
7
- import { inject, Injectable } from '@angular/core';
8
- import { addDoc, collection, collectionData, deleteDoc, doc, docData, Firestore, getDoc, getDocs, limit, orderBy, query, serverTimestamp, setDoc, startAfter, startAt, endBefore, endAt, Timestamp, updateDoc, where, writeBatch, increment, arrayUnion, arrayRemove, } from '@angular/fire/firestore';
9
- import { map } from 'rxjs';
10
- import { buildPath } from './utils/path-builder';
11
- import * as i0 from "@angular/core";
12
- /**
13
- * Servicio para operaciones CRUD en Firestore.
14
- *
15
- * @example
16
- * ```typescript
17
- * interface User extends FirestoreDocument {
18
- * name: string;
19
- * email: string;
20
- * role: 'admin' | 'user';
21
- * }
22
- *
23
- * @Component({...})
24
- * export class UsersComponent {
25
- * private firestore = inject(FirestoreService);
26
- *
27
- * // Lectura one-time
28
- * async loadUser(id: string) {
29
- * const user = await this.firestore.getDoc<User>('users', id);
30
- * }
31
- *
32
- * // Subscripción real-time
33
- * users$ = this.firestore.collectionChanges<User>('users', {
34
- * where: [{ field: 'role', operator: '==', value: 'admin' }],
35
- * orderBy: [{ field: 'name', direction: 'asc' }]
36
- * });
37
- *
38
- * // Crear documento
39
- * async createUser(data: Omit<User, 'id'>) {
40
- * const user = await this.firestore.addDoc<User>('users', data);
41
- * }
42
- * }
43
- * ```
44
- */
45
- export class FirestoreService {
46
- constructor() {
47
- this.firestore = inject(Firestore);
48
- }
49
- // ===========================================================================
50
- // LECTURAS ONE-TIME (Promise)
51
- // ===========================================================================
52
- /**
53
- * Obtiene un documento por ID (lectura única).
54
- *
55
- * @param collectionPath - Ruta de la colección
56
- * @param docId - ID del documento
57
- * @returns Documento o null si no existe
58
- *
59
- * @example
60
- * ```typescript
61
- * const user = await firestoreService.getDoc<User>('users', 'abc123');
62
- * if (user) {
63
- * console.log(user.name);
64
- * }
65
- * ```
66
- */
67
- async getDoc(collectionPath, docId) {
68
- const docRef = doc(this.firestore, collectionPath, docId);
69
- const snapshot = await getDoc(docRef);
70
- if (!snapshot.exists()) {
71
- return null;
72
- }
73
- return this.mapDocument(snapshot);
74
- }
75
- /**
76
- * Obtiene múltiples documentos con opciones de query.
77
- *
78
- * @param collectionPath - Ruta de la colección
79
- * @param options - Opciones de query (where, orderBy, limit)
80
- * @returns Array de documentos
81
- *
82
- * @example
83
- * ```typescript
84
- * // Todos los usuarios activos ordenados por nombre
85
- * const users = await firestoreService.getDocs<User>('users', {
86
- * where: [{ field: 'active', operator: '==', value: true }],
87
- * orderBy: [{ field: 'name', direction: 'asc' }],
88
- * limit: 50
89
- * });
90
- * ```
91
- */
92
- async getDocs(collectionPath, options) {
93
- const collectionRef = collection(this.firestore, collectionPath);
94
- const constraints = this.buildQueryConstraints(options);
95
- const q = query(collectionRef, ...constraints);
96
- const snapshot = await getDocs(q);
97
- return snapshot.docs.map((doc) => this.mapDocument(doc));
98
- }
99
- /**
100
- * Obtiene documentos con paginación basada en cursores.
101
- *
102
- * @param collectionPath - Ruta de la colección
103
- * @param options - Opciones de query (debe incluir limit)
104
- * @returns Resultado paginado con cursor para la siguiente página
105
- *
106
- * @example
107
- * ```typescript
108
- * // Primera página
109
- * const page1 = await firestoreService.getPaginated<User>('users', {
110
- * orderBy: [{ field: 'createdAt', direction: 'desc' }],
111
- * limit: 10
112
- * });
113
- *
114
- * // Siguiente página
115
- * if (page1.hasMore) {
116
- * const page2 = await firestoreService.getPaginated<User>('users', {
117
- * orderBy: [{ field: 'createdAt', direction: 'desc' }],
118
- * limit: 10,
119
- * startAfter: page1.lastDoc
120
- * });
121
- * }
122
- * ```
123
- */
124
- async getPaginated(collectionPath, options) {
125
- const collectionRef = collection(this.firestore, collectionPath);
126
- const constraints = this.buildQueryConstraints(options);
127
- // Pedir uno más para saber si hay más páginas
128
- const q = query(collectionRef, ...constraints, limit(options.limit + 1));
129
- const snapshot = await getDocs(q);
130
- const docs = snapshot.docs;
131
- const hasMore = docs.length > options.limit;
132
- // Si hay más, remover el documento extra
133
- const resultDocs = hasMore ? docs.slice(0, -1) : docs;
134
- const lastDoc = resultDocs.length > 0 ? resultDocs[resultDocs.length - 1] : null;
135
- return {
136
- data: resultDocs.map((doc) => this.mapDocument(doc)),
137
- hasMore,
138
- lastDoc,
139
- };
140
- }
141
- /**
142
- * Verifica si un documento existe.
143
- *
144
- * @param collectionPath - Ruta de la colección
145
- * @param docId - ID del documento
146
- * @returns true si el documento existe
147
- */
148
- async exists(collectionPath, docId) {
149
- const docRef = doc(this.firestore, collectionPath, docId);
150
- const snapshot = await getDoc(docRef);
151
- return snapshot.exists();
152
- }
153
- // ===========================================================================
154
- // SUBSCRIPCIONES REAL-TIME (Observable)
155
- // ===========================================================================
156
- /**
157
- * Suscribe a cambios de un documento (real-time).
158
- *
159
- * @param collectionPath - Ruta de la colección
160
- * @param docId - ID del documento
161
- * @returns Observable que emite cuando el documento cambia
162
- *
163
- * @example
164
- * ```typescript
165
- * // En el componente
166
- * user$ = this.firestoreService.docChanges<User>('users', this.userId);
167
- *
168
- * // En el template
169
- * @if (user$ | async; as user) {
170
- * <p>{{ user.name }}</p>
171
- * }
172
- * ```
173
- */
174
- docChanges(collectionPath, docId) {
175
- const docRef = doc(this.firestore, collectionPath, docId);
176
- return docData(docRef, { idField: 'id' }).pipe(map((data) => {
177
- if (!data)
178
- return null;
179
- return this.convertTimestamps(data);
180
- }));
181
- }
182
- /**
183
- * Suscribe a cambios de una colección (real-time).
184
- *
185
- * @param collectionPath - Ruta de la colección
186
- * @param options - Opciones de query
187
- * @returns Observable que emite cuando la colección cambia
188
- *
189
- * @example
190
- * ```typescript
191
- * // Usuarios activos en tiempo real
192
- * activeUsers$ = this.firestoreService.collectionChanges<User>('users', {
193
- * where: [{ field: 'status', operator: '==', value: 'online' }]
194
- * });
195
- * ```
196
- */
197
- collectionChanges(collectionPath, options) {
198
- const collectionRef = collection(this.firestore, collectionPath);
199
- const constraints = this.buildQueryConstraints(options);
200
- const q = query(collectionRef, ...constraints);
201
- return collectionData(q, { idField: 'id' }).pipe(map((docs) => docs.map((doc) => this.convertTimestamps(doc))));
202
- }
203
- // ===========================================================================
204
- // ESCRITURA
205
- // ===========================================================================
206
- /**
207
- * Agrega un documento con ID auto-generado.
208
- *
209
- * @param collectionPath - Ruta de la colección
210
- * @param data - Datos del documento (sin id, createdAt, updatedAt)
211
- * @returns Documento creado con su ID
212
- *
213
- * @example
214
- * ```typescript
215
- * const newUser = await firestoreService.addDoc<User>('users', {
216
- * name: 'John Doe',
217
- * email: 'john@example.com',
218
- * role: 'user'
219
- * });
220
- * console.log('Created user with ID:', newUser.id);
221
- * ```
222
- */
223
- async addDoc(collectionPath, data) {
224
- const collectionRef = collection(this.firestore, collectionPath);
225
- const timestamp = serverTimestamp();
226
- const docData = {
227
- ...data,
228
- createdAt: timestamp,
229
- updatedAt: timestamp,
230
- };
231
- const docRef = await addDoc(collectionRef, docData);
232
- // Obtener el documento creado para retornarlo con timestamps resueltos
233
- const snapshot = await getDoc(docRef);
234
- return this.mapDocument(snapshot);
235
- }
236
- /**
237
- * Crea o sobrescribe un documento con ID específico.
238
- *
239
- * @param collectionPath - Ruta de la colección
240
- * @param docId - ID del documento
241
- * @param data - Datos del documento
242
- * @param options - Opciones (merge: true para merge en lugar de sobrescribir)
243
- *
244
- * @example
245
- * ```typescript
246
- * // Sobrescribir completamente
247
- * await firestoreService.setDoc<User>('users', 'user123', userData);
248
- *
249
- * // Merge con datos existentes
250
- * await firestoreService.setDoc<User>('users', 'user123', { name: 'New Name' }, { merge: true });
251
- * ```
252
- */
253
- async setDoc(collectionPath, docId, data, options) {
254
- const docRef = doc(this.firestore, collectionPath, docId);
255
- const timestamp = serverTimestamp();
256
- const docData = {
257
- ...data,
258
- updatedAt: timestamp,
259
- ...(options?.merge ? {} : { createdAt: timestamp }),
260
- };
261
- await setDoc(docRef, docData, { merge: options?.merge ?? false });
262
- }
263
- /**
264
- * Actualiza campos específicos de un documento.
265
- *
266
- * @param collectionPath - Ruta de la colección
267
- * @param docId - ID del documento
268
- * @param data - Campos a actualizar
269
- *
270
- * @example
271
- * ```typescript
272
- * await firestoreService.updateDoc<User>('users', 'user123', {
273
- * name: 'Updated Name',
274
- * lastLogin: new Date()
275
- * });
276
- * ```
277
- */
278
- async updateDoc(collectionPath, docId, data) {
279
- const docRef = doc(this.firestore, collectionPath, docId);
280
- await updateDoc(docRef, {
281
- ...data,
282
- updatedAt: serverTimestamp(),
283
- });
284
- }
285
- /**
286
- * Elimina un documento.
287
- *
288
- * @param collectionPath - Ruta de la colección
289
- * @param docId - ID del documento
290
- *
291
- * @example
292
- * ```typescript
293
- * await firestoreService.deleteDoc('users', 'user123');
294
- * ```
295
- */
296
- async deleteDoc(collectionPath, docId) {
297
- const docRef = doc(this.firestore, collectionPath, docId);
298
- await deleteDoc(docRef);
299
- }
300
- // ===========================================================================
301
- // OPERACIONES EN LOTE
302
- // ===========================================================================
303
- /**
304
- * Ejecuta múltiples operaciones de escritura de forma atómica.
305
- *
306
- * @param operations - Función que recibe el batch y agrega operaciones
307
- *
308
- * @example
309
- * ```typescript
310
- * await firestoreService.batch((batch) => {
311
- * batch.set('users/user1', { name: 'User 1' });
312
- * batch.update('users/user2', { status: 'inactive' });
313
- * batch.delete('users/user3');
314
- * });
315
- * ```
316
- */
317
- async batch(operations) {
318
- const batch = writeBatch(this.firestore);
319
- const batchApi = {
320
- set: (path, data) => {
321
- const [collectionPath, docId] = this.splitPath(path);
322
- const docRef = doc(this.firestore, collectionPath, docId);
323
- batch.set(docRef, {
324
- ...data,
325
- createdAt: serverTimestamp(),
326
- updatedAt: serverTimestamp(),
327
- });
328
- },
329
- update: (path, data) => {
330
- const [collectionPath, docId] = this.splitPath(path);
331
- const docRef = doc(this.firestore, collectionPath, docId);
332
- batch.update(docRef, {
333
- ...data,
334
- updatedAt: serverTimestamp(),
335
- });
336
- },
337
- delete: (path) => {
338
- const [collectionPath, docId] = this.splitPath(path);
339
- const docRef = doc(this.firestore, collectionPath, docId);
340
- batch.delete(docRef);
341
- },
342
- };
343
- operations(batchApi);
344
- await batch.commit();
345
- }
346
- // ===========================================================================
347
- // UTILIDADES
348
- // ===========================================================================
349
- /**
350
- * Construye una ruta a partir de un template.
351
- *
352
- * @param template - Template con placeholders {param}
353
- * @param params - Valores para los placeholders
354
- * @returns Ruta construida
355
- *
356
- * @example
357
- * ```typescript
358
- * const path = firestoreService.buildPath('users/{userId}/documents/{docId}', {
359
- * userId: 'user123',
360
- * docId: 'doc456'
361
- * });
362
- * // => 'users/user123/documents/doc456'
363
- * ```
364
- */
365
- buildPath(template, params) {
366
- return buildPath(template, params);
367
- }
368
- /**
369
- * Genera un ID único para un documento (sin crearlo).
370
- *
371
- * @param collectionPath - Ruta de la colección
372
- * @returns ID único generado por Firestore
373
- */
374
- generateId(collectionPath) {
375
- const collectionRef = collection(this.firestore, collectionPath);
376
- return doc(collectionRef).id;
377
- }
378
- /**
379
- * Retorna un valor de timestamp del servidor.
380
- * Usar en campos de fecha para que Firestore asigne el timestamp.
381
- */
382
- serverTimestamp() {
383
- return serverTimestamp();
384
- }
385
- /**
386
- * Retorna un valor para agregar elementos a un array.
387
- *
388
- * @example
389
- * ```typescript
390
- * await firestoreService.updateDoc('users', 'user123', {
391
- * tags: firestoreService.arrayUnion('new-tag')
392
- * });
393
- * ```
394
- */
395
- arrayUnion(...elements) {
396
- return arrayUnion(...elements);
397
- }
398
- /**
399
- * Retorna un valor para remover elementos de un array.
400
- */
401
- arrayRemove(...elements) {
402
- return arrayRemove(...elements);
403
- }
404
- /**
405
- * Retorna un valor para incrementar un campo numérico.
406
- *
407
- * @example
408
- * ```typescript
409
- * await firestoreService.updateDoc('users', 'user123', {
410
- * loginCount: firestoreService.increment(1)
411
- * });
412
- * ```
413
- */
414
- increment(n) {
415
- return increment(n);
416
- }
417
- // ===========================================================================
418
- // MÉTODOS PRIVADOS
419
- // ===========================================================================
420
- /**
421
- * Construye los QueryConstraints a partir de QueryOptions
422
- */
423
- buildQueryConstraints(options) {
424
- const constraints = [];
425
- if (!options)
426
- return constraints;
427
- // Where clauses
428
- if (options.where) {
429
- for (const clause of options.where) {
430
- constraints.push(where(clause.field, clause.operator, clause.value));
431
- }
432
- }
433
- // OrderBy clauses
434
- if (options.orderBy) {
435
- for (const clause of options.orderBy) {
436
- constraints.push(orderBy(clause.field, clause.direction));
437
- }
438
- }
439
- // Cursors para paginación
440
- if (options.startAfter) {
441
- constraints.push(startAfter(options.startAfter));
442
- }
443
- if (options.startAt) {
444
- constraints.push(startAt(options.startAt));
445
- }
446
- if (options.endBefore) {
447
- constraints.push(endBefore(options.endBefore));
448
- }
449
- if (options.endAt) {
450
- constraints.push(endAt(options.endAt));
451
- }
452
- // Limit (se agrega al final)
453
- if (options.limit) {
454
- constraints.push(limit(options.limit));
455
- }
456
- return constraints;
457
- }
458
- /**
459
- * Mapea un DocumentSnapshot a nuestro tipo
460
- */
461
- mapDocument(snapshot) {
462
- const data = snapshot.data();
463
- if (!data) {
464
- throw new Error('Documento no tiene datos');
465
- }
466
- return {
467
- id: snapshot.id,
468
- ...this.convertTimestamps(data),
469
- };
470
- }
471
- /**
472
- * Convierte Timestamps de Firestore a Date de JavaScript
473
- */
474
- convertTimestamps(data) {
475
- const result = {};
476
- for (const [key, value] of Object.entries(data)) {
477
- if (value instanceof Timestamp) {
478
- result[key] = value.toDate();
479
- }
480
- else if (value && typeof value === 'object' && !Array.isArray(value)) {
481
- result[key] = this.convertTimestamps(value);
482
- }
483
- else {
484
- result[key] = value;
485
- }
486
- }
487
- return result;
488
- }
489
- /**
490
- * Divide una ruta de documento en colección e ID
491
- */
492
- splitPath(path) {
493
- const segments = path.split('/');
494
- if (segments.length < 2 || segments.length % 2 !== 0) {
495
- throw new Error(`Ruta de documento inválida: ${path}`);
496
- }
497
- const docId = segments.pop();
498
- const collectionPath = segments.join('/');
499
- return [collectionPath, docId];
500
- }
501
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
502
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, providedIn: 'root' }); }
503
- }
504
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreService, decorators: [{
505
- type: Injectable,
506
- args: [{ providedIn: 'root' }]
507
- }] });
508
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firestore Service\n *\n * Servicio genérico para operaciones CRUD en Firestore.\n * Soporta lecturas one-time, subscripciones real-time, paginación y queries complejas.\n */\n\nimport { inject, Injectable } from '@angular/core';\nimport {\n  addDoc,\n  collection,\n  collectionData,\n  deleteDoc,\n  doc,\n  docData,\n  DocumentData,\n  DocumentReference,\n  DocumentSnapshot,\n  Firestore,\n  getDoc,\n  getDocs,\n  limit,\n  orderBy,\n  query,\n  QueryConstraint,\n  serverTimestamp,\n  setDoc,\n  startAfter,\n  startAt,\n  endBefore,\n  endAt,\n  Timestamp,\n  updateDoc,\n  where,\n  writeBatch,\n  increment,\n  arrayUnion,\n  arrayRemove,\n  FieldValue,\n} from '@angular/fire/firestore';\nimport { map, Observable } from 'rxjs';\n\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\nimport { buildPath } from './utils/path-builder';\n\n/**\n * Servicio para operaciones CRUD en Firestore.\n *\n * @example\n * ```typescript\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n *   role: 'admin' | 'user';\n * }\n *\n * @Component({...})\n * export class UsersComponent {\n *   private firestore = inject(FirestoreService);\n *\n *   // Lectura one-time\n *   async loadUser(id: string) {\n *     const user = await this.firestore.getDoc<User>('users', id);\n *   }\n *\n *   // Subscripción real-time\n *   users$ = this.firestore.collectionChanges<User>('users', {\n *     where: [{ field: 'role', operator: '==', value: 'admin' }],\n *     orderBy: [{ field: 'name', direction: 'asc' }]\n *   });\n *\n *   // Crear documento\n *   async createUser(data: Omit<User, 'id'>) {\n *     const user = await this.firestore.addDoc<User>('users', data);\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreService {\n  private firestore = inject(Firestore);\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME (Promise)\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID (lectura única).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Documento o null si no existe\n   *\n   * @example\n   * ```typescript\n   * const user = await firestoreService.getDoc<User>('users', 'abc123');\n   * if (user) {\n   *   console.log(user.name);\n   * }\n   * ```\n   */\n  async getDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Promise<T | null> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const snapshot = await getDoc(docRef);\n\n    if (!snapshot.exists()) {\n      return null;\n    }\n\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Obtiene múltiples documentos con opciones de query.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (where, orderBy, limit)\n   * @returns Array de documentos\n   *\n   * @example\n   * ```typescript\n   * // Todos los usuarios activos ordenados por nombre\n   * const users = await firestoreService.getDocs<User>('users', {\n   *   where: [{ field: 'active', operator: '==', value: true }],\n   *   orderBy: [{ field: 'name', direction: 'asc' }],\n   *   limit: 50\n   * });\n   * ```\n   */\n  async getDocs<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Promise<T[]> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n    const snapshot = await getDocs(q);\n\n    return snapshot.docs.map((doc) => this.mapDocument<T>(doc));\n  }\n\n  /**\n   * Obtiene documentos con paginación basada en cursores.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query (debe incluir limit)\n   * @returns Resultado paginado con cursor para la siguiente página\n   *\n   * @example\n   * ```typescript\n   * // Primera página\n   * const page1 = await firestoreService.getPaginated<User>('users', {\n   *   orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *   limit: 10\n   * });\n   *\n   * // Siguiente página\n   * if (page1.hasMore) {\n   *   const page2 = await firestoreService.getPaginated<User>('users', {\n   *     orderBy: [{ field: 'createdAt', direction: 'desc' }],\n   *     limit: 10,\n   *     startAfter: page1.lastDoc\n   *   });\n   * }\n   * ```\n   */\n  async getPaginated<T extends FirestoreDocument>(\n    collectionPath: string,\n    options: QueryOptions & { limit: number }\n  ): Promise<PaginatedResult<T>> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n\n    // Pedir uno más para saber si hay más páginas\n    const q = query(collectionRef, ...constraints, limit(options.limit + 1));\n    const snapshot = await getDocs(q);\n\n    const docs = snapshot.docs;\n    const hasMore = docs.length > options.limit;\n\n    // Si hay más, remover el documento extra\n    const resultDocs = hasMore ? docs.slice(0, -1) : docs;\n    const lastDoc = resultDocs.length > 0 ? resultDocs[resultDocs.length - 1] : null;\n\n    return {\n      data: resultDocs.map((doc) => this.mapDocument<T>(doc)),\n      hasMore,\n      lastDoc,\n    };\n  }\n\n  /**\n   * Verifica si un documento existe.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns true si el documento existe\n   */\n  async exists(collectionPath: string, docId: string): Promise<boolean> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const snapshot = await getDoc(docRef);\n    return snapshot.exists();\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME (Observable)\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @returns Observable que emite cuando el documento cambia\n   *\n   * @example\n   * ```typescript\n   * // En el componente\n   * user$ = this.firestoreService.docChanges<User>('users', this.userId);\n   *\n   * // En el template\n   * @if (user$ | async; as user) {\n   *   <p>{{ user.name }}</p>\n   * }\n   * ```\n   */\n  docChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string\n  ): Observable<T | null> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    return docData(docRef, { idField: 'id' }).pipe(\n      map((data) => {\n        if (!data) return null;\n        return this.convertTimestamps(data as DocumentData) as T;\n      })\n    );\n  }\n\n  /**\n   * Suscribe a cambios de una colección (real-time).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param options - Opciones de query\n   * @returns Observable que emite cuando la colección cambia\n   *\n   * @example\n   * ```typescript\n   * // Usuarios activos en tiempo real\n   * activeUsers$ = this.firestoreService.collectionChanges<User>('users', {\n   *   where: [{ field: 'status', operator: '==', value: 'online' }]\n   * });\n   * ```\n   */\n  collectionChanges<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: QueryOptions\n  ): Observable<T[]> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const constraints = this.buildQueryConstraints(options);\n    const q = query(collectionRef, ...constraints);\n\n    return collectionData(q, { idField: 'id' }).pipe(\n      map((docs) => docs.map((doc) => this.convertTimestamps(doc) as T))\n    );\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Agrega un documento con ID auto-generado.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param data - Datos del documento (sin id, createdAt, updatedAt)\n   * @returns Documento creado con su ID\n   *\n   * @example\n   * ```typescript\n   * const newUser = await firestoreService.addDoc<User>('users', {\n   *   name: 'John Doe',\n   *   email: 'john@example.com',\n   *   role: 'user'\n   * });\n   * console.log('Created user with ID:', newUser.id);\n   * ```\n   */\n  async addDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>\n  ): Promise<T> {\n    const collectionRef = collection(this.firestore, collectionPath);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      createdAt: timestamp,\n      updatedAt: timestamp,\n    };\n\n    const docRef = await addDoc(collectionRef, docData);\n\n    // Obtener el documento creado para retornarlo con timestamps resueltos\n    const snapshot = await getDoc(docRef);\n    return this.mapDocument<T>(snapshot);\n  }\n\n  /**\n   * Crea o sobrescribe un documento con ID específico.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Datos del documento\n   * @param options - Opciones (merge: true para merge en lugar de sobrescribir)\n   *\n   * @example\n   * ```typescript\n   * // Sobrescribir completamente\n   * await firestoreService.setDoc<User>('users', 'user123', userData);\n   *\n   * // Merge con datos existentes\n   * await firestoreService.setDoc<User>('users', 'user123', { name: 'New Name' }, { merge: true });\n   * ```\n   */\n  async setDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Omit<T, 'id'>,\n    options?: { merge?: boolean }\n  ): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    const timestamp = serverTimestamp();\n\n    const docData = {\n      ...data,\n      updatedAt: timestamp,\n      ...(options?.merge ? {} : { createdAt: timestamp }),\n    };\n\n    await setDoc(docRef, docData, { merge: options?.merge ?? false });\n  }\n\n  /**\n   * Actualiza campos específicos de un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   * @param data - Campos a actualizar\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc<User>('users', 'user123', {\n   *   name: 'Updated Name',\n   *   lastLogin: new Date()\n   * });\n   * ```\n   */\n  async updateDoc<T extends FirestoreDocument>(\n    collectionPath: string,\n    docId: string,\n    data: Partial<Omit<T, 'id' | 'createdAt'>>\n  ): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n\n    await updateDoc(docRef, {\n      ...data,\n      updatedAt: serverTimestamp(),\n    });\n  }\n\n  /**\n   * Elimina un documento.\n   *\n   * @param collectionPath - Ruta de la colección\n   * @param docId - ID del documento\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.deleteDoc('users', 'user123');\n   * ```\n   */\n  async deleteDoc(collectionPath: string, docId: string): Promise<void> {\n    const docRef = doc(this.firestore, collectionPath, docId);\n    await deleteDoc(docRef);\n  }\n\n  // ===========================================================================\n  // OPERACIONES EN LOTE\n  // ===========================================================================\n\n  /**\n   * Ejecuta múltiples operaciones de escritura de forma atómica.\n   *\n   * @param operations - Función que recibe el batch y agrega operaciones\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.batch((batch) => {\n   *   batch.set('users/user1', { name: 'User 1' });\n   *   batch.update('users/user2', { status: 'inactive' });\n   *   batch.delete('users/user3');\n   * });\n   * ```\n   */\n  async batch(\n    operations: (batch: {\n      set: <T>(path: string, data: T) => void;\n      update: <T>(path: string, data: Partial<T>) => void;\n      delete: (path: string) => void;\n    }) => void\n  ): Promise<void> {\n    const batch = writeBatch(this.firestore);\n\n    const batchApi = {\n      set: <T>(path: string, data: T) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.set(docRef, {\n          ...data,\n          createdAt: serverTimestamp(),\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      update: <T>(path: string, data: Partial<T>) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.update(docRef, {\n          ...data,\n          updatedAt: serverTimestamp(),\n        } as DocumentData);\n      },\n      delete: (path: string) => {\n        const [collectionPath, docId] = this.splitPath(path);\n        const docRef = doc(this.firestore, collectionPath, docId);\n        batch.delete(docRef);\n      },\n    };\n\n    operations(batchApi);\n    await batch.commit();\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Construye una ruta a partir de un template.\n   *\n   * @param template - Template con placeholders {param}\n   * @param params - Valores para los placeholders\n   * @returns Ruta construida\n   *\n   * @example\n   * ```typescript\n   * const path = firestoreService.buildPath('users/{userId}/documents/{docId}', {\n   *   userId: 'user123',\n   *   docId: 'doc456'\n   * });\n   * // => 'users/user123/documents/doc456'\n   * ```\n   */\n  buildPath(template: string, params: Record<string, string>): string {\n    return buildPath(template, params);\n  }\n\n  /**\n   * Genera un ID único para un documento (sin crearlo).\n   *\n   * @param collectionPath - Ruta de la colección\n   * @returns ID único generado por Firestore\n   */\n  generateId(collectionPath: string): string {\n    const collectionRef = collection(this.firestore, collectionPath);\n    return doc(collectionRef).id;\n  }\n\n  /**\n   * Retorna un valor de timestamp del servidor.\n   * Usar en campos de fecha para que Firestore asigne el timestamp.\n   */\n  serverTimestamp(): FieldValue {\n    return serverTimestamp();\n  }\n\n  /**\n   * Retorna un valor para agregar elementos a un array.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   tags: firestoreService.arrayUnion('new-tag')\n   * });\n   * ```\n   */\n  arrayUnion(...elements: unknown[]): FieldValue {\n    return arrayUnion(...elements);\n  }\n\n  /**\n   * Retorna un valor para remover elementos de un array.\n   */\n  arrayRemove(...elements: unknown[]): FieldValue {\n    return arrayRemove(...elements);\n  }\n\n  /**\n   * Retorna un valor para incrementar un campo numérico.\n   *\n   * @example\n   * ```typescript\n   * await firestoreService.updateDoc('users', 'user123', {\n   *   loginCount: firestoreService.increment(1)\n   * });\n   * ```\n   */\n  increment(n: number): FieldValue {\n    return increment(n);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Construye los QueryConstraints a partir de QueryOptions\n   */\n  private buildQueryConstraints(options?: QueryOptions): QueryConstraint[] {\n    const constraints: QueryConstraint[] = [];\n\n    if (!options) return constraints;\n\n    // Where clauses\n    if (options.where) {\n      for (const clause of options.where) {\n        constraints.push(where(clause.field, clause.operator, clause.value));\n      }\n    }\n\n    // OrderBy clauses\n    if (options.orderBy) {\n      for (const clause of options.orderBy) {\n        constraints.push(orderBy(clause.field, clause.direction));\n      }\n    }\n\n    // Cursors para paginación\n    if (options.startAfter) {\n      constraints.push(startAfter(options.startAfter));\n    }\n    if (options.startAt) {\n      constraints.push(startAt(options.startAt));\n    }\n    if (options.endBefore) {\n      constraints.push(endBefore(options.endBefore));\n    }\n    if (options.endAt) {\n      constraints.push(endAt(options.endAt));\n    }\n\n    // Limit (se agrega al final)\n    if (options.limit) {\n      constraints.push(limit(options.limit));\n    }\n\n    return constraints;\n  }\n\n  /**\n   * Mapea un DocumentSnapshot a nuestro tipo\n   */\n  private mapDocument<T extends FirestoreDocument>(\n    snapshot: DocumentSnapshot<DocumentData>\n  ): T {\n    const data = snapshot.data();\n    if (!data) {\n      throw new Error('Documento no tiene datos');\n    }\n\n    return {\n      id: snapshot.id,\n      ...this.convertTimestamps(data),\n    } as T;\n  }\n\n  /**\n   * Convierte Timestamps de Firestore a Date de JavaScript\n   */\n  private convertTimestamps(data: DocumentData): DocumentData {\n    const result: DocumentData = {};\n\n    for (const [key, value] of Object.entries(data)) {\n      if (value instanceof Timestamp) {\n        result[key] = value.toDate();\n      } else if (value && typeof value === 'object' && !Array.isArray(value)) {\n        result[key] = this.convertTimestamps(value);\n      } else {\n        result[key] = value;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Divide una ruta de documento en colección e ID\n   */\n  private splitPath(path: string): [string, string] {\n    const segments = path.split('/');\n    if (segments.length < 2 || segments.length % 2 !== 0) {\n      throw new Error(`Ruta de documento inválida: ${path}`);\n    }\n    const docId = segments.pop()!;\n    const collectionPath = segments.join('/');\n    return [collectionPath, docId];\n  }\n}\n"]}
@@ -1,46 +0,0 @@
1
- /**
2
- * Firebase Services
3
- *
4
- * Servicios reutilizables para integración con Firebase.
5
- *
6
- * @example
7
- * ```typescript
8
- * // En main.ts
9
- * import { provideValtechFirebase } from 'valtech-components';
10
- *
11
- * bootstrapApplication(AppComponent, {
12
- * providers: [
13
- * provideValtechFirebase({
14
- * firebase: environment.firebase,
15
- * persistence: true,
16
- * }),
17
- * ],
18
- * });
19
- *
20
- * // En componentes
21
- * import { FirebaseService, FirestoreService } from 'valtech-components';
22
- *
23
- * @Component({...})
24
- * export class MyComponent {
25
- * private firebase = inject(FirebaseService);
26
- * private firestore = inject(FirestoreService);
27
- * }
28
- * ```
29
- */
30
- // Tipos
31
- export * from './types';
32
- // Configuración
33
- export { VALTECH_FIREBASE_CONFIG, hasEmulators, provideValtechFirebase } from './config';
34
- // Servicios
35
- export { FirebaseService } from './firebase.service';
36
- // Firestore
37
- export { FirestoreService } from './firestore.service';
38
- export { FirestoreCollection } from './firestore-collection';
39
- // Utilidades
40
- export { QueryBuilder, query } from './utils/query-builder';
41
- export { buildPath, extractPathParams, getCollectionPath, getDocumentId, isCollectionPath, isDocumentPath, isValidPath, joinPath, } from './utils/path-builder';
42
- // Storage
43
- export { StorageService } from './storage.service';
44
- // Messaging (FCM)
45
- export { MessagingService } from './messaging.service';
46
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZpcmViYXNlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBRUgsUUFBUTtBQUNSLGNBQWMsU0FBUyxDQUFDO0FBRXhCLGdCQUFnQjtBQUNoQixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsWUFBWSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXpGLFlBQVk7QUFDWixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFckQsWUFBWTtBQUNaLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3ZELE9BQU8sRUFBcUIsbUJBQW1CLEVBQW9CLE1BQU0sd0JBQXdCLENBQUM7QUFFbEcsYUFBYTtBQUNiLE9BQU8sRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDNUQsT0FBTyxFQUNMLFNBQVMsRUFDVCxpQkFBaUIsRUFDakIsaUJBQWlCLEVBQ2pCLGFBQWEsRUFDYixnQkFBZ0IsRUFDaEIsY0FBYyxFQUNkLFdBQVcsRUFDWCxRQUFRLEdBQ1QsTUFBTSxzQkFBc0IsQ0FBQztBQUU5QixVQUFVO0FBQ1YsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBRW5ELGtCQUFrQjtBQUNsQixPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogRmlyZWJhc2UgU2VydmljZXNcbiAqXG4gKiBTZXJ2aWNpb3MgcmV1dGlsaXphYmxlcyBwYXJhIGludGVncmFjacOzbiBjb24gRmlyZWJhc2UuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIEVuIG1haW4udHNcbiAqIGltcG9ydCB7IHByb3ZpZGVWYWx0ZWNoRmlyZWJhc2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIGJvb3RzdHJhcEFwcGxpY2F0aW9uKEFwcENvbXBvbmVudCwge1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICBwcm92aWRlVmFsdGVjaEZpcmViYXNlKHtcbiAqICAgICAgIGZpcmViYXNlOiBlbnZpcm9ubWVudC5maXJlYmFzZSxcbiAqICAgICAgIHBlcnNpc3RlbmNlOiB0cnVlLFxuICogICAgIH0pLFxuICogICBdLFxuICogfSk7XG4gKlxuICogLy8gRW4gY29tcG9uZW50ZXNcbiAqIGltcG9ydCB7IEZpcmViYXNlU2VydmljZSwgRmlyZXN0b3JlU2VydmljZSB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogQENvbXBvbmVudCh7Li4ufSlcbiAqIGV4cG9ydCBjbGFzcyBNeUNvbXBvbmVudCB7XG4gKiAgIHByaXZhdGUgZmlyZWJhc2UgPSBpbmplY3QoRmlyZWJhc2VTZXJ2aWNlKTtcbiAqICAgcHJpdmF0ZSBmaXJlc3RvcmUgPSBpbmplY3QoRmlyZXN0b3JlU2VydmljZSk7XG4gKiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBUaXBvc1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG5cbi8vIENvbmZpZ3VyYWNpw7NuXG5leHBvcnQgeyBWQUxURUNIX0ZJUkVCQVNFX0NPTkZJRywgaGFzRW11bGF0b3JzLCBwcm92aWRlVmFsdGVjaEZpcmViYXNlIH0gZnJvbSAnLi9jb25maWcnO1xuXG4vLyBTZXJ2aWNpb3NcbmV4cG9ydCB7IEZpcmViYXNlU2VydmljZSB9IGZyb20gJy4vZmlyZWJhc2Uuc2VydmljZSc7XG5cbi8vIEZpcmVzdG9yZVxuZXhwb3J0IHsgRmlyZXN0b3JlU2VydmljZSB9IGZyb20gJy4vZmlyZXN0b3JlLnNlcnZpY2UnO1xuZXhwb3J0IHsgQ29sbGVjdGlvbk9wdGlvbnMsIEZpcmVzdG9yZUNvbGxlY3Rpb24sIFN1YkNvbGxlY3Rpb25SZWYgfSBmcm9tICcuL2ZpcmVzdG9yZS1jb2xsZWN0aW9uJztcblxuLy8gVXRpbGlkYWRlc1xuZXhwb3J0IHsgUXVlcnlCdWlsZGVyLCBxdWVyeSB9IGZyb20gJy4vdXRpbHMvcXVlcnktYnVpbGRlcic7XG5leHBvcnQge1xuICBidWlsZFBhdGgsXG4gIGV4dHJhY3RQYXRoUGFyYW1zLFxuICBnZXRDb2xsZWN0aW9uUGF0aCxcbiAgZ2V0RG9jdW1lbnRJZCxcbiAgaXNDb2xsZWN0aW9uUGF0aCxcbiAgaXNEb2N1bWVudFBhdGgsXG4gIGlzVmFsaWRQYXRoLFxuICBqb2luUGF0aCxcbn0gZnJvbSAnLi91dGlscy9wYXRoLWJ1aWxkZXInO1xuXG4vLyBTdG9yYWdlXG5leHBvcnQgeyBTdG9yYWdlU2VydmljZSB9IGZyb20gJy4vc3RvcmFnZS5zZXJ2aWNlJztcblxuLy8gTWVzc2FnaW5nIChGQ00pXG5leHBvcnQgeyBNZXNzYWdpbmdTZXJ2aWNlIH0gZnJvbSAnLi9tZXNzYWdpbmcuc2VydmljZSc7XG4iXX0=