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