valtech-components 2.0.452 → 2.0.453

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 (47) hide show
  1. package/esm2022/lib/services/auth/auth-state.service.mjs +173 -0
  2. package/esm2022/lib/services/auth/auth.service.mjs +454 -0
  3. package/esm2022/lib/services/auth/config.mjs +76 -0
  4. package/esm2022/lib/services/auth/guards.mjs +194 -0
  5. package/esm2022/lib/services/auth/index.mjs +70 -0
  6. package/esm2022/lib/services/auth/interceptor.mjs +98 -0
  7. package/esm2022/lib/services/auth/storage.service.mjs +141 -0
  8. package/esm2022/lib/services/auth/sync.service.mjs +149 -0
  9. package/esm2022/lib/services/auth/token.service.mjs +113 -0
  10. package/esm2022/lib/services/auth/types.mjs +29 -0
  11. package/esm2022/lib/services/firebase/config.mjs +108 -0
  12. package/esm2022/lib/services/firebase/firebase.service.mjs +288 -0
  13. package/esm2022/lib/services/firebase/firestore-collection.mjs +254 -0
  14. package/esm2022/lib/services/firebase/firestore.service.mjs +509 -0
  15. package/esm2022/lib/services/firebase/index.mjs +49 -0
  16. package/esm2022/lib/services/firebase/messaging.service.mjs +512 -0
  17. package/esm2022/lib/services/firebase/shared-config.mjs +138 -0
  18. package/esm2022/lib/services/firebase/storage.service.mjs +422 -0
  19. package/esm2022/lib/services/firebase/types.mjs +8 -0
  20. package/esm2022/lib/services/firebase/utils/path-builder.mjs +195 -0
  21. package/esm2022/lib/services/firebase/utils/query-builder.mjs +302 -0
  22. package/esm2022/public-api.mjs +3 -5
  23. package/fesm2022/valtech-components.mjs +4195 -4
  24. package/fesm2022/valtech-components.mjs.map +1 -1
  25. package/lib/services/auth/auth-state.service.d.ts +85 -0
  26. package/lib/services/auth/auth.service.d.ts +146 -0
  27. package/lib/services/auth/config.d.ts +38 -0
  28. package/lib/services/auth/guards.d.ts +123 -0
  29. package/lib/services/auth/index.d.ts +63 -0
  30. package/lib/services/auth/interceptor.d.ts +22 -0
  31. package/lib/services/auth/storage.service.d.ts +48 -0
  32. package/lib/services/auth/sync.service.d.ts +49 -0
  33. package/lib/services/auth/token.service.d.ts +51 -0
  34. package/lib/services/auth/types.d.ts +315 -0
  35. package/lib/services/firebase/config.d.ts +49 -0
  36. package/lib/services/firebase/firebase.service.d.ts +140 -0
  37. package/lib/services/firebase/firestore-collection.d.ts +175 -0
  38. package/lib/services/firebase/firestore.service.d.ts +304 -0
  39. package/lib/services/firebase/index.d.ts +39 -0
  40. package/lib/services/firebase/messaging.service.d.ts +263 -0
  41. package/lib/services/firebase/shared-config.d.ts +126 -0
  42. package/lib/services/firebase/storage.service.d.ts +206 -0
  43. package/lib/services/firebase/types.d.ts +281 -0
  44. package/lib/services/firebase/utils/path-builder.d.ts +132 -0
  45. package/lib/services/firebase/utils/query-builder.d.ts +210 -0
  46. package/package.json +1 -1
  47. package/public-api.d.ts +2 -0
@@ -0,0 +1,138 @@
1
+ /**
2
+ * Firebase Shared Configuration
3
+ *
4
+ * Configuración base de Firebase compartida entre todas las apps del monorepo.
5
+ * Los secrets (apiKey, appId) se inyectan en build time via environment.
6
+ */
7
+ export const APP_IDS = {
8
+ DEMO: 'demo',
9
+ SHOWCASE: 'showcase',
10
+ ADMIN_PORTAL: 'admin-portal',
11
+ APP: 'app',
12
+ };
13
+ // ============================================================================
14
+ // FIREBASE PROJECT IDS
15
+ // ============================================================================
16
+ /**
17
+ * IDs de los proyectos Firebase por ambiente.
18
+ * Deben coincidir con los proyectos creados en Firebase Console.
19
+ */
20
+ export const FIREBASE_PROJECTS = {
21
+ development: 'myvaltech-dev',
22
+ production: 'myvaltech-prod',
23
+ };
24
+ // ============================================================================
25
+ // EMULATOR CONFIGURATION (shared across all apps)
26
+ // ============================================================================
27
+ /**
28
+ * Configuración de emuladores compartida.
29
+ * Todos los puertos deben coincidir con frontend/firebase/firebase.json
30
+ */
31
+ export const SHARED_EMULATOR_CONFIG = {
32
+ firestore: { host: 'localhost', port: 9080 },
33
+ storage: { host: 'localhost', port: 9199 },
34
+ auth: { host: 'localhost', port: 9099 },
35
+ };
36
+ // ============================================================================
37
+ // STORAGE PATH BUILDERS
38
+ // ============================================================================
39
+ /**
40
+ * Genera paths de Storage con namespace por app.
41
+ *
42
+ * @example
43
+ * // Path específico de la app
44
+ * storagePaths.forApp('showcase', 'uploads', 'image.jpg')
45
+ * // => 'showcase/uploads/image.jpg'
46
+ *
47
+ * // Path compartido
48
+ * storagePaths.shared.profilePhoto('user123', 'avatar.jpg')
49
+ * // => 'profile-photos/user123/avatar.jpg'
50
+ */
51
+ export const storagePaths = {
52
+ /** Carpeta específica de la app: {appId}/{...paths} */
53
+ forApp: (appId, ...paths) => [appId, ...paths].join('/'),
54
+ /** Carpetas compartidas (sin namespace) */
55
+ shared: {
56
+ /** Foto de perfil de usuario */
57
+ profilePhoto: (userId, fileName) => `profile-photos/${userId}/${fileName}`,
58
+ /** Archivos públicos accesibles sin autenticación */
59
+ public: (...paths) => `public/${paths.join('/')}`,
60
+ },
61
+ /** Carpetas de desarrollo (acceso libre en emuladores) */
62
+ demo: (...paths) => `demo/${paths.join('/')}`,
63
+ };
64
+ // ============================================================================
65
+ // FIRESTORE COLLECTION BUILDERS
66
+ // ============================================================================
67
+ /**
68
+ * Genera paths de colecciones con namespace por app.
69
+ *
70
+ * IMPORTANTE: La estructura es /apps/{appId}/{collection}/{docId}
71
+ * Firestore requiere número impar de segmentos para paths de colección.
72
+ *
73
+ * @example
74
+ * // Colección específica de la app
75
+ * collections.forApp('showcase', 'items')
76
+ * // => 'apps/showcase/items'
77
+ *
78
+ * // Colección de desarrollo
79
+ * collections.forApp('demo', 'items')
80
+ * // => 'apps/demo/items'
81
+ *
82
+ * // Colección compartida
83
+ * collections.shared.users
84
+ * // => 'users'
85
+ */
86
+ export const collections = {
87
+ /** Colección específica de la app: apps/{appId}/{collection} */
88
+ forApp: (appId, collectionName) => `apps/${appId}/${collectionName}`,
89
+ /** Colecciones compartidas (sin namespace, nivel raíz) */
90
+ shared: {
91
+ /** Usuarios del sistema */
92
+ users: 'users',
93
+ /** Perfiles públicos */
94
+ profiles: 'profiles',
95
+ /** Notificaciones de usuarios */
96
+ notifications: 'notifications',
97
+ },
98
+ };
99
+ /**
100
+ * Crea la configuración completa de Firebase desde variables de entorno.
101
+ * Usa esto en el environment.ts de cada app.
102
+ *
103
+ * @example
104
+ * // environment.ts
105
+ * export const environment = {
106
+ * firebase: createFirebaseConfig(
107
+ * {
108
+ * apiKey: 'AIza...',
109
+ * authDomain: 'myvaltech-dev.firebaseapp.com',
110
+ * projectId: 'myvaltech-dev',
111
+ * storageBucket: 'myvaltech-dev.appspot.com',
112
+ * messagingSenderId: '123456789',
113
+ * appId: '1:123456789:web:abc123',
114
+ * },
115
+ * { useEmulators: true, persistence: true }
116
+ * ),
117
+ * };
118
+ */
119
+ export function createFirebaseConfig(envConfig, options = {}) {
120
+ const { useEmulators = false, persistence = true, enableMessaging = false, messagingVapidKey } = options;
121
+ return {
122
+ firebase: envConfig,
123
+ persistence,
124
+ enableMessaging,
125
+ messagingVapidKey,
126
+ emulator: useEmulators ? SHARED_EMULATOR_CONFIG : undefined,
127
+ };
128
+ }
129
+ // ============================================================================
130
+ // UTILITY: Check if running in emulator mode
131
+ // ============================================================================
132
+ /**
133
+ * Verifica si la configuración tiene emuladores habilitados
134
+ */
135
+ export function isEmulatorMode(config) {
136
+ return config.emulator !== undefined;
137
+ }
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shared-config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/shared-config.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAcH,MAAM,CAAC,MAAM,OAAO,GAAG;IACrB,IAAI,EAAE,MAAe;IACrB,QAAQ,EAAE,UAAmB;IAC7B,YAAY,EAAE,cAAuB;IACrC,GAAG,EAAE,KAAc;CACX,CAAC;AAEX,+EAA+E;AAC/E,uBAAuB;AACvB,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,CAAC,MAAM,iBAAiB,GAAG;IAC/B,WAAW,EAAE,eAAe;IAC5B,UAAU,EAAE,gBAAgB;CACpB,CAAC;AAEX,+EAA+E;AAC/E,kDAAkD;AAClD,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,CAAC,MAAM,sBAAsB,GAAmB;IACpD,SAAS,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;IAC5C,OAAO,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;IAC1C,IAAI,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;CACxC,CAAC;AAEF,+EAA+E;AAC/E,wBAAwB;AACxB,+EAA+E;AAE/E;;;;;;;;;;;GAWG;AACH,MAAM,CAAC,MAAM,YAAY,GAAG;IAC1B,uDAAuD;IACvD,MAAM,EAAE,CAAC,KAAY,EAAE,GAAG,KAAe,EAAU,EAAE,CAAC,CAAC,KAAK,EAAE,GAAG,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;IAEjF,2CAA2C;IAC3C,MAAM,EAAE;QACN,gCAAgC;QAChC,YAAY,EAAE,CAAC,MAAc,EAAE,QAAgB,EAAU,EAAE,CACzD,kBAAkB,MAAM,IAAI,QAAQ,EAAE;QAExC,qDAAqD;QACrD,MAAM,EAAE,CAAC,GAAG,KAAe,EAAU,EAAE,CAAC,UAAU,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;KACpE;IAED,0DAA0D;IAC1D,IAAI,EAAE,CAAC,GAAG,KAAe,EAAU,EAAE,CAAC,QAAQ,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;CAChE,CAAC;AAEF,+EAA+E;AAC/E,gCAAgC;AAChC,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,CAAC,MAAM,WAAW,GAAG;IACzB,gEAAgE;IAChE,MAAM,EAAE,CAAC,KAAY,EAAE,cAAsB,EAAU,EAAE,CAAC,QAAQ,KAAK,IAAI,cAAc,EAAE;IAE3F,0DAA0D;IAC1D,MAAM,EAAE;QACN,2BAA2B;QAC3B,KAAK,EAAE,OAAO;QACd,wBAAwB;QACxB,QAAQ,EAAE,UAAU;QACpB,iCAAiC;QACjC,aAAa,EAAE,eAAe;KAC/B;CACF,CAAC;AAoBF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,oBAAoB,CAClC,SAAyB,EACzB,UAAuC,EAAE;IAEzC,MAAM,EAAE,YAAY,GAAG,KAAK,EAAE,WAAW,GAAG,IAAI,EAAE,eAAe,GAAG,KAAK,EAAE,iBAAiB,EAAE,GAC5F,OAAO,CAAC;IAEV,OAAO;QACL,QAAQ,EAAE,SAAS;QACnB,WAAW;QACX,eAAe;QACf,iBAAiB;QACjB,QAAQ,EAAE,YAAY,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC,CAAC,SAAS;KAC5D,CAAC;AACJ,CAAC;AAED,+EAA+E;AAC/E,6CAA6C;AAC7C,+EAA+E;AAE/E;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,MAA6B;IAC1D,OAAO,MAAM,CAAC,QAAQ,KAAK,SAAS,CAAC;AACvC,CAAC","sourcesContent":["/**\n * Firebase Shared Configuration\n *\n * Configuración base de Firebase compartida entre todas las apps del monorepo.\n * Los secrets (apiKey, appId) se inyectan en build time via environment.\n */\n\nimport { EmulatorConfig, FirebaseConfig, ValtechFirebaseConfig } from './types';\n\n// ============================================================================\n// APP IDS - Identificadores únicos por aplicación\n// ============================================================================\n\n/**\n * Identificadores de las apps del monorepo.\n * Usados para namespacing de colecciones Firestore y paths de Storage.\n */\nexport type AppId = 'demo' | 'showcase' | 'admin-portal' | 'app';\n\nexport const APP_IDS = {\n  DEMO: 'demo' as AppId,\n  SHOWCASE: 'showcase' as AppId,\n  ADMIN_PORTAL: 'admin-portal' as AppId,\n  APP: 'app' as AppId,\n} as const;\n\n// ============================================================================\n// FIREBASE PROJECT IDS\n// ============================================================================\n\n/**\n * IDs de los proyectos Firebase por ambiente.\n * Deben coincidir con los proyectos creados en Firebase Console.\n */\nexport const FIREBASE_PROJECTS = {\n  development: 'myvaltech-dev',\n  production: 'myvaltech-prod',\n} as const;\n\n// ============================================================================\n// EMULATOR CONFIGURATION (shared across all apps)\n// ============================================================================\n\n/**\n * Configuración de emuladores compartida.\n * Todos los puertos deben coincidir con frontend/firebase/firebase.json\n */\nexport const SHARED_EMULATOR_CONFIG: EmulatorConfig = {\n  firestore: { host: 'localhost', port: 9080 },\n  storage: { host: 'localhost', port: 9199 },\n  auth: { host: 'localhost', port: 9099 },\n};\n\n// ============================================================================\n// STORAGE PATH BUILDERS\n// ============================================================================\n\n/**\n * Genera paths de Storage con namespace por app.\n *\n * @example\n * // Path específico de la app\n * storagePaths.forApp('showcase', 'uploads', 'image.jpg')\n * // => 'showcase/uploads/image.jpg'\n *\n * // Path compartido\n * storagePaths.shared.profilePhoto('user123', 'avatar.jpg')\n * // => 'profile-photos/user123/avatar.jpg'\n */\nexport const storagePaths = {\n  /** Carpeta específica de la app: {appId}/{...paths} */\n  forApp: (appId: AppId, ...paths: string[]): string => [appId, ...paths].join('/'),\n\n  /** Carpetas compartidas (sin namespace) */\n  shared: {\n    /** Foto de perfil de usuario */\n    profilePhoto: (userId: string, fileName: string): string =>\n      `profile-photos/${userId}/${fileName}`,\n\n    /** Archivos públicos accesibles sin autenticación */\n    public: (...paths: string[]): string => `public/${paths.join('/')}`,\n  },\n\n  /** Carpetas de desarrollo (acceso libre en emuladores) */\n  demo: (...paths: string[]): string => `demo/${paths.join('/')}`,\n};\n\n// ============================================================================\n// FIRESTORE COLLECTION BUILDERS\n// ============================================================================\n\n/**\n * Genera paths de colecciones con namespace por app.\n *\n * IMPORTANTE: La estructura es /apps/{appId}/{collection}/{docId}\n * Firestore requiere número impar de segmentos para paths de colección.\n *\n * @example\n * // Colección específica de la app\n * collections.forApp('showcase', 'items')\n * // => 'apps/showcase/items'\n *\n * // Colección de desarrollo\n * collections.forApp('demo', 'items')\n * // => 'apps/demo/items'\n *\n * // Colección compartida\n * collections.shared.users\n * // => 'users'\n */\nexport const collections = {\n  /** Colección específica de la app: apps/{appId}/{collection} */\n  forApp: (appId: AppId, collectionName: string): string => `apps/${appId}/${collectionName}`,\n\n  /** Colecciones compartidas (sin namespace, nivel raíz) */\n  shared: {\n    /** Usuarios del sistema */\n    users: 'users',\n    /** Perfiles públicos */\n    profiles: 'profiles',\n    /** Notificaciones de usuarios */\n    notifications: 'notifications',\n  },\n};\n\n// ============================================================================\n// HELPER: Create Firebase Config from Environment\n// ============================================================================\n\n/**\n * Opciones para crear la configuración de Firebase\n */\nexport interface CreateFirebaseConfigOptions {\n  /** Usar emuladores locales (para desarrollo) */\n  useEmulators?: boolean;\n  /** Habilitar persistencia offline de Firestore */\n  persistence?: boolean;\n  /** Habilitar Firebase Cloud Messaging */\n  enableMessaging?: boolean;\n  /** VAPID key para FCM (requerido si enableMessaging es true) */\n  messagingVapidKey?: string;\n}\n\n/**\n * Crea la configuración completa de Firebase desde variables de entorno.\n * Usa esto en el environment.ts de cada app.\n *\n * @example\n * // environment.ts\n * export const environment = {\n *   firebase: createFirebaseConfig(\n *     {\n *       apiKey: 'AIza...',\n *       authDomain: 'myvaltech-dev.firebaseapp.com',\n *       projectId: 'myvaltech-dev',\n *       storageBucket: 'myvaltech-dev.appspot.com',\n *       messagingSenderId: '123456789',\n *       appId: '1:123456789:web:abc123',\n *     },\n *     { useEmulators: true, persistence: true }\n *   ),\n * };\n */\nexport function createFirebaseConfig(\n  envConfig: FirebaseConfig,\n  options: CreateFirebaseConfigOptions = {}\n): ValtechFirebaseConfig {\n  const { useEmulators = false, persistence = true, enableMessaging = false, messagingVapidKey } =\n    options;\n\n  return {\n    firebase: envConfig,\n    persistence,\n    enableMessaging,\n    messagingVapidKey,\n    emulator: useEmulators ? SHARED_EMULATOR_CONFIG : undefined,\n  };\n}\n\n// ============================================================================\n// UTILITY: Check if running in emulator mode\n// ============================================================================\n\n/**\n * Verifica si la configuración tiene emuladores habilitados\n */\nexport function isEmulatorMode(config: ValtechFirebaseConfig): boolean {\n  return config.emulator !== undefined;\n}\n"]}
@@ -0,0 +1,422 @@
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 { Injectable } from '@angular/core';
8
+ import { deleteObject, getDownloadURL, getMetadata, listAll, ref, uploadBytesResumable, } from '@angular/fire/storage';
9
+ import { BehaviorSubject } from 'rxjs';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "@angular/fire/storage";
12
+ /**
13
+ * Servicio para Firebase Storage.
14
+ *
15
+ * @example
16
+ * ```typescript
17
+ * @Component({...})
18
+ * export class FileUploadComponent {
19
+ * private storage = inject(StorageService);
20
+ *
21
+ * uploadProgress = signal<number>(0);
22
+ * downloadUrl = signal<string | null>(null);
23
+ *
24
+ * async onFileSelected(event: Event) {
25
+ * const file = (event.target as HTMLInputElement).files?.[0];
26
+ * if (!file) return;
27
+ *
28
+ * // Upload con progreso
29
+ * this.storage.upload(`uploads/${file.name}`, file).subscribe({
30
+ * next: (progress) => this.uploadProgress.set(progress.percentage),
31
+ * complete: async () => {
32
+ * const url = await this.storage.getDownloadUrl(`uploads/${file.name}`);
33
+ * this.downloadUrl.set(url);
34
+ * }
35
+ * });
36
+ * }
37
+ * }
38
+ * ```
39
+ */
40
+ export class StorageService {
41
+ constructor(storage) {
42
+ this.storage = storage;
43
+ }
44
+ // ===========================================================================
45
+ // UPLOAD
46
+ // ===========================================================================
47
+ /**
48
+ * Sube un archivo con tracking de progreso.
49
+ *
50
+ * @param path - Ruta en Storage donde guardar el archivo
51
+ * @param file - Archivo a subir (File o Blob)
52
+ * @param metadata - Metadata opcional (contentType, customMetadata)
53
+ * @returns Observable que emite el progreso y completa cuando termina
54
+ *
55
+ * @example
56
+ * ```typescript
57
+ * // Upload básico
58
+ * storage.upload('images/photo.jpg', file).subscribe({
59
+ * next: (progress) => console.log(`${progress.percentage}%`),
60
+ * complete: () => console.log('Upload completado')
61
+ * });
62
+ *
63
+ * // Con metadata
64
+ * storage.upload('docs/report.pdf', file, {
65
+ * contentType: 'application/pdf',
66
+ * customMetadata: { uploadedBy: 'user123' }
67
+ * }).subscribe(...);
68
+ * ```
69
+ */
70
+ upload(path, file, metadata) {
71
+ const storageRef = ref(this.storage, path);
72
+ const uploadMetadata = {
73
+ contentType: metadata?.contentType || (file instanceof File ? file.type : undefined),
74
+ customMetadata: metadata?.customMetadata,
75
+ cacheControl: metadata?.cacheControl,
76
+ };
77
+ const task = uploadBytesResumable(storageRef, file, uploadMetadata);
78
+ const progress$ = new BehaviorSubject({
79
+ bytesTransferred: 0,
80
+ totalBytes: file.size,
81
+ percentage: 0,
82
+ state: 'running',
83
+ });
84
+ task.on('state_changed', (snapshot) => {
85
+ progress$.next({
86
+ bytesTransferred: snapshot.bytesTransferred,
87
+ totalBytes: snapshot.totalBytes,
88
+ percentage: Math.round((snapshot.bytesTransferred / snapshot.totalBytes) * 100),
89
+ state: this.mapTaskState(snapshot.state),
90
+ });
91
+ }, (error) => {
92
+ progress$.next({
93
+ bytesTransferred: 0,
94
+ totalBytes: file.size,
95
+ percentage: 0,
96
+ state: 'error',
97
+ });
98
+ progress$.error(this.getErrorMessage(error));
99
+ }, () => {
100
+ progress$.next({
101
+ bytesTransferred: file.size,
102
+ totalBytes: file.size,
103
+ percentage: 100,
104
+ state: 'success',
105
+ });
106
+ progress$.complete();
107
+ });
108
+ return progress$.asObservable();
109
+ }
110
+ /**
111
+ * Sube un archivo y retorna la URL de descarga al completar.
112
+ *
113
+ * @param path - Ruta en Storage
114
+ * @param file - Archivo a subir
115
+ * @param metadata - Metadata opcional
116
+ * @returns Resultado del upload con URL de descarga
117
+ *
118
+ * @example
119
+ * ```typescript
120
+ * const result = await storage.uploadAndGetUrl('avatars/user123.jpg', file);
121
+ * console.log('URL:', result.downloadUrl);
122
+ * ```
123
+ */
124
+ async uploadAndGetUrl(path, file, metadata) {
125
+ return new Promise((resolve, reject) => {
126
+ this.upload(path, file, metadata).subscribe({
127
+ complete: async () => {
128
+ try {
129
+ const storageRef = ref(this.storage, path);
130
+ const downloadUrl = await getDownloadURL(storageRef);
131
+ const storedMetadata = await getMetadata(storageRef);
132
+ resolve({
133
+ downloadUrl,
134
+ fullPath: storedMetadata.fullPath,
135
+ name: storedMetadata.name,
136
+ size: storedMetadata.size,
137
+ contentType: storedMetadata.contentType || 'application/octet-stream',
138
+ metadata: storedMetadata.customMetadata || {},
139
+ });
140
+ }
141
+ catch (error) {
142
+ reject(this.getErrorMessage(error));
143
+ }
144
+ },
145
+ error: (error) => reject(error),
146
+ });
147
+ });
148
+ }
149
+ /**
150
+ * Sube un archivo desde una Data URL (base64).
151
+ *
152
+ * @param path - Ruta en Storage
153
+ * @param dataUrl - Data URL (ej: 'data:image/png;base64,...')
154
+ * @param metadata - Metadata opcional
155
+ * @returns Resultado del upload
156
+ *
157
+ * @example
158
+ * ```typescript
159
+ * // Desde canvas
160
+ * const dataUrl = canvas.toDataURL('image/png');
161
+ * const result = await storage.uploadFromDataUrl('images/drawing.png', dataUrl);
162
+ * ```
163
+ */
164
+ async uploadFromDataUrl(path, dataUrl, metadata) {
165
+ // Extraer content type y datos base64
166
+ const matches = dataUrl.match(/^data:([^;]+);base64,(.+)$/);
167
+ if (!matches) {
168
+ throw new Error('Data URL inválida');
169
+ }
170
+ const contentType = matches[1];
171
+ const base64Data = matches[2];
172
+ // Convertir base64 a Blob
173
+ const byteCharacters = atob(base64Data);
174
+ const byteNumbers = new Array(byteCharacters.length);
175
+ for (let i = 0; i < byteCharacters.length; i++) {
176
+ byteNumbers[i] = byteCharacters.charCodeAt(i);
177
+ }
178
+ const byteArray = new Uint8Array(byteNumbers);
179
+ const blob = new Blob([byteArray], { type: contentType });
180
+ return this.uploadAndGetUrl(path, blob, {
181
+ contentType,
182
+ ...metadata,
183
+ });
184
+ }
185
+ // ===========================================================================
186
+ // DOWNLOAD
187
+ // ===========================================================================
188
+ /**
189
+ * Obtiene la URL de descarga de un archivo.
190
+ *
191
+ * @param path - Ruta del archivo en Storage
192
+ * @returns URL de descarga
193
+ *
194
+ * @example
195
+ * ```typescript
196
+ * const url = await storage.getDownloadUrl('images/photo.jpg');
197
+ * // Usar en <img [src]="url">
198
+ * ```
199
+ */
200
+ async getDownloadUrl(path) {
201
+ try {
202
+ const storageRef = ref(this.storage, path);
203
+ return await getDownloadURL(storageRef);
204
+ }
205
+ catch (error) {
206
+ throw new Error(this.getErrorMessage(error));
207
+ }
208
+ }
209
+ /**
210
+ * Obtiene la metadata de un archivo.
211
+ *
212
+ * @param path - Ruta del archivo
213
+ * @returns Metadata del archivo
214
+ */
215
+ async getMetadata(path) {
216
+ try {
217
+ const storageRef = ref(this.storage, path);
218
+ const metadata = await getMetadata(storageRef);
219
+ return {
220
+ contentType: metadata.contentType,
221
+ customMetadata: metadata.customMetadata,
222
+ cacheControl: metadata.cacheControl,
223
+ size: metadata.size,
224
+ name: metadata.name,
225
+ };
226
+ }
227
+ catch (error) {
228
+ throw new Error(this.getErrorMessage(error));
229
+ }
230
+ }
231
+ // ===========================================================================
232
+ // DELETE
233
+ // ===========================================================================
234
+ /**
235
+ * Elimina un archivo.
236
+ *
237
+ * @param path - Ruta del archivo a eliminar
238
+ *
239
+ * @example
240
+ * ```typescript
241
+ * await storage.delete('images/old-photo.jpg');
242
+ * ```
243
+ */
244
+ async delete(path) {
245
+ try {
246
+ const storageRef = ref(this.storage, path);
247
+ await deleteObject(storageRef);
248
+ }
249
+ catch (error) {
250
+ throw new Error(this.getErrorMessage(error));
251
+ }
252
+ }
253
+ /**
254
+ * Elimina múltiples archivos.
255
+ *
256
+ * @param paths - Array de rutas a eliminar
257
+ *
258
+ * @example
259
+ * ```typescript
260
+ * await storage.deleteMultiple([
261
+ * 'images/photo1.jpg',
262
+ * 'images/photo2.jpg'
263
+ * ]);
264
+ * ```
265
+ */
266
+ async deleteMultiple(paths) {
267
+ await Promise.all(paths.map((path) => this.delete(path)));
268
+ }
269
+ // ===========================================================================
270
+ // LIST
271
+ // ===========================================================================
272
+ /**
273
+ * Lista archivos en un directorio.
274
+ *
275
+ * @param path - Ruta del directorio
276
+ * @returns Lista de rutas de archivos
277
+ *
278
+ * @example
279
+ * ```typescript
280
+ * const result = await storage.list('images/');
281
+ * console.log(result.items); // ['images/photo1.jpg', 'images/photo2.jpg']
282
+ * ```
283
+ */
284
+ async list(path) {
285
+ try {
286
+ const storageRef = ref(this.storage, path);
287
+ const result = await listAll(storageRef);
288
+ return {
289
+ items: result.items.map((item) => item.fullPath),
290
+ nextPageToken: undefined, // listAll no soporta paginación
291
+ };
292
+ }
293
+ catch (error) {
294
+ throw new Error(this.getErrorMessage(error));
295
+ }
296
+ }
297
+ // ===========================================================================
298
+ // UTILIDADES
299
+ // ===========================================================================
300
+ /**
301
+ * Genera un nombre de archivo único con timestamp.
302
+ *
303
+ * @param originalName - Nombre original del archivo
304
+ * @param prefix - Prefijo opcional
305
+ * @returns Nombre único
306
+ *
307
+ * @example
308
+ * ```typescript
309
+ * const uniqueName = storage.generateFileName('photo.jpg', 'user123');
310
+ * // => 'user123_1703091234567_photo.jpg'
311
+ * ```
312
+ */
313
+ generateFileName(originalName, prefix) {
314
+ const timestamp = Date.now();
315
+ const sanitizedName = originalName.replace(/[^a-zA-Z0-9.-]/g, '_');
316
+ if (prefix) {
317
+ return `${prefix}_${timestamp}_${sanitizedName}`;
318
+ }
319
+ return `${timestamp}_${sanitizedName}`;
320
+ }
321
+ /**
322
+ * Genera una ruta única para un archivo.
323
+ *
324
+ * @param directory - Directorio base
325
+ * @param originalName - Nombre original
326
+ * @param prefix - Prefijo opcional
327
+ * @returns Ruta completa única
328
+ *
329
+ * @example
330
+ * ```typescript
331
+ * const path = storage.generatePath('uploads', 'photo.jpg', 'user123');
332
+ * // => 'uploads/user123_1703091234567_photo.jpg'
333
+ * ```
334
+ */
335
+ generatePath(directory, originalName, prefix) {
336
+ const fileName = this.generateFileName(originalName, prefix);
337
+ const cleanDir = directory.replace(/\/+$/, ''); // Remover / final
338
+ return `${cleanDir}/${fileName}`;
339
+ }
340
+ /**
341
+ * Obtiene la extensión de un archivo.
342
+ *
343
+ * @param filename - Nombre del archivo
344
+ * @returns Extensión (sin el punto)
345
+ */
346
+ getExtension(filename) {
347
+ const parts = filename.split('.');
348
+ return parts.length > 1 ? parts.pop().toLowerCase() : '';
349
+ }
350
+ /**
351
+ * Verifica si un archivo es una imagen basándose en su extensión.
352
+ */
353
+ isImage(filename) {
354
+ const ext = this.getExtension(filename);
355
+ return ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg', 'bmp'].includes(ext);
356
+ }
357
+ /**
358
+ * Verifica si un archivo es un documento.
359
+ */
360
+ isDocument(filename) {
361
+ const ext = this.getExtension(filename);
362
+ return ['pdf', 'doc', 'docx', 'xls', 'xlsx', 'ppt', 'pptx', 'txt'].includes(ext);
363
+ }
364
+ // ===========================================================================
365
+ // MÉTODOS PRIVADOS
366
+ // ===========================================================================
367
+ /**
368
+ * Mapea el estado de la tarea de upload
369
+ */
370
+ mapTaskState(state) {
371
+ switch (state) {
372
+ case 'running':
373
+ return 'running';
374
+ case 'paused':
375
+ return 'paused';
376
+ case 'success':
377
+ return 'success';
378
+ case 'canceled':
379
+ return 'canceled';
380
+ case 'error':
381
+ return 'error';
382
+ default:
383
+ return 'running';
384
+ }
385
+ }
386
+ /**
387
+ * Convierte errores de Storage a mensajes en español
388
+ */
389
+ getErrorMessage(error) {
390
+ if (error instanceof Error) {
391
+ const code = error.code;
392
+ switch (code) {
393
+ case 'storage/object-not-found':
394
+ return 'El archivo no existe';
395
+ case 'storage/unauthorized':
396
+ return 'No tienes permiso para acceder a este archivo';
397
+ case 'storage/canceled':
398
+ return 'La operación fue cancelada';
399
+ case 'storage/quota-exceeded':
400
+ return 'Se ha excedido la cuota de almacenamiento';
401
+ case 'storage/invalid-checksum':
402
+ return 'El archivo está corrupto';
403
+ case 'storage/retry-limit-exceeded':
404
+ return 'Error de conexión. Intenta de nuevo';
405
+ case 'storage/invalid-url':
406
+ return 'URL de archivo inválida';
407
+ case 'storage/invalid-argument':
408
+ return 'Argumento inválido';
409
+ default:
410
+ return error.message || 'Error de almacenamiento desconocido';
411
+ }
412
+ }
413
+ return 'Error de almacenamiento desconocido';
414
+ }
415
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, deps: [{ token: i1.Storage }], target: i0.ɵɵFactoryTarget.Injectable }); }
416
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, providedIn: 'root' }); }
417
+ }
418
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StorageService, decorators: [{
419
+ type: Injectable,
420
+ args: [{ providedIn: 'root' }]
421
+ }], ctorParameters: () => [{ type: i1.Storage }] });
422
+ //# 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 { 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  constructor(private storage: 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"]}