valtech-components 2.0.449 → 2.0.451

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 (39) hide show
  1. package/esm2022/lib/components/atoms/fab/fab.component.mjs +4 -5
  2. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +5 -3
  3. package/esm2022/lib/services/link-processor.service.mjs +43 -61
  4. package/esm2022/lib/services/modal/modal.service.mjs +9 -8
  5. package/esm2022/public-api.mjs +4 -24
  6. package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs +136 -0
  7. package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs.map +1 -0
  8. package/fesm2022/valtech-components.mjs +59 -3093
  9. package/fesm2022/valtech-components.mjs.map +1 -1
  10. package/lib/components/atoms/fab/fab.component.d.ts +1 -2
  11. package/lib/services/modal/modal.service.d.ts +0 -2
  12. package/package.json +1 -3
  13. package/public-api.d.ts +0 -8
  14. package/esm2022/lib/components/organisms/tabbed-content/tabbed-content.component.mjs +0 -170
  15. package/esm2022/lib/components/organisms/tabbed-content/types.mjs +0 -2
  16. package/esm2022/lib/services/firebase/config.mjs +0 -108
  17. package/esm2022/lib/services/firebase/firebase.service.mjs +0 -285
  18. package/esm2022/lib/services/firebase/firestore-collection.mjs +0 -254
  19. package/esm2022/lib/services/firebase/firestore.service.mjs +0 -508
  20. package/esm2022/lib/services/firebase/index.mjs +0 -49
  21. package/esm2022/lib/services/firebase/messaging.service.mjs +0 -503
  22. package/esm2022/lib/services/firebase/shared-config.mjs +0 -138
  23. package/esm2022/lib/services/firebase/storage.service.mjs +0 -421
  24. package/esm2022/lib/services/firebase/types.mjs +0 -8
  25. package/esm2022/lib/services/firebase/utils/path-builder.mjs +0 -195
  26. package/esm2022/lib/services/firebase/utils/query-builder.mjs +0 -302
  27. package/lib/components/organisms/tabbed-content/tabbed-content.component.d.ts +0 -65
  28. package/lib/components/organisms/tabbed-content/types.d.ts +0 -53
  29. package/lib/services/firebase/config.d.ts +0 -49
  30. package/lib/services/firebase/firebase.service.d.ts +0 -140
  31. package/lib/services/firebase/firestore-collection.d.ts +0 -174
  32. package/lib/services/firebase/firestore.service.d.ts +0 -303
  33. package/lib/services/firebase/index.d.ts +0 -39
  34. package/lib/services/firebase/messaging.service.d.ts +0 -254
  35. package/lib/services/firebase/shared-config.d.ts +0 -126
  36. package/lib/services/firebase/storage.service.d.ts +0 -204
  37. package/lib/services/firebase/types.d.ts +0 -281
  38. package/lib/services/firebase/utils/path-builder.d.ts +0 -132
  39. package/lib/services/firebase/utils/query-builder.d.ts +0 -210
@@ -1,285 +0,0 @@
1
- /**
2
- * Firebase Service
3
- *
4
- * Servicio principal para la autenticación con Firebase usando Custom Tokens.
5
- * Permite que usuarios autenticados con tu backend (Cognito, etc.) accedan
6
- * a servicios de Firebase (Firestore, Storage, FCM) de manera segura.
7
- */
8
- import { inject, Injectable } from '@angular/core';
9
- import { Auth, authState, signInWithCustomToken, signOut } from '@angular/fire/auth';
10
- import { BehaviorSubject, distinctUntilChanged, map } from 'rxjs';
11
- import { VALTECH_FIREBASE_CONFIG } from './config';
12
- import * as i0 from "@angular/core";
13
- /**
14
- * Servicio de autenticación de Firebase.
15
- *
16
- * Este servicio NO maneja el login de usuarios directamente.
17
- * En su lugar, trabaja con Custom Tokens generados por tu backend.
18
- *
19
- * @example
20
- * ```typescript
21
- * // Después de autenticarte con tu backend (ej: Cognito)
22
- * @Component({...})
23
- * export class LoginComponent {
24
- * private authService = inject(AuthService); // Tu servicio de auth
25
- * private firebase = inject(FirebaseService); // Este servicio
26
- *
27
- * async login(email: string, password: string) {
28
- * // 1. Autenticar con tu backend
29
- * const response = await this.authService.login(email, password);
30
- *
31
- * // 2. El backend devuelve un Firebase Custom Token
32
- * if (response.firebaseToken) {
33
- * await this.firebase.signInWithCustomToken(response.firebaseToken);
34
- * }
35
- *
36
- * // Ahora el usuario puede acceder a Firestore, Storage, etc.
37
- * }
38
- *
39
- * async logout() {
40
- * await this.authService.logout();
41
- * await this.firebase.signOut();
42
- * }
43
- * }
44
- * ```
45
- */
46
- export class FirebaseService {
47
- constructor() {
48
- this.auth = inject(Auth);
49
- this.config = inject(VALTECH_FIREBASE_CONFIG);
50
- /** Estado interno de la sesión */
51
- this.sessionState = new BehaviorSubject({
52
- user: null,
53
- isAuthenticated: false,
54
- isLoading: true,
55
- error: null,
56
- });
57
- /** Estado actual de la sesión como Observable */
58
- this.state$ = this.sessionState.asObservable();
59
- /** Usuario actual de Firebase como Observable */
60
- this.user$ = authState(this.auth).pipe(map((user) => (user ? this.mapUser(user) : null)), distinctUntilChanged((a, b) => a?.uid === b?.uid));
61
- /** Indica si el usuario está autenticado en Firebase */
62
- this.isAuthenticated$ = this.user$.pipe(map((user) => !!user), distinctUntilChanged());
63
- // Escuchar cambios en el estado de autenticación
64
- authState(this.auth).subscribe({
65
- next: (user) => {
66
- this.sessionState.next({
67
- user: user ? this.mapUser(user) : null,
68
- isAuthenticated: !!user,
69
- isLoading: false,
70
- error: null,
71
- });
72
- },
73
- error: (error) => {
74
- this.sessionState.next({
75
- user: null,
76
- isAuthenticated: false,
77
- isLoading: false,
78
- error,
79
- });
80
- },
81
- });
82
- }
83
- // ===========================================================================
84
- // AUTENTICACIÓN
85
- // ===========================================================================
86
- /**
87
- * Autentica al usuario con un Custom Token generado por el backend.
88
- *
89
- * @param token - Firebase Custom Token generado por tu backend
90
- * @returns UserCredential con la información del usuario
91
- * @throws Error si el token es inválido o expiró
92
- *
93
- * @example
94
- * ```typescript
95
- * // Después de login exitoso con tu backend
96
- * const { firebaseToken } = await backendAuth.login(email, password);
97
- * await firebaseService.signInWithCustomToken(firebaseToken);
98
- * ```
99
- */
100
- async signInWithCustomToken(token) {
101
- try {
102
- const credential = await signInWithCustomToken(this.auth, token);
103
- return credential;
104
- }
105
- catch (error) {
106
- const message = this.getErrorMessage(error);
107
- throw new Error(message);
108
- }
109
- }
110
- /**
111
- * Cierra la sesión de Firebase.
112
- * Llamar junto con el logout de tu sistema de autenticación principal.
113
- *
114
- * @example
115
- * ```typescript
116
- * async logout() {
117
- * await this.backendAuth.logout(); // Tu auth
118
- * await this.firebase.signOut(); // Firebase
119
- * }
120
- * ```
121
- */
122
- async signOut() {
123
- try {
124
- await signOut(this.auth);
125
- }
126
- catch (error) {
127
- const message = this.getErrorMessage(error);
128
- throw new Error(message);
129
- }
130
- }
131
- // ===========================================================================
132
- // GETTERS SÍNCRONOS
133
- // ===========================================================================
134
- /**
135
- * Obtiene el usuario actual de Firebase (síncrono).
136
- * Retorna null si no hay usuario autenticado.
137
- */
138
- get currentUser() {
139
- const user = this.auth.currentUser;
140
- return user ? this.mapUser(user) : null;
141
- }
142
- /**
143
- * Obtiene el UID del usuario actual.
144
- * Retorna null si no hay usuario autenticado.
145
- */
146
- get uid() {
147
- return this.auth.currentUser?.uid ?? null;
148
- }
149
- /**
150
- * Indica si hay un usuario autenticado actualmente.
151
- */
152
- get isAuthenticated() {
153
- return !!this.auth.currentUser;
154
- }
155
- // ===========================================================================
156
- // TOKENS
157
- // ===========================================================================
158
- /**
159
- * Obtiene el ID Token de Firebase para el usuario actual.
160
- * Útil para validar el usuario en tu backend.
161
- *
162
- * @param forceRefresh - Si true, fuerza la renovación del token
163
- * @returns ID Token o null si no hay usuario
164
- */
165
- async getIdToken(forceRefresh = false) {
166
- const user = this.auth.currentUser;
167
- if (!user)
168
- return null;
169
- try {
170
- return await user.getIdToken(forceRefresh);
171
- }
172
- catch {
173
- return null;
174
- }
175
- }
176
- /**
177
- * Obtiene los claims personalizados del token del usuario.
178
- * Los claims son establecidos por tu backend al crear el Custom Token.
179
- *
180
- * @returns Objeto con los claims o vacío si no hay usuario
181
- */
182
- async getClaims() {
183
- const user = this.auth.currentUser;
184
- if (!user)
185
- return {};
186
- try {
187
- const result = await user.getIdTokenResult();
188
- return result.claims;
189
- }
190
- catch {
191
- return {};
192
- }
193
- }
194
- /**
195
- * Verifica si el usuario tiene un rol específico.
196
- * El rol debe estar definido en los claims del Custom Token.
197
- *
198
- * @param role - Nombre del rol a verificar
199
- * @returns true si el usuario tiene el rol
200
- */
201
- async hasRole(role) {
202
- const claims = await this.getClaims();
203
- return claims['role'] === role || (Array.isArray(claims['roles']) && claims['roles'].includes(role));
204
- }
205
- // ===========================================================================
206
- // UTILIDADES
207
- // ===========================================================================
208
- /**
209
- * Espera a que el estado de autenticación esté determinado.
210
- * Útil en guards o al inicializar la app.
211
- *
212
- * @returns Usuario actual o null
213
- */
214
- waitForAuth() {
215
- return new Promise((resolve) => {
216
- const subscription = this.state$.subscribe((state) => {
217
- if (!state.isLoading) {
218
- subscription.unsubscribe();
219
- resolve(state.user);
220
- }
221
- });
222
- });
223
- }
224
- /**
225
- * Obtiene la configuración actual de Firebase.
226
- */
227
- getConfig() {
228
- return this.config;
229
- }
230
- /**
231
- * Indica si los emuladores están habilitados.
232
- */
233
- isUsingEmulators() {
234
- return !!(this.config.emulator?.firestore || this.config.emulator?.auth || this.config.emulator?.storage);
235
- }
236
- // ===========================================================================
237
- // MÉTODOS PRIVADOS
238
- // ===========================================================================
239
- /**
240
- * Mapea un User de Firebase a nuestra interface FirebaseUser
241
- */
242
- mapUser(user) {
243
- return {
244
- uid: user.uid,
245
- email: user.email,
246
- displayName: user.displayName,
247
- photoURL: user.photoURL,
248
- emailVerified: user.emailVerified,
249
- isAnonymous: user.isAnonymous,
250
- providerId: user.providerId,
251
- };
252
- }
253
- /**
254
- * Convierte errores de Firebase a mensajes en español
255
- */
256
- getErrorMessage(error) {
257
- if (error instanceof Error) {
258
- const code = error.code;
259
- switch (code) {
260
- case 'auth/invalid-custom-token':
261
- return 'Token de autenticación inválido';
262
- case 'auth/custom-token-mismatch':
263
- return 'El token no corresponde a este proyecto';
264
- case 'auth/network-request-failed':
265
- return 'Error de conexión. Verifica tu conexión a internet';
266
- case 'auth/too-many-requests':
267
- return 'Demasiados intentos. Intenta de nuevo más tarde';
268
- case 'auth/user-disabled':
269
- return 'Esta cuenta ha sido deshabilitada';
270
- case 'auth/user-not-found':
271
- return 'Usuario no encontrado';
272
- default:
273
- return error.message || 'Error de autenticación desconocido';
274
- }
275
- }
276
- return 'Error de autenticación desconocido';
277
- }
278
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
279
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, providedIn: 'root' }); }
280
- }
281
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirebaseService, decorators: [{
282
- type: Injectable,
283
- args: [{ providedIn: 'root' }]
284
- }], ctorParameters: () => [] });
285
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firebase.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firebase.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firebase Service\n *\n * Servicio principal para la autenticación con Firebase usando Custom Tokens.\n * Permite que usuarios autenticados con tu backend (Cognito, etc.) accedan\n * a servicios de Firebase (Firestore, Storage, FCM) de manera segura.\n */\n\nimport { inject, Injectable } from '@angular/core';\nimport { Auth, authState, signInWithCustomToken, signOut, User, UserCredential } from '@angular/fire/auth';\nimport { BehaviorSubject, distinctUntilChanged, map, Observable } from 'rxjs';\n\nimport { VALTECH_FIREBASE_CONFIG } from './config';\nimport { FirebaseUser, SessionState, ValtechFirebaseConfig } from './types';\n\n/**\n * Servicio de autenticación de Firebase.\n *\n * Este servicio NO maneja el login de usuarios directamente.\n * En su lugar, trabaja con Custom Tokens generados por tu backend.\n *\n * @example\n * ```typescript\n * // Después de autenticarte con tu backend (ej: Cognito)\n * @Component({...})\n * export class LoginComponent {\n *   private authService = inject(AuthService);     // Tu servicio de auth\n *   private firebase = inject(FirebaseService);    // Este servicio\n *\n *   async login(email: string, password: string) {\n *     // 1. Autenticar con tu backend\n *     const response = await this.authService.login(email, password);\n *\n *     // 2. El backend devuelve un Firebase Custom Token\n *     if (response.firebaseToken) {\n *       await this.firebase.signInWithCustomToken(response.firebaseToken);\n *     }\n *\n *     // Ahora el usuario puede acceder a Firestore, Storage, etc.\n *   }\n *\n *   async logout() {\n *     await this.authService.logout();\n *     await this.firebase.signOut();\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FirebaseService {\n  private auth = inject(Auth);\n  private config = inject(VALTECH_FIREBASE_CONFIG);\n\n  /** Estado interno de la sesión */\n  private sessionState = new BehaviorSubject<SessionState>({\n    user: null,\n    isAuthenticated: false,\n    isLoading: true,\n    error: null,\n  });\n\n  /** Estado actual de la sesión como Observable */\n  readonly state$: Observable<SessionState> = this.sessionState.asObservable();\n\n  /** Usuario actual de Firebase como Observable */\n  readonly user$: Observable<FirebaseUser | null> = authState(this.auth).pipe(\n    map((user) => (user ? this.mapUser(user) : null)),\n    distinctUntilChanged((a, b) => a?.uid === b?.uid)\n  );\n\n  /** Indica si el usuario está autenticado en Firebase */\n  readonly isAuthenticated$: Observable<boolean> = this.user$.pipe(\n    map((user) => !!user),\n    distinctUntilChanged()\n  );\n\n  constructor() {\n    // Escuchar cambios en el estado de autenticación\n    authState(this.auth).subscribe({\n      next: (user) => {\n        this.sessionState.next({\n          user: user ? this.mapUser(user) : null,\n          isAuthenticated: !!user,\n          isLoading: false,\n          error: null,\n        });\n      },\n      error: (error) => {\n        this.sessionState.next({\n          user: null,\n          isAuthenticated: false,\n          isLoading: false,\n          error,\n        });\n      },\n    });\n  }\n\n  // ===========================================================================\n  // AUTENTICACIÓN\n  // ===========================================================================\n\n  /**\n   * Autentica al usuario con un Custom Token generado por el backend.\n   *\n   * @param token - Firebase Custom Token generado por tu backend\n   * @returns UserCredential con la información del usuario\n   * @throws Error si el token es inválido o expiró\n   *\n   * @example\n   * ```typescript\n   * // Después de login exitoso con tu backend\n   * const { firebaseToken } = await backendAuth.login(email, password);\n   * await firebaseService.signInWithCustomToken(firebaseToken);\n   * ```\n   */\n  async signInWithCustomToken(token: string): Promise<UserCredential> {\n    try {\n      const credential = await signInWithCustomToken(this.auth, token);\n      return credential;\n    } catch (error) {\n      const message = this.getErrorMessage(error);\n      throw new Error(message);\n    }\n  }\n\n  /**\n   * Cierra la sesión de Firebase.\n   * Llamar junto con el logout de tu sistema de autenticación principal.\n   *\n   * @example\n   * ```typescript\n   * async logout() {\n   *   await this.backendAuth.logout();    // Tu auth\n   *   await this.firebase.signOut();       // Firebase\n   * }\n   * ```\n   */\n  async signOut(): Promise<void> {\n    try {\n      await signOut(this.auth);\n    } catch (error) {\n      const message = this.getErrorMessage(error);\n      throw new Error(message);\n    }\n  }\n\n  // ===========================================================================\n  // GETTERS SÍNCRONOS\n  // ===========================================================================\n\n  /**\n   * Obtiene el usuario actual de Firebase (síncrono).\n   * Retorna null si no hay usuario autenticado.\n   */\n  get currentUser(): FirebaseUser | null {\n    const user = this.auth.currentUser;\n    return user ? this.mapUser(user) : null;\n  }\n\n  /**\n   * Obtiene el UID del usuario actual.\n   * Retorna null si no hay usuario autenticado.\n   */\n  get uid(): string | null {\n    return this.auth.currentUser?.uid ?? null;\n  }\n\n  /**\n   * Indica si hay un usuario autenticado actualmente.\n   */\n  get isAuthenticated(): boolean {\n    return !!this.auth.currentUser;\n  }\n\n  // ===========================================================================\n  // TOKENS\n  // ===========================================================================\n\n  /**\n   * Obtiene el ID Token de Firebase para el usuario actual.\n   * Útil para validar el usuario en tu backend.\n   *\n   * @param forceRefresh - Si true, fuerza la renovación del token\n   * @returns ID Token o null si no hay usuario\n   */\n  async getIdToken(forceRefresh = false): Promise<string | null> {\n    const user = this.auth.currentUser;\n    if (!user) return null;\n\n    try {\n      return await user.getIdToken(forceRefresh);\n    } catch {\n      return null;\n    }\n  }\n\n  /**\n   * Obtiene los claims personalizados del token del usuario.\n   * Los claims son establecidos por tu backend al crear el Custom Token.\n   *\n   * @returns Objeto con los claims o vacío si no hay usuario\n   */\n  async getClaims(): Promise<Record<string, unknown>> {\n    const user = this.auth.currentUser;\n    if (!user) return {};\n\n    try {\n      const result = await user.getIdTokenResult();\n      return result.claims;\n    } catch {\n      return {};\n    }\n  }\n\n  /**\n   * Verifica si el usuario tiene un rol específico.\n   * El rol debe estar definido en los claims del Custom Token.\n   *\n   * @param role - Nombre del rol a verificar\n   * @returns true si el usuario tiene el rol\n   */\n  async hasRole(role: string): Promise<boolean> {\n    const claims = await this.getClaims();\n    return claims['role'] === role || (Array.isArray(claims['roles']) && claims['roles'].includes(role));\n  }\n\n  // ===========================================================================\n  // UTILIDADES\n  // ===========================================================================\n\n  /**\n   * Espera a que el estado de autenticación esté determinado.\n   * Útil en guards o al inicializar la app.\n   *\n   * @returns Usuario actual o null\n   */\n  waitForAuth(): Promise<FirebaseUser | null> {\n    return new Promise((resolve) => {\n      const subscription = this.state$.subscribe((state) => {\n        if (!state.isLoading) {\n          subscription.unsubscribe();\n          resolve(state.user);\n        }\n      });\n    });\n  }\n\n  /**\n   * Obtiene la configuración actual de Firebase.\n   */\n  getConfig(): ValtechFirebaseConfig {\n    return this.config;\n  }\n\n  /**\n   * Indica si los emuladores están habilitados.\n   */\n  isUsingEmulators(): boolean {\n    return !!(this.config.emulator?.firestore || this.config.emulator?.auth || this.config.emulator?.storage);\n  }\n\n  // ===========================================================================\n  // MÉTODOS PRIVADOS\n  // ===========================================================================\n\n  /**\n   * Mapea un User de Firebase a nuestra interface FirebaseUser\n   */\n  private mapUser(user: User): FirebaseUser {\n    return {\n      uid: user.uid,\n      email: user.email,\n      displayName: user.displayName,\n      photoURL: user.photoURL,\n      emailVerified: user.emailVerified,\n      isAnonymous: user.isAnonymous,\n      providerId: user.providerId,\n    };\n  }\n\n  /**\n   * Convierte errores de Firebase 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 'auth/invalid-custom-token':\n          return 'Token de autenticación inválido';\n        case 'auth/custom-token-mismatch':\n          return 'El token no corresponde a este proyecto';\n        case 'auth/network-request-failed':\n          return 'Error de conexión. Verifica tu conexión a internet';\n        case 'auth/too-many-requests':\n          return 'Demasiados intentos. Intenta de nuevo más tarde';\n        case 'auth/user-disabled':\n          return 'Esta cuenta ha sido deshabilitada';\n        case 'auth/user-not-found':\n          return 'Usuario no encontrado';\n        default:\n          return error.message || 'Error de autenticación desconocido';\n      }\n    }\n\n    return 'Error de autenticación desconocido';\n  }\n}\n"]}
@@ -1,254 +0,0 @@
1
- /**
2
- * Firestore Collection Factory
3
- *
4
- * Patrón factory para crear instancias de colección tipadas.
5
- * Reemplaza la clase abstracta para evitar problemas con inject() en clases no-injectable.
6
- */
7
- import { Injectable, inject } from '@angular/core';
8
- import { FirestoreService } from './firestore.service';
9
- import * as i0 from "@angular/core";
10
- /**
11
- * Factory para crear instancias de colección tipadas.
12
- *
13
- * @example
14
- * ```typescript
15
- * @Injectable({ providedIn: 'root' })
16
- * export class UsersService {
17
- * private users = inject(FirestoreCollectionFactory).create<User>('users');
18
- *
19
- * getAll = () => this.users.getAll();
20
- * getById = (id: string) => this.users.getById(id);
21
- * create = (data: Omit<User, 'id'>) => this.users.create(data);
22
- *
23
- * // Métodos personalizados
24
- * async getActiveUsers(): Promise<User[]> {
25
- * return this.users.query({
26
- * where: [{ field: 'active', operator: '==', value: true }]
27
- * });
28
- * }
29
- * }
30
- * ```
31
- */
32
- export class FirestoreCollectionFactory {
33
- constructor() {
34
- this.firestore = inject(FirestoreService);
35
- }
36
- /**
37
- * Crea una instancia de colección tipada.
38
- *
39
- * @param collectionPath - Ruta de la colección en Firestore
40
- * @param options - Opciones de configuración
41
- * @returns Instancia de TypedCollection
42
- */
43
- create(collectionPath, options) {
44
- return new TypedCollection(this.firestore, collectionPath, options);
45
- }
46
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
47
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, providedIn: 'root' }); }
48
- }
49
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FirestoreCollectionFactory, decorators: [{
50
- type: Injectable,
51
- args: [{ providedIn: 'root' }]
52
- }] });
53
- /**
54
- * Colección tipada con métodos CRUD.
55
- *
56
- * NO usa inject() - recibe FirestoreService por constructor.
57
- * Esto evita el error NG0203.
58
- */
59
- export class TypedCollection {
60
- constructor(firestore, collectionPath, options = {}) {
61
- this.firestore = firestore;
62
- this.collectionPath = collectionPath;
63
- this.options = {
64
- softDelete: false,
65
- timestamps: true,
66
- ...options,
67
- };
68
- }
69
- // ===========================================================================
70
- // LECTURAS ONE-TIME
71
- // ===========================================================================
72
- /**
73
- * Obtiene un documento por ID.
74
- */
75
- async getById(id) {
76
- return this.firestore.getDoc(this.collectionPath, id);
77
- }
78
- /**
79
- * Obtiene todos los documentos de la colección.
80
- */
81
- async getAll(options) {
82
- const queryOptions = this.applyDefaultFilters(options);
83
- return this.firestore.getDocs(this.collectionPath, queryOptions);
84
- }
85
- /**
86
- * Ejecuta una query personalizada.
87
- */
88
- async query(options) {
89
- const queryOptions = this.applyDefaultFilters(options);
90
- return this.firestore.getDocs(this.collectionPath, queryOptions);
91
- }
92
- /**
93
- * Obtiene documentos con paginación.
94
- */
95
- async paginate(options) {
96
- const queryOptions = this.applyDefaultFilters(options);
97
- return this.firestore.getPaginated(this.collectionPath, queryOptions);
98
- }
99
- /**
100
- * Obtiene el primer documento que coincida con la query.
101
- */
102
- async getFirst(options) {
103
- const queryOptions = this.applyDefaultFilters({
104
- ...options,
105
- limit: 1,
106
- });
107
- const results = await this.firestore.getDocs(this.collectionPath, queryOptions);
108
- return results[0] ?? null;
109
- }
110
- /**
111
- * Cuenta los documentos que coinciden con la query.
112
- * Nota: Esto carga todos los documentos, usar con cuidado en colecciones grandes.
113
- */
114
- async count(options) {
115
- const queryOptions = this.applyDefaultFilters(options);
116
- const results = await this.firestore.getDocs(this.collectionPath, queryOptions);
117
- return results.length;
118
- }
119
- /**
120
- * Verifica si un documento existe.
121
- */
122
- async exists(id) {
123
- return this.firestore.exists(this.collectionPath, id);
124
- }
125
- // ===========================================================================
126
- // SUBSCRIPCIONES REAL-TIME
127
- // ===========================================================================
128
- /**
129
- * Suscribe a cambios de un documento.
130
- */
131
- watch(id) {
132
- return this.firestore.docChanges(this.collectionPath, id);
133
- }
134
- /**
135
- * Suscribe a cambios de la colección.
136
- */
137
- watchAll(options) {
138
- const queryOptions = this.applyDefaultFilters(options);
139
- return this.firestore.collectionChanges(this.collectionPath, queryOptions);
140
- }
141
- /**
142
- * Suscribe a una query personalizada.
143
- */
144
- watchQuery(options) {
145
- const queryOptions = this.applyDefaultFilters(options);
146
- return this.firestore.collectionChanges(this.collectionPath, queryOptions);
147
- }
148
- // ===========================================================================
149
- // ESCRITURA
150
- // ===========================================================================
151
- /**
152
- * Crea un nuevo documento con ID auto-generado.
153
- */
154
- async create(data) {
155
- return this.firestore.addDoc(this.collectionPath, data);
156
- }
157
- /**
158
- * Crea un documento con ID específico.
159
- */
160
- async createWithId(id, data) {
161
- return this.firestore.setDoc(this.collectionPath, id, data);
162
- }
163
- /**
164
- * Actualiza campos de un documento.
165
- */
166
- async update(id, data) {
167
- return this.firestore.updateDoc(this.collectionPath, id, data);
168
- }
169
- /**
170
- * Elimina un documento.
171
- * Si softDelete está habilitado, marca como eliminado en lugar de borrar.
172
- */
173
- async delete(id) {
174
- if (this.options.softDelete) {
175
- return this.firestore.updateDoc(this.collectionPath, id, {
176
- deletedAt: new Date(),
177
- });
178
- }
179
- return this.firestore.deleteDoc(this.collectionPath, id);
180
- }
181
- /**
182
- * Restaura un documento soft-deleted.
183
- */
184
- async restore(id) {
185
- if (!this.options.softDelete) {
186
- throw new Error('Soft delete no está habilitado para esta colección');
187
- }
188
- return this.firestore.updateDoc(this.collectionPath, id, {
189
- deletedAt: null,
190
- });
191
- }
192
- // ===========================================================================
193
- // SUB-COLECCIONES
194
- // ===========================================================================
195
- /**
196
- * Obtiene una referencia a una sub-colección.
197
- *
198
- * @example
199
- * ```typescript
200
- * // En UsersService
201
- * getUserDocuments(userId: string) {
202
- * return this.users.subcollection<Document>(userId, 'documents');
203
- * }
204
- *
205
- * // Uso
206
- * const docs = await users.getUserDocuments('user123').getAll();
207
- * ```
208
- */
209
- subcollection(parentId, subcollectionName) {
210
- const subPath = `${this.collectionPath}/${parentId}/${subcollectionName}`;
211
- return {
212
- getById: (id) => this.firestore.getDoc(subPath, id),
213
- getAll: (options) => this.firestore.getDocs(subPath, options),
214
- watch: (id) => this.firestore.docChanges(subPath, id),
215
- watchAll: (options) => this.firestore.collectionChanges(subPath, options),
216
- create: (data) => this.firestore.addDoc(subPath, data),
217
- update: (id, data) => this.firestore.updateDoc(subPath, id, data),
218
- delete: (id) => this.firestore.deleteDoc(subPath, id),
219
- };
220
- }
221
- // ===========================================================================
222
- // MÉTODOS PRIVADOS
223
- // ===========================================================================
224
- /**
225
- * Aplica filtros por defecto a las queries.
226
- */
227
- applyDefaultFilters(options) {
228
- if (!this.options.softDelete) {
229
- return options ?? {};
230
- }
231
- // Excluir documentos soft-deleted por defecto
232
- const whereClause = { field: 'deletedAt', operator: '==', value: null };
233
- return {
234
- ...options,
235
- where: [...(options?.where ?? []), whereClause],
236
- };
237
- }
238
- // ===========================================================================
239
- // UTILIDADES
240
- // ===========================================================================
241
- /**
242
- * Genera un nuevo ID sin crear el documento.
243
- */
244
- generateId() {
245
- return this.firestore.generateId(this.collectionPath);
246
- }
247
- /**
248
- * Obtiene la ruta de la colección.
249
- */
250
- getPath() {
251
- return this.collectionPath;
252
- }
253
- }
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, inject } 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  private firestore = inject(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"]}