valtech-components 2.0.445 → 2.0.446
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/public-api.mjs +9 -6
- package/fesm2022/valtech-components.mjs +5 -3237
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/organisms/article/article.component.d.ts +5 -5
- package/package.json +1 -1
- package/public-api.d.ts +0 -5
- package/esm2022/lib/components/molecules/feedback-form/feedback-form.component.mjs +0 -352
- package/esm2022/lib/components/molecules/feedback-form/types.mjs +0 -2
- package/esm2022/lib/services/auth/auth-state.service.mjs +0 -173
- package/esm2022/lib/services/auth/auth.service.mjs +0 -454
- package/esm2022/lib/services/auth/config.mjs +0 -76
- package/esm2022/lib/services/auth/guards.mjs +0 -194
- package/esm2022/lib/services/auth/index.mjs +0 -70
- package/esm2022/lib/services/auth/interceptor.mjs +0 -98
- package/esm2022/lib/services/auth/storage.service.mjs +0 -138
- package/esm2022/lib/services/auth/sync.service.mjs +0 -146
- package/esm2022/lib/services/auth/token.service.mjs +0 -113
- package/esm2022/lib/services/auth/types.mjs +0 -29
- package/esm2022/lib/services/content/content-types/blog.mjs +0 -275
- package/esm2022/lib/services/content/content-types/documentation.mjs +0 -303
- package/esm2022/lib/services/content/content-types/news.mjs +0 -277
- package/esm2022/lib/services/content/index.mjs +0 -51
- package/esm2022/lib/services/content/transformer.mjs +0 -265
- package/esm2022/lib/services/content/types.mjs +0 -41
- package/esm2022/lib/services/feedback/config.mjs +0 -49
- package/esm2022/lib/services/feedback/feedback.service.mjs +0 -174
- package/esm2022/lib/services/feedback/index.mjs +0 -44
- package/esm2022/lib/services/feedback/types.mjs +0 -30
- package/lib/components/molecules/feedback-form/feedback-form.component.d.ts +0 -56
- package/lib/components/molecules/feedback-form/types.d.ts +0 -54
- package/lib/services/auth/auth-state.service.d.ts +0 -85
- package/lib/services/auth/auth.service.d.ts +0 -146
- package/lib/services/auth/config.d.ts +0 -38
- package/lib/services/auth/guards.d.ts +0 -123
- package/lib/services/auth/index.d.ts +0 -63
- package/lib/services/auth/interceptor.d.ts +0 -22
- package/lib/services/auth/storage.service.d.ts +0 -48
- package/lib/services/auth/sync.service.d.ts +0 -49
- package/lib/services/auth/token.service.d.ts +0 -51
- package/lib/services/auth/types.d.ts +0 -315
- package/lib/services/content/content-types/blog.d.ts +0 -148
- package/lib/services/content/content-types/documentation.d.ts +0 -183
- package/lib/services/content/content-types/news.d.ts +0 -162
- package/lib/services/content/index.d.ts +0 -49
- package/lib/services/content/transformer.d.ts +0 -96
- package/lib/services/content/types.d.ts +0 -220
- package/lib/services/feedback/config.d.ts +0 -35
- package/lib/services/feedback/feedback.service.d.ts +0 -76
- package/lib/services/feedback/index.d.ts +0 -40
- package/lib/services/feedback/types.d.ts +0 -107
|
@@ -1,174 +0,0 @@
|
|
|
1
|
-
import { Injectable, inject } from '@angular/core';
|
|
2
|
-
import { HttpClient } from '@angular/common/http';
|
|
3
|
-
import { firstValueFrom } from 'rxjs';
|
|
4
|
-
import { VALTECH_FEEDBACK_CONFIG } from './config';
|
|
5
|
-
import * as i0 from "@angular/core";
|
|
6
|
-
/**
|
|
7
|
-
* Servicio para gestionar feedback de usuarios.
|
|
8
|
-
*
|
|
9
|
-
* @example
|
|
10
|
-
* ```typescript
|
|
11
|
-
* @Component({...})
|
|
12
|
-
* export class MyComponent {
|
|
13
|
-
* private feedbackService = inject(FeedbackService);
|
|
14
|
-
*
|
|
15
|
-
* async submitFeedback() {
|
|
16
|
-
* const response = await this.feedbackService.createAsync(
|
|
17
|
-
* 'feedback',
|
|
18
|
-
* 'Mi comentario',
|
|
19
|
-
* 'Descripción detallada...'
|
|
20
|
-
* );
|
|
21
|
-
* console.log('Feedback enviado:', response.feedbackId);
|
|
22
|
-
* }
|
|
23
|
-
* }
|
|
24
|
-
* ```
|
|
25
|
-
*/
|
|
26
|
-
export class FeedbackService {
|
|
27
|
-
constructor() {
|
|
28
|
-
this.config = inject(VALTECH_FEEDBACK_CONFIG);
|
|
29
|
-
this.http = inject(HttpClient);
|
|
30
|
-
}
|
|
31
|
-
/**
|
|
32
|
-
* URL base para endpoints de feedback.
|
|
33
|
-
*/
|
|
34
|
-
get baseUrl() {
|
|
35
|
-
return `${this.config.apiUrl}${this.config.feedbackPrefix}`;
|
|
36
|
-
}
|
|
37
|
-
/**
|
|
38
|
-
* Captura el contexto del dispositivo automáticamente.
|
|
39
|
-
*/
|
|
40
|
-
captureDeviceContext() {
|
|
41
|
-
const ua = navigator.userAgent;
|
|
42
|
-
return {
|
|
43
|
-
browser: this.detectBrowser(ua),
|
|
44
|
-
os: this.detectOS(ua),
|
|
45
|
-
viewport: `${window.innerWidth}x${window.innerHeight}`,
|
|
46
|
-
language: navigator.language,
|
|
47
|
-
userAgent: ua,
|
|
48
|
-
pageUrl: window.location.href,
|
|
49
|
-
};
|
|
50
|
-
}
|
|
51
|
-
/**
|
|
52
|
-
* Crea un nuevo feedback.
|
|
53
|
-
*
|
|
54
|
-
* @param type - Tipo de feedback
|
|
55
|
-
* @param title - Título del feedback
|
|
56
|
-
* @param description - Descripción detallada
|
|
57
|
-
* @param attachments - URLs de archivos adjuntos (opcional)
|
|
58
|
-
* @param contentRef - Referencia a contenido específico (opcional)
|
|
59
|
-
* @returns Observable con la respuesta
|
|
60
|
-
*/
|
|
61
|
-
create(type, title, description, attachments = [], contentRef) {
|
|
62
|
-
const request = {
|
|
63
|
-
type,
|
|
64
|
-
title,
|
|
65
|
-
description,
|
|
66
|
-
attachments,
|
|
67
|
-
contentRef,
|
|
68
|
-
deviceContext: this.captureDeviceContext(),
|
|
69
|
-
appId: this.config.appId,
|
|
70
|
-
};
|
|
71
|
-
return this.http.post(this.baseUrl, request);
|
|
72
|
-
}
|
|
73
|
-
/**
|
|
74
|
-
* Crea un nuevo feedback (versión async/await).
|
|
75
|
-
*/
|
|
76
|
-
async createAsync(type, title, description, attachments = [], contentRef) {
|
|
77
|
-
return firstValueFrom(this.create(type, title, description, attachments, contentRef));
|
|
78
|
-
}
|
|
79
|
-
/**
|
|
80
|
-
* Obtiene un feedback por ID (solo el propietario).
|
|
81
|
-
*
|
|
82
|
-
* @param feedbackId - ID del feedback
|
|
83
|
-
* @returns Observable con la respuesta
|
|
84
|
-
*/
|
|
85
|
-
getById(feedbackId) {
|
|
86
|
-
return this.http.get(`${this.baseUrl}/${feedbackId}`);
|
|
87
|
-
}
|
|
88
|
-
/**
|
|
89
|
-
* Obtiene un feedback por ID (versión async/await).
|
|
90
|
-
*/
|
|
91
|
-
async getByIdAsync(feedbackId) {
|
|
92
|
-
return firstValueFrom(this.getById(feedbackId));
|
|
93
|
-
}
|
|
94
|
-
/**
|
|
95
|
-
* Valida si un archivo cumple con las restricciones.
|
|
96
|
-
*/
|
|
97
|
-
validateFile(file) {
|
|
98
|
-
// Verificar tamaño
|
|
99
|
-
if (file.size > this.config.maxFileSize) {
|
|
100
|
-
const maxSizeMB = Math.round(this.config.maxFileSize / (1024 * 1024));
|
|
101
|
-
return {
|
|
102
|
-
valid: false,
|
|
103
|
-
error: `El archivo excede el tamaño máximo de ${maxSizeMB}MB`,
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
// Verificar tipo
|
|
107
|
-
const allowedTypes = this.config.allowedFileTypes || [];
|
|
108
|
-
const isAllowed = allowedTypes.some((pattern) => {
|
|
109
|
-
if (pattern.endsWith('/*')) {
|
|
110
|
-
const baseType = pattern.replace('/*', '');
|
|
111
|
-
return file.type.startsWith(baseType);
|
|
112
|
-
}
|
|
113
|
-
return file.type === pattern;
|
|
114
|
-
});
|
|
115
|
-
if (!isAllowed) {
|
|
116
|
-
return {
|
|
117
|
-
valid: false,
|
|
118
|
-
error: 'Tipo de archivo no permitido',
|
|
119
|
-
};
|
|
120
|
-
}
|
|
121
|
-
return { valid: true };
|
|
122
|
-
}
|
|
123
|
-
/**
|
|
124
|
-
* Obtiene la configuración actual del servicio.
|
|
125
|
-
*/
|
|
126
|
-
getConfig() {
|
|
127
|
-
return this.config;
|
|
128
|
-
}
|
|
129
|
-
// =========================================================================
|
|
130
|
-
// Helpers privados para detección de browser/OS
|
|
131
|
-
// =========================================================================
|
|
132
|
-
detectBrowser(ua) {
|
|
133
|
-
if (ua.includes('Edg/'))
|
|
134
|
-
return 'Edge';
|
|
135
|
-
if (ua.includes('Chrome/'))
|
|
136
|
-
return 'Chrome';
|
|
137
|
-
if (ua.includes('Firefox/'))
|
|
138
|
-
return 'Firefox';
|
|
139
|
-
if (ua.includes('Safari/') && !ua.includes('Chrome'))
|
|
140
|
-
return 'Safari';
|
|
141
|
-
if (ua.includes('Opera') || ua.includes('OPR/'))
|
|
142
|
-
return 'Opera';
|
|
143
|
-
return 'Unknown';
|
|
144
|
-
}
|
|
145
|
-
detectOS(ua) {
|
|
146
|
-
if (ua.includes('Windows NT 10'))
|
|
147
|
-
return 'Windows 10';
|
|
148
|
-
if (ua.includes('Windows NT 11'))
|
|
149
|
-
return 'Windows 11';
|
|
150
|
-
if (ua.includes('Windows'))
|
|
151
|
-
return 'Windows';
|
|
152
|
-
if (ua.includes('Mac OS X')) {
|
|
153
|
-
const match = ua.match(/Mac OS X (\d+[._]\d+)/);
|
|
154
|
-
if (match) {
|
|
155
|
-
return `macOS ${match[1].replace('_', '.')}`;
|
|
156
|
-
}
|
|
157
|
-
return 'macOS';
|
|
158
|
-
}
|
|
159
|
-
if (ua.includes('Android'))
|
|
160
|
-
return 'Android';
|
|
161
|
-
if (ua.includes('iPhone') || ua.includes('iPad'))
|
|
162
|
-
return 'iOS';
|
|
163
|
-
if (ua.includes('Linux'))
|
|
164
|
-
return 'Linux';
|
|
165
|
-
return 'Unknown';
|
|
166
|
-
}
|
|
167
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
168
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, providedIn: 'root' }); }
|
|
169
|
-
}
|
|
170
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FeedbackService, decorators: [{
|
|
171
|
-
type: Injectable,
|
|
172
|
-
args: [{ providedIn: 'root' }]
|
|
173
|
-
}] });
|
|
174
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feedback.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/feedback/feedback.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { Observable, firstValueFrom } from 'rxjs';\nimport { VALTECH_FEEDBACK_CONFIG } from './config';\nimport {\n  CreateFeedbackRequest,\n  CreateFeedbackResponse,\n  GetFeedbackResponse,\n  DeviceContext,\n  FeedbackType,\n  ContentRef,\n} from './types';\n\n/**\n * Servicio para gestionar feedback de usuarios.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private feedbackService = inject(FeedbackService);\n *\n *   async submitFeedback() {\n *     const response = await this.feedbackService.createAsync(\n *       'feedback',\n *       'Mi comentario',\n *       'Descripción detallada...'\n *     );\n *     console.log('Feedback enviado:', response.feedbackId);\n *   }\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class FeedbackService {\n  private config = inject(VALTECH_FEEDBACK_CONFIG);\n  private http = inject(HttpClient);\n\n  /**\n   * URL base para endpoints de feedback.\n   */\n  private get baseUrl(): string {\n    return `${this.config.apiUrl}${this.config.feedbackPrefix}`;\n  }\n\n  /**\n   * Captura el contexto del dispositivo automáticamente.\n   */\n  captureDeviceContext(): DeviceContext {\n    const ua = navigator.userAgent;\n    return {\n      browser: this.detectBrowser(ua),\n      os: this.detectOS(ua),\n      viewport: `${window.innerWidth}x${window.innerHeight}`,\n      language: navigator.language,\n      userAgent: ua,\n      pageUrl: window.location.href,\n    };\n  }\n\n  /**\n   * Crea un nuevo feedback.\n   *\n   * @param type - Tipo de feedback\n   * @param title - Título del feedback\n   * @param description - Descripción detallada\n   * @param attachments - URLs de archivos adjuntos (opcional)\n   * @param contentRef - Referencia a contenido específico (opcional)\n   * @returns Observable con la respuesta\n   */\n  create(\n    type: FeedbackType,\n    title: string,\n    description: string,\n    attachments: string[] = [],\n    contentRef?: ContentRef\n  ): Observable<CreateFeedbackResponse> {\n    const request: CreateFeedbackRequest = {\n      type,\n      title,\n      description,\n      attachments,\n      contentRef,\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return this.http.post<CreateFeedbackResponse>(this.baseUrl, request);\n  }\n\n  /**\n   * Crea un nuevo feedback (versión async/await).\n   */\n  async createAsync(\n    type: FeedbackType,\n    title: string,\n    description: string,\n    attachments: string[] = [],\n    contentRef?: ContentRef\n  ): Promise<CreateFeedbackResponse> {\n    return firstValueFrom(\n      this.create(type, title, description, attachments, contentRef)\n    );\n  }\n\n  /**\n   * Obtiene un feedback por ID (solo el propietario).\n   *\n   * @param feedbackId - ID del feedback\n   * @returns Observable con la respuesta\n   */\n  getById(feedbackId: string): Observable<GetFeedbackResponse> {\n    return this.http.get<GetFeedbackResponse>(`${this.baseUrl}/${feedbackId}`);\n  }\n\n  /**\n   * Obtiene un feedback por ID (versión async/await).\n   */\n  async getByIdAsync(feedbackId: string): Promise<GetFeedbackResponse> {\n    return firstValueFrom(this.getById(feedbackId));\n  }\n\n  /**\n   * Valida si un archivo cumple con las restricciones.\n   */\n  validateFile(file: File): { valid: boolean; error?: string } {\n    // Verificar tamaño\n    if (file.size > this.config.maxFileSize!) {\n      const maxSizeMB = Math.round(this.config.maxFileSize! / (1024 * 1024));\n      return {\n        valid: false,\n        error: `El archivo excede el tamaño máximo de ${maxSizeMB}MB`,\n      };\n    }\n\n    // Verificar tipo\n    const allowedTypes = this.config.allowedFileTypes || [];\n    const isAllowed = allowedTypes.some((pattern) => {\n      if (pattern.endsWith('/*')) {\n        const baseType = pattern.replace('/*', '');\n        return file.type.startsWith(baseType);\n      }\n      return file.type === pattern;\n    });\n\n    if (!isAllowed) {\n      return {\n        valid: false,\n        error: 'Tipo de archivo no permitido',\n      };\n    }\n\n    return { valid: true };\n  }\n\n  /**\n   * Obtiene la configuración actual del servicio.\n   */\n  getConfig(): Readonly<typeof this.config> {\n    return this.config;\n  }\n\n  // =========================================================================\n  // Helpers privados para detección de browser/OS\n  // =========================================================================\n\n  private detectBrowser(ua: string): string {\n    if (ua.includes('Edg/')) return 'Edge';\n    if (ua.includes('Chrome/')) return 'Chrome';\n    if (ua.includes('Firefox/')) return 'Firefox';\n    if (ua.includes('Safari/') && !ua.includes('Chrome')) return 'Safari';\n    if (ua.includes('Opera') || ua.includes('OPR/')) return 'Opera';\n    return 'Unknown';\n  }\n\n  private detectOS(ua: string): string {\n    if (ua.includes('Windows NT 10')) return 'Windows 10';\n    if (ua.includes('Windows NT 11')) return 'Windows 11';\n    if (ua.includes('Windows')) return 'Windows';\n    if (ua.includes('Mac OS X')) {\n      const match = ua.match(/Mac OS X (\\d+[._]\\d+)/);\n      if (match) {\n        return `macOS ${match[1].replace('_', '.')}`;\n      }\n      return 'macOS';\n    }\n    if (ua.includes('Android')) return 'Android';\n    if (ua.includes('iPhone') || ua.includes('iPad')) return 'iOS';\n    if (ua.includes('Linux')) return 'Linux';\n    return 'Unknown';\n  }\n}\n"]}
|
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Valtech Feedback Service
|
|
3
|
-
*
|
|
4
|
-
* Servicio para gestionar feedback de usuarios a nivel de plataforma.
|
|
5
|
-
*
|
|
6
|
-
* @example
|
|
7
|
-
* ```typescript
|
|
8
|
-
* // main.ts - Configuración
|
|
9
|
-
* import { provideValtechFeedback } from 'valtech-components';
|
|
10
|
-
*
|
|
11
|
-
* bootstrapApplication(AppComponent, {
|
|
12
|
-
* providers: [
|
|
13
|
-
* provideValtechAuth({ apiUrl: environment.apiUrl }),
|
|
14
|
-
* provideValtechFeedback({
|
|
15
|
-
* apiUrl: environment.apiUrl,
|
|
16
|
-
* appId: 'my-app-name',
|
|
17
|
-
* }),
|
|
18
|
-
* ],
|
|
19
|
-
* });
|
|
20
|
-
*
|
|
21
|
-
* // component.ts - Uso
|
|
22
|
-
* import { FeedbackService } from 'valtech-components';
|
|
23
|
-
*
|
|
24
|
-
* @Component({...})
|
|
25
|
-
* export class MyComponent {
|
|
26
|
-
* private feedbackService = inject(FeedbackService);
|
|
27
|
-
*
|
|
28
|
-
* async submitFeedback() {
|
|
29
|
-
* const response = await this.feedbackService.createAsync(
|
|
30
|
-
* 'feedback',
|
|
31
|
-
* 'Título',
|
|
32
|
-
* 'Descripción...'
|
|
33
|
-
* );
|
|
34
|
-
* }
|
|
35
|
-
* }
|
|
36
|
-
* ```
|
|
37
|
-
*/
|
|
38
|
-
// Configuration
|
|
39
|
-
export { VALTECH_FEEDBACK_CONFIG, provideValtechFeedback, DEFAULT_FEEDBACK_CONFIG } from './config';
|
|
40
|
-
// Service
|
|
41
|
-
export { FeedbackService } from './feedback.service';
|
|
42
|
-
// Types
|
|
43
|
-
export { DEFAULT_FEEDBACK_TYPE_OPTIONS, } from './types';
|
|
44
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZlZWRiYWNrL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FvQ0c7QUFFSCxnQkFBZ0I7QUFDaEIsT0FBTyxFQUFFLHVCQUF1QixFQUFFLHNCQUFzQixFQUFFLHVCQUF1QixFQUFFLE1BQU0sVUFBVSxDQUFDO0FBRXBHLFVBQVU7QUFDVixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFckQsUUFBUTtBQUNSLE9BQU8sRUFZTCw2QkFBNkIsR0FDOUIsTUFBTSxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFZhbHRlY2ggRmVlZGJhY2sgU2VydmljZVxuICpcbiAqIFNlcnZpY2lvIHBhcmEgZ2VzdGlvbmFyIGZlZWRiYWNrIGRlIHVzdWFyaW9zIGEgbml2ZWwgZGUgcGxhdGFmb3JtYS5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gbWFpbi50cyAtIENvbmZpZ3VyYWNpw7NuXG4gKiBpbXBvcnQgeyBwcm92aWRlVmFsdGVjaEZlZWRiYWNrIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hBdXRoKHsgYXBpVXJsOiBlbnZpcm9ubWVudC5hcGlVcmwgfSksXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGZWVkYmFjayh7XG4gKiAgICAgICBhcGlVcmw6IGVudmlyb25tZW50LmFwaVVybCxcbiAqICAgICAgIGFwcElkOiAnbXktYXBwLW5hbWUnLFxuICogICAgIH0pLFxuICogICBdLFxuICogfSk7XG4gKlxuICogLy8gY29tcG9uZW50LnRzIC0gVXNvXG4gKiBpbXBvcnQgeyBGZWVkYmFja1NlcnZpY2UgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICpcbiAqIEBDb21wb25lbnQoey4uLn0pXG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGZlZWRiYWNrU2VydmljZSA9IGluamVjdChGZWVkYmFja1NlcnZpY2UpO1xuICpcbiAqICAgYXN5bmMgc3VibWl0RmVlZGJhY2soKSB7XG4gKiAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLmZlZWRiYWNrU2VydmljZS5jcmVhdGVBc3luYyhcbiAqICAgICAgICdmZWVkYmFjaycsXG4gKiAgICAgICAnVMOtdHVsbycsXG4gKiAgICAgICAnRGVzY3JpcGNpw7NuLi4uJ1xuICogICAgICk7XG4gKiAgIH1cbiAqIH1cbiAqIGBgYFxuICovXG5cbi8vIENvbmZpZ3VyYXRpb25cbmV4cG9ydCB7IFZBTFRFQ0hfRkVFREJBQ0tfQ09ORklHLCBwcm92aWRlVmFsdGVjaEZlZWRiYWNrLCBERUZBVUxUX0ZFRURCQUNLX0NPTkZJRyB9IGZyb20gJy4vY29uZmlnJztcblxuLy8gU2VydmljZVxuZXhwb3J0IHsgRmVlZGJhY2tTZXJ2aWNlIH0gZnJvbSAnLi9mZWVkYmFjay5zZXJ2aWNlJztcblxuLy8gVHlwZXNcbmV4cG9ydCB7XG4gIFZhbHRlY2hGZWVkYmFja0NvbmZpZyxcbiAgRmVlZGJhY2tUeXBlLFxuICBGZWVkYmFja1N0YXR1cyxcbiAgQ29udGVudFR5cGUsXG4gIENvbnRlbnRSZWYsXG4gIERldmljZUNvbnRleHQsXG4gIEZlZWRiYWNrLFxuICBDcmVhdGVGZWVkYmFja1JlcXVlc3QsXG4gIENyZWF0ZUZlZWRiYWNrUmVzcG9uc2UsXG4gIEdldEZlZWRiYWNrUmVzcG9uc2UsXG4gIEZlZWRiYWNrVHlwZU9wdGlvbixcbiAgREVGQVVMVF9GRUVEQkFDS19UWVBFX09QVElPTlMsXG59IGZyb20gJy4vdHlwZXMnO1xuIl19
|
|
@@ -1,30 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Configuración por defecto de tipos de feedback.
|
|
3
|
-
*/
|
|
4
|
-
export const DEFAULT_FEEDBACK_TYPE_OPTIONS = [
|
|
5
|
-
{
|
|
6
|
-
value: 'issue',
|
|
7
|
-
label: 'Reportar problema',
|
|
8
|
-
description: 'Algo no funciona correctamente',
|
|
9
|
-
icon: 'bug-outline',
|
|
10
|
-
},
|
|
11
|
-
{
|
|
12
|
-
value: 'poor-content',
|
|
13
|
-
label: 'Contenido incorrecto',
|
|
14
|
-
description: 'Información incorrecta o desactualizada',
|
|
15
|
-
icon: 'document-text-outline',
|
|
16
|
-
},
|
|
17
|
-
{
|
|
18
|
-
value: 'feedback',
|
|
19
|
-
label: 'Comentario general',
|
|
20
|
-
description: 'Tu opinión o experiencia',
|
|
21
|
-
icon: 'chatbubble-outline',
|
|
22
|
-
},
|
|
23
|
-
{
|
|
24
|
-
value: 'suggestion',
|
|
25
|
-
label: 'Sugerencia',
|
|
26
|
-
description: 'Propuesta de mejora o nueva funcionalidad',
|
|
27
|
-
icon: 'bulb-outline',
|
|
28
|
-
},
|
|
29
|
-
];
|
|
30
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2ZlZWRiYWNrL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXlIQTs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLDZCQUE2QixHQUF5QjtJQUNqRTtRQUNFLEtBQUssRUFBRSxPQUFPO1FBQ2QsS0FBSyxFQUFFLG1CQUFtQjtRQUMxQixXQUFXLEVBQUUsZ0NBQWdDO1FBQzdDLElBQUksRUFBRSxhQUFhO0tBQ3BCO0lBQ0Q7UUFDRSxLQUFLLEVBQUUsY0FBYztRQUNyQixLQUFLLEVBQUUsc0JBQXNCO1FBQzdCLFdBQVcsRUFBRSx5Q0FBeUM7UUFDdEQsSUFBSSxFQUFFLHVCQUF1QjtLQUM5QjtJQUNEO1FBQ0UsS0FBSyxFQUFFLFVBQVU7UUFDakIsS0FBSyxFQUFFLG9CQUFvQjtRQUMzQixXQUFXLEVBQUUsMEJBQTBCO1FBQ3ZDLElBQUksRUFBRSxvQkFBb0I7S0FDM0I7SUFDRDtRQUNFLEtBQUssRUFBRSxZQUFZO1FBQ25CLEtBQUssRUFBRSxZQUFZO1FBQ25CLFdBQVcsRUFBRSwyQ0FBMkM7UUFDeEQsSUFBSSxFQUFFLGNBQWM7S0FDckI7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb25maWd1cmFjacOzbiBkZWwgc2VydmljaW8gZGUgRmVlZGJhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVmFsdGVjaEZlZWRiYWNrQ29uZmlnIHtcbiAgLyoqIFVSTCBiYXNlIGRlIGxhIEFQSSAqL1xuICBhcGlVcmw6IHN0cmluZztcbiAgLyoqIElEIGRlIGxhIGFwbGljYWNpw7NuIChlajogJ215LXZhbHRlY2gtYXBwJykgKi9cbiAgYXBwSWQ6IHN0cmluZztcbiAgLyoqIFByZWZpam8gcGFyYSBlbmRwb2ludHMgKGRlZmF1bHQ6ICcvdjEvZmVlZGJhY2snKSAqL1xuICBmZWVkYmFja1ByZWZpeD86IHN0cmluZztcbiAgLyoqIE7Dum1lcm8gbcOheGltbyBkZSBhZGp1bnRvcyAoZGVmYXVsdDogNSkgKi9cbiAgbWF4QXR0YWNobWVudHM/OiBudW1iZXI7XG4gIC8qKiBUYW1hw7FvIG3DoXhpbW8gcG9yIGFyY2hpdm8gZW4gYnl0ZXMgKGRlZmF1bHQ6IDEwTUIpICovXG4gIG1heEZpbGVTaXplPzogbnVtYmVyO1xuICAvKiogVGlwb3MgZGUgYXJjaGl2byBwZXJtaXRpZG9zIChkZWZhdWx0OiBbJ2ltYWdlLyonLCAndmlkZW8vKicsICdhcHBsaWNhdGlvbi9wZGYnXSkgKi9cbiAgYWxsb3dlZEZpbGVUeXBlcz86IHN0cmluZ1tdO1xuICAvKiogUnV0YSBlbiBGaXJlYmFzZSBTdG9yYWdlIHBhcmEgYWRqdW50b3MgKGRlZmF1bHQ6ICdmZWVkYmFjaycpICovXG4gIHN0b3JhZ2VQYXRoPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIFRpcG9zIGRlIGZlZWRiYWNrIGRpc3BvbmlibGVzLlxuICovXG5leHBvcnQgdHlwZSBGZWVkYmFja1R5cGUgPSAnaXNzdWUnIHwgJ3Bvb3ItY29udGVudCcgfCAnZmVlZGJhY2snIHwgJ3N1Z2dlc3Rpb24nO1xuXG4vKipcbiAqIEVzdGFkbyBkZSB1biBmZWVkYmFjay5cbiAqL1xuZXhwb3J0IHR5cGUgRmVlZGJhY2tTdGF0dXMgPSAnbmV3JyB8ICdyZXZpZXdlZCcgfCAncmVzb2x2ZWQnO1xuXG4vKipcbiAqIFRpcG9zIGRlIGNvbnRlbmlkbyBwYXJhIHJlZmVyZW5jaWEuXG4gKi9cbmV4cG9ydCB0eXBlIENvbnRlbnRUeXBlID1cbiAgfCAnYXJ0aWNsZSdcbiAgfCAnZmFxJ1xuICB8ICduZXdzJ1xuICB8ICdwYWdlJ1xuICB8ICdwcm9kdWN0J1xuICB8ICdldmVudCdcbiAgfCAnb3RoZXInO1xuXG4vKipcbiAqIFJlZmVyZW5jaWEgYSBjb250ZW5pZG8gZXNwZWPDrWZpY28uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29udGVudFJlZiB7XG4gIGNvbnRlbnRJZDogc3RyaW5nO1xuICBjb250ZW50VHlwZTogQ29udGVudFR5cGU7XG59XG5cbi8qKlxuICogQ29udGV4dG8gZGVsIGRpc3Bvc2l0aXZvIGRlbCB1c3VhcmlvLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIERldmljZUNvbnRleHQge1xuICBicm93c2VyOiBzdHJpbmc7XG4gIG9zOiBzdHJpbmc7XG4gIHZpZXdwb3J0OiBzdHJpbmc7XG4gIGxhbmd1YWdlOiBzdHJpbmc7XG4gIHVzZXJBZ2VudDogc3RyaW5nO1xuICBwYWdlVXJsOiBzdHJpbmc7XG59XG5cbi8qKlxuICogRW50cmFkYSBkZSBmZWVkYmFjayBjb21wbGV0YS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBGZWVkYmFjayB7XG4gIGZlZWRiYWNrSWQ6IHN0cmluZztcbiAgYXBwSWQ6IHN0cmluZztcbiAgdXNlcklkOiBzdHJpbmc7XG4gIHR5cGU6IEZlZWRiYWNrVHlwZTtcbiAgdGl0bGU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgYXR0YWNobWVudHM6IHN0cmluZ1tdO1xuICBjb250ZW50UmVmPzogQ29udGVudFJlZjtcbiAgZGV2aWNlQ29udGV4dDogRGV2aWNlQ29udGV4dDtcbiAgc3RhdHVzOiBGZWVkYmFja1N0YXR1cztcbiAgY3JlYXRlZEF0OiBzdHJpbmc7XG4gIHVwZGF0ZWRBdDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlcXVlc3QgcGFyYSBjcmVhciBmZWVkYmFjay5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDcmVhdGVGZWVkYmFja1JlcXVlc3Qge1xuICB0eXBlOiBGZWVkYmFja1R5cGU7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGF0dGFjaG1lbnRzPzogc3RyaW5nW107XG4gIGNvbnRlbnRSZWY/OiBDb250ZW50UmVmO1xuICBkZXZpY2VDb250ZXh0OiBEZXZpY2VDb250ZXh0O1xuICBhcHBJZDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlc3BvbnNlIGFsIGNyZWFyIGZlZWRiYWNrLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENyZWF0ZUZlZWRiYWNrUmVzcG9uc2Uge1xuICBvcGVyYXRpb25JZDogc3RyaW5nO1xuICBmZWVkYmFja0lkOiBzdHJpbmc7XG4gIHN0YXR1czogRmVlZGJhY2tTdGF0dXM7XG4gIGNyZWF0ZWRBdDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlc3BvbnNlIGFsIG9idGVuZXIgZmVlZGJhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgR2V0RmVlZGJhY2tSZXNwb25zZSB7XG4gIG9wZXJhdGlvbklkOiBzdHJpbmc7XG4gIGZlZWRiYWNrOiBGZWVkYmFjaztcbn1cblxuLyoqXG4gKiBPcGNpb25lcyBkZSB0aXBvIGRlIGZlZWRiYWNrIHBhcmEgVUkuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRmVlZGJhY2tUeXBlT3B0aW9uIHtcbiAgdmFsdWU6IEZlZWRiYWNrVHlwZTtcbiAgbGFiZWw6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGljb24/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogQ29uZmlndXJhY2nDs24gcG9yIGRlZmVjdG8gZGUgdGlwb3MgZGUgZmVlZGJhY2suXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX0ZFRURCQUNLX1RZUEVfT1BUSU9OUzogRmVlZGJhY2tUeXBlT3B0aW9uW10gPSBbXG4gIHtcbiAgICB2YWx1ZTogJ2lzc3VlJyxcbiAgICBsYWJlbDogJ1JlcG9ydGFyIHByb2JsZW1hJyxcbiAgICBkZXNjcmlwdGlvbjogJ0FsZ28gbm8gZnVuY2lvbmEgY29ycmVjdGFtZW50ZScsXG4gICAgaWNvbjogJ2J1Zy1vdXRsaW5lJyxcbiAgfSxcbiAge1xuICAgIHZhbHVlOiAncG9vci1jb250ZW50JyxcbiAgICBsYWJlbDogJ0NvbnRlbmlkbyBpbmNvcnJlY3RvJyxcbiAgICBkZXNjcmlwdGlvbjogJ0luZm9ybWFjacOzbiBpbmNvcnJlY3RhIG8gZGVzYWN0dWFsaXphZGEnLFxuICAgIGljb246ICdkb2N1bWVudC10ZXh0LW91dGxpbmUnLFxuICB9LFxuICB7XG4gICAgdmFsdWU6ICdmZWVkYmFjaycsXG4gICAgbGFiZWw6ICdDb21lbnRhcmlvIGdlbmVyYWwnLFxuICAgIGRlc2NyaXB0aW9uOiAnVHUgb3BpbmnDs24gbyBleHBlcmllbmNpYScsXG4gICAgaWNvbjogJ2NoYXRidWJibGUtb3V0bGluZScsXG4gIH0sXG4gIHtcbiAgICB2YWx1ZTogJ3N1Z2dlc3Rpb24nLFxuICAgIGxhYmVsOiAnU3VnZXJlbmNpYScsXG4gICAgZGVzY3JpcHRpb246ICdQcm9wdWVzdGEgZGUgbWVqb3JhIG8gbnVldmEgZnVuY2lvbmFsaWRhZCcsXG4gICAgaWNvbjogJ2J1bGItb3V0bGluZScsXG4gIH0sXG5dO1xuIl19
|
|
@@ -1,56 +0,0 @@
|
|
|
1
|
-
import { EventEmitter, OnInit } from '@angular/core';
|
|
2
|
-
import { FormGroup } from '@angular/forms';
|
|
3
|
-
import { FeedbackFormMetadata, FeedbackSubmitEvent } from './types';
|
|
4
|
-
import * as i0 from "@angular/core";
|
|
5
|
-
/**
|
|
6
|
-
* val-feedback-form
|
|
7
|
-
*
|
|
8
|
-
* Formulario reutilizable para enviar feedback desde cualquier parte de la aplicación.
|
|
9
|
-
*
|
|
10
|
-
* @example
|
|
11
|
-
* ```html
|
|
12
|
-
* <!-- Feedback general -->
|
|
13
|
-
* <val-feedback-form
|
|
14
|
-
* [props]="{ defaultType: 'feedback', showTypeSelector: true }"
|
|
15
|
-
* (onSubmit)="handleSuccess($event)"
|
|
16
|
-
* (onCancel)="closeModal()"
|
|
17
|
-
* />
|
|
18
|
-
*
|
|
19
|
-
* <!-- Reportar contenido incorrecto -->
|
|
20
|
-
* <val-feedback-form
|
|
21
|
-
* [props]="{
|
|
22
|
-
* defaultType: 'poor-content',
|
|
23
|
-
* showTypeSelector: false,
|
|
24
|
-
* contentRef: { contentId: article.id, contentType: 'article' },
|
|
25
|
-
* submitButtonText: 'Reportar contenido'
|
|
26
|
-
* }"
|
|
27
|
-
* />
|
|
28
|
-
* ```
|
|
29
|
-
*/
|
|
30
|
-
export declare class FeedbackFormComponent implements OnInit {
|
|
31
|
-
/**
|
|
32
|
-
* Configuración del formulario.
|
|
33
|
-
*/
|
|
34
|
-
props: FeedbackFormMetadata;
|
|
35
|
-
/**
|
|
36
|
-
* Evento emitido cuando el feedback se envía exitosamente.
|
|
37
|
-
*/
|
|
38
|
-
onSubmit: EventEmitter<FeedbackSubmitEvent>;
|
|
39
|
-
/**
|
|
40
|
-
* Evento emitido cuando el usuario cancela.
|
|
41
|
-
*/
|
|
42
|
-
onCancel: EventEmitter<void>;
|
|
43
|
-
private fb;
|
|
44
|
-
private feedbackService;
|
|
45
|
-
form: FormGroup;
|
|
46
|
-
typeOptions: import("../../../services/feedback").FeedbackTypeOption[];
|
|
47
|
-
isSubmitting: import("@angular/core").WritableSignal<boolean>;
|
|
48
|
-
isSuccess: import("@angular/core").WritableSignal<boolean>;
|
|
49
|
-
error: import("@angular/core").WritableSignal<string>;
|
|
50
|
-
constructor();
|
|
51
|
-
ngOnInit(): void;
|
|
52
|
-
handleSubmit(): Promise<void>;
|
|
53
|
-
onCancelClick(): void;
|
|
54
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<FeedbackFormComponent, never>;
|
|
55
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<FeedbackFormComponent, "val-feedback-form", never, { "props": { "alias": "props"; "required": false; }; }, { "onSubmit": "onSubmit"; "onCancel": "onCancel"; }, never, never, true, never>;
|
|
56
|
-
}
|
|
@@ -1,54 +0,0 @@
|
|
|
1
|
-
import { FeedbackType, ContentRef, CreateFeedbackResponse, FeedbackTypeOption } from '../../../services/feedback/types';
|
|
2
|
-
/**
|
|
3
|
-
* Metadata para el componente FeedbackForm.
|
|
4
|
-
*/
|
|
5
|
-
export interface FeedbackFormMetadata {
|
|
6
|
-
/** Tipo de feedback preseleccionado */
|
|
7
|
-
defaultType?: FeedbackType;
|
|
8
|
-
/** Referencia a contenido (si aplica) */
|
|
9
|
-
contentRef?: ContentRef;
|
|
10
|
-
/** Mostrar selector de tipo (default: true) */
|
|
11
|
-
showTypeSelector?: boolean;
|
|
12
|
-
/** Tipos de feedback habilitados (por defecto todos) */
|
|
13
|
-
enabledTypes?: FeedbackType[];
|
|
14
|
-
/** Opciones personalizadas para tipos de feedback */
|
|
15
|
-
typeOptions?: FeedbackTypeOption[];
|
|
16
|
-
/** Placeholder para título */
|
|
17
|
-
titlePlaceholder?: string;
|
|
18
|
-
/** Placeholder para descripción */
|
|
19
|
-
descriptionPlaceholder?: string;
|
|
20
|
-
/** Label para campo de título */
|
|
21
|
-
titleLabel?: string;
|
|
22
|
-
/** Label para campo de descripción */
|
|
23
|
-
descriptionLabel?: string;
|
|
24
|
-
/** Texto del botón de envío */
|
|
25
|
-
submitButtonText?: string;
|
|
26
|
-
/** Texto del botón cancelar (si no se proporciona, no se muestra) */
|
|
27
|
-
cancelButtonText?: string;
|
|
28
|
-
/** Mostrar campo de adjuntos (default: true) */
|
|
29
|
-
showAttachments?: boolean;
|
|
30
|
-
/** Label para campo de adjuntos */
|
|
31
|
-
attachmentsLabel?: string;
|
|
32
|
-
/** Mensaje de éxito personalizado */
|
|
33
|
-
successMessage?: string;
|
|
34
|
-
/** CSS class adicional para el contenedor */
|
|
35
|
-
cssClass?: string;
|
|
36
|
-
/** Modo compacto (menos espaciado) */
|
|
37
|
-
compact?: boolean;
|
|
38
|
-
}
|
|
39
|
-
/**
|
|
40
|
-
* Evento emitido cuando el feedback se envía exitosamente.
|
|
41
|
-
*/
|
|
42
|
-
export interface FeedbackSubmitEvent {
|
|
43
|
-
response: CreateFeedbackResponse;
|
|
44
|
-
type: FeedbackType;
|
|
45
|
-
title: string;
|
|
46
|
-
}
|
|
47
|
-
/**
|
|
48
|
-
* Estado del formulario.
|
|
49
|
-
*/
|
|
50
|
-
export interface FeedbackFormState {
|
|
51
|
-
isSubmitting: boolean;
|
|
52
|
-
isSuccess: boolean;
|
|
53
|
-
error: string | null;
|
|
54
|
-
}
|
|
@@ -1,85 +0,0 @@
|
|
|
1
|
-
import { AuthState, AuthUser, AuthError, MFAPendingState, StoredAuthState } from './types';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
/**
|
|
4
|
-
* Servicio para manejo de estado de autenticación con Angular Signals.
|
|
5
|
-
* Proporciona estado reactivo inmutable.
|
|
6
|
-
*/
|
|
7
|
-
export declare class AuthStateService {
|
|
8
|
-
private _state;
|
|
9
|
-
private _mfaPending;
|
|
10
|
-
/** Estado completo de autenticación */
|
|
11
|
-
readonly state: import("@angular/core").Signal<AuthState>;
|
|
12
|
-
/** Estado de MFA pendiente */
|
|
13
|
-
readonly mfaPending: import("@angular/core").Signal<MFAPendingState>;
|
|
14
|
-
/** Usuario está autenticado */
|
|
15
|
-
readonly isAuthenticated: import("@angular/core").Signal<boolean>;
|
|
16
|
-
/** Estado de carga */
|
|
17
|
-
readonly isLoading: import("@angular/core").Signal<boolean>;
|
|
18
|
-
/** Token de acceso */
|
|
19
|
-
readonly accessToken: import("@angular/core").Signal<string>;
|
|
20
|
-
/** Roles del usuario */
|
|
21
|
-
readonly roles: import("@angular/core").Signal<string[]>;
|
|
22
|
-
/** Permisos del usuario */
|
|
23
|
-
readonly permissions: import("@angular/core").Signal<string[]>;
|
|
24
|
-
/** Usuario es super admin */
|
|
25
|
-
readonly isSuperAdmin: import("@angular/core").Signal<boolean>;
|
|
26
|
-
/** Error actual */
|
|
27
|
-
readonly error: import("@angular/core").Signal<AuthError>;
|
|
28
|
-
/** Información del usuario */
|
|
29
|
-
readonly user: import("@angular/core").Signal<AuthUser>;
|
|
30
|
-
/**
|
|
31
|
-
* Establece el estado de carga.
|
|
32
|
-
*/
|
|
33
|
-
setLoading(isLoading: boolean): void;
|
|
34
|
-
/**
|
|
35
|
-
* Establece el estado de autenticación exitosa.
|
|
36
|
-
*/
|
|
37
|
-
setAuthenticated(data: {
|
|
38
|
-
accessToken: string;
|
|
39
|
-
refreshToken: string;
|
|
40
|
-
userId?: string;
|
|
41
|
-
email?: string;
|
|
42
|
-
roles: string[];
|
|
43
|
-
permissions: string[];
|
|
44
|
-
isSuperAdmin: boolean;
|
|
45
|
-
expiresAt: number;
|
|
46
|
-
}): void;
|
|
47
|
-
/**
|
|
48
|
-
* Actualiza solo el access token (después de refresh).
|
|
49
|
-
*/
|
|
50
|
-
updateAccessToken(accessToken: string, expiresIn: number): void;
|
|
51
|
-
/**
|
|
52
|
-
* Actualiza los permisos.
|
|
53
|
-
*/
|
|
54
|
-
updatePermissions(roles: string[], permissions: string[], isSuperAdmin: boolean): void;
|
|
55
|
-
/**
|
|
56
|
-
* Establece un error de autenticación.
|
|
57
|
-
*/
|
|
58
|
-
setError(error: AuthError): void;
|
|
59
|
-
/**
|
|
60
|
-
* Limpia el error.
|
|
61
|
-
*/
|
|
62
|
-
clearError(): void;
|
|
63
|
-
/**
|
|
64
|
-
* Establece estado de MFA pendiente.
|
|
65
|
-
*/
|
|
66
|
-
setMFAPending(mfaState: MFAPendingState): void;
|
|
67
|
-
/**
|
|
68
|
-
* Limpia el estado de MFA pendiente.
|
|
69
|
-
*/
|
|
70
|
-
clearMFAPending(): void;
|
|
71
|
-
/**
|
|
72
|
-
* Resetea todo el estado a valores iniciales.
|
|
73
|
-
*/
|
|
74
|
-
reset(): void;
|
|
75
|
-
/**
|
|
76
|
-
* Restaura estado desde datos almacenados.
|
|
77
|
-
*/
|
|
78
|
-
restoreFromStorage(stored: Partial<StoredAuthState>): void;
|
|
79
|
-
/**
|
|
80
|
-
* Actualiza el userId y email (después de parsear el token).
|
|
81
|
-
*/
|
|
82
|
-
updateUserInfo(userId: string, email: string): void;
|
|
83
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<AuthStateService, never>;
|
|
84
|
-
static ɵprov: i0.ɵɵInjectableDeclaration<AuthStateService>;
|
|
85
|
-
}
|
|
@@ -1,146 +0,0 @@
|
|
|
1
|
-
import { OnDestroy } from '@angular/core';
|
|
2
|
-
import { HttpClient } from '@angular/common/http';
|
|
3
|
-
import { Router } from '@angular/router';
|
|
4
|
-
import { Observable } from 'rxjs';
|
|
5
|
-
import { AuthStateService } from './auth-state.service';
|
|
6
|
-
import { TokenService } from './token.service';
|
|
7
|
-
import { AuthStorageService } from './storage.service';
|
|
8
|
-
import { AuthSyncService } from './sync.service';
|
|
9
|
-
import { SigninRequest, SigninResponse, SignupRequest, SignupResponse, VerifyEmailRequest, VerifyEmailResponse, ResendCodeRequest, ResendCodeResponse, MFAVerifyResponse, RefreshResponse, GetPermissionsResponse, MFASetupResponse, MFAConfirmResponse, MFADisableResponse, MFAMethod, AuthError, ValtechAuthConfig } from './types';
|
|
10
|
-
import { FirebaseService } from '../firebase';
|
|
11
|
-
import * as i0 from "@angular/core";
|
|
12
|
-
/**
|
|
13
|
-
* Servicio principal de autenticación.
|
|
14
|
-
*
|
|
15
|
-
* @example
|
|
16
|
-
* ```typescript
|
|
17
|
-
* import { AuthService } from 'valtech-components';
|
|
18
|
-
*
|
|
19
|
-
* @Component({...})
|
|
20
|
-
* export class LoginComponent {
|
|
21
|
-
* private auth = inject(AuthService);
|
|
22
|
-
*
|
|
23
|
-
* async login() {
|
|
24
|
-
* await firstValueFrom(this.auth.signin({ email, password }));
|
|
25
|
-
* if (this.auth.mfaPending().required) {
|
|
26
|
-
* // Mostrar UI de MFA
|
|
27
|
-
* } else {
|
|
28
|
-
* this.router.navigate(['/']);
|
|
29
|
-
* }
|
|
30
|
-
* }
|
|
31
|
-
* }
|
|
32
|
-
* ```
|
|
33
|
-
*/
|
|
34
|
-
export declare class AuthService implements OnDestroy {
|
|
35
|
-
private config;
|
|
36
|
-
private http;
|
|
37
|
-
private router;
|
|
38
|
-
private stateService;
|
|
39
|
-
private tokenService;
|
|
40
|
-
private storageService;
|
|
41
|
-
private syncService;
|
|
42
|
-
private firebaseService;
|
|
43
|
-
private refreshTimerId;
|
|
44
|
-
private syncSubscription;
|
|
45
|
-
constructor(config: ValtechAuthConfig, http: HttpClient, router: Router, stateService: AuthStateService, tokenService: TokenService, storageService: AuthStorageService, syncService: AuthSyncService, firebaseService: FirebaseService);
|
|
46
|
-
/** Estado completo de autenticación */
|
|
47
|
-
readonly state: import("@angular/core").Signal<import("./types").AuthState>;
|
|
48
|
-
/** Usuario está autenticado */
|
|
49
|
-
readonly isAuthenticated: import("@angular/core").Signal<boolean>;
|
|
50
|
-
/** Estado de carga */
|
|
51
|
-
readonly isLoading: import("@angular/core").Signal<boolean>;
|
|
52
|
-
/** Información del usuario */
|
|
53
|
-
readonly user: import("@angular/core").Signal<import("./types").AuthUser>;
|
|
54
|
-
/** Token de acceso */
|
|
55
|
-
readonly accessToken: import("@angular/core").Signal<string>;
|
|
56
|
-
/** Roles del usuario */
|
|
57
|
-
readonly roles: import("@angular/core").Signal<string[]>;
|
|
58
|
-
/** Permisos del usuario */
|
|
59
|
-
readonly permissions: import("@angular/core").Signal<string[]>;
|
|
60
|
-
/** Usuario es super admin */
|
|
61
|
-
readonly isSuperAdmin: import("@angular/core").Signal<boolean>;
|
|
62
|
-
/** Estado de MFA pendiente */
|
|
63
|
-
readonly mfaPending: import("@angular/core").Signal<import("./types").MFAPendingState>;
|
|
64
|
-
/** Error actual */
|
|
65
|
-
readonly error: import("@angular/core").Signal<AuthError>;
|
|
66
|
-
/**
|
|
67
|
-
* Inicializa el servicio de autenticación.
|
|
68
|
-
* Llamado automáticamente por provideValtechAuth.
|
|
69
|
-
*/
|
|
70
|
-
initialize(): Promise<void>;
|
|
71
|
-
ngOnDestroy(): void;
|
|
72
|
-
/**
|
|
73
|
-
* Inicia sesión con email y contraseña.
|
|
74
|
-
*/
|
|
75
|
-
signin(request: SigninRequest): Observable<SigninResponse>;
|
|
76
|
-
/**
|
|
77
|
-
* Registra un nuevo usuario.
|
|
78
|
-
* El usuario queda en estado PENDING hasta verificar su email.
|
|
79
|
-
*/
|
|
80
|
-
signup(request: SignupRequest): Observable<SignupResponse>;
|
|
81
|
-
/**
|
|
82
|
-
* Verifica email con código de 6 dígitos.
|
|
83
|
-
* Si es exitoso, hace auto-login y retorna tokens.
|
|
84
|
-
*/
|
|
85
|
-
verifyEmail(request: VerifyEmailRequest): Observable<VerifyEmailResponse>;
|
|
86
|
-
/**
|
|
87
|
-
* Reenvía código de verificación al email.
|
|
88
|
-
*/
|
|
89
|
-
resendCode(request: ResendCodeRequest): Observable<ResendCodeResponse>;
|
|
90
|
-
/**
|
|
91
|
-
* Verifica código MFA.
|
|
92
|
-
*/
|
|
93
|
-
verifyMFA(code: string): Observable<MFAVerifyResponse>;
|
|
94
|
-
/**
|
|
95
|
-
* Refresca el token de acceso.
|
|
96
|
-
*/
|
|
97
|
-
refreshAccessToken(): Observable<RefreshResponse>;
|
|
98
|
-
/**
|
|
99
|
-
* Cierra sesión.
|
|
100
|
-
*/
|
|
101
|
-
logout(): void;
|
|
102
|
-
/**
|
|
103
|
-
* Configura MFA para el usuario.
|
|
104
|
-
*/
|
|
105
|
-
setupMFA(method: MFAMethod, phone?: string): Observable<MFASetupResponse>;
|
|
106
|
-
/**
|
|
107
|
-
* Confirma la configuración de MFA.
|
|
108
|
-
*/
|
|
109
|
-
confirmMFA(code: string): Observable<MFAConfirmResponse>;
|
|
110
|
-
/**
|
|
111
|
-
* Deshabilita MFA.
|
|
112
|
-
*/
|
|
113
|
-
disableMFA(password: string): Observable<MFADisableResponse>;
|
|
114
|
-
/**
|
|
115
|
-
* Obtiene los permisos actualizados del backend.
|
|
116
|
-
*/
|
|
117
|
-
fetchPermissions(): Observable<GetPermissionsResponse>;
|
|
118
|
-
/**
|
|
119
|
-
* Verifica si el usuario tiene un permiso específico.
|
|
120
|
-
* Formato: "resource:action" (ej: "templates:edit")
|
|
121
|
-
*/
|
|
122
|
-
hasPermission(permission: string): boolean;
|
|
123
|
-
/**
|
|
124
|
-
* Verifica si el usuario tiene alguno de los permisos dados.
|
|
125
|
-
*/
|
|
126
|
-
hasAnyPermission(permissions: string[]): boolean;
|
|
127
|
-
/**
|
|
128
|
-
* Verifica si el usuario tiene todos los permisos dados.
|
|
129
|
-
*/
|
|
130
|
-
hasAllPermissions(permissions: string[]): boolean;
|
|
131
|
-
/**
|
|
132
|
-
* Verifica si el usuario tiene un rol específico.
|
|
133
|
-
*/
|
|
134
|
-
hasRole(role: string): boolean;
|
|
135
|
-
private get baseUrl();
|
|
136
|
-
private handleSuccessfulAuth;
|
|
137
|
-
private clearState;
|
|
138
|
-
private startRefreshTimer;
|
|
139
|
-
private stopRefreshTimer;
|
|
140
|
-
private handleSyncEvent;
|
|
141
|
-
private handleAuthError;
|
|
142
|
-
private signInWithFirebase;
|
|
143
|
-
private signOutFirebase;
|
|
144
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<AuthService, never>;
|
|
145
|
-
static ɵprov: i0.ɵɵInjectableDeclaration<AuthService>;
|
|
146
|
-
}
|
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
import { EnvironmentProviders, InjectionToken } from '@angular/core';
|
|
2
|
-
import { ValtechAuthConfig } from './types';
|
|
3
|
-
/**
|
|
4
|
-
* Token de inyección para la configuración de Auth.
|
|
5
|
-
*/
|
|
6
|
-
export declare const VALTECH_AUTH_CONFIG: InjectionToken<ValtechAuthConfig>;
|
|
7
|
-
/**
|
|
8
|
-
* Configuración por defecto.
|
|
9
|
-
*/
|
|
10
|
-
export declare const DEFAULT_AUTH_CONFIG: Partial<ValtechAuthConfig>;
|
|
11
|
-
/**
|
|
12
|
-
* Provee el servicio de autenticación a la aplicación Angular.
|
|
13
|
-
*
|
|
14
|
-
* @param config - Configuración de autenticación
|
|
15
|
-
* @returns EnvironmentProviders para usar en bootstrapApplication
|
|
16
|
-
*
|
|
17
|
-
* @example
|
|
18
|
-
* ```typescript
|
|
19
|
-
* // main.ts
|
|
20
|
-
* import { bootstrapApplication } from '@angular/platform-browser';
|
|
21
|
-
* import { provideValtechAuth } from 'valtech-components';
|
|
22
|
-
* import { environment } from './environments/environment';
|
|
23
|
-
*
|
|
24
|
-
* bootstrapApplication(AppComponent, {
|
|
25
|
-
* providers: [
|
|
26
|
-
* provideValtechAuth({
|
|
27
|
-
* apiUrl: environment.apiUrl,
|
|
28
|
-
* enableFirebaseIntegration: true,
|
|
29
|
-
* }),
|
|
30
|
-
* ],
|
|
31
|
-
* });
|
|
32
|
-
* ```
|
|
33
|
-
*/
|
|
34
|
-
export declare function provideValtechAuth(config: ValtechAuthConfig): EnvironmentProviders;
|
|
35
|
-
/**
|
|
36
|
-
* Provee solo el interceptor (para apps que ya tienen AuthService configurado manualmente).
|
|
37
|
-
*/
|
|
38
|
-
export declare function provideValtechAuthInterceptor(): EnvironmentProviders;
|