valtech-components 2.0.686 → 2.0.688

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.
@@ -116,6 +116,14 @@ export class TypedCollection {
116
116
  const results = await this.firestore.getDocs(this.collectionPath, queryOptions);
117
117
  return results.length;
118
118
  }
119
+ /**
120
+ * Obtiene todos los documentos (one-time fetch sin listener).
121
+ * A diferencia de watchAll(), no mantiene subscripciones activas.
122
+ */
123
+ async getAllOnce(options) {
124
+ const queryOptions = this.applyDefaultFilters(options);
125
+ return this.firestore.getDocs(this.collectionPath, queryOptions);
126
+ }
119
127
  /**
120
128
  * Verifica si un documento existe.
121
129
  */
@@ -251,4 +259,4 @@ export class TypedCollection {
251
259
  return this.collectionPath;
252
260
  }
253
261
  }
254
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore-collection.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore-collection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firestore Collection Factory\n *\n * Patrón factory para crear instancias de colección tipadas.\n * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.\n */\n\nimport { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\n\nimport { FirestoreService } from './firestore.service';\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\n\n/**\n * Opciones de configuración para una colección.\n */\nexport interface CollectionOptions {\n  /**\n   * Si true, usa soft delete (marca deletedAt en lugar de eliminar).\n   * Default: false\n   */\n  softDelete?: boolean;\n\n  /**\n   * Si true, maneja automáticamente createdAt/updatedAt.\n   * Default: true\n   */\n  timestamps?: boolean;\n}\n\n/**\n * Referencia a una sub-colección tipada.\n */\nexport interface SubCollectionRef<T extends FirestoreDocument> {\n  getById(id: string): Promise<T | null>;\n  getAll(options?: QueryOptions): Promise<T[]>;\n  watch(id: string): Observable<T | null>;\n  watchAll(options?: QueryOptions): Observable<T[]>;\n  create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T>;\n  update(id: string, data: Partial<T>): Promise<void>;\n  delete(id: string): Promise<void>;\n}\n\n/**\n * Factory para crear instancias de colección tipadas.\n *\n * @example\n * ```typescript\n * @Injectable({ providedIn: 'root' })\n * export class UsersService {\n *   private users = inject(FirestoreCollectionFactory).create<User>('users');\n *\n *   getAll = () => this.users.getAll();\n *   getById = (id: string) => this.users.getById(id);\n *   create = (data: Omit<User, 'id'>) => this.users.create(data);\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.users.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreCollectionFactory {\n  constructor(private firestore: FirestoreService) {}\n\n  /**\n   * Crea una instancia de colección tipada.\n   *\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   * @returns Instancia de TypedCollection\n   */\n  create<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: CollectionOptions\n  ): TypedCollection<T> {\n    return new TypedCollection<T>(this.firestore, collectionPath, options);\n  }\n}\n\n/**\n * Colección tipada con métodos CRUD.\n *\n * NO usa inject() - recibe FirestoreService por constructor.\n * Esto evita el error NG0203.\n */\nexport class TypedCollection<T extends FirestoreDocument> {\n  private readonly options: CollectionOptions;\n\n  constructor(\n    private firestore: FirestoreService,\n    private collectionPath: string,\n    options: CollectionOptions = {}\n  ) {\n    this.options = {\n      softDelete: false,\n      timestamps: true,\n      ...options,\n    };\n  }\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID.\n   */\n  async getById(id: string): Promise<T | null> {\n    return this.firestore.getDoc<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Obtiene todos los documentos de la colección.\n   */\n  async getAll(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Ejecuta una query personalizada.\n   */\n  async query(options: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene documentos con paginación.\n   */\n  async paginate(options: QueryOptions & { limit: number }): Promise<PaginatedResult<T>> {\n    const queryOptions = this.applyDefaultFilters(options) as QueryOptions & { limit: number };\n    return this.firestore.getPaginated<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene el primer documento que coincida con la query.\n   */\n  async getFirst(options?: QueryOptions): Promise<T | null> {\n    const queryOptions = this.applyDefaultFilters({\n      ...options,\n      limit: 1,\n    });\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results[0] ?? null;\n  }\n\n  /**\n   * Cuenta los documentos que coinciden con la query.\n   * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.\n   */\n  async count(options?: QueryOptions): Promise<number> {\n    const queryOptions = this.applyDefaultFilters(options);\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results.length;\n  }\n\n  /**\n   * Verifica si un documento existe.\n   */\n  async exists(id: string): Promise<boolean> {\n    return this.firestore.exists(this.collectionPath, id);\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento.\n   */\n  watch(id: string): Observable<T | null> {\n    return this.firestore.docChanges<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Suscribe a cambios de la colección.\n   */\n  watchAll(options?: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Suscribe a una query personalizada.\n   */\n  watchQuery(options: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Crea un nuevo documento con ID auto-generado.\n   */\n  async create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T> {\n    return this.firestore.addDoc<T>(this.collectionPath, data);\n  }\n\n  /**\n   * Crea un documento con ID específico.\n   */\n  async createWithId(id: string, data: Omit<T, 'id'>): Promise<void> {\n    return this.firestore.setDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Actualiza campos de un documento.\n   */\n  async update(id: string, data: Partial<Omit<T, 'id' | 'createdAt'>>): Promise<void> {\n    return this.firestore.updateDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Elimina un documento.\n   * Si softDelete está habilitado, marca como eliminado en lugar de borrar.\n   */\n  async delete(id: string): Promise<void> {\n    if (this.options.softDelete) {\n      return this.firestore.updateDoc<T>(this.collectionPath, id, {\n        deletedAt: new Date(),\n      } as unknown as Partial<T>);\n    }\n    return this.firestore.deleteDoc(this.collectionPath, id);\n  }\n\n  /**\n   * Restaura un documento soft-deleted.\n   */\n  async restore(id: string): Promise<void> {\n    if (!this.options.softDelete) {\n      throw new Error('Soft delete no está habilitado para esta colección');\n    }\n    return this.firestore.updateDoc<T>(this.collectionPath, id, {\n      deletedAt: null,\n    } as unknown as Partial<T>);\n  }\n\n  // ===========================================================================\n  // SUB-COLECCIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene una referencia a una sub-colección.\n   *\n   * @example\n   * ```typescript\n   * // En UsersService\n   * getUserDocuments(userId: string) {\n   *   return this.users.subcollection<Document>(userId, 'documents');\n   * }\n   *\n   * // Uso\n   * const docs = await users.getUserDocuments('user123').getAll();\n   * ```\n   */\n  subcollection<S extends FirestoreDocument>(\n    parentId: string,\n    subcollectionName: string\n  ): SubCollectionRef<S> {\n    const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;\n\n    return {\n      getById: (id: string) => this.firestore.getDoc<S>(subPath, id),\n      getAll: (options?: QueryOptions) => this.firestore.getDocs<S>(subPath, options),\n      watch: (id: string) => this.firestore.docChanges<S>(subPath, id),\n      watchAll: (options?: QueryOptions) => this.firestore.collectionChanges<S>(subPath, options),\n      create: (data: Omit<S, 'id' | 'createdAt' | 'updatedAt'>) =>\n        this.firestore.addDoc<S>(subPath, data),\n      update: (id: string, data: Partial<S>) => this.firestore.updateDoc<S>(subPath, id, data),\n      delete: (id: string) => this.firestore.deleteDoc(subPath, id),\n    };\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   */\n  private applyDefaultFilters(options?: QueryOptions): QueryOptions {\n    if (!this.options.softDelete) {\n      return options ?? {};\n    }\n\n    // Excluir documentos soft-deleted por defecto\n    const whereClause = { field: 'deletedAt', operator: '==' as const, value: null };\n\n    return {\n      ...options,\n      where: [...(options?.where ?? []), whereClause],\n    };\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Genera un nuevo ID sin crear el documento.\n   */\n  generateId(): string {\n    return this.firestore.generateId(this.collectionPath);\n  }\n\n  /**\n   * Obtiene la ruta de la colección.\n   */\n  getPath(): string {\n    return this.collectionPath;\n  }\n}\n\n/**\n * @deprecated Use FirestoreCollectionFactory.create() instead.\n * Type alias for backwards compatibility.\n */\nexport type FirestoreCollection<T extends FirestoreDocument> = TypedCollection<T>;\n"]}
262
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore-collection.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore-collection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firestore Collection Factory\n *\n * Patrón factory para crear instancias de colección tipadas.\n * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.\n */\n\nimport { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\n\nimport { FirestoreService } from './firestore.service';\nimport { FirestoreDocument, PaginatedResult, QueryOptions } from './types';\n\n/**\n * Opciones de configuración para una colección.\n */\nexport interface CollectionOptions {\n  /**\n   * Si true, usa soft delete (marca deletedAt en lugar de eliminar).\n   * Default: false\n   */\n  softDelete?: boolean;\n\n  /**\n   * Si true, maneja automáticamente createdAt/updatedAt.\n   * Default: true\n   */\n  timestamps?: boolean;\n}\n\n/**\n * Referencia a una sub-colección tipada.\n */\nexport interface SubCollectionRef<T extends FirestoreDocument> {\n  getById(id: string): Promise<T | null>;\n  getAll(options?: QueryOptions): Promise<T[]>;\n  watch(id: string): Observable<T | null>;\n  watchAll(options?: QueryOptions): Observable<T[]>;\n  create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T>;\n  update(id: string, data: Partial<T>): Promise<void>;\n  delete(id: string): Promise<void>;\n}\n\n/**\n * Factory para crear instancias de colección tipadas.\n *\n * @example\n * ```typescript\n * @Injectable({ providedIn: 'root' })\n * export class UsersService {\n *   private users = inject(FirestoreCollectionFactory).create<User>('users');\n *\n *   getAll = () => this.users.getAll();\n *   getById = (id: string) => this.users.getById(id);\n *   create = (data: Omit<User, 'id'>) => this.users.create(data);\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.users.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirestoreCollectionFactory {\n  constructor(private firestore: FirestoreService) {}\n\n  /**\n   * Crea una instancia de colección tipada.\n   *\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   * @returns Instancia de TypedCollection\n   */\n  create<T extends FirestoreDocument>(\n    collectionPath: string,\n    options?: CollectionOptions\n  ): TypedCollection<T> {\n    return new TypedCollection<T>(this.firestore, collectionPath, options);\n  }\n}\n\n/**\n * Colección tipada con métodos CRUD.\n *\n * NO usa inject() - recibe FirestoreService por constructor.\n * Esto evita el error NG0203.\n */\nexport class TypedCollection<T extends FirestoreDocument> {\n  private readonly options: CollectionOptions;\n\n  constructor(\n    private firestore: FirestoreService,\n    private collectionPath: string,\n    options: CollectionOptions = {}\n  ) {\n    this.options = {\n      softDelete: false,\n      timestamps: true,\n      ...options,\n    };\n  }\n\n  // ===========================================================================\n  // LECTURAS ONE-TIME\n  // ===========================================================================\n\n  /**\n   * Obtiene un documento por ID.\n   */\n  async getById(id: string): Promise<T | null> {\n    return this.firestore.getDoc<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Obtiene todos los documentos de la colección.\n   */\n  async getAll(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Ejecuta una query personalizada.\n   */\n  async query(options: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene documentos con paginación.\n   */\n  async paginate(options: QueryOptions & { limit: number }): Promise<PaginatedResult<T>> {\n    const queryOptions = this.applyDefaultFilters(options) as QueryOptions & { limit: number };\n    return this.firestore.getPaginated<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Obtiene el primer documento que coincida con la query.\n   */\n  async getFirst(options?: QueryOptions): Promise<T | null> {\n    const queryOptions = this.applyDefaultFilters({\n      ...options,\n      limit: 1,\n    });\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results[0] ?? null;\n  }\n\n  /**\n   * Cuenta los documentos que coinciden con la query.\n   * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.\n   */\n  async count(options?: QueryOptions): Promise<number> {\n    const queryOptions = this.applyDefaultFilters(options);\n    const results = await this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n    return results.length;\n  }\n\n  /**\n   * Obtiene todos los documentos (one-time fetch sin listener).\n   * A diferencia de watchAll(), no mantiene subscripciones activas.\n   */\n  async getAllOnce(options?: QueryOptions): Promise<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.getDocs<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Verifica si un documento existe.\n   */\n  async exists(id: string): Promise<boolean> {\n    return this.firestore.exists(this.collectionPath, id);\n  }\n\n  // ===========================================================================\n  // SUBSCRIPCIONES REAL-TIME\n  // ===========================================================================\n\n  /**\n   * Suscribe a cambios de un documento.\n   */\n  watch(id: string): Observable<T | null> {\n    return this.firestore.docChanges<T>(this.collectionPath, id);\n  }\n\n  /**\n   * Suscribe a cambios de la colección.\n   */\n  watchAll(options?: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  /**\n   * Suscribe a una query personalizada.\n   */\n  watchQuery(options: QueryOptions): Observable<T[]> {\n    const queryOptions = this.applyDefaultFilters(options);\n    return this.firestore.collectionChanges<T>(this.collectionPath, queryOptions);\n  }\n\n  // ===========================================================================\n  // ESCRITURA\n  // ===========================================================================\n\n  /**\n   * Crea un nuevo documento con ID auto-generado.\n   */\n  async create(data: Omit<T, 'id' | 'createdAt' | 'updatedAt'>): Promise<T> {\n    return this.firestore.addDoc<T>(this.collectionPath, data);\n  }\n\n  /**\n   * Crea un documento con ID específico.\n   */\n  async createWithId(id: string, data: Omit<T, 'id'>): Promise<void> {\n    return this.firestore.setDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Actualiza campos de un documento.\n   */\n  async update(id: string, data: Partial<Omit<T, 'id' | 'createdAt'>>): Promise<void> {\n    return this.firestore.updateDoc<T>(this.collectionPath, id, data);\n  }\n\n  /**\n   * Elimina un documento.\n   * Si softDelete está habilitado, marca como eliminado en lugar de borrar.\n   */\n  async delete(id: string): Promise<void> {\n    if (this.options.softDelete) {\n      return this.firestore.updateDoc<T>(this.collectionPath, id, {\n        deletedAt: new Date(),\n      } as unknown as Partial<T>);\n    }\n    return this.firestore.deleteDoc(this.collectionPath, id);\n  }\n\n  /**\n   * Restaura un documento soft-deleted.\n   */\n  async restore(id: string): Promise<void> {\n    if (!this.options.softDelete) {\n      throw new Error('Soft delete no está habilitado para esta colección');\n    }\n    return this.firestore.updateDoc<T>(this.collectionPath, id, {\n      deletedAt: null,\n    } as unknown as Partial<T>);\n  }\n\n  // ===========================================================================\n  // SUB-COLECCIONES\n  // ===========================================================================\n\n  /**\n   * Obtiene una referencia a una sub-colección.\n   *\n   * @example\n   * ```typescript\n   * // En UsersService\n   * getUserDocuments(userId: string) {\n   *   return this.users.subcollection<Document>(userId, 'documents');\n   * }\n   *\n   * // Uso\n   * const docs = await users.getUserDocuments('user123').getAll();\n   * ```\n   */\n  subcollection<S extends FirestoreDocument>(\n    parentId: string,\n    subcollectionName: string\n  ): SubCollectionRef<S> {\n    const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;\n\n    return {\n      getById: (id: string) => this.firestore.getDoc<S>(subPath, id),\n      getAll: (options?: QueryOptions) => this.firestore.getDocs<S>(subPath, options),\n      watch: (id: string) => this.firestore.docChanges<S>(subPath, id),\n      watchAll: (options?: QueryOptions) => this.firestore.collectionChanges<S>(subPath, options),\n      create: (data: Omit<S, 'id' | 'createdAt' | 'updatedAt'>) =>\n        this.firestore.addDoc<S>(subPath, data),\n      update: (id: string, data: Partial<S>) => this.firestore.updateDoc<S>(subPath, id, data),\n      delete: (id: string) => this.firestore.deleteDoc(subPath, id),\n    };\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   */\n  private applyDefaultFilters(options?: QueryOptions): QueryOptions {\n    if (!this.options.softDelete) {\n      return options ?? {};\n    }\n\n    // Excluir documentos soft-deleted por defecto\n    const whereClause = { field: 'deletedAt', operator: '==' as const, value: null };\n\n    return {\n      ...options,\n      where: [...(options?.where ?? []), whereClause],\n    };\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Genera un nuevo ID sin crear el documento.\n   */\n  generateId(): string {\n    return this.firestore.generateId(this.collectionPath);\n  }\n\n  /**\n   * Obtiene la ruta de la colección.\n   */\n  getPath(): string {\n    return this.collectionPath;\n  }\n}\n\n/**\n * @deprecated Use FirestoreCollectionFactory.create() instead.\n * Type alias for backwards compatibility.\n */\nexport type FirestoreCollection<T extends FirestoreDocument> = TypedCollection<T>;\n"]}
@@ -119,6 +119,17 @@ export class NotificationsService {
119
119
  orderBy: [{ field: 'createdAt', direction: 'desc' }],
120
120
  });
121
121
  }
122
+ /**
123
+ * Obtiene todas las notificaciones (one-time fetch sin listener).
124
+ * Útil para cargas iniciales sin necesidad de updates en tiempo real.
125
+ */
126
+ async getAllOnce() {
127
+ if (!this.collection)
128
+ return [];
129
+ return this.collection.getAllOnce({
130
+ orderBy: [{ field: 'createdAt', direction: 'desc' }],
131
+ });
132
+ }
122
133
  /**
123
134
  * Obtiene solo notificaciones no leídas.
124
135
  */
@@ -196,4 +207,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
196
207
  }], ctorParameters: () => [{ type: i0.Injector }, { type: i1.FirestoreCollectionFactory, decorators: [{
197
208
  type: Optional
198
209
  }] }] });
199
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notifications.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/notifications.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Notifications Service\n *\n * Servicio para leer notificaciones desde Firestore.\n * El backend escribe las notificaciones, el frontend solo las lee y actualiza estado.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * También puede inicializarse manualmente con `initialize(userId)`.\n */\n\nimport { effect, Injectable, Injector, Optional, runInInjectionContext } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { map } from 'rxjs/operators';\n\nimport { AuthService } from '../auth/auth.service';\nimport { FirestoreCollectionFactory, TypedCollection } from './firestore-collection';\nimport { FirestoreDocument } from './types';\n\n/**\n * Documento de notificación en Firestore.\n * Estructura escrita por el backend.\n */\nexport interface NotificationDocument extends FirestoreDocument {\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 (ej: route, actionType) */\n  data?: Record<string, string>;\n  /** Tipo de notificación */\n  type?: 'fcm' | 'system' | 'reminder' | string;\n  /** Si la notificación fue leída */\n  isRead: boolean;\n}\n\n/**\n * Servicio para leer notificaciones desde Firestore.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * No requiere llamar a `initialize()` manualmente si AuthService está configurado.\n *\n * @example\n * ```typescript\n * // Con AuthService configurado: auto-inicialización\n * // Solo inyectar y usar directamente\n * private notifications = inject(NotificationsService);\n *\n * notifications$ = this.notifications.getAll();\n * unreadCount$ = this.notifications.getUnreadCount();\n *\n * // Sin AuthService: inicialización manual\n * this.notifications.initialize(userId);\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class NotificationsService {\n  private collection: TypedCollection<NotificationDocument> | null = null;\n  private currentUserId: string | null = null;\n\n  // Inyección opcional - AuthService puede no estar configurado\n  private authService: AuthService | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Optional() private collectionFactory: FirestoreCollectionFactory\n  ) {\n    // Intentar obtener AuthService de forma lazy (puede no estar configurado)\n    this.setupAutoInitialization();\n  }\n\n  /**\n   * Configura auto-inicialización observando el estado de AuthService.\n   * Se ejecuta en el contexto del injector para poder usar effect().\n   */\n  private setupAutoInitialization(): void {\n    // Obtener AuthService de forma lazy\n    try {\n      this.authService = this.injector.get(AuthService, null);\n    } catch {\n      // AuthService no está configurado, no hay auto-inicialización\n      return;\n    }\n\n    if (!this.authService) {\n      return;\n    }\n\n    // Usar runInInjectionContext para poder crear el effect\n    runInInjectionContext(this.injector, () => {\n      effect(() => {\n        const user = this.authService!.user();\n\n        if (user?.userId && this.currentUserId !== user.userId) {\n          // Usuario autenticado: inicializar con su ID\n          this.initialize(user.userId);\n        } else if (!user && this.currentUserId) {\n          // Logout: limpiar estado\n          this.reset();\n        }\n      });\n    });\n  }\n\n  /**\n   * Inicializa el servicio para un usuario específico.\n   *\n   * NOTA: Se llama automáticamente si AuthService está configurado.\n   * Solo usar manualmente si AuthService no está disponible o se necesita\n   * un userId diferente al del usuario autenticado.\n   */\n  initialize(userId: string): void {\n    if (!this.collectionFactory) {\n      console.warn('[Notifications] FirestoreCollectionFactory not available. Ensure provideValtechFirebase() is configured.');\n      return;\n    }\n\n    this.currentUserId = userId;\n\n    // Path relativo - FirestoreService agrega automáticamente apps/{appId}/\n    // NO agregar apps/ aquí para evitar doble prefijo\n    const basePath = `users/${userId}/notifications`;\n\n    this.collection = this.collectionFactory.create<NotificationDocument>(\n      basePath,\n      { timestamps: true }\n    );\n\n    console.log('[Notifications] Initialized with path:', basePath);\n  }\n\n  /**\n   * Verifica si el servicio está inicializado.\n   */\n  get isReady(): boolean {\n    return this.collection !== null;\n  }\n\n  /**\n   * Obtiene el ID del usuario actual.\n   */\n  get userId(): string | null {\n    return this.currentUserId;\n  }\n\n  // ===========================================================================\n  // LECTURA\n  // ===========================================================================\n\n  /**\n   * Obtiene todas las notificaciones ordenadas por fecha descendente.\n   * Real-time: se actualiza automáticamente cuando cambian los datos.\n   */\n  getAll(): Observable<NotificationDocument[]> {\n    if (!this.collection) return of([]);\n\n    return this.collection.watchAll({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n    });\n  }\n\n  /**\n   * Obtiene solo notificaciones no leídas.\n   */\n  getUnread(): Observable<NotificationDocument[]> {\n    return this.getAll().pipe(\n      map(notifications => notifications.filter(n => !n.isRead))\n    );\n  }\n\n  /**\n   * Cuenta notificaciones no leídas.\n   * Útil para badges en UI.\n   */\n  getUnreadCount(): Observable<number> {\n    return this.getUnread().pipe(map(n => n.length));\n  }\n\n  /**\n   * Obtiene una notificación por ID.\n   */\n  async getById(notificationId: string): Promise<NotificationDocument | null> {\n    if (!this.collection) return null;\n    return this.collection.getById(notificationId);\n  }\n\n  // ===========================================================================\n  // ACCIONES\n  // ===========================================================================\n\n  /**\n   * Marca una notificación como leída.\n   */\n  async markAsRead(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.update(notificationId, { isRead: true });\n  }\n\n  /**\n   * Marca todas las notificaciones como leídas.\n   */\n  async markAllAsRead(): Promise<void> {\n    if (!this.collection) return;\n\n    const unread = await this.collection.query({\n      where: [{ field: 'isRead', operator: '==', value: false }],\n    });\n\n    await Promise.all(\n      unread.map(n => this.collection!.update(n.id!, { isRead: true }))\n    );\n  }\n\n  /**\n   * Elimina una notificación.\n   */\n  async delete(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.delete(notificationId);\n  }\n\n  /**\n   * Elimina todas las notificaciones del usuario.\n   */\n  async deleteAll(): Promise<void> {\n    if (!this.collection) return;\n\n    const all = await this.collection.getAll();\n    await Promise.all(all.map(n => this.collection!.delete(n.id!)));\n  }\n\n  /**\n   * Limpia el estado del servicio.\n   * Útil para logout.\n   */\n  reset(): void {\n    this.collection = null;\n    this.currentUserId = null;\n  }\n}\n"]}
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notifications.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/notifications.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Notifications Service\n *\n * Servicio para leer notificaciones desde Firestore.\n * El backend escribe las notificaciones, el frontend solo las lee y actualiza estado.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * También puede inicializarse manualmente con `initialize(userId)`.\n */\n\nimport { effect, Injectable, Injector, Optional, runInInjectionContext } from '@angular/core';\nimport { Observable, of } from 'rxjs';\nimport { map } from 'rxjs/operators';\n\nimport { AuthService } from '../auth/auth.service';\nimport { FirestoreCollectionFactory, TypedCollection } from './firestore-collection';\nimport { FirestoreDocument } from './types';\n\n/**\n * Documento de notificación en Firestore.\n * Estructura escrita por el backend.\n */\nexport interface NotificationDocument extends FirestoreDocument {\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 (ej: route, actionType) */\n  data?: Record<string, string>;\n  /** Tipo de notificación */\n  type?: 'fcm' | 'system' | 'reminder' | string;\n  /** Si la notificación fue leída */\n  isRead: boolean;\n}\n\n/**\n * Servicio para leer notificaciones desde Firestore.\n *\n * Se auto-inicializa cuando AuthService tiene un usuario autenticado.\n * No requiere llamar a `initialize()` manualmente si AuthService está configurado.\n *\n * @example\n * ```typescript\n * // Con AuthService configurado: auto-inicialización\n * // Solo inyectar y usar directamente\n * private notifications = inject(NotificationsService);\n *\n * notifications$ = this.notifications.getAll();\n * unreadCount$ = this.notifications.getUnreadCount();\n *\n * // Sin AuthService: inicialización manual\n * this.notifications.initialize(userId);\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class NotificationsService {\n  private collection: TypedCollection<NotificationDocument> | null = null;\n  private currentUserId: string | null = null;\n\n  // Inyección opcional - AuthService puede no estar configurado\n  private authService: AuthService | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Optional() private collectionFactory: FirestoreCollectionFactory\n  ) {\n    // Intentar obtener AuthService de forma lazy (puede no estar configurado)\n    this.setupAutoInitialization();\n  }\n\n  /**\n   * Configura auto-inicialización observando el estado de AuthService.\n   * Se ejecuta en el contexto del injector para poder usar effect().\n   */\n  private setupAutoInitialization(): void {\n    // Obtener AuthService de forma lazy\n    try {\n      this.authService = this.injector.get(AuthService, null);\n    } catch {\n      // AuthService no está configurado, no hay auto-inicialización\n      return;\n    }\n\n    if (!this.authService) {\n      return;\n    }\n\n    // Usar runInInjectionContext para poder crear el effect\n    runInInjectionContext(this.injector, () => {\n      effect(() => {\n        const user = this.authService!.user();\n\n        if (user?.userId && this.currentUserId !== user.userId) {\n          // Usuario autenticado: inicializar con su ID\n          this.initialize(user.userId);\n        } else if (!user && this.currentUserId) {\n          // Logout: limpiar estado\n          this.reset();\n        }\n      });\n    });\n  }\n\n  /**\n   * Inicializa el servicio para un usuario específico.\n   *\n   * NOTA: Se llama automáticamente si AuthService está configurado.\n   * Solo usar manualmente si AuthService no está disponible o se necesita\n   * un userId diferente al del usuario autenticado.\n   */\n  initialize(userId: string): void {\n    if (!this.collectionFactory) {\n      console.warn('[Notifications] FirestoreCollectionFactory not available. Ensure provideValtechFirebase() is configured.');\n      return;\n    }\n\n    this.currentUserId = userId;\n\n    // Path relativo - FirestoreService agrega automáticamente apps/{appId}/\n    // NO agregar apps/ aquí para evitar doble prefijo\n    const basePath = `users/${userId}/notifications`;\n\n    this.collection = this.collectionFactory.create<NotificationDocument>(\n      basePath,\n      { timestamps: true }\n    );\n\n    console.log('[Notifications] Initialized with path:', basePath);\n  }\n\n  /**\n   * Verifica si el servicio está inicializado.\n   */\n  get isReady(): boolean {\n    return this.collection !== null;\n  }\n\n  /**\n   * Obtiene el ID del usuario actual.\n   */\n  get userId(): string | null {\n    return this.currentUserId;\n  }\n\n  // ===========================================================================\n  // LECTURA\n  // ===========================================================================\n\n  /**\n   * Obtiene todas las notificaciones ordenadas por fecha descendente.\n   * Real-time: se actualiza automáticamente cuando cambian los datos.\n   */\n  getAll(): Observable<NotificationDocument[]> {\n    if (!this.collection) return of([]);\n\n    return this.collection.watchAll({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n    });\n  }\n\n  /**\n   * Obtiene todas las notificaciones (one-time fetch sin listener).\n   * Útil para cargas iniciales sin necesidad de updates en tiempo real.\n   */\n  async getAllOnce(): Promise<NotificationDocument[]> {\n    if (!this.collection) return [];\n\n    return this.collection.getAllOnce({\n      orderBy: [{ field: 'createdAt', direction: 'desc' }],\n    });\n  }\n\n  /**\n   * Obtiene solo notificaciones no leídas.\n   */\n  getUnread(): Observable<NotificationDocument[]> {\n    return this.getAll().pipe(\n      map(notifications => notifications.filter(n => !n.isRead))\n    );\n  }\n\n  /**\n   * Cuenta notificaciones no leídas.\n   * Útil para badges en UI.\n   */\n  getUnreadCount(): Observable<number> {\n    return this.getUnread().pipe(map(n => n.length));\n  }\n\n  /**\n   * Obtiene una notificación por ID.\n   */\n  async getById(notificationId: string): Promise<NotificationDocument | null> {\n    if (!this.collection) return null;\n    return this.collection.getById(notificationId);\n  }\n\n  // ===========================================================================\n  // ACCIONES\n  // ===========================================================================\n\n  /**\n   * Marca una notificación como leída.\n   */\n  async markAsRead(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.update(notificationId, { isRead: true });\n  }\n\n  /**\n   * Marca todas las notificaciones como leídas.\n   */\n  async markAllAsRead(): Promise<void> {\n    if (!this.collection) return;\n\n    const unread = await this.collection.query({\n      where: [{ field: 'isRead', operator: '==', value: false }],\n    });\n\n    await Promise.all(\n      unread.map(n => this.collection!.update(n.id!, { isRead: true }))\n    );\n  }\n\n  /**\n   * Elimina una notificación.\n   */\n  async delete(notificationId: string): Promise<void> {\n    if (!this.collection) return;\n    await this.collection.delete(notificationId);\n  }\n\n  /**\n   * Elimina todas las notificaciones del usuario.\n   */\n  async deleteAll(): Promise<void> {\n    if (!this.collection) return;\n\n    const all = await this.collection.getAll();\n    await Promise.all(all.map(n => this.collection!.delete(n.id!)));\n  }\n\n  /**\n   * Limpia el estado del servicio.\n   * Útil para logout.\n   */\n  reset(): void {\n    this.collection = null;\n    this.currentUserId = null;\n  }\n}\n"]}
@@ -45,12 +45,14 @@ export class StorageService {
45
45
  }
46
46
  /**
47
47
  * Prefija el path de storage con el appId si está configurado.
48
- * Si no hay appId, retorna el path sin modificar (backward compatible).
48
+ * Si no hay appId o skipPrefix es true, retorna el path sin modificar.
49
49
  *
50
+ * @param path - Ruta original
51
+ * @param skipPrefix - Si es true, no aplica el prefix (útil para paths compartidos como avatares)
50
52
  * @internal
51
53
  */
52
- prefixStoragePath(path) {
53
- if (!this.config?.appId)
54
+ prefixStoragePath(path, skipPrefix = false) {
55
+ if (skipPrefix || !this.config?.appId)
54
56
  return path;
55
57
  return `${this.config.appId}/${path}`;
56
58
  }
@@ -81,7 +83,7 @@ export class StorageService {
81
83
  * ```
82
84
  */
83
85
  upload(path, file, metadata) {
84
- const prefixedPath = this.prefixStoragePath(path);
86
+ const prefixedPath = this.prefixStoragePath(path, metadata?.skipPrefix);
85
87
  const storageRef = ref(this.storage, prefixedPath);
86
88
  const uploadMetadata = {
87
89
  contentType: metadata?.contentType || (file instanceof File ? file.type : undefined),
@@ -136,7 +138,7 @@ export class StorageService {
136
138
  * ```
137
139
  */
138
140
  async uploadAndGetUrl(path, file, metadata) {
139
- const prefixedPath = this.prefixStoragePath(path);
141
+ const prefixedPath = this.prefixStoragePath(path, metadata?.skipPrefix);
140
142
  return new Promise((resolve, reject) => {
141
143
  this.upload(path, file, metadata).subscribe({
142
144
  complete: async () => {
@@ -467,4 +469,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
467
469
  type: Injectable,
468
470
  args: [{ providedIn: 'root' }]
469
471
  }], ctorParameters: () => [{ type: i1.Storage }] });
470
- //# 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 { VALTECH_FIREBASE_CONFIG } from './config';\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 config = inject(VALTECH_FIREBASE_CONFIG, { optional: true });\n\n  constructor(private storage: Storage) {}\n\n  /**\n   * Prefija el path de storage con el appId si está configurado.\n   * Si no hay appId, retorna el path sin modificar (backward compatible).\n   *\n   * @internal\n   */\n  private prefixStoragePath(path: string): string {\n    if (!this.config?.appId) return path;\n    return `${this.config.appId}/${path}`;\n  }\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 prefixedPath = this.prefixStoragePath(path);\n    const storageRef = ref(this.storage, prefixedPath);\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    const prefixedPath = this.prefixStoragePath(path);\n    return new Promise((resolve, reject) => {\n      this.upload(path, file, metadata).subscribe({\n        complete: async () => {\n          try {\n            const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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  // USER AVATAR HELPERS\n  // ===========================================================================\n\n  /**\n   * Construye la URL de avatar de un usuario.\n   *\n   * Los avatares se almacenan en paths predecibles: `users/{userId}/avatar.jpg`\n   * Esto permite que cualquier app pueda mostrar avatares sin pedir la URL al backend,\n   * solo necesita el userId.\n   *\n   * @param userId - ID del usuario (ULID)\n   * @param type - 'avatar' para imagen completa, 'thumb' para thumbnail\n   * @returns URL directa de Firebase Storage\n   *\n   * @example\n   * ```typescript\n   * // Mostrar avatar de cualquier usuario\n   * const avatarUrl = storage.getUserAvatarUrl('01ABCD...');\n   * const thumbUrl = storage.getUserAvatarUrl('01ABCD...', 'thumb');\n   *\n   * // Con cache-busting\n   * const freshUrl = storage.getUserAvatarUrl(userId) + `&t=${Date.now()}`;\n   * ```\n   */\n  getUserAvatarUrl(userId: string, type: 'avatar' | 'thumb' = 'avatar'): string {\n    const bucket = this.config?.firebase?.storageBucket || 'myvaltech-dev.firebasestorage.app';\n    const path = `users/${userId}/${type}.jpg`;\n    return `https://firebasestorage.googleapis.com/v0/b/${bucket}/o/${encodeURIComponent(path)}?alt=media`;\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"]}
472
+ //# 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 { VALTECH_FIREBASE_CONFIG } from './config';\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 config = inject(VALTECH_FIREBASE_CONFIG, { optional: true });\n\n  constructor(private storage: Storage) {}\n\n  /**\n   * Prefija el path de storage con el appId si está configurado.\n   * Si no hay appId o skipPrefix es true, retorna el path sin modificar.\n   *\n   * @param path - Ruta original\n   * @param skipPrefix - Si es true, no aplica el prefix (útil para paths compartidos como avatares)\n   * @internal\n   */\n  private prefixStoragePath(path: string, skipPrefix = false): string {\n    if (skipPrefix || !this.config?.appId) return path;\n    return `${this.config.appId}/${path}`;\n  }\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 prefixedPath = this.prefixStoragePath(path, metadata?.skipPrefix);\n    const storageRef = ref(this.storage, prefixedPath);\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    const prefixedPath = this.prefixStoragePath(path, metadata?.skipPrefix);\n    return new Promise((resolve, reject) => {\n      this.upload(path, file, metadata).subscribe({\n        complete: async () => {\n          try {\n            const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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 prefixedPath = this.prefixStoragePath(path);\n      const storageRef = ref(this.storage, prefixedPath);\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  // USER AVATAR HELPERS\n  // ===========================================================================\n\n  /**\n   * Construye la URL de avatar de un usuario.\n   *\n   * Los avatares se almacenan en paths predecibles: `users/{userId}/avatar.jpg`\n   * Esto permite que cualquier app pueda mostrar avatares sin pedir la URL al backend,\n   * solo necesita el userId.\n   *\n   * @param userId - ID del usuario (ULID)\n   * @param type - 'avatar' para imagen completa, 'thumb' para thumbnail\n   * @returns URL directa de Firebase Storage\n   *\n   * @example\n   * ```typescript\n   * // Mostrar avatar de cualquier usuario\n   * const avatarUrl = storage.getUserAvatarUrl('01ABCD...');\n   * const thumbUrl = storage.getUserAvatarUrl('01ABCD...', 'thumb');\n   *\n   * // Con cache-busting\n   * const freshUrl = storage.getUserAvatarUrl(userId) + `&t=${Date.now()}`;\n   * ```\n   */\n  getUserAvatarUrl(userId: string, type: 'avatar' | 'thumb' = 'avatar'): string {\n    const bucket = this.config?.firebase?.storageBucket || 'myvaltech-dev.firebasestorage.app';\n    const path = `users/${userId}/${type}.jpg`;\n    return `https://firebasestorage.googleapis.com/v0/b/${bucket}/o/${encodeURIComponent(path)}?alt=media`;\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"]}
@@ -2,5 +2,5 @@
2
2
  * Current version of valtech-components.
3
3
  * This is automatically updated during the publish process.
4
4
  */
5
- export const VERSION = '2.0.684';
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC42ODQnO1xuIl19
5
+ export const VERSION = '2.0.688';
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC42ODgnO1xuIl19