valtech-components 2.0.427 → 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 +28 -3
  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 +2886 -230
  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,421 @@
1
+ /**
2
+ * Storage Service
3
+ *
4
+ * Servicio para operaciones de Firebase Storage.
5
+ * Soporta upload con tracking de progreso, download y gestión de archivos.
6
+ */
7
+ import { inject, Injectable } from '@angular/core';
8
+ import { deleteObject, getDownloadURL, getMetadata, listAll, ref, Storage, uploadBytesResumable, } from '@angular/fire/storage';
9
+ import { BehaviorSubject } from 'rxjs';
10
+ import * as i0 from "@angular/core";
11
+ /**
12
+ * Servicio para Firebase Storage.
13
+ *
14
+ * @example
15
+ * ```typescript
16
+ * @Component({...})
17
+ * export class FileUploadComponent {
18
+ * private storage = inject(StorageService);
19
+ *
20
+ * uploadProgress = signal<number>(0);
21
+ * downloadUrl = signal<string | null>(null);
22
+ *
23
+ * async onFileSelected(event: Event) {
24
+ * const file = (event.target as HTMLInputElement).files?.[0];
25
+ * if (!file) return;
26
+ *
27
+ * // Upload con progreso
28
+ * this.storage.upload(`uploads/${file.name}`, file).subscribe({
29
+ * next: (progress) => this.uploadProgress.set(progress.percentage),
30
+ * complete: async () => {
31
+ * const url = await this.storage.getDownloadUrl(`uploads/${file.name}`);
32
+ * this.downloadUrl.set(url);
33
+ * }
34
+ * });
35
+ * }
36
+ * }
37
+ * ```
38
+ */
39
+ export class StorageService {
40
+ constructor() {
41
+ this.storage = inject(Storage);
42
+ }
43
+ // ===========================================================================
44
+ // UPLOAD
45
+ // ===========================================================================
46
+ /**
47
+ * Sube un archivo con tracking de progreso.
48
+ *
49
+ * @param path - Ruta en Storage donde guardar el archivo
50
+ * @param file - Archivo a subir (File o Blob)
51
+ * @param metadata - Metadata opcional (contentType, customMetadata)
52
+ * @returns Observable que emite el progreso y completa cuando termina
53
+ *
54
+ * @example
55
+ * ```typescript
56
+ * // Upload básico
57
+ * storage.upload('images/photo.jpg', file).subscribe({
58
+ * next: (progress) => console.log(`${progress.percentage}%`),
59
+ * complete: () => console.log('Upload completado')
60
+ * });
61
+ *
62
+ * // Con metadata
63
+ * storage.upload('docs/report.pdf', file, {
64
+ * contentType: 'application/pdf',
65
+ * customMetadata: { uploadedBy: 'user123' }
66
+ * }).subscribe(...);
67
+ * ```
68
+ */
69
+ upload(path, file, metadata) {
70
+ const storageRef = ref(this.storage, path);
71
+ const uploadMetadata = {
72
+ contentType: metadata?.contentType || (file instanceof File ? file.type : undefined),
73
+ customMetadata: metadata?.customMetadata,
74
+ cacheControl: metadata?.cacheControl,
75
+ };
76
+ const task = uploadBytesResumable(storageRef, file, uploadMetadata);
77
+ const progress$ = new BehaviorSubject({
78
+ bytesTransferred: 0,
79
+ totalBytes: file.size,
80
+ percentage: 0,
81
+ state: 'running',
82
+ });
83
+ task.on('state_changed', (snapshot) => {
84
+ progress$.next({
85
+ bytesTransferred: snapshot.bytesTransferred,
86
+ totalBytes: snapshot.totalBytes,
87
+ percentage: Math.round((snapshot.bytesTransferred / snapshot.totalBytes) * 100),
88
+ state: this.mapTaskState(snapshot.state),
89
+ });
90
+ }, (error) => {
91
+ progress$.next({
92
+ bytesTransferred: 0,
93
+ totalBytes: file.size,
94
+ percentage: 0,
95
+ state: 'error',
96
+ });
97
+ progress$.error(this.getErrorMessage(error));
98
+ }, () => {
99
+ progress$.next({
100
+ bytesTransferred: file.size,
101
+ totalBytes: file.size,
102
+ percentage: 100,
103
+ state: 'success',
104
+ });
105
+ progress$.complete();
106
+ });
107
+ return progress$.asObservable();
108
+ }
109
+ /**
110
+ * Sube un archivo y retorna la URL de descarga al completar.
111
+ *
112
+ * @param path - Ruta en Storage
113
+ * @param file - Archivo a subir
114
+ * @param metadata - Metadata opcional
115
+ * @returns Resultado del upload con URL de descarga
116
+ *
117
+ * @example
118
+ * ```typescript
119
+ * const result = await storage.uploadAndGetUrl('avatars/user123.jpg', file);
120
+ * console.log('URL:', result.downloadUrl);
121
+ * ```
122
+ */
123
+ async uploadAndGetUrl(path, file, metadata) {
124
+ return new Promise((resolve, reject) => {
125
+ this.upload(path, file, metadata).subscribe({
126
+ complete: async () => {
127
+ try {
128
+ const storageRef = ref(this.storage, path);
129
+ const downloadUrl = await getDownloadURL(storageRef);
130
+ const storedMetadata = await getMetadata(storageRef);
131
+ resolve({
132
+ downloadUrl,
133
+ fullPath: storedMetadata.fullPath,
134
+ name: storedMetadata.name,
135
+ size: storedMetadata.size,
136
+ contentType: storedMetadata.contentType || 'application/octet-stream',
137
+ metadata: storedMetadata.customMetadata || {},
138
+ });
139
+ }
140
+ catch (error) {
141
+ reject(this.getErrorMessage(error));
142
+ }
143
+ },
144
+ error: (error) => reject(error),
145
+ });
146
+ });
147
+ }
148
+ /**
149
+ * Sube un archivo desde una Data URL (base64).
150
+ *
151
+ * @param path - Ruta en Storage
152
+ * @param dataUrl - Data URL (ej: 'data:image/png;base64,...')
153
+ * @param metadata - Metadata opcional
154
+ * @returns Resultado del upload
155
+ *
156
+ * @example
157
+ * ```typescript
158
+ * // Desde canvas
159
+ * const dataUrl = canvas.toDataURL('image/png');
160
+ * const result = await storage.uploadFromDataUrl('images/drawing.png', dataUrl);
161
+ * ```
162
+ */
163
+ async uploadFromDataUrl(path, dataUrl, metadata) {
164
+ // Extraer content type y datos base64
165
+ const matches = dataUrl.match(/^data:([^;]+);base64,(.+)$/);
166
+ if (!matches) {
167
+ throw new Error('Data URL inválida');
168
+ }
169
+ const contentType = matches[1];
170
+ const base64Data = matches[2];
171
+ // Convertir base64 a Blob
172
+ const byteCharacters = atob(base64Data);
173
+ const byteNumbers = new Array(byteCharacters.length);
174
+ for (let i = 0; i < byteCharacters.length; i++) {
175
+ byteNumbers[i] = byteCharacters.charCodeAt(i);
176
+ }
177
+ const byteArray = new Uint8Array(byteNumbers);
178
+ const blob = new Blob([byteArray], { type: contentType });
179
+ return this.uploadAndGetUrl(path, blob, {
180
+ contentType,
181
+ ...metadata,
182
+ });
183
+ }
184
+ // ===========================================================================
185
+ // DOWNLOAD
186
+ // ===========================================================================
187
+ /**
188
+ * Obtiene la URL de descarga de un archivo.
189
+ *
190
+ * @param path - Ruta del archivo en Storage
191
+ * @returns URL de descarga
192
+ *
193
+ * @example
194
+ * ```typescript
195
+ * const url = await storage.getDownloadUrl('images/photo.jpg');
196
+ * // Usar en <img [src]="url">
197
+ * ```
198
+ */
199
+ async getDownloadUrl(path) {
200
+ try {
201
+ const storageRef = ref(this.storage, path);
202
+ return await getDownloadURL(storageRef);
203
+ }
204
+ catch (error) {
205
+ throw new Error(this.getErrorMessage(error));
206
+ }
207
+ }
208
+ /**
209
+ * Obtiene la metadata de un archivo.
210
+ *
211
+ * @param path - Ruta del archivo
212
+ * @returns Metadata del archivo
213
+ */
214
+ async getMetadata(path) {
215
+ try {
216
+ const storageRef = ref(this.storage, path);
217
+ const metadata = await getMetadata(storageRef);
218
+ return {
219
+ contentType: metadata.contentType,
220
+ customMetadata: metadata.customMetadata,
221
+ cacheControl: metadata.cacheControl,
222
+ size: metadata.size,
223
+ name: metadata.name,
224
+ };
225
+ }
226
+ catch (error) {
227
+ throw new Error(this.getErrorMessage(error));
228
+ }
229
+ }
230
+ // ===========================================================================
231
+ // DELETE
232
+ // ===========================================================================
233
+ /**
234
+ * Elimina un archivo.
235
+ *
236
+ * @param path - Ruta del archivo a eliminar
237
+ *
238
+ * @example
239
+ * ```typescript
240
+ * await storage.delete('images/old-photo.jpg');
241
+ * ```
242
+ */
243
+ async delete(path) {
244
+ try {
245
+ const storageRef = ref(this.storage, path);
246
+ await deleteObject(storageRef);
247
+ }
248
+ catch (error) {
249
+ throw new Error(this.getErrorMessage(error));
250
+ }
251
+ }
252
+ /**
253
+ * Elimina múltiples archivos.
254
+ *
255
+ * @param paths - Array de rutas a eliminar
256
+ *
257
+ * @example
258
+ * ```typescript
259
+ * await storage.deleteMultiple([
260
+ * 'images/photo1.jpg',
261
+ * 'images/photo2.jpg'
262
+ * ]);
263
+ * ```
264
+ */
265
+ async deleteMultiple(paths) {
266
+ await Promise.all(paths.map((path) => this.delete(path)));
267
+ }
268
+ // ===========================================================================
269
+ // LIST
270
+ // ===========================================================================
271
+ /**
272
+ * Lista archivos en un directorio.
273
+ *
274
+ * @param path - Ruta del directorio
275
+ * @returns Lista de rutas de archivos
276
+ *
277
+ * @example
278
+ * ```typescript
279
+ * const result = await storage.list('images/');
280
+ * console.log(result.items); // ['images/photo1.jpg', 'images/photo2.jpg']
281
+ * ```
282
+ */
283
+ async list(path) {
284
+ try {
285
+ const storageRef = ref(this.storage, path);
286
+ const result = await listAll(storageRef);
287
+ return {
288
+ items: result.items.map((item) => item.fullPath),
289
+ nextPageToken: undefined, // listAll no soporta paginación
290
+ };
291
+ }
292
+ catch (error) {
293
+ throw new Error(this.getErrorMessage(error));
294
+ }
295
+ }
296
+ // ===========================================================================
297
+ // UTILIDADES
298
+ // ===========================================================================
299
+ /**
300
+ * Genera un nombre de archivo único con timestamp.
301
+ *
302
+ * @param originalName - Nombre original del archivo
303
+ * @param prefix - Prefijo opcional
304
+ * @returns Nombre único
305
+ *
306
+ * @example
307
+ * ```typescript
308
+ * const uniqueName = storage.generateFileName('photo.jpg', 'user123');
309
+ * // => 'user123_1703091234567_photo.jpg'
310
+ * ```
311
+ */
312
+ generateFileName(originalName, prefix) {
313
+ const timestamp = Date.now();
314
+ const sanitizedName = originalName.replace(/[^a-zA-Z0-9.-]/g, '_');
315
+ if (prefix) {
316
+ return `${prefix}_${timestamp}_${sanitizedName}`;
317
+ }
318
+ return `${timestamp}_${sanitizedName}`;
319
+ }
320
+ /**
321
+ * Genera una ruta única para un archivo.
322
+ *
323
+ * @param directory - Directorio base
324
+ * @param originalName - Nombre original
325
+ * @param prefix - Prefijo opcional
326
+ * @returns Ruta completa única
327
+ *
328
+ * @example
329
+ * ```typescript
330
+ * const path = storage.generatePath('uploads', 'photo.jpg', 'user123');
331
+ * // => 'uploads/user123_1703091234567_photo.jpg'
332
+ * ```
333
+ */
334
+ generatePath(directory, originalName, prefix) {
335
+ const fileName = this.generateFileName(originalName, prefix);
336
+ const cleanDir = directory.replace(/\/+$/, ''); // Remover / final
337
+ return `${cleanDir}/${fileName}`;
338
+ }
339
+ /**
340
+ * Obtiene la extensión de un archivo.
341
+ *
342
+ * @param filename - Nombre del archivo
343
+ * @returns Extensión (sin el punto)
344
+ */
345
+ getExtension(filename) {
346
+ const parts = filename.split('.');
347
+ return parts.length > 1 ? parts.pop().toLowerCase() : '';
348
+ }
349
+ /**
350
+ * Verifica si un archivo es una imagen basándose en su extensión.
351
+ */
352
+ isImage(filename) {
353
+ const ext = this.getExtension(filename);
354
+ return ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg', 'bmp'].includes(ext);
355
+ }
356
+ /**
357
+ * Verifica si un archivo es un documento.
358
+ */
359
+ isDocument(filename) {
360
+ const ext = this.getExtension(filename);
361
+ return ['pdf', 'doc', 'docx', 'xls', 'xlsx', 'ppt', 'pptx', 'txt'].includes(ext);
362
+ }
363
+ // ===========================================================================
364
+ // MÉTODOS PRIVADOS
365
+ // ===========================================================================
366
+ /**
367
+ * Mapea el estado de la tarea de upload
368
+ */
369
+ mapTaskState(state) {
370
+ switch (state) {
371
+ case 'running':
372
+ return 'running';
373
+ case 'paused':
374
+ return 'paused';
375
+ case 'success':
376
+ return 'success';
377
+ case 'canceled':
378
+ return 'canceled';
379
+ case 'error':
380
+ return 'error';
381
+ default:
382
+ return 'running';
383
+ }
384
+ }
385
+ /**
386
+ * Convierte errores de Storage a mensajes en español
387
+ */
388
+ getErrorMessage(error) {
389
+ if (error instanceof Error) {
390
+ const code = error.code;
391
+ switch (code) {
392
+ case 'storage/object-not-found':
393
+ return 'El archivo no existe';
394
+ case 'storage/unauthorized':
395
+ return 'No tienes permiso para acceder a este archivo';
396
+ case 'storage/canceled':
397
+ return 'La operación fue cancelada';
398
+ case 'storage/quota-exceeded':
399
+ return 'Se ha excedido la cuota de almacenamiento';
400
+ case 'storage/invalid-checksum':
401
+ return 'El archivo está corrupto';
402
+ case 'storage/retry-limit-exceeded':
403
+ return 'Error de conexión. Intenta de nuevo';
404
+ case 'storage/invalid-url':
405
+ return 'URL de archivo inválida';
406
+ case 'storage/invalid-argument':
407
+ return 'Argumento inválido';
408
+ default:
409
+ return error.message || 'Error de almacenamiento desconocido';
410
+ }
411
+ }
412
+ return 'Error de almacenamiento desconocido';
413
+ }
414
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
415
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, providedIn: 'root' }); }
416
+ }
417
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, decorators: [{
418
+ type: Injectable,
419
+ args: [{ providedIn: 'root' }]
420
+ }] });
421
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/storage.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Storage Service\n *\n * Servicio para operaciones de Firebase Storage.\n * Soporta upload con tracking de progreso, download y gestión de archivos.\n */\n\nimport { inject, Injectable } from '@angular/core';\nimport {\n  deleteObject,\n  getDownloadURL,\n  getMetadata,\n  listAll,\n  ref,\n  Storage,\n  uploadBytesResumable,\n  UploadMetadata,\n  UploadTaskSnapshot,\n} from '@angular/fire/storage';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\nimport { StorageListResult, StorageMetadata, UploadProgress, UploadResult, UploadState } from './types';\n\n/**\n * Servicio para Firebase Storage.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class FileUploadComponent {\n *   private storage = inject(StorageService);\n *\n *   uploadProgress = signal<number>(0);\n *   downloadUrl = signal<string | null>(null);\n *\n *   async onFileSelected(event: Event) {\n *     const file = (event.target as HTMLInputElement).files?.[0];\n *     if (!file) return;\n *\n *     // Upload con progreso\n *     this.storage.upload(`uploads/${file.name}`, file).subscribe({\n *       next: (progress) => this.uploadProgress.set(progress.percentage),\n *       complete: async () => {\n *         const url = await this.storage.getDownloadUrl(`uploads/${file.name}`);\n *         this.downloadUrl.set(url);\n *       }\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class StorageService {\n  private storage = inject(Storage);\n\n  // ===========================================================================\n  // UPLOAD\n  // ===========================================================================\n\n  /**\n   * Sube un archivo con tracking de progreso.\n   *\n   * @param path - Ruta en Storage donde guardar el archivo\n   * @param file - Archivo a subir (File o Blob)\n   * @param metadata - Metadata opcional (contentType, customMetadata)\n   * @returns Observable que emite el progreso y completa cuando termina\n   *\n   * @example\n   * ```typescript\n   * // Upload básico\n   * storage.upload('images/photo.jpg', file).subscribe({\n   *   next: (progress) => console.log(`${progress.percentage}%`),\n   *   complete: () => console.log('Upload completado')\n   * });\n   *\n   * // Con metadata\n   * storage.upload('docs/report.pdf', file, {\n   *   contentType: 'application/pdf',\n   *   customMetadata: { uploadedBy: 'user123' }\n   * }).subscribe(...);\n   * ```\n   */\n  upload(path: string, file: File | Blob, metadata?: StorageMetadata): Observable<UploadProgress> {\n    const storageRef = ref(this.storage, path);\n    const uploadMetadata: UploadMetadata = {\n      contentType: metadata?.contentType || (file instanceof File ? file.type : undefined),\n      customMetadata: metadata?.customMetadata,\n      cacheControl: metadata?.cacheControl,\n    };\n\n    const task = uploadBytesResumable(storageRef, file, uploadMetadata);\n    const progress$ = new BehaviorSubject<UploadProgress>({\n      bytesTransferred: 0,\n      totalBytes: file.size,\n      percentage: 0,\n      state: 'running',\n    });\n\n    task.on(\n      'state_changed',\n      (snapshot: UploadTaskSnapshot) => {\n        progress$.next({\n          bytesTransferred: snapshot.bytesTransferred,\n          totalBytes: snapshot.totalBytes,\n          percentage: Math.round((snapshot.bytesTransferred / snapshot.totalBytes) * 100),\n          state: this.mapTaskState(snapshot.state),\n        });\n      },\n      (error) => {\n        progress$.next({\n          bytesTransferred: 0,\n          totalBytes: file.size,\n          percentage: 0,\n          state: 'error',\n        });\n        progress$.error(this.getErrorMessage(error));\n      },\n      () => {\n        progress$.next({\n          bytesTransferred: file.size,\n          totalBytes: file.size,\n          percentage: 100,\n          state: 'success',\n        });\n        progress$.complete();\n      }\n    );\n\n    return progress$.asObservable();\n  }\n\n  /**\n   * Sube un archivo y retorna la URL de descarga al completar.\n   *\n   * @param path - Ruta en Storage\n   * @param file - Archivo a subir\n   * @param metadata - Metadata opcional\n   * @returns Resultado del upload con URL de descarga\n   *\n   * @example\n   * ```typescript\n   * const result = await storage.uploadAndGetUrl('avatars/user123.jpg', file);\n   * console.log('URL:', result.downloadUrl);\n   * ```\n   */\n  async uploadAndGetUrl(\n    path: string,\n    file: File | Blob,\n    metadata?: StorageMetadata\n  ): Promise<UploadResult> {\n    return new Promise((resolve, reject) => {\n      this.upload(path, file, metadata).subscribe({\n        complete: async () => {\n          try {\n            const storageRef = ref(this.storage, path);\n            const downloadUrl = await getDownloadURL(storageRef);\n            const storedMetadata = await getMetadata(storageRef);\n\n            resolve({\n              downloadUrl,\n              fullPath: storedMetadata.fullPath,\n              name: storedMetadata.name,\n              size: storedMetadata.size,\n              contentType: storedMetadata.contentType || 'application/octet-stream',\n              metadata: storedMetadata.customMetadata || {},\n            });\n          } catch (error) {\n            reject(this.getErrorMessage(error));\n          }\n        },\n        error: (error) => reject(error),\n      });\n    });\n  }\n\n  /**\n   * Sube un archivo desde una Data URL (base64).\n   *\n   * @param path - Ruta en Storage\n   * @param dataUrl - Data URL (ej: 'data:image/png;base64,...')\n   * @param metadata - Metadata opcional\n   * @returns Resultado del upload\n   *\n   * @example\n   * ```typescript\n   * // Desde canvas\n   * const dataUrl = canvas.toDataURL('image/png');\n   * const result = await storage.uploadFromDataUrl('images/drawing.png', dataUrl);\n   * ```\n   */\n  async uploadFromDataUrl(\n    path: string,\n    dataUrl: string,\n    metadata?: StorageMetadata\n  ): Promise<UploadResult> {\n    // Extraer content type y datos base64\n    const matches = dataUrl.match(/^data:([^;]+);base64,(.+)$/);\n    if (!matches) {\n      throw new Error('Data URL inválida');\n    }\n\n    const contentType = matches[1];\n    const base64Data = matches[2];\n\n    // Convertir base64 a Blob\n    const byteCharacters = atob(base64Data);\n    const byteNumbers = new Array(byteCharacters.length);\n    for (let i = 0; i < byteCharacters.length; i++) {\n      byteNumbers[i] = byteCharacters.charCodeAt(i);\n    }\n    const byteArray = new Uint8Array(byteNumbers);\n    const blob = new Blob([byteArray], { type: contentType });\n\n    return this.uploadAndGetUrl(path, blob, {\n      contentType,\n      ...metadata,\n    });\n  }\n\n  // ===========================================================================\n  // DOWNLOAD\n  // ===========================================================================\n\n  /**\n   * Obtiene la URL de descarga de un archivo.\n   *\n   * @param path - Ruta del archivo en Storage\n   * @returns URL de descarga\n   *\n   * @example\n   * ```typescript\n   * const url = await storage.getDownloadUrl('images/photo.jpg');\n   * // Usar en <img [src]=\"url\">\n   * ```\n   */\n  async getDownloadUrl(path: string): Promise<string> {\n    try {\n      const storageRef = ref(this.storage, path);\n      return await getDownloadURL(storageRef);\n    } catch (error) {\n      throw new Error(this.getErrorMessage(error));\n    }\n  }\n\n  /**\n   * Obtiene la metadata de un archivo.\n   *\n   * @param path - Ruta del archivo\n   * @returns Metadata del archivo\n   */\n  async getMetadata(path: string): Promise<StorageMetadata & { size: number; name: string }> {\n    try {\n      const storageRef = ref(this.storage, path);\n      const metadata = await getMetadata(storageRef);\n      return {\n        contentType: metadata.contentType,\n        customMetadata: metadata.customMetadata,\n        cacheControl: metadata.cacheControl,\n        size: metadata.size,\n        name: metadata.name,\n      };\n    } catch (error) {\n      throw new Error(this.getErrorMessage(error));\n    }\n  }\n\n  // ===========================================================================\n  // DELETE\n  // ===========================================================================\n\n  /**\n   * Elimina un archivo.\n   *\n   * @param path - Ruta del archivo a eliminar\n   *\n   * @example\n   * ```typescript\n   * await storage.delete('images/old-photo.jpg');\n   * ```\n   */\n  async delete(path: string): Promise<void> {\n    try {\n      const storageRef = ref(this.storage, path);\n      await deleteObject(storageRef);\n    } catch (error) {\n      throw new Error(this.getErrorMessage(error));\n    }\n  }\n\n  /**\n   * Elimina múltiples archivos.\n   *\n   * @param paths - Array de rutas a eliminar\n   *\n   * @example\n   * ```typescript\n   * await storage.deleteMultiple([\n   *   'images/photo1.jpg',\n   *   'images/photo2.jpg'\n   * ]);\n   * ```\n   */\n  async deleteMultiple(paths: string[]): Promise<void> {\n    await Promise.all(paths.map((path) => this.delete(path)));\n  }\n\n  // ===========================================================================\n  // LIST\n  // ===========================================================================\n\n  /**\n   * Lista archivos en un directorio.\n   *\n   * @param path - Ruta del directorio\n   * @returns Lista de rutas de archivos\n   *\n   * @example\n   * ```typescript\n   * const result = await storage.list('images/');\n   * console.log(result.items); // ['images/photo1.jpg', 'images/photo2.jpg']\n   * ```\n   */\n  async list(path: string): Promise<StorageListResult> {\n    try {\n      const storageRef = ref(this.storage, path);\n      const result = await listAll(storageRef);\n\n      return {\n        items: result.items.map((item) => item.fullPath),\n        nextPageToken: undefined, // listAll no soporta paginación\n      };\n    } catch (error) {\n      throw new Error(this.getErrorMessage(error));\n    }\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Genera un nombre de archivo único con timestamp.\n   *\n   * @param originalName - Nombre original del archivo\n   * @param prefix - Prefijo opcional\n   * @returns Nombre único\n   *\n   * @example\n   * ```typescript\n   * const uniqueName = storage.generateFileName('photo.jpg', 'user123');\n   * // => 'user123_1703091234567_photo.jpg'\n   * ```\n   */\n  generateFileName(originalName: string, prefix?: string): string {\n    const timestamp = Date.now();\n    const sanitizedName = originalName.replace(/[^a-zA-Z0-9.-]/g, '_');\n\n    if (prefix) {\n      return `${prefix}_${timestamp}_${sanitizedName}`;\n    }\n    return `${timestamp}_${sanitizedName}`;\n  }\n\n  /**\n   * Genera una ruta única para un archivo.\n   *\n   * @param directory - Directorio base\n   * @param originalName - Nombre original\n   * @param prefix - Prefijo opcional\n   * @returns Ruta completa única\n   *\n   * @example\n   * ```typescript\n   * const path = storage.generatePath('uploads', 'photo.jpg', 'user123');\n   * // => 'uploads/user123_1703091234567_photo.jpg'\n   * ```\n   */\n  generatePath(directory: string, originalName: string, prefix?: string): string {\n    const fileName = this.generateFileName(originalName, prefix);\n    const cleanDir = directory.replace(/\\/+$/, ''); // Remover / final\n    return `${cleanDir}/${fileName}`;\n  }\n\n  /**\n   * Obtiene la extensión de un archivo.\n   *\n   * @param filename - Nombre del archivo\n   * @returns Extensión (sin el punto)\n   */\n  getExtension(filename: string): string {\n    const parts = filename.split('.');\n    return parts.length > 1 ? parts.pop()!.toLowerCase() : '';\n  }\n\n  /**\n   * Verifica si un archivo es una imagen basándose en su extensión.\n   */\n  isImage(filename: string): boolean {\n    const ext = this.getExtension(filename);\n    return ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg', 'bmp'].includes(ext);\n  }\n\n  /**\n   * Verifica si un archivo es un documento.\n   */\n  isDocument(filename: string): boolean {\n    const ext = this.getExtension(filename);\n    return ['pdf', 'doc', 'docx', 'xls', 'xlsx', 'ppt', 'pptx', 'txt'].includes(ext);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Mapea el estado de la tarea de upload\n   */\n  private mapTaskState(state: string): UploadState {\n    switch (state) {\n      case 'running':\n        return 'running';\n      case 'paused':\n        return 'paused';\n      case 'success':\n        return 'success';\n      case 'canceled':\n        return 'canceled';\n      case 'error':\n        return 'error';\n      default:\n        return 'running';\n    }\n  }\n\n  /**\n   * Convierte errores de Storage a mensajes en español\n   */\n  private getErrorMessage(error: unknown): string {\n    if (error instanceof Error) {\n      const code = (error as { code?: string }).code;\n\n      switch (code) {\n        case 'storage/object-not-found':\n          return 'El archivo no existe';\n        case 'storage/unauthorized':\n          return 'No tienes permiso para acceder a este archivo';\n        case 'storage/canceled':\n          return 'La operación fue cancelada';\n        case 'storage/quota-exceeded':\n          return 'Se ha excedido la cuota de almacenamiento';\n        case 'storage/invalid-checksum':\n          return 'El archivo está corrupto';\n        case 'storage/retry-limit-exceeded':\n          return 'Error de conexión. Intenta de nuevo';\n        case 'storage/invalid-url':\n          return 'URL de archivo inválida';\n        case 'storage/invalid-argument':\n          return 'Argumento inválido';\n        default:\n          return error.message || 'Error de almacenamiento desconocido';\n      }\n    }\n\n    return 'Error de almacenamiento desconocido';\n  }\n}\n"]}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Firebase Types
3
+ *
4
+ * Tipos e interfaces para la integración de Firebase en valtech-components.
5
+ * Todos los modelos de Firestore deben extender FirestoreDocument.
6
+ */
7
+ export {};
8
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG","sourcesContent":["/**\n * Firebase Types\n *\n * Tipos e interfaces para la integración de Firebase en valtech-components.\n * Todos los modelos de Firestore deben extender FirestoreDocument.\n */\n\n// ============================================================================\n// CONFIGURACIÓN\n// ============================================================================\n\n/**\n * Configuración de Firebase (valores de firebaseConfig)\n */\nexport interface FirebaseConfig {\n  apiKey: string;\n  authDomain: string;\n  projectId: string;\n  storageBucket: string;\n  messagingSenderId: string;\n  appId: string;\n  measurementId?: string;\n}\n\n/**\n * Configuración de emuladores para desarrollo local\n */\nexport interface EmulatorConfig {\n  firestore?: {\n    host: string;\n    port: number;\n  };\n  auth?: {\n    host: string;\n    port: number;\n  };\n  storage?: {\n    host: string;\n    port: number;\n  };\n}\n\n/**\n * Configuración completa de Valtech Firebase\n */\nexport interface ValtechFirebaseConfig {\n  /** Configuración de Firebase */\n  firebase: FirebaseConfig;\n  /** Configuración de emuladores (opcional, para desarrollo) */\n  emulator?: EmulatorConfig;\n  /** Habilitar persistencia offline de Firestore (default: false) */\n  persistence?: boolean;\n  /** Habilitar Firebase Cloud Messaging (default: false) - requiere Service Worker */\n  enableMessaging?: boolean;\n  /** VAPID key para Firebase Cloud Messaging */\n  messagingVapidKey?: string;\n}\n\n// ============================================================================\n// FIRESTORE - DOCUMENTOS\n// ============================================================================\n\n/**\n * Interface base para todos los documentos de Firestore.\n * Todos los modelos deben extender esta interface.\n *\n * @example\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n * }\n */\nexport interface FirestoreDocument {\n  /** ID del documento (asignado por Firestore) */\n  id?: string;\n  /** Fecha de creación (manejada automáticamente) */\n  createdAt?: Date;\n  /** Fecha de última actualización (manejada automáticamente) */\n  updatedAt?: Date;\n}\n\n// ============================================================================\n// FIRESTORE - QUERIES\n// ============================================================================\n\n/**\n * Operadores disponibles para cláusulas where\n */\nexport type WhereOperator =\n  | '=='\n  | '!='\n  | '<'\n  | '<='\n  | '>'\n  | '>='\n  | 'array-contains'\n  | 'array-contains-any'\n  | 'in'\n  | 'not-in';\n\n/**\n * Cláusula where para filtrar documentos\n */\nexport interface WhereClause {\n  /** Campo a filtrar */\n  field: string;\n  /** Operador de comparación */\n  operator: WhereOperator;\n  /** Valor a comparar */\n  value: unknown;\n}\n\n/**\n * Dirección de ordenamiento\n */\nexport type OrderDirection = 'asc' | 'desc';\n\n/**\n * Cláusula orderBy para ordenar resultados\n */\nexport interface OrderByClause {\n  /** Campo por el cual ordenar */\n  field: string;\n  /** Dirección del ordenamiento */\n  direction: OrderDirection;\n}\n\n/**\n * Opciones para queries de Firestore\n */\nexport interface QueryOptions {\n  /** Filtros where (AND entre todos) */\n  where?: WhereClause[];\n  /** Ordenamiento de resultados */\n  orderBy?: OrderByClause[];\n  /** Límite de documentos a retornar */\n  limit?: number;\n  /** Cursor para paginación: empezar después de este documento */\n  startAfter?: unknown;\n  /** Cursor para paginación: empezar en este documento */\n  startAt?: unknown;\n  /** Cursor para paginación: terminar antes de este documento */\n  endBefore?: unknown;\n  /** Cursor para paginación: terminar en este documento */\n  endAt?: unknown;\n}\n\n/**\n * Opciones adicionales para subscripciones real-time\n */\nexport interface SubscriptionOptions extends QueryOptions {\n  /** Incluir cambios de metadata (ej: pendingWrites) */\n  includeMetadataChanges?: boolean;\n}\n\n/**\n * Resultado de una query paginada\n */\nexport interface PaginatedResult<T> {\n  /** Documentos de la página actual */\n  data: T[];\n  /** Indica si hay más páginas disponibles */\n  hasMore: boolean;\n  /** Cursor para la siguiente página (pasar a startAfter) */\n  lastDoc: unknown;\n  /** Total de documentos (opcional, requiere query adicional) */\n  total?: number;\n}\n\n// ============================================================================\n// STORAGE\n// ============================================================================\n\n/**\n * Estado de una operación de upload\n */\nexport type UploadState = 'running' | 'paused' | 'success' | 'canceled' | 'error';\n\n/**\n * Progreso de upload de archivo\n */\nexport interface UploadProgress {\n  /** Bytes transferidos hasta ahora */\n  bytesTransferred: number;\n  /** Total de bytes a transferir */\n  totalBytes: number;\n  /** Porcentaje completado (0-100) */\n  percentage: number;\n  /** Estado actual del upload */\n  state: UploadState;\n}\n\n/**\n * Resultado de un upload completado\n */\nexport interface UploadResult {\n  /** URL de descarga del archivo */\n  downloadUrl: string;\n  /** Ruta completa en Storage */\n  fullPath: string;\n  /** Nombre del archivo */\n  name: string;\n  /** Tamaño en bytes */\n  size: number;\n  /** Tipo MIME del archivo */\n  contentType: string;\n  /** Metadata personalizada */\n  metadata: Record<string, string>;\n}\n\n/**\n * Metadata para archivos en Storage\n */\nexport interface StorageMetadata {\n  /** Tipo MIME del archivo */\n  contentType?: string;\n  /** Metadata personalizada (key-value) */\n  customMetadata?: Record<string, string>;\n  /** Control de caché HTTP */\n  cacheControl?: string;\n}\n\n/**\n * Resultado de listar archivos en Storage\n */\nexport interface StorageListResult {\n  /** Rutas de los archivos encontrados */\n  items: string[];\n  /** Token para la siguiente página (si hay más) */\n  nextPageToken?: string;\n}\n\n// ============================================================================\n// AUTH / SESSION\n// ============================================================================\n\n/**\n * Información del usuario de Firebase (simplificada)\n */\nexport interface FirebaseUser {\n  /** UID único del usuario */\n  uid: string;\n  /** Email del usuario */\n  email: string | null;\n  /** Nombre para mostrar */\n  displayName: string | null;\n  /** URL de foto de perfil */\n  photoURL: string | null;\n  /** Email verificado */\n  emailVerified: boolean;\n  /** Usuario anónimo */\n  isAnonymous: boolean;\n  /** Proveedor de autenticación */\n  providerId: string;\n}\n\n/**\n * Estado de la sesión de Firebase\n */\nexport interface SessionState {\n  /** Usuario actual (null si no autenticado) */\n  user: FirebaseUser | null;\n  /** Indica si el usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Indica si se está cargando el estado de auth */\n  isLoading: boolean;\n  /** Error de autenticación (si lo hay) */\n  error: Error | null;\n}\n\n// ============================================================================\n// MESSAGING (FCM)\n// ============================================================================\n\n/**\n * Estado del permiso de notificaciones\n */\nexport type NotificationPermission = 'granted' | 'denied' | 'default';\n\n/**\n * Payload de una notificación push\n */\nexport interface NotificationPayload {\n  /** Título de la notificación */\n  title?: string;\n  /** Cuerpo del mensaje */\n  body?: string;\n  /** URL de imagen */\n  image?: string;\n  /** Datos personalizados */\n  data?: Record<string, string>;\n  /** ID del mensaje de FCM */\n  messageId?: string;\n}\n\n/**\n * Acción de navegación desde una notificación\n */\nexport interface NotificationAction {\n  /** Ruta interna de la app (ej: '/orders/123') */\n  route?: string;\n  /** URL externa (ej: 'https://example.com') */\n  url?: string;\n  /** Parámetros de query string */\n  queryParams?: Record<string, string>;\n  /** Tipo de acción personalizada */\n  actionType?: string;\n  /** Datos adicionales para la acción */\n  actionData?: Record<string, unknown>;\n}\n\n/**\n * Evento de click en una notificación\n */\nexport interface NotificationClickEvent {\n  /** Payload original de la notificación */\n  notification: NotificationPayload;\n  /** Acción de navegación extraída */\n  action: NotificationAction;\n  /** Timestamp del click */\n  timestamp: Date;\n}\n\n// ============================================================================\n// ERRORES\n// ============================================================================\n\n/**\n * Códigos de error de Firebase\n */\nexport type FirebaseErrorCode =\n  | 'permission-denied'\n  | 'not-found'\n  | 'already-exists'\n  | 'resource-exhausted'\n  | 'cancelled'\n  | 'unknown'\n  | 'invalid-argument'\n  | 'deadline-exceeded'\n  | 'unauthenticated';\n\n/**\n * Error de Firebase tipado\n */\nexport interface FirebaseError {\n  /** Código del error */\n  code: FirebaseErrorCode;\n  /** Mensaje de error (en español) */\n  message: string;\n  /** Error original de Firebase */\n  originalError?: unknown;\n}\n"]}