valtech-components 2.0.442 → 2.0.444
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/services/auth/auth.service.mjs +4 -5
- package/esm2022/lib/services/firebase/firestore-collection.mjs +1 -1
- package/esm2022/lib/services/link-processor.service.mjs +54 -54
- package/fesm2022/valtech-components.mjs +56 -57
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/article/article.component.d.ts +2 -2
- package/lib/services/auth/auth.service.d.ts +2 -3
- package/package.json +1 -1
|
@@ -33,11 +33,10 @@ import * as i7 from "../firebase";
|
|
|
33
33
|
* ```
|
|
34
34
|
*/
|
|
35
35
|
export class AuthService {
|
|
36
|
-
constructor(config, http, router,
|
|
36
|
+
constructor(config, http, router, stateService, tokenService, storageService, syncService, firebaseService) {
|
|
37
37
|
this.config = config;
|
|
38
38
|
this.http = http;
|
|
39
39
|
this.router = router;
|
|
40
|
-
this.injector = injector;
|
|
41
40
|
this.stateService = stateService;
|
|
42
41
|
this.tokenService = tokenService;
|
|
43
42
|
this.storageService = storageService;
|
|
@@ -442,7 +441,7 @@ export class AuthService {
|
|
|
442
441
|
console.warn('[ValtechAuth] Firebase signout failed:', error);
|
|
443
442
|
}
|
|
444
443
|
}
|
|
445
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, deps: [{ token: VALTECH_AUTH_CONFIG }, { token: i1.HttpClient }, { token: i2.Router }, { token:
|
|
444
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, deps: [{ token: VALTECH_AUTH_CONFIG }, { token: i1.HttpClient }, { token: i2.Router }, { token: i3.AuthStateService }, { token: i4.TokenService }, { token: i5.AuthStorageService }, { token: i6.AuthSyncService }, { token: i7.FirebaseService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
446
445
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, providedIn: 'root' }); }
|
|
447
446
|
}
|
|
448
447
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AuthService, decorators: [{
|
|
@@ -451,5 +450,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
451
450
|
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
|
452
451
|
type: Inject,
|
|
453
452
|
args: [VALTECH_AUTH_CONFIG]
|
|
454
|
-
}] }, { type: i1.HttpClient }, { type: i2.Router }, { type:
|
|
455
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy, Injector } from '@angular/core';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Router } from '@angular/router';\nimport { Observable, throwError, of, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthStateService } from './auth-state.service';\nimport { TokenService } from './token.service';\nimport { AuthStorageService } from './storage.service';\nimport { AuthSyncService } from './sync.service';\nimport {\n  SigninRequest,\n  SigninResponse,\n  SignupRequest,\n  SignupResponse,\n  VerifyEmailRequest,\n  VerifyEmailResponse,\n  ResendCodeRequest,\n  ResendCodeResponse,\n  MFAVerifyResponse,\n  RefreshResponse,\n  GetPermissionsResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  MFAMethod,\n  AuthError,\n  AuthSyncEvent,\n  ValtechAuthConfig,\n} from './types';\nimport { FirebaseService } from '../firebase';\n\n\n/**\n * Servicio principal de autenticación.\n *\n * @example\n * ```typescript\n * import { AuthService } from 'valtech-components';\n *\n * @Component({...})\n * export class LoginComponent {\n *   private auth = inject(AuthService);\n *\n *   async login() {\n *     await firstValueFrom(this.auth.signin({ email, password }));\n *     if (this.auth.mfaPending().required) {\n *       // Mostrar UI de MFA\n *     } else {\n *       this.router.navigate(['/']);\n *     }\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthService implements OnDestroy {\n  // Timer para refresh proactivo\n  private refreshTimerId: ReturnType<typeof setTimeout> | null = null;\n  private syncSubscription: Subscription | null = null;\n\n  constructor(\n    @Inject(VALTECH_AUTH_CONFIG) private config: ValtechAuthConfig,\n    private http: HttpClient,\n    private router: Router,\n    private injector: Injector,\n    private stateService: AuthStateService,\n    private tokenService: TokenService,\n    private storageService: AuthStorageService,\n    private syncService: AuthSyncService,\n    private firebaseService: FirebaseService,\n  ) {}\n\n  // =============================================\n  // ESTADO PÚBLICO (Signals readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this.stateService.state;\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = this.stateService.isAuthenticated;\n\n  /** Estado de carga */\n  readonly isLoading = this.stateService.isLoading;\n\n  /** Información del usuario */\n  readonly user = this.stateService.user;\n\n  /** Token de acceso */\n  readonly accessToken = this.stateService.accessToken;\n\n  /** Roles del usuario */\n  readonly roles = this.stateService.roles;\n\n  /** Permisos del usuario */\n  readonly permissions = this.stateService.permissions;\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = this.stateService.isSuperAdmin;\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this.stateService.mfaPending;\n\n  /** Error actual */\n  readonly error = this.stateService.error;\n\n  // =============================================\n  // INICIALIZACIÓN\n  // =============================================\n\n  /**\n   * Inicializa el servicio de autenticación.\n   * Llamado automáticamente por provideValtechAuth.\n   */\n  async initialize(): Promise<void> {\n    // 1. Cargar estado desde storage\n    const storedState = this.storageService.loadState();\n\n    if (storedState.accessToken) {\n      // 2. Verificar si token es válido\n      if (this.tokenService.isTokenValid(storedState.accessToken)) {\n        this.stateService.restoreFromStorage(storedState);\n\n        // Extraer info del token\n        const claims = this.tokenService.parseToken(storedState.accessToken);\n        if (claims) {\n          this.stateService.updateUserInfo(claims.uid, claims.email);\n        }\n\n        // 3. Iniciar timer de refresco proactivo\n        this.startRefreshTimer();\n      } else if (storedState.refreshToken) {\n        // 4. Token expirado pero hay refresh token - intentar refrescar\n        try {\n          await firstValueFrom(this.refreshAccessToken());\n        } catch {\n          this.clearState();\n        }\n      } else {\n        this.clearState();\n      }\n    }\n\n    // 5. Iniciar sincronización entre pestañas\n    if (this.config.enableTabSync) {\n      this.syncService.start();\n      this.syncSubscription = this.syncService.onEvent$.subscribe(event =>\n        this.handleSyncEvent(event)\n      );\n    }\n\n    this.stateService.setLoading(false);\n  }\n\n  ngOnDestroy(): void {\n    this.stopRefreshTimer();\n    this.syncSubscription?.unsubscribe();\n  }\n\n  // =============================================\n  // AUTENTICACIÓN\n  // =============================================\n\n  /**\n   * Inicia sesión con email y contraseña.\n   */\n  signin(request: SigninRequest): Observable<SigninResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<SigninResponse>(`${this.baseUrl}/signin`, request).pipe(\n      tap(response => {\n        if (response.mfaRequired) {\n          // MFA requerido - guardar estado temporal\n          this.stateService.setMFAPending({\n            required: true,\n            mfaToken: response.mfaToken!,\n            method: response.mfaMethod!,\n          });\n        } else if (response.accessToken) {\n          // Login exitoso sin MFA\n          this.handleSuccessfulAuth(response);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Registra un nuevo usuario.\n   * El usuario queda en estado PENDING hasta verificar su email.\n   */\n  signup(request: SignupRequest): Observable<SignupResponse> {\n    this.stateService.clearError();\n\n    return this.http\n      .post<SignupResponse>(`${this.baseUrl}/signup`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica email con código de 6 dígitos.\n   * Si es exitoso, hace auto-login y retorna tokens.\n   */\n  verifyEmail(request: VerifyEmailRequest): Observable<VerifyEmailResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<VerifyEmailResponse>(`${this.baseUrl}/verify-email`, request).pipe(\n      tap(response => {\n        if (response.verified && response.accessToken) {\n          // Auto-login: guardar tokens y actualizar estado\n          this.handleSuccessfulAuth(response as unknown as SigninResponse);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Reenvía código de verificación al email.\n   */\n  resendCode(request: ResendCodeRequest): Observable<ResendCodeResponse> {\n    return this.http\n      .post<ResendCodeResponse>(`${this.baseUrl}/resend-code`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica código MFA.\n   */\n  verifyMFA(code: string): Observable<MFAVerifyResponse> {\n    const mfaState = this.mfaPending();\n    if (!mfaState.mfaToken) {\n      return throwError(() => ({\n        code: 'MFA_NOT_PENDING',\n        message: 'No hay verificación MFA pendiente',\n      }));\n    }\n\n    return this.http\n      .post<MFAVerifyResponse>(`${this.baseUrl}/mfa/verify`, {\n        mfaToken: mfaState.mfaToken,\n        code,\n      })\n      .pipe(\n        tap(response => {\n          this.stateService.clearMFAPending();\n          this.handleSuccessfulAuth(response);\n        }),\n        catchError(error => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Refresca el token de acceso.\n   */\n  refreshAccessToken(): Observable<RefreshResponse> {\n    const refreshToken = this.state().refreshToken;\n    if (!refreshToken) {\n      return throwError(() => ({\n        code: 'NO_REFRESH_TOKEN',\n        message: 'No hay token de refresco',\n      }));\n    }\n\n    return this.http.post<RefreshResponse>(`${this.baseUrl}/refresh`, { refreshToken }).pipe(\n      tap(response => {\n        const expiresAt = Date.now() + response.expiresIn * 1000;\n        this.stateService.updateAccessToken(response.accessToken, response.expiresIn);\n        this.storageService.saveAccessToken(response.accessToken, expiresAt);\n        this.startRefreshTimer();\n        this.syncService.broadcast({\n          type: 'TOKEN_REFRESH',\n          payload: { accessToken: response.accessToken, expiresAt },\n        });\n      }),\n      catchError(error => {\n        this.logout();\n        return throwError(() => error);\n      })\n    );\n  }\n\n  /**\n   * Cierra sesión.\n   */\n  logout(): void {\n    const refreshToken = this.state().refreshToken;\n\n    // Notificar al backend (fire and forget)\n    if (refreshToken) {\n      this.http\n        .post(`${this.baseUrl}/logout`, { refreshToken })\n        .pipe(catchError(() => of(null)))\n        .subscribe();\n    }\n\n    // Cerrar sesión de Firebase si está integrado\n    this.signOutFirebase();\n\n    this.clearState();\n    this.syncService.broadcast({ type: 'LOGOUT' });\n    this.router.navigate([this.config.loginRoute]);\n  }\n\n  // =============================================\n  // MFA SETUP (usuario autenticado)\n  // =============================================\n\n  /**\n   * Configura MFA para el usuario.\n   */\n  setupMFA(method: MFAMethod, phone?: string): Observable<MFASetupResponse> {\n    return this.http\n      .post<MFASetupResponse>(`${this.baseUrl}/mfa/setup`, { method, phone })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Confirma la configuración de MFA.\n   */\n  confirmMFA(code: string): Observable<MFAConfirmResponse> {\n    return this.http\n      .post<MFAConfirmResponse>(`${this.baseUrl}/mfa/confirm`, { code })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Deshabilita MFA.\n   */\n  disableMFA(password: string): Observable<MFADisableResponse> {\n    return this.http\n      .post<MFADisableResponse>(`${this.baseUrl}/mfa/disable`, { password })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // PERMISOS\n  // =============================================\n\n  /**\n   * Obtiene los permisos actualizados del backend.\n   */\n  fetchPermissions(): Observable<GetPermissionsResponse> {\n    return this.http.get<GetPermissionsResponse>(`${this.baseUrl}/permissions`).pipe(\n      tap(response => {\n        this.stateService.updatePermissions(\n          response.roles,\n          response.permissions,\n          response.isSuperAdmin\n        );\n        this.storageService.savePermissions(response);\n        this.syncService.broadcast({ type: 'PERMISSIONS_UPDATE' });\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Verifica si el usuario tiene un permiso específico.\n   * Formato: \"resource:action\" (ej: \"templates:edit\")\n   */\n  hasPermission(permission: string): boolean {\n    if (this.isSuperAdmin()) return true;\n\n    const [resource, action] = permission.split(':');\n    return this.permissions().some(p => {\n      const [pResource, pAction] = p.split(':');\n      return (\n        (pResource === '*' || pResource === resource) && (pAction === '*' || pAction === action)\n      );\n    });\n  }\n\n  /**\n   * Verifica si el usuario tiene alguno de los permisos dados.\n   */\n  hasAnyPermission(permissions: string[]): boolean {\n    return permissions.some(p => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene todos los permisos dados.\n   */\n  hasAllPermissions(permissions: string[]): boolean {\n    return permissions.every(p => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene un rol específico.\n   */\n  hasRole(role: string): boolean {\n    return this.roles().some(r => r.toLowerCase() === role.toLowerCase());\n  }\n\n  // =============================================\n  // PRIVATE METHODS\n  // =============================================\n\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}${this.config.authPrefix}`;\n  }\n\n  private handleSuccessfulAuth(response: SigninResponse | MFAVerifyResponse): void {\n    const expiresAt = Date.now() + response.expiresIn! * 1000;\n    const tokenData = this.tokenService.parseToken(response.accessToken!);\n\n    this.stateService.setAuthenticated({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      userId: tokenData?.uid,\n      email: tokenData?.email,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.storageService.saveState({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.startRefreshTimer();\n    this.syncService.broadcast({ type: 'LOGIN' });\n\n    // Integración con Firebase\n    if (\n      this.config.enableFirebaseIntegration &&\n      'firebaseToken' in response &&\n      response.firebaseToken\n    ) {\n      this.signInWithFirebase(response.firebaseToken);\n    }\n  }\n\n  private clearState(): void {\n    this.stopRefreshTimer();\n    this.stateService.reset();\n    this.storageService.clear();\n  }\n\n  private startRefreshTimer(): void {\n    this.stopRefreshTimer();\n\n    const state = this.stateService.state();\n    if (!state.expiresAt) return;\n\n    const refreshBeforeMs = (this.config.refreshBeforeExpiry || 60) * 1000;\n    const refreshAt = state.expiresAt - refreshBeforeMs;\n    const delay = refreshAt - Date.now();\n\n    if (delay > 0) {\n      this.refreshTimerId = setTimeout(() => {\n        this.refreshAccessToken().subscribe({\n          error: () => this.logout(),\n        });\n      }, delay);\n    } else if (state.refreshToken) {\n      // Token ya debería refrescarse, intentar ahora\n      this.refreshAccessToken().subscribe({\n        error: () => this.logout(),\n      });\n    }\n  }\n\n  private stopRefreshTimer(): void {\n    if (this.refreshTimerId) {\n      clearTimeout(this.refreshTimerId);\n      this.refreshTimerId = null;\n    }\n  }\n\n  private handleSyncEvent(event: AuthSyncEvent): void {\n    switch (event.type) {\n      case 'LOGIN':\n      case 'TOKEN_REFRESH': {\n        // Recargar estado desde storage\n        const state = this.storageService.loadState();\n        if (state.accessToken) {\n          this.stateService.restoreFromStorage(state);\n          const claims = this.tokenService.parseToken(state.accessToken);\n          if (claims) {\n            this.stateService.updateUserInfo(claims.uid, claims.email);\n          }\n          this.startRefreshTimer();\n        }\n        break;\n      }\n      case 'LOGOUT':\n        this.stateService.reset();\n        this.stopRefreshTimer();\n        this.router.navigate([this.config.loginRoute]);\n        break;\n      case 'PERMISSIONS_UPDATE': {\n        const perms = this.storageService.loadPermissions();\n        this.stateService.updatePermissions(perms.roles, perms.permissions, perms.isSuperAdmin);\n        break;\n      }\n    }\n  }\n\n  private handleAuthError(error: HttpErrorResponse): Observable<never> {\n    const authError: AuthError = {\n      code: error.error?.code || 'UNKNOWN_ERROR',\n      message: error.error?.message || 'Error de autenticación desconocido',\n    };\n    this.stateService.setError(authError);\n    return throwError(() => authError);\n  }\n\n  // =============================================\n  // FIREBASE INTEGRATION\n  // =============================================\n\n  private async signInWithFirebase(firebaseToken: string): Promise<void> {\n    try {\n      if (this.firebaseService) {\n        await this.firebaseService.signInWithCustomToken(firebaseToken);\n        console.log('[ValtechAuth] Firebase signin successful');\n      } else {\n        console.warn(\n          '[ValtechAuth] FirebaseService not provided. Add provideValtechFirebase() to your providers.'\n        );\n      }\n    } catch (error) {\n      // No bloquear el login principal si Firebase falla\n      console.error('[ValtechAuth] Firebase signin failed:', error);\n    }\n  }\n\n  private async signOutFirebase(): Promise<void> {\n    if (!this.config.enableFirebaseIntegration) return;\n\n    try {\n      if (this.firebaseService) {\n        await this.firebaseService.signOut();\n        console.log('[ValtechAuth] Firebase signout successful');\n      }\n    } catch (error) {\n      // Ignorar errores de Firebase signout\n      console.warn('[ValtechAuth] Firebase signout failed:', error);\n    }\n  }\n}\n"]}
|
|
453
|
+
}] }, { type: i1.HttpClient }, { type: i2.Router }, { type: i3.AuthStateService }, { type: i4.TokenService }, { type: i5.AuthStorageService }, { type: i6.AuthSyncService }, { type: i7.FirebaseService }] });
|
|
454
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/auth/auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy, Injector } from '@angular/core';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Router } from '@angular/router';\nimport { Observable, throwError, of, firstValueFrom, Subscription } from 'rxjs';\nimport { tap, catchError } from 'rxjs/operators';\n\nimport { VALTECH_AUTH_CONFIG } from './config';\nimport { AuthStateService } from './auth-state.service';\nimport { TokenService } from './token.service';\nimport { AuthStorageService } from './storage.service';\nimport { AuthSyncService } from './sync.service';\nimport {\n  SigninRequest,\n  SigninResponse,\n  SignupRequest,\n  SignupResponse,\n  VerifyEmailRequest,\n  VerifyEmailResponse,\n  ResendCodeRequest,\n  ResendCodeResponse,\n  MFAVerifyResponse,\n  RefreshResponse,\n  GetPermissionsResponse,\n  MFASetupResponse,\n  MFAConfirmResponse,\n  MFADisableResponse,\n  MFAMethod,\n  AuthError,\n  AuthSyncEvent,\n  ValtechAuthConfig,\n} from './types';\nimport { FirebaseService } from '../firebase';\n\n\n/**\n * Servicio principal de autenticación.\n *\n * @example\n * ```typescript\n * import { AuthService } from 'valtech-components';\n *\n * @Component({...})\n * export class LoginComponent {\n *   private auth = inject(AuthService);\n *\n *   async login() {\n *     await firstValueFrom(this.auth.signin({ email, password }));\n *     if (this.auth.mfaPending().required) {\n *       // Mostrar UI de MFA\n *     } else {\n *       this.router.navigate(['/']);\n *     }\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AuthService implements OnDestroy {\n  // Timer para refresh proactivo\n  private refreshTimerId: ReturnType<typeof setTimeout> | null = null;\n  private syncSubscription: Subscription | null = null;\n\n  constructor(\n    @Inject(VALTECH_AUTH_CONFIG) private config: ValtechAuthConfig,\n    private http: HttpClient,\n    private router: Router,\n    private stateService: AuthStateService,\n    private tokenService: TokenService,\n    private storageService: AuthStorageService,\n    private syncService: AuthSyncService,\n    private firebaseService: FirebaseService,\n  ) {}\n\n  // =============================================\n  // ESTADO PÚBLICO (Signals readonly)\n  // =============================================\n\n  /** Estado completo de autenticación */\n  readonly state = this.stateService.state;\n\n  /** Usuario está autenticado */\n  readonly isAuthenticated = this.stateService.isAuthenticated;\n\n  /** Estado de carga */\n  readonly isLoading = this.stateService.isLoading;\n\n  /** Información del usuario */\n  readonly user = this.stateService.user;\n\n  /** Token de acceso */\n  readonly accessToken = this.stateService.accessToken;\n\n  /** Roles del usuario */\n  readonly roles = this.stateService.roles;\n\n  /** Permisos del usuario */\n  readonly permissions = this.stateService.permissions;\n\n  /** Usuario es super admin */\n  readonly isSuperAdmin = this.stateService.isSuperAdmin;\n\n  /** Estado de MFA pendiente */\n  readonly mfaPending = this.stateService.mfaPending;\n\n  /** Error actual */\n  readonly error = this.stateService.error;\n\n  // =============================================\n  // INICIALIZACIÓN\n  // =============================================\n\n  /**\n   * Inicializa el servicio de autenticación.\n   * Llamado automáticamente por provideValtechAuth.\n   */\n  async initialize(): Promise<void> {\n    // 1. Cargar estado desde storage\n    const storedState = this.storageService.loadState();\n\n    if (storedState.accessToken) {\n      // 2. Verificar si token es válido\n      if (this.tokenService.isTokenValid(storedState.accessToken)) {\n        this.stateService.restoreFromStorage(storedState);\n\n        // Extraer info del token\n        const claims = this.tokenService.parseToken(storedState.accessToken);\n        if (claims) {\n          this.stateService.updateUserInfo(claims.uid, claims.email);\n        }\n\n        // 3. Iniciar timer de refresco proactivo\n        this.startRefreshTimer();\n      } else if (storedState.refreshToken) {\n        // 4. Token expirado pero hay refresh token - intentar refrescar\n        try {\n          await firstValueFrom(this.refreshAccessToken());\n        } catch {\n          this.clearState();\n        }\n      } else {\n        this.clearState();\n      }\n    }\n\n    // 5. Iniciar sincronización entre pestañas\n    if (this.config.enableTabSync) {\n      this.syncService.start();\n      this.syncSubscription = this.syncService.onEvent$.subscribe(event =>\n        this.handleSyncEvent(event)\n      );\n    }\n\n    this.stateService.setLoading(false);\n  }\n\n  ngOnDestroy(): void {\n    this.stopRefreshTimer();\n    this.syncSubscription?.unsubscribe();\n  }\n\n  // =============================================\n  // AUTENTICACIÓN\n  // =============================================\n\n  /**\n   * Inicia sesión con email y contraseña.\n   */\n  signin(request: SigninRequest): Observable<SigninResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<SigninResponse>(`${this.baseUrl}/signin`, request).pipe(\n      tap(response => {\n        if (response.mfaRequired) {\n          // MFA requerido - guardar estado temporal\n          this.stateService.setMFAPending({\n            required: true,\n            mfaToken: response.mfaToken!,\n            method: response.mfaMethod!,\n          });\n        } else if (response.accessToken) {\n          // Login exitoso sin MFA\n          this.handleSuccessfulAuth(response);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Registra un nuevo usuario.\n   * El usuario queda en estado PENDING hasta verificar su email.\n   */\n  signup(request: SignupRequest): Observable<SignupResponse> {\n    this.stateService.clearError();\n\n    return this.http\n      .post<SignupResponse>(`${this.baseUrl}/signup`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica email con código de 6 dígitos.\n   * Si es exitoso, hace auto-login y retorna tokens.\n   */\n  verifyEmail(request: VerifyEmailRequest): Observable<VerifyEmailResponse> {\n    this.stateService.clearError();\n\n    return this.http.post<VerifyEmailResponse>(`${this.baseUrl}/verify-email`, request).pipe(\n      tap(response => {\n        if (response.verified && response.accessToken) {\n          // Auto-login: guardar tokens y actualizar estado\n          this.handleSuccessfulAuth(response as unknown as SigninResponse);\n        }\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Reenvía código de verificación al email.\n   */\n  resendCode(request: ResendCodeRequest): Observable<ResendCodeResponse> {\n    return this.http\n      .post<ResendCodeResponse>(`${this.baseUrl}/resend-code`, request)\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Verifica código MFA.\n   */\n  verifyMFA(code: string): Observable<MFAVerifyResponse> {\n    const mfaState = this.mfaPending();\n    if (!mfaState.mfaToken) {\n      return throwError(() => ({\n        code: 'MFA_NOT_PENDING',\n        message: 'No hay verificación MFA pendiente',\n      }));\n    }\n\n    return this.http\n      .post<MFAVerifyResponse>(`${this.baseUrl}/mfa/verify`, {\n        mfaToken: mfaState.mfaToken,\n        code,\n      })\n      .pipe(\n        tap(response => {\n          this.stateService.clearMFAPending();\n          this.handleSuccessfulAuth(response);\n        }),\n        catchError(error => this.handleAuthError(error))\n      );\n  }\n\n  /**\n   * Refresca el token de acceso.\n   */\n  refreshAccessToken(): Observable<RefreshResponse> {\n    const refreshToken = this.state().refreshToken;\n    if (!refreshToken) {\n      return throwError(() => ({\n        code: 'NO_REFRESH_TOKEN',\n        message: 'No hay token de refresco',\n      }));\n    }\n\n    return this.http.post<RefreshResponse>(`${this.baseUrl}/refresh`, { refreshToken }).pipe(\n      tap(response => {\n        const expiresAt = Date.now() + response.expiresIn * 1000;\n        this.stateService.updateAccessToken(response.accessToken, response.expiresIn);\n        this.storageService.saveAccessToken(response.accessToken, expiresAt);\n        this.startRefreshTimer();\n        this.syncService.broadcast({\n          type: 'TOKEN_REFRESH',\n          payload: { accessToken: response.accessToken, expiresAt },\n        });\n      }),\n      catchError(error => {\n        this.logout();\n        return throwError(() => error);\n      })\n    );\n  }\n\n  /**\n   * Cierra sesión.\n   */\n  logout(): void {\n    const refreshToken = this.state().refreshToken;\n\n    // Notificar al backend (fire and forget)\n    if (refreshToken) {\n      this.http\n        .post(`${this.baseUrl}/logout`, { refreshToken })\n        .pipe(catchError(() => of(null)))\n        .subscribe();\n    }\n\n    // Cerrar sesión de Firebase si está integrado\n    this.signOutFirebase();\n\n    this.clearState();\n    this.syncService.broadcast({ type: 'LOGOUT' });\n    this.router.navigate([this.config.loginRoute]);\n  }\n\n  // =============================================\n  // MFA SETUP (usuario autenticado)\n  // =============================================\n\n  /**\n   * Configura MFA para el usuario.\n   */\n  setupMFA(method: MFAMethod, phone?: string): Observable<MFASetupResponse> {\n    return this.http\n      .post<MFASetupResponse>(`${this.baseUrl}/mfa/setup`, { method, phone })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Confirma la configuración de MFA.\n   */\n  confirmMFA(code: string): Observable<MFAConfirmResponse> {\n    return this.http\n      .post<MFAConfirmResponse>(`${this.baseUrl}/mfa/confirm`, { code })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  /**\n   * Deshabilita MFA.\n   */\n  disableMFA(password: string): Observable<MFADisableResponse> {\n    return this.http\n      .post<MFADisableResponse>(`${this.baseUrl}/mfa/disable`, { password })\n      .pipe(catchError(error => this.handleAuthError(error)));\n  }\n\n  // =============================================\n  // PERMISOS\n  // =============================================\n\n  /**\n   * Obtiene los permisos actualizados del backend.\n   */\n  fetchPermissions(): Observable<GetPermissionsResponse> {\n    return this.http.get<GetPermissionsResponse>(`${this.baseUrl}/permissions`).pipe(\n      tap(response => {\n        this.stateService.updatePermissions(\n          response.roles,\n          response.permissions,\n          response.isSuperAdmin\n        );\n        this.storageService.savePermissions(response);\n        this.syncService.broadcast({ type: 'PERMISSIONS_UPDATE' });\n      }),\n      catchError(error => this.handleAuthError(error))\n    );\n  }\n\n  /**\n   * Verifica si el usuario tiene un permiso específico.\n   * Formato: \"resource:action\" (ej: \"templates:edit\")\n   */\n  hasPermission(permission: string): boolean {\n    if (this.isSuperAdmin()) return true;\n\n    const [resource, action] = permission.split(':');\n    return this.permissions().some(p => {\n      const [pResource, pAction] = p.split(':');\n      return (\n        (pResource === '*' || pResource === resource) && (pAction === '*' || pAction === action)\n      );\n    });\n  }\n\n  /**\n   * Verifica si el usuario tiene alguno de los permisos dados.\n   */\n  hasAnyPermission(permissions: string[]): boolean {\n    return permissions.some(p => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene todos los permisos dados.\n   */\n  hasAllPermissions(permissions: string[]): boolean {\n    return permissions.every(p => this.hasPermission(p));\n  }\n\n  /**\n   * Verifica si el usuario tiene un rol específico.\n   */\n  hasRole(role: string): boolean {\n    return this.roles().some(r => r.toLowerCase() === role.toLowerCase());\n  }\n\n  // =============================================\n  // PRIVATE METHODS\n  // =============================================\n\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}${this.config.authPrefix}`;\n  }\n\n  private handleSuccessfulAuth(response: SigninResponse | MFAVerifyResponse): void {\n    const expiresAt = Date.now() + response.expiresIn! * 1000;\n    const tokenData = this.tokenService.parseToken(response.accessToken!);\n\n    this.stateService.setAuthenticated({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      userId: tokenData?.uid,\n      email: tokenData?.email,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.storageService.saveState({\n      accessToken: response.accessToken!,\n      refreshToken: response.refreshToken!,\n      roles: response.roles || [],\n      permissions: response.permissions || [],\n      isSuperAdmin: response.permissions?.includes('*:*') || false,\n      expiresAt,\n    });\n\n    this.startRefreshTimer();\n    this.syncService.broadcast({ type: 'LOGIN' });\n\n    // Integración con Firebase\n    if (\n      this.config.enableFirebaseIntegration &&\n      'firebaseToken' in response &&\n      response.firebaseToken\n    ) {\n      this.signInWithFirebase(response.firebaseToken);\n    }\n  }\n\n  private clearState(): void {\n    this.stopRefreshTimer();\n    this.stateService.reset();\n    this.storageService.clear();\n  }\n\n  private startRefreshTimer(): void {\n    this.stopRefreshTimer();\n\n    const state = this.stateService.state();\n    if (!state.expiresAt) return;\n\n    const refreshBeforeMs = (this.config.refreshBeforeExpiry || 60) * 1000;\n    const refreshAt = state.expiresAt - refreshBeforeMs;\n    const delay = refreshAt - Date.now();\n\n    if (delay > 0) {\n      this.refreshTimerId = setTimeout(() => {\n        this.refreshAccessToken().subscribe({\n          error: () => this.logout(),\n        });\n      }, delay);\n    } else if (state.refreshToken) {\n      // Token ya debería refrescarse, intentar ahora\n      this.refreshAccessToken().subscribe({\n        error: () => this.logout(),\n      });\n    }\n  }\n\n  private stopRefreshTimer(): void {\n    if (this.refreshTimerId) {\n      clearTimeout(this.refreshTimerId);\n      this.refreshTimerId = null;\n    }\n  }\n\n  private handleSyncEvent(event: AuthSyncEvent): void {\n    switch (event.type) {\n      case 'LOGIN':\n      case 'TOKEN_REFRESH': {\n        // Recargar estado desde storage\n        const state = this.storageService.loadState();\n        if (state.accessToken) {\n          this.stateService.restoreFromStorage(state);\n          const claims = this.tokenService.parseToken(state.accessToken);\n          if (claims) {\n            this.stateService.updateUserInfo(claims.uid, claims.email);\n          }\n          this.startRefreshTimer();\n        }\n        break;\n      }\n      case 'LOGOUT':\n        this.stateService.reset();\n        this.stopRefreshTimer();\n        this.router.navigate([this.config.loginRoute]);\n        break;\n      case 'PERMISSIONS_UPDATE': {\n        const perms = this.storageService.loadPermissions();\n        this.stateService.updatePermissions(perms.roles, perms.permissions, perms.isSuperAdmin);\n        break;\n      }\n    }\n  }\n\n  private handleAuthError(error: HttpErrorResponse): Observable<never> {\n    const authError: AuthError = {\n      code: error.error?.code || 'UNKNOWN_ERROR',\n      message: error.error?.message || 'Error de autenticación desconocido',\n    };\n    this.stateService.setError(authError);\n    return throwError(() => authError);\n  }\n\n  // =============================================\n  // FIREBASE INTEGRATION\n  // =============================================\n\n  private async signInWithFirebase(firebaseToken: string): Promise<void> {\n    try {\n      if (this.firebaseService) {\n        await this.firebaseService.signInWithCustomToken(firebaseToken);\n        console.log('[ValtechAuth] Firebase signin successful');\n      } else {\n        console.warn(\n          '[ValtechAuth] FirebaseService not provided. Add provideValtechFirebase() to your providers.'\n        );\n      }\n    } catch (error) {\n      // No bloquear el login principal si Firebase falla\n      console.error('[ValtechAuth] Firebase signin failed:', error);\n    }\n  }\n\n  private async signOutFirebase(): Promise<void> {\n    if (!this.config.enableFirebaseIntegration) return;\n\n    try {\n      if (this.firebaseService) {\n        await this.firebaseService.signOut();\n        console.log('[ValtechAuth] Firebase signout successful');\n      }\n    } catch (error) {\n      // Ignorar errores de Firebase signout\n      console.warn('[ValtechAuth] Firebase signout failed:', error);\n    }\n  }\n}\n"]}
|
|
@@ -263,4 +263,4 @@ export class FirestoreCollection {
|
|
|
263
263
|
return this.collectionPath;
|
|
264
264
|
}
|
|
265
265
|
}
|
|
266
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"firestore-collection.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/firestore-collection.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAGvC,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AAiCvD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwDG;AACH,MAAM,OAAgB,mBAAmB;IAKvC;;;OAGG;IACH,YAAY,cAAsB,EAAE,UAA6B,EAAE;QARzD,cAAS,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAS7C,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC;QACrC,IAAI,CAAC,OAAO,GAAG;YACb,UAAU,EAAE,KAAK;YACjB,UAAU,EAAE,IAAI;YAChB,GAAG,OAAO;SACX,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,oBAAoB;IACpB,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,EAAU;QACtB,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAAsB;QACjC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAK,CAAC,OAAqB;QAC/B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IACtE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,OAAyC;QACtD,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAqC,CAAC;QAC3F,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAC3E,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,OAAsB;QACnC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC;YAC5C,GAAG,OAAO;YACV,KAAK,EAAE,CAAC;SACT,CAAC,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;QACnF,OAAO,OAAO,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,KAAK,CAAC,OAAsB;QAChC,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;QACnF,OAAO,OAAO,CAAC,MAAM,CAAC;IACxB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU;QACrB,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IACxD,CAAC;IAED,8EAA8E;IAC9E,2BAA2B;IAC3B,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,EAAU;QACd,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED;;OAEG;IACH,QAAQ,CAAC,OAAsB;QAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAChF,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,OAAqB;QAC9B,MAAM,YAAY,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACvD,OAAO,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,IAAI,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAChF,CAAC;IAED,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,IAA+C;QAC1D,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,CAAC;IAC7D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,EAAU,EAAE,IAAmB;QAChD,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU,EAAE,IAA0C;QACjE,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;IACpE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,MAAM,CAAC,EAAU;QACrB,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC5B,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE;gBAC1D,SAAS,EAAE,IAAI,IAAI,EAAE;aACG,CAAC,CAAC;QAC9B,CAAC;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,EAAU;QACtB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;QACxE,CAAC;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,IAAI,CAAC,cAAc,EAAE,EAAE,EAAE;YAC1D,SAAS,EAAE,IAAI;SACS,CAAC,CAAC;IAC9B,CAAC;IAED,8EAA8E;IAC9E,kBAAkB;IAClB,8EAA8E;IAE9E;;;;;;;;;;;;;OAaG;IACH,aAAa,CACX,QAAgB,EAChB,iBAAyB;QAEzB,MAAM,OAAO,GAAG,GAAG,IAAI,CAAC,cAAc,IAAI,QAAQ,IAAI,iBAAiB,EAAE,CAAC;QAE1E,OAAO;YACL,OAAO,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,OAAO,EAAE,EAAE,CAAC;YAC9D,MAAM,EAAE,CAAC,OAAsB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAI,OAAO,EAAE,OAAO,CAAC;YAC/E,KAAK,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAI,OAAO,EAAE,EAAE,CAAC;YAChE,QAAQ,EAAE,CAAC,OAAsB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAI,OAAO,EAAE,OAAO,CAAC;YAC3F,MAAM,EAAE,CAAC,IAA+C,EAAE,EAAE,CAC1D,IAAI,CAAC,SAAS,CAAC,MAAM,CAAI,OAAO,EAAE,IAAI,CAAC;YACzC,MAAM,EAAE,CAAC,EAAU,EAAE,IAAgB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAI,OAAO,EAAE,EAAE,EAAE,IAAI,CAAC;YACxF,MAAM,EAAE,CAAC,EAAU,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,OAAO,EAAE,EAAE,CAAC;SAC9D,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,kDAAkD;IAClD,8EAA8E;IAE9E;;;OAGG;IACO,mBAAmB,CAAC,OAAsB;QAClD,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YAC7B,OAAO,OAAO,IAAI,EAAE,CAAC;QACvB,CAAC;QAED,8CAA8C;QAC9C,MAAM,WAAW,GAAG,EAAE,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAE,IAAa,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QAEjF,OAAO;YACL,GAAG,OAAO;YACV,KAAK,EAAE,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE,CAAC,EAAE,WAAW,CAAC;SAChD,CAAC;IACJ,CAAC;IAED,8EAA8E;IAC9E,aAAa;IACb,8EAA8E;IAE9E;;OAEG;IACH,UAAU;QACR,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACxD,CAAC;IAED;;OAEG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;CACF","sourcesContent":["/**\n * Firestore Collection\n *\n * Clase base abstracta para crear servicios de colección tipados.\n * Extiende esta clase para tener un servicio dedicado por entidad.\n */\n\nimport { 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 * Clase base para servicios de colección tipados.\n *\n * Extiende esta clase para crear un servicio dedicado para cada entidad,\n * con métodos personalizados y tipado fuerte.\n *\n * @example\n * ```typescript\n * // Definir el modelo\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n *   role: 'admin' | 'user';\n *   active: boolean;\n * }\n *\n * // Crear el servicio\n * @Injectable({ providedIn: 'root' })\n * export class UsersCollection extends FirestoreCollection<User> {\n *   constructor() {\n *     super('users');\n *   }\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n *\n *   async getAdmins(): Promise<User[]> {\n *     return this.query({\n *       where: [{ field: 'role', operator: '==', value: 'admin' }]\n *     });\n *   }\n *\n *   watchOnlineUsers(): Observable<User[]> {\n *     return this.watchQuery({\n *       where: [{ field: 'status', operator: '==', value: 'online' }]\n *     });\n *   }\n * }\n *\n * // Usar en componentes\n * @Component({...})\n * export class UsersComponent {\n *   private users = inject(UsersCollection);\n *\n *   admins$ = this.users.getAdmins();\n *   onlineUsers$ = this.users.watchOnlineUsers();\n *\n *   async createUser(data: Omit<User, 'id'>) {\n *     const user = await this.users.create(data);\n *   }\n * }\n * ```\n */\nexport abstract class FirestoreCollection<T extends FirestoreDocument> {\n  protected firestore = inject(FirestoreService);\n  protected readonly collectionPath: string;\n  protected readonly options: CollectionOptions;\n\n  /**\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   */\n  constructor(collectionPath: string, options: CollectionOptions = {}) {\n    this.collectionPath = collectionPath;\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 UsersCollection\n   * getUserDocuments(userId: string) {\n   *   return this.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 PROTEGIDOS (para override en subclases)\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   * Override este método para agregar filtros globales (ej: excluir soft-deleted).\n   */\n  protected 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"]}
|
|
266
|
+
//# 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\n *\n * Clase base abstracta para crear servicios de colección tipados.\n * Extiende esta clase para tener un servicio dedicado por entidad.\n */\n\nimport { 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 * Clase base para servicios de colección tipados.\n *\n * Extiende esta clase para crear un servicio dedicado para cada entidad,\n * con métodos personalizados y tipado fuerte.\n *\n * @example\n * ```typescript\n * // Definir el modelo\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n *   role: 'admin' | 'user';\n *   active: boolean;\n * }\n *\n * // Crear el servicio\n * @Injectable({ providedIn: 'root' })\n * export class UsersCollection extends FirestoreCollection<User> {\n *   constructor() {\n *     super('users');\n *   }\n *\n *   // Métodos personalizados\n *   async getActiveUsers(): Promise<User[]> {\n *     return this.query({\n *       where: [{ field: 'active', operator: '==', value: true }]\n *     });\n *   }\n *\n *   async getAdmins(): Promise<User[]> {\n *     return this.query({\n *       where: [{ field: 'role', operator: '==', value: 'admin' }]\n *     });\n *   }\n *\n *   watchOnlineUsers(): Observable<User[]> {\n *     return this.watchQuery({\n *       where: [{ field: 'status', operator: '==', value: 'online' }]\n *     });\n *   }\n * }\n *\n * // Usar en componentes\n * @Component({...})\n * export class UsersComponent {\n *   private users = inject(UsersCollection);\n *\n *   admins$ = this.users.getAdmins();\n *   onlineUsers$ = this.users.watchOnlineUsers();\n *\n *   async createUser(data: Omit<User, 'id'>) {\n *     const user = await this.users.create(data);\n *   }\n * }\n * ```\n */\nexport abstract class FirestoreCollection<T extends FirestoreDocument> {\n  protected firestore: FirestoreService;\n  protected readonly collectionPath: string;\n  protected readonly options: CollectionOptions;\n\n  /**\n   * @param collectionPath - Ruta de la colección en Firestore\n   * @param options - Opciones de configuración\n   */\n  constructor(collectionPath: string, options: CollectionOptions = {}) {\n    this.firestore = inject(FirestoreService);\n    this.collectionPath = collectionPath;\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 UsersCollection\n   * getUserDocuments(userId: string) {\n   *   return this.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 PROTEGIDOS (para override en subclases)\n  // ===========================================================================\n\n  /**\n   * Aplica filtros por defecto a las queries.\n   * Override este método para agregar filtros globales (ej: excluir soft-deleted).\n   */\n  protected 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"]}
|
|
@@ -76,32 +76,32 @@ export class LinkProcessorService {
|
|
|
76
76
|
let processedText = text;
|
|
77
77
|
// 1. Procesar enlaces estilo Markdown [texto](url) primero
|
|
78
78
|
if (processMarkdownLinks) {
|
|
79
|
-
// Usar exec en bucle (compatible con ES2018)
|
|
80
|
-
const markdownMatches = [];
|
|
81
|
-
this.markdownLinkRegex.lastIndex = 0;
|
|
82
|
-
let mdMatch;
|
|
83
|
-
while ((mdMatch = this.markdownLinkRegex.exec(processedText)) !== null) {
|
|
84
|
-
|
|
85
|
-
}
|
|
79
|
+
// // Usar exec en bucle (compatible con ES2018)
|
|
80
|
+
// const markdownMatches: RegExpExecArray[] = [];
|
|
81
|
+
// this.markdownLinkRegex.lastIndex = 0;
|
|
82
|
+
// let mdMatch: RegExpExecArray | null;
|
|
83
|
+
// while ((mdMatch = this.markdownLinkRegex.exec(processedText)) !== null) {
|
|
84
|
+
// markdownMatches.push(mdMatch);
|
|
85
|
+
// }
|
|
86
86
|
// Procesar de atrás hacia adelante para mantener las posiciones
|
|
87
|
-
for (let i = markdownMatches.length - 1; i >= 0; i--) {
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
}
|
|
87
|
+
// for (let i = markdownMatches.length - 1; i >= 0; i--) {
|
|
88
|
+
// const match = markdownMatches[i];
|
|
89
|
+
// const [fullMatch, linkText, url] = match;
|
|
90
|
+
// const startIndex = match.index!;
|
|
91
|
+
// const endIndex = startIndex + fullMatch.length;
|
|
92
|
+
// hasLinks = true;
|
|
93
|
+
// const isExternal = /^https?:\/\//.test(url);
|
|
94
|
+
// const target = (isExternal ? openExternalInNewTab : openInternalInNewTab)
|
|
95
|
+
// ? isExternal
|
|
96
|
+
// ? ' target="_blank" rel="noopener noreferrer"'
|
|
97
|
+
// : ' target="_blank"'
|
|
98
|
+
// : '';
|
|
99
|
+
// const typeClass = isExternal ? externalLinkClass : internalLinkClass;
|
|
100
|
+
// const classes = `${linkClass} ${typeClass}`.trim();
|
|
101
|
+
// const linkHtml = `<a href="${url}"${target} class="${classes}">${linkText}</a>`;
|
|
102
|
+
// processedText =
|
|
103
|
+
// processedText.substring(0, startIndex) + linkHtml + processedText.substring(endIndex);
|
|
104
|
+
// }
|
|
105
105
|
}
|
|
106
106
|
// 2. Procesar URLs externas directas
|
|
107
107
|
// Usar exec en bucle (compatible con ES2018)
|
|
@@ -139,35 +139,35 @@ export class LinkProcessorService {
|
|
|
139
139
|
processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);
|
|
140
140
|
}
|
|
141
141
|
// 3. Procesar rutas internas
|
|
142
|
-
// Usar exec en bucle (compatible con ES2018)
|
|
143
|
-
const internalMatches = [];
|
|
144
|
-
this.internalRouteRegex.lastIndex = 0;
|
|
145
|
-
let internalMatch;
|
|
146
|
-
while ((internalMatch = this.internalRouteRegex.exec(processedText)) !== null) {
|
|
147
|
-
|
|
148
|
-
}
|
|
142
|
+
// // Usar exec en bucle (compatible con ES2018)
|
|
143
|
+
// const internalMatches: RegExpExecArray[] = [];
|
|
144
|
+
// this.internalRouteRegex.lastIndex = 0;
|
|
145
|
+
// let internalMatch: RegExpExecArray | null;
|
|
146
|
+
// while ((internalMatch = this.internalRouteRegex.exec(processedText)) !== null) {
|
|
147
|
+
// internalMatches.push(internalMatch);
|
|
148
|
+
// }
|
|
149
149
|
// Procesar de atrás hacia adelante para mantener las posiciones
|
|
150
|
-
for (let i = internalMatches.length - 1; i >= 0; i--) {
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
}
|
|
150
|
+
// for (let i = internalMatches.length - 1; i >= 0; i--) {
|
|
151
|
+
// const match = internalMatches[i];
|
|
152
|
+
// const [fullMatch, prefix, route] = match;
|
|
153
|
+
// const startIndex = match.index!;
|
|
154
|
+
// const endIndex = startIndex + fullMatch.length;
|
|
155
|
+
// // Verificar que no esté ya dentro de un enlace HTML existente
|
|
156
|
+
// const textBefore = processedText.substring(0, startIndex);
|
|
157
|
+
// const lastOpenTag = textBefore.lastIndexOf('<a ');
|
|
158
|
+
// const lastCloseTag = textBefore.lastIndexOf('</a>');
|
|
159
|
+
// // Si hay un tag <a abierto sin cerrar, no procesamos
|
|
160
|
+
// if (lastOpenTag > lastCloseTag) {
|
|
161
|
+
// continue;
|
|
162
|
+
// }
|
|
163
|
+
// hasLinks = true;
|
|
164
|
+
// const target = openInternalInNewTab ? ' target="_blank"' : '';
|
|
165
|
+
// const classes = `${linkClass} ${internalLinkClass}`.trim();
|
|
166
|
+
// const linkHtml = `<a href="${route}"${target} class="${classes}">${route}</a>`;
|
|
167
|
+
// const replacement = `${prefix}${linkHtml}`;
|
|
168
|
+
// processedText =
|
|
169
|
+
// processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);
|
|
170
|
+
// }
|
|
171
171
|
// Si hay enlaces, sanitizar el HTML
|
|
172
172
|
if (hasLinks) {
|
|
173
173
|
return this.sanitizer.bypassSecurityTrustHtml(processedText);
|
|
@@ -256,4 +256,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
256
256
|
providedIn: 'root',
|
|
257
257
|
}]
|
|
258
258
|
}], ctorParameters: () => [{ type: i1.DomSanitizer }] });
|
|
259
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-processor.service.js","sourceRoot":"","sources":["../../../../../src/lib/services/link-processor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\nexport interface LinkProcessorConfig {\n  /** Whether to open external links in new tab (default: true) */\n  openExternalInNewTab?: boolean;\n  /** Whether to open internal links in new tab (default: false) */\n  openInternalInNewTab?: boolean;\n  /** Custom CSS classes for links */\n  linkClass?: string;\n  /** Custom CSS classes for external links */\n  externalLinkClass?: string;\n  /** Custom CSS classes for internal links */\n  internalLinkClass?: string;\n  /** Whether to process Markdown-style links [text](url) (default: true) */\n  processMarkdownLinks?: boolean;\n}\n\n/**\n * LinkProcessorService - Service for processing text content to convert URLs and internal routes into clickable links.\n *\n * This service automatically detects external URLs (http/https), internal routes (starting with /),\n * and Markdown-style links [text](url) and converts them into HTML anchor elements with appropriate attributes.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private linkProcessor: LinkProcessorService) {}\n *\n * processText() {\n *   const text = 'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)';\n *   const processed = this.linkProcessor.processLinks(text);\n *   // Returns SafeHtml with clickable links\n * }\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LinkProcessorService {\n  // Regex para detectar URLs completas (http/https) - captura toda la URL y luego limpiamos puntuación\n  private readonly urlRegex = /(https?:\\/\\/[^\\s]+)/g;\n\n  // Regex para detectar rutas internas - captura toda la ruta y luego limpiamos puntuación\n  private readonly internalRouteRegex = /(\\s|^)(\\/[^\\s]*)/g;\n\n  // Regex para detectar enlaces estilo Markdown [texto](url)\n  private readonly markdownLinkRegex = /\\[([^\\]]+)\\]\\(([^)]+)\\)/g;\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  /**\n   * Limpia la puntuación del final de una URL.\n   * Mantiene caracteres válidos de URL pero remueve signos de puntuación comunes al final.\n   * Preserva parámetros de consulta, fragmentos y caracteres válidos en URLs.\n   */\n  private cleanUrlPunctuation(url: string): string {\n    // Caracteres que consideramos puntuación al final de oración, pero NO parte de URLs\n    // No incluimos & o = que son parte de query params, ni # que es parte de fragmentos\n    const trailingPunctuation = /[.,;!?)]+$/;\n\n    // Casos especiales: si la URL termina con paréntesis pero no tiene paréntesis de apertura\n    // probablemente el paréntesis no es parte de la URL\n    const hasOpeningParen = url.includes('(');\n    const endsWithClosingParen = url.endsWith(')');\n\n    if (endsWithClosingParen && !hasOpeningParen) {\n      // Remover el paréntesis de cierre si no hay uno de apertura\n      url = url.replace(/\\)$/, '');\n    }\n\n    return url.replace(trailingPunctuation, '');\n  }\n\n  /**\n   * Procesa texto para convertir enlaces en elementos <a> clickeables.\n   * Detecta automáticamente URLs externas, rutas internas y enlaces estilo Markdown.\n   *\n   * @param text - Texto a procesar\n   * @param config - Configuración del procesamiento\n   * @returns SafeHtml con enlaces procesados o string original\n   *\n   * @example\n   * ```typescript\n   * const result = this.linkProcessor.processLinks(\n   *   'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)',\n   *   {\n   *     openExternalInNewTab: true,\n   *     openInternalInNewTab: false,\n   *     processMarkdownLinks: true,\n   *     linkClass: 'custom-link'\n   *   }\n   * );\n   * ```\n   */\n  processLinks(text: string, config: LinkProcessorConfig = {}): SafeHtml | string {\n    if (!text) return '';\n\n    const {\n      openExternalInNewTab = true,\n      openInternalInNewTab = false,\n      linkClass = 'processed-link',\n      externalLinkClass = 'external-link',\n      internalLinkClass = 'internal-link',\n      processMarkdownLinks = true,\n    } = config;\n\n    let hasLinks = false;\n    let processedText = text;\n\n    // 1. Procesar enlaces estilo Markdown [texto](url) primero\n    if (processMarkdownLinks) {\n      // Usar exec en bucle (compatible con ES2018)\n      const markdownMatches: RegExpExecArray[] = [];\n      this.markdownLinkRegex.lastIndex = 0;\n      let mdMatch: RegExpExecArray | null;\n      while ((mdMatch = this.markdownLinkRegex.exec(processedText)) !== null) {\n        markdownMatches.push(mdMatch);\n      }\n\n      // Procesar de atrás hacia adelante para mantener las posiciones\n      for (let i = markdownMatches.length - 1; i >= 0; i--) {\n        const match = markdownMatches[i];\n        const [fullMatch, linkText, url] = match;\n        const startIndex = match.index!;\n        const endIndex = startIndex + fullMatch.length;\n\n        hasLinks = true;\n        const isExternal = /^https?:\\/\\//.test(url);\n        const target = (isExternal ? openExternalInNewTab : openInternalInNewTab)\n          ? isExternal\n            ? ' target=\"_blank\" rel=\"noopener noreferrer\"'\n            : ' target=\"_blank\"'\n          : '';\n        const typeClass = isExternal ? externalLinkClass : internalLinkClass;\n        const classes = `${linkClass} ${typeClass}`.trim();\n        const linkHtml = `<a href=\"${url}\"${target} class=\"${classes}\">${linkText}</a>`;\n\n        processedText =\n          processedText.substring(0, startIndex) + linkHtml + processedText.substring(endIndex);\n      }\n    }\n\n    // 2. Procesar URLs externas directas\n    // Usar exec en bucle (compatible con ES2018)\n    const urlMatches: RegExpExecArray[] = [];\n    this.urlRegex.lastIndex = 0;\n    let urlMatch: RegExpExecArray | null;\n    while ((urlMatch = this.urlRegex.exec(processedText)) !== null) {\n      urlMatches.push(urlMatch);\n    }\n\n    // Procesar de atrás hacia adelante para mantener las posiciones\n    for (let i = urlMatches.length - 1; i >= 0; i--) {\n      const match = urlMatches[i];\n      const [fullMatch, url] = match;\n      const startIndex = match.index!;\n      const endIndex = startIndex + fullMatch.length;\n\n      // Verificar que no esté ya dentro de un enlace HTML existente\n      const textBefore = processedText.substring(0, startIndex);\n      const lastOpenTag = textBefore.lastIndexOf('<a ');\n      const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n      // Si hay un tag <a abierto sin cerrar, no procesamos\n      if (lastOpenTag > lastCloseTag) {\n        continue;\n      }\n\n      // Limpiar puntuación del final de la URL\n      const cleanUrl = this.cleanUrlPunctuation(url);\n      const punctuationRemoved = url !== cleanUrl;\n      const punctuation = punctuationRemoved ? url.substring(cleanUrl.length) : '';\n\n      hasLinks = true;\n      const target = openExternalInNewTab ? ' target=\"_blank\" rel=\"noopener noreferrer\"' : '';\n      const classes = `${linkClass} ${externalLinkClass}`.trim();\n      const linkHtml = `<a href=\"${cleanUrl}\"${target} class=\"${classes}\">${cleanUrl}</a>`;\n\n      // Reemplazar el URL original con el enlace + puntuación si existía\n      const replacement = punctuationRemoved ? linkHtml + punctuation : linkHtml;\n      processedText =\n        processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);\n    }\n\n    // 3. Procesar rutas internas\n    // Usar exec en bucle (compatible con ES2018)\n    const internalMatches: RegExpExecArray[] = [];\n    this.internalRouteRegex.lastIndex = 0;\n    let internalMatch: RegExpExecArray | null;\n    while ((internalMatch = this.internalRouteRegex.exec(processedText)) !== null) {\n      internalMatches.push(internalMatch);\n    }\n\n    // Procesar de atrás hacia adelante para mantener las posiciones\n    for (let i = internalMatches.length - 1; i >= 0; i--) {\n      const match = internalMatches[i];\n      const [fullMatch, prefix, route] = match;\n      const startIndex = match.index!;\n      const endIndex = startIndex + fullMatch.length;\n\n      // Verificar que no esté ya dentro de un enlace HTML existente\n      const textBefore = processedText.substring(0, startIndex);\n      const lastOpenTag = textBefore.lastIndexOf('<a ');\n      const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n      // Si hay un tag <a abierto sin cerrar, no procesamos\n      if (lastOpenTag > lastCloseTag) {\n        continue;\n      }\n\n      hasLinks = true;\n      const target = openInternalInNewTab ? ' target=\"_blank\"' : '';\n      const classes = `${linkClass} ${internalLinkClass}`.trim();\n      const linkHtml = `<a href=\"${route}\"${target} class=\"${classes}\">${route}</a>`;\n\n      const replacement = `${prefix}${linkHtml}`;\n      processedText =\n        processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);\n    }\n\n    // Si hay enlaces, sanitizar el HTML\n    if (hasLinks) {\n      return this.sanitizer.bypassSecurityTrustHtml(processedText);\n    }\n\n    return text;\n  }\n\n  /**\n   * Detecta si un texto contiene enlaces (URLs, rutas internas o enlaces Markdown).\n   *\n   * @param text - Texto a analizar\n   * @returns true si contiene enlaces\n   *\n   * @example\n   * ```typescript\n   * const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com or [docs](https://docs.com)');\n   * // Returns: true\n   * ```\n   */\n  hasLinks(text: string): boolean {\n    if (!text) return false;\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    return (\n      this.urlRegex.test(text) ||\n      this.internalRouteRegex.test(text) ||\n      this.markdownLinkRegex.test(text)\n    );\n  }\n\n  /**\n   * Extrae todos los enlaces de un texto.\n   *\n   * @param text - Texto a analizar\n   * @returns Array de enlaces encontrados con su tipo y texto (si es Markdown)\n   *\n   * @example\n   * ```typescript\n   * const links = this.linkProcessor.extractLinks('Visit https://example.com, /profile, or [docs](https://docs.com)');\n   * // Returns: [\n   * //   { url: 'https://example.com', type: 'external', text: 'https://example.com' },\n   * //   { url: '/profile', type: 'internal', text: '/profile' },\n   * //   { url: 'https://docs.com', type: 'external', text: 'docs' }\n   * // ]\n   * ```\n   */\n  extractLinks(text: string): Array<{ url: string; type: 'external' | 'internal'; text: string }> {\n    if (!text) return [];\n\n    const links: Array<{ url: string; type: 'external' | 'internal'; text: string }> = [];\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    // Extraer enlaces Markdown primero\n    let match;\n    while ((match = this.markdownLinkRegex.exec(text)) !== null) {\n      const url = match[2];\n      const linkText = match[1];\n      const type = /^https?:\\/\\//.test(url) ? 'external' : 'internal';\n      links.push({ url, type, text: linkText });\n    }\n\n    // Extraer URLs externas directas\n    while ((match = this.urlRegex.exec(text)) !== null) {\n      const url = match[1];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'external', text: url });\n      }\n    }\n\n    // Extraer rutas internas directas\n    while ((match = this.internalRouteRegex.exec(text)) !== null) {\n      const url = match[2];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'internal', text: url });\n      }\n    }\n\n    return links;\n  }\n}\n"]}
|
|
259
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-processor.service.js","sourceRoot":"","sources":["../../../../../src/lib/services/link-processor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\nexport interface LinkProcessorConfig {\n  /** Whether to open external links in new tab (default: true) */\n  openExternalInNewTab?: boolean;\n  /** Whether to open internal links in new tab (default: false) */\n  openInternalInNewTab?: boolean;\n  /** Custom CSS classes for links */\n  linkClass?: string;\n  /** Custom CSS classes for external links */\n  externalLinkClass?: string;\n  /** Custom CSS classes for internal links */\n  internalLinkClass?: string;\n  /** Whether to process Markdown-style links [text](url) (default: true) */\n  processMarkdownLinks?: boolean;\n}\n\n/**\n * LinkProcessorService - Service for processing text content to convert URLs and internal routes into clickable links.\n *\n * This service automatically detects external URLs (http/https), internal routes (starting with /),\n * and Markdown-style links [text](url) and converts them into HTML anchor elements with appropriate attributes.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private linkProcessor: LinkProcessorService) {}\n *\n * processText() {\n *   const text = 'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)';\n *   const processed = this.linkProcessor.processLinks(text);\n *   // Returns SafeHtml with clickable links\n * }\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LinkProcessorService {\n  // Regex para detectar URLs completas (http/https) - captura toda la URL y luego limpiamos puntuación\n  private readonly urlRegex = /(https?:\\/\\/[^\\s]+)/g;\n\n  // Regex para detectar rutas internas - captura toda la ruta y luego limpiamos puntuación\n  private readonly internalRouteRegex = /(\\s|^)(\\/[^\\s]*)/g;\n\n  // Regex para detectar enlaces estilo Markdown [texto](url)\n  private readonly markdownLinkRegex = /\\[([^\\]]+)\\]\\(([^)]+)\\)/g;\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  /**\n   * Limpia la puntuación del final de una URL.\n   * Mantiene caracteres válidos de URL pero remueve signos de puntuación comunes al final.\n   * Preserva parámetros de consulta, fragmentos y caracteres válidos en URLs.\n   */\n  private cleanUrlPunctuation(url: string): string {\n    // Caracteres que consideramos puntuación al final de oración, pero NO parte de URLs\n    // No incluimos & o = que son parte de query params, ni # que es parte de fragmentos\n    const trailingPunctuation = /[.,;!?)]+$/;\n\n    // Casos especiales: si la URL termina con paréntesis pero no tiene paréntesis de apertura\n    // probablemente el paréntesis no es parte de la URL\n    const hasOpeningParen = url.includes('(');\n    const endsWithClosingParen = url.endsWith(')');\n\n    if (endsWithClosingParen && !hasOpeningParen) {\n      // Remover el paréntesis de cierre si no hay uno de apertura\n      url = url.replace(/\\)$/, '');\n    }\n\n    return url.replace(trailingPunctuation, '');\n  }\n\n  /**\n   * Procesa texto para convertir enlaces en elementos <a> clickeables.\n   * Detecta automáticamente URLs externas, rutas internas y enlaces estilo Markdown.\n   *\n   * @param text - Texto a procesar\n   * @param config - Configuración del procesamiento\n   * @returns SafeHtml con enlaces procesados o string original\n   *\n   * @example\n   * ```typescript\n   * const result = this.linkProcessor.processLinks(\n   *   'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)',\n   *   {\n   *     openExternalInNewTab: true,\n   *     openInternalInNewTab: false,\n   *     processMarkdownLinks: true,\n   *     linkClass: 'custom-link'\n   *   }\n   * );\n   * ```\n   */\n  processLinks(text: string, config: LinkProcessorConfig = {}): SafeHtml | string {\n    if (!text) return '';\n\n    const {\n      openExternalInNewTab = true,\n      openInternalInNewTab = false,\n      linkClass = 'processed-link',\n      externalLinkClass = 'external-link',\n      internalLinkClass = 'internal-link',\n      processMarkdownLinks = true,\n    } = config;\n\n    let hasLinks = false;\n    let processedText = text;\n\n    // 1. Procesar enlaces estilo Markdown [texto](url) primero\n    if (processMarkdownLinks) {\n      // // Usar exec en bucle (compatible con ES2018)\n      // const markdownMatches: RegExpExecArray[] = [];\n      // this.markdownLinkRegex.lastIndex = 0;\n      // let mdMatch: RegExpExecArray | null;\n      // while ((mdMatch = this.markdownLinkRegex.exec(processedText)) !== null) {\n      //   markdownMatches.push(mdMatch);\n      // }\n\n      // Procesar de atrás hacia adelante para mantener las posiciones\n      // for (let i = markdownMatches.length - 1; i >= 0; i--) {\n      //   const match = markdownMatches[i];\n      //   const [fullMatch, linkText, url] = match;\n      //   const startIndex = match.index!;\n      //   const endIndex = startIndex + fullMatch.length;\n\n      //   hasLinks = true;\n      //   const isExternal = /^https?:\\/\\//.test(url);\n      //   const target = (isExternal ? openExternalInNewTab : openInternalInNewTab)\n      //     ? isExternal\n      //       ? ' target=\"_blank\" rel=\"noopener noreferrer\"'\n      //       : ' target=\"_blank\"'\n      //     : '';\n      //   const typeClass = isExternal ? externalLinkClass : internalLinkClass;\n      //   const classes = `${linkClass} ${typeClass}`.trim();\n      //   const linkHtml = `<a href=\"${url}\"${target} class=\"${classes}\">${linkText}</a>`;\n\n      //   processedText =\n      //     processedText.substring(0, startIndex) + linkHtml + processedText.substring(endIndex);\n      // }\n    }\n\n    // 2. Procesar URLs externas directas\n    // Usar exec en bucle (compatible con ES2018)\n    const urlMatches: RegExpExecArray[] = [];\n    this.urlRegex.lastIndex = 0;\n    let urlMatch: RegExpExecArray | null;\n    while ((urlMatch = this.urlRegex.exec(processedText)) !== null) {\n      urlMatches.push(urlMatch);\n    }\n\n    // Procesar de atrás hacia adelante para mantener las posiciones\n    for (let i = urlMatches.length - 1; i >= 0; i--) {\n      const match = urlMatches[i];\n      const [fullMatch, url] = match;\n      const startIndex = match.index!;\n      const endIndex = startIndex + fullMatch.length;\n\n      // Verificar que no esté ya dentro de un enlace HTML existente\n      const textBefore = processedText.substring(0, startIndex);\n      const lastOpenTag = textBefore.lastIndexOf('<a ');\n      const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n      // Si hay un tag <a abierto sin cerrar, no procesamos\n      if (lastOpenTag > lastCloseTag) {\n        continue;\n      }\n\n      // Limpiar puntuación del final de la URL\n      const cleanUrl = this.cleanUrlPunctuation(url);\n      const punctuationRemoved = url !== cleanUrl;\n      const punctuation = punctuationRemoved ? url.substring(cleanUrl.length) : '';\n\n      hasLinks = true;\n      const target = openExternalInNewTab ? ' target=\"_blank\" rel=\"noopener noreferrer\"' : '';\n      const classes = `${linkClass} ${externalLinkClass}`.trim();\n      const linkHtml = `<a href=\"${cleanUrl}\"${target} class=\"${classes}\">${cleanUrl}</a>`;\n\n      // Reemplazar el URL original con el enlace + puntuación si existía\n      const replacement = punctuationRemoved ? linkHtml + punctuation : linkHtml;\n      processedText =\n        processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);\n    }\n\n    // 3. Procesar rutas internas\n    // // Usar exec en bucle (compatible con ES2018)\n    // const internalMatches: RegExpExecArray[] = [];\n    // this.internalRouteRegex.lastIndex = 0;\n    // let internalMatch: RegExpExecArray | null;\n    // while ((internalMatch = this.internalRouteRegex.exec(processedText)) !== null) {\n    //   internalMatches.push(internalMatch);\n    // }\n\n    // Procesar de atrás hacia adelante para mantener las posiciones\n    // for (let i = internalMatches.length - 1; i >= 0; i--) {\n    //   const match = internalMatches[i];\n    //   const [fullMatch, prefix, route] = match;\n    //   const startIndex = match.index!;\n    //   const endIndex = startIndex + fullMatch.length;\n\n    //   // Verificar que no esté ya dentro de un enlace HTML existente\n    //   const textBefore = processedText.substring(0, startIndex);\n    //   const lastOpenTag = textBefore.lastIndexOf('<a ');\n    //   const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n    //   // Si hay un tag <a abierto sin cerrar, no procesamos\n    //   if (lastOpenTag > lastCloseTag) {\n    //     continue;\n    //   }\n\n    //   hasLinks = true;\n    //   const target = openInternalInNewTab ? ' target=\"_blank\"' : '';\n    //   const classes = `${linkClass} ${internalLinkClass}`.trim();\n    //   const linkHtml = `<a href=\"${route}\"${target} class=\"${classes}\">${route}</a>`;\n\n    //   const replacement = `${prefix}${linkHtml}`;\n    //   processedText =\n    //     processedText.substring(0, startIndex) + replacement + processedText.substring(endIndex);\n    // }\n\n    // Si hay enlaces, sanitizar el HTML\n    if (hasLinks) {\n      return this.sanitizer.bypassSecurityTrustHtml(processedText);\n    }\n\n    return text;\n  }\n\n  /**\n   * Detecta si un texto contiene enlaces (URLs, rutas internas o enlaces Markdown).\n   *\n   * @param text - Texto a analizar\n   * @returns true si contiene enlaces\n   *\n   * @example\n   * ```typescript\n   * const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com or [docs](https://docs.com)');\n   * // Returns: true\n   * ```\n   */\n  hasLinks(text: string): boolean {\n    if (!text) return false;\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    return (\n      this.urlRegex.test(text) ||\n      this.internalRouteRegex.test(text) ||\n      this.markdownLinkRegex.test(text)\n    );\n  }\n\n  /**\n   * Extrae todos los enlaces de un texto.\n   *\n   * @param text - Texto a analizar\n   * @returns Array de enlaces encontrados con su tipo y texto (si es Markdown)\n   *\n   * @example\n   * ```typescript\n   * const links = this.linkProcessor.extractLinks('Visit https://example.com, /profile, or [docs](https://docs.com)');\n   * // Returns: [\n   * //   { url: 'https://example.com', type: 'external', text: 'https://example.com' },\n   * //   { url: '/profile', type: 'internal', text: '/profile' },\n   * //   { url: 'https://docs.com', type: 'external', text: 'docs' }\n   * // ]\n   * ```\n   */\n  extractLinks(text: string): Array<{ url: string; type: 'external' | 'internal'; text: string }> {\n    if (!text) return [];\n\n    const links: Array<{ url: string; type: 'external' | 'internal'; text: string }> = [];\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    // Extraer enlaces Markdown primero\n    let match;\n    while ((match = this.markdownLinkRegex.exec(text)) !== null) {\n      const url = match[2];\n      const linkText = match[1];\n      const type = /^https?:\\/\\//.test(url) ? 'external' : 'internal';\n      links.push({ url, type, text: linkText });\n    }\n\n    // Extraer URLs externas directas\n    while ((match = this.urlRegex.exec(text)) !== null) {\n      const url = match[1];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'external', text: url });\n      }\n    }\n\n    // Extraer rutas internas directas\n    while ((match = this.internalRouteRegex.exec(text)) !== null) {\n      const url = match[2];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'internal', text: url });\n      }\n    }\n\n    return links;\n  }\n}\n"]}
|