valtech-components 2.0.682 → 2.0.684
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/feedback/feedback.service.mjs +24 -21
- package/esm2022/lib/services/i18n/default-content.mjs +21 -1
- package/esm2022/lib/version.mjs +2 -2
- package/fesm2022/valtech-components.mjs +44 -21
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
- package/lib/components/organisms/article/article.component.d.ts +2 -2
- package/lib/components/organisms/toolbar/toolbar.component.d.ts +1 -1
- package/lib/version.d.ts +1 -1
- package/package.json +1 -1
|
@@ -152,35 +152,38 @@ export class FeedbackService {
|
|
|
152
152
|
* ```
|
|
153
153
|
*/
|
|
154
154
|
async checkFeedback(entityType, entityId) {
|
|
155
|
+
// Si no hay usuario autenticado, no puede haber feedback previo
|
|
156
|
+
// Retornar inmediatamente sin llamar al API (evita 401 y redirect a login)
|
|
157
|
+
const userId = this.auth?.user()?.userId;
|
|
158
|
+
if (!userId) {
|
|
159
|
+
return { operationId: '', hasFeedback: false };
|
|
160
|
+
}
|
|
155
161
|
// 1. Intentar Firebase primero (si está disponible)
|
|
156
|
-
if (this.firestore
|
|
162
|
+
if (this.firestore) {
|
|
157
163
|
try {
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
createdAt: doc.createdAt?.toISOString(),
|
|
172
|
-
};
|
|
173
|
-
}
|
|
174
|
-
// Doc no existe = no hay feedback
|
|
175
|
-
return { operationId: '', hasFeedback: false };
|
|
164
|
+
// Path: feedback/{entityType}/{entityId}/{userId}
|
|
165
|
+
// FirestoreService agrega automáticamente el prefijo apps/{appId}/
|
|
166
|
+
const collectionPath = `feedback/${entityType}/${entityId}`;
|
|
167
|
+
const doc = await this.firestore.getDoc(collectionPath, userId);
|
|
168
|
+
if (doc) {
|
|
169
|
+
return {
|
|
170
|
+
operationId: '',
|
|
171
|
+
hasFeedback: true,
|
|
172
|
+
feedbackId: doc.feedbackId,
|
|
173
|
+
type: doc.type,
|
|
174
|
+
reactionValue: doc.reactionValue,
|
|
175
|
+
createdAt: doc.createdAt?.toISOString(),
|
|
176
|
+
};
|
|
176
177
|
}
|
|
178
|
+
// Doc no existe = no hay feedback
|
|
179
|
+
return { operationId: '', hasFeedback: false };
|
|
177
180
|
}
|
|
178
181
|
catch (error) {
|
|
179
182
|
console.warn('[FeedbackService] Firebase check failed, falling back to API:', error);
|
|
180
183
|
// Fallback a API
|
|
181
184
|
}
|
|
182
185
|
}
|
|
183
|
-
// 2. Fallback: llamar API
|
|
186
|
+
// 2. Fallback: llamar API (solo si hay usuario autenticado)
|
|
184
187
|
const params = new URLSearchParams({
|
|
185
188
|
appId: this.config.appId,
|
|
186
189
|
entityType,
|
|
@@ -290,4 +293,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
290
293
|
type: Injectable,
|
|
291
294
|
args: [{ providedIn: 'root' }]
|
|
292
295
|
}] });
|
|
293
|
-
//# 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  CheckFeedbackResponse,\n  DeviceContext,\n  FeedbackType,\n  ContentRef,\n  EntityRef,\n  ReactionValue,\n  FeedbackFirestoreDoc,\n} from './types';\nimport { FirestoreService } from '../firebase/firestore.service';\nimport { AuthService } from '../auth/auth.service';\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  private firestore = inject(FirestoreService, { optional: true });\n  private auth = inject(AuthService, { optional: true });\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  // Reaction Methods (Content feedback with emojis)\n  // =========================================================================\n\n  /**\n   * Verifica si el usuario ya dio feedback para una entidad específica.\n   *\n   * Primero intenta leer de Firebase (rápido, sin latencia de red al backend).\n   * Si Firebase no está disponible o falla, hace fallback a la API.\n   *\n   * @param entityType - Tipo de entidad (article, docs, feature, etc.)\n   * @param entityId - ID de la entidad\n   * @returns Promise con la respuesta de verificación\n   *\n   * @example\n   * ```typescript\n   * const check = await this.feedbackService.checkFeedback('article', 'art-123');\n   * if (check.hasFeedback) {\n   *   console.log('Ya dio feedback:', check.reactionValue);\n   * }\n   * ```\n   */\n  async checkFeedback(\n    entityType: string,\n    entityId: string\n  ): Promise<CheckFeedbackResponse> {\n    // 1. Intentar Firebase primero (si está disponible)\n    if (this.firestore && this.auth) {\n      try {\n        const userId = this.auth.user()?.userId;\n        if (userId) {\n          // Path: feedback/{entityType}/{entityId}/{userId}\n          // FirestoreService agrega automáticamente el prefijo apps/{appId}/\n          const collectionPath = `feedback/${entityType}/${entityId}`;\n          const doc = await this.firestore.getDoc<FeedbackFirestoreDoc>(\n            collectionPath,\n            userId\n          );\n\n          if (doc) {\n            return {\n              operationId: '',\n              hasFeedback: true,\n              feedbackId: doc.feedbackId,\n              type: doc.type as FeedbackType,\n              reactionValue: doc.reactionValue as ReactionValue,\n              createdAt: doc.createdAt?.toISOString(),\n            };\n          }\n\n          // Doc no existe = no hay feedback\n          return { operationId: '', hasFeedback: false };\n        }\n      } catch (error) {\n        console.warn('[FeedbackService] Firebase check failed, falling back to API:', error);\n        // Fallback a API\n      }\n    }\n\n    // 2. Fallback: llamar API\n    const params = new URLSearchParams({\n      appId: this.config.appId,\n      entityType,\n      entityId,\n    });\n\n    return firstValueFrom(\n      this.http.get<CheckFeedbackResponse>(`${this.baseUrl}/check?${params}`)\n    );\n  }\n\n  /**\n   * Crea o actualiza una reacción (feedback con emoji).\n   *\n   * @param entityRef - Referencia a la entidad\n   * @param value - Valor de la reacción (negative, neutral, positive)\n   * @param comment - Comentario opcional (máx 500 caracteres)\n   * @returns Promise con la respuesta\n   *\n   * @example\n   * ```typescript\n   * const response = await this.feedbackService.createReaction(\n   *   { entityType: 'article', entityId: 'art-123' },\n   *   'positive',\n   *   'Muy útil!'\n   * );\n   * ```\n   */\n  async createReaction(\n    entityRef: EntityRef,\n    value: ReactionValue,\n    comment?: string\n  ): Promise<CreateFeedbackResponse> {\n    const request: CreateFeedbackRequest = {\n      type: 'reaction',\n      entityRef,\n      reactionValue: value,\n      description: comment || '',\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return firstValueFrom(\n      this.http.post<CreateFeedbackResponse>(this.baseUrl, request)\n    );\n  }\n\n  /**\n   * Crea feedback anónimo (sin autenticación requerida).\n   * Usado para blogs, FAQs y contenido público.\n   *\n   * @param entityRef - Referencia a la entidad\n   * @param value - Valor de la reacción (negative, neutral, positive)\n   * @param comment - Comentario opcional (máx 500 caracteres)\n   * @returns Promise con la respuesta\n   *\n   * @example\n   * ```typescript\n   * // En un blog público\n   * const response = await this.feedbackService.createAnonymousReaction(\n   *   { entityType: 'blog', entityId: 'my-post-slug' },\n   *   'positive'\n   * );\n   * ```\n   */\n  async createAnonymousReaction(\n    entityRef: EntityRef,\n    value: ReactionValue,\n    comment?: string\n  ): Promise<CreateFeedbackResponse> {\n    const request = {\n      type: 'reaction' as const,\n      entityRef,\n      reactionValue: value,\n      description: comment || '',\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return firstValueFrom(\n      this.http.post<CreateFeedbackResponse>(`${this.baseUrl}/anonymous`, request)\n    );\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"]}
|
|
296
|
+
//# 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  CheckFeedbackResponse,\n  DeviceContext,\n  FeedbackType,\n  ContentRef,\n  EntityRef,\n  ReactionValue,\n  FeedbackFirestoreDoc,\n} from './types';\nimport { FirestoreService } from '../firebase/firestore.service';\nimport { AuthService } from '../auth/auth.service';\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  private firestore = inject(FirestoreService, { optional: true });\n  private auth = inject(AuthService, { optional: true });\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  // Reaction Methods (Content feedback with emojis)\n  // =========================================================================\n\n  /**\n   * Verifica si el usuario ya dio feedback para una entidad específica.\n   *\n   * Primero intenta leer de Firebase (rápido, sin latencia de red al backend).\n   * Si Firebase no está disponible o falla, hace fallback a la API.\n   *\n   * @param entityType - Tipo de entidad (article, docs, feature, etc.)\n   * @param entityId - ID de la entidad\n   * @returns Promise con la respuesta de verificación\n   *\n   * @example\n   * ```typescript\n   * const check = await this.feedbackService.checkFeedback('article', 'art-123');\n   * if (check.hasFeedback) {\n   *   console.log('Ya dio feedback:', check.reactionValue);\n   * }\n   * ```\n   */\n  async checkFeedback(\n    entityType: string,\n    entityId: string\n  ): Promise<CheckFeedbackResponse> {\n    // Si no hay usuario autenticado, no puede haber feedback previo\n    // Retornar inmediatamente sin llamar al API (evita 401 y redirect a login)\n    const userId = this.auth?.user()?.userId;\n    if (!userId) {\n      return { operationId: '', hasFeedback: false };\n    }\n\n    // 1. Intentar Firebase primero (si está disponible)\n    if (this.firestore) {\n      try {\n        // Path: feedback/{entityType}/{entityId}/{userId}\n        // FirestoreService agrega automáticamente el prefijo apps/{appId}/\n        const collectionPath = `feedback/${entityType}/${entityId}`;\n        const doc = await this.firestore.getDoc<FeedbackFirestoreDoc>(\n          collectionPath,\n          userId\n        );\n\n        if (doc) {\n          return {\n            operationId: '',\n            hasFeedback: true,\n            feedbackId: doc.feedbackId,\n            type: doc.type as FeedbackType,\n            reactionValue: doc.reactionValue as ReactionValue,\n            createdAt: doc.createdAt?.toISOString(),\n          };\n        }\n\n        // Doc no existe = no hay feedback\n        return { operationId: '', hasFeedback: false };\n      } catch (error) {\n        console.warn('[FeedbackService] Firebase check failed, falling back to API:', error);\n        // Fallback a API\n      }\n    }\n\n    // 2. Fallback: llamar API (solo si hay usuario autenticado)\n    const params = new URLSearchParams({\n      appId: this.config.appId,\n      entityType,\n      entityId,\n    });\n\n    return firstValueFrom(\n      this.http.get<CheckFeedbackResponse>(`${this.baseUrl}/check?${params}`)\n    );\n  }\n\n  /**\n   * Crea o actualiza una reacción (feedback con emoji).\n   *\n   * @param entityRef - Referencia a la entidad\n   * @param value - Valor de la reacción (negative, neutral, positive)\n   * @param comment - Comentario opcional (máx 500 caracteres)\n   * @returns Promise con la respuesta\n   *\n   * @example\n   * ```typescript\n   * const response = await this.feedbackService.createReaction(\n   *   { entityType: 'article', entityId: 'art-123' },\n   *   'positive',\n   *   'Muy útil!'\n   * );\n   * ```\n   */\n  async createReaction(\n    entityRef: EntityRef,\n    value: ReactionValue,\n    comment?: string\n  ): Promise<CreateFeedbackResponse> {\n    const request: CreateFeedbackRequest = {\n      type: 'reaction',\n      entityRef,\n      reactionValue: value,\n      description: comment || '',\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return firstValueFrom(\n      this.http.post<CreateFeedbackResponse>(this.baseUrl, request)\n    );\n  }\n\n  /**\n   * Crea feedback anónimo (sin autenticación requerida).\n   * Usado para blogs, FAQs y contenido público.\n   *\n   * @param entityRef - Referencia a la entidad\n   * @param value - Valor de la reacción (negative, neutral, positive)\n   * @param comment - Comentario opcional (máx 500 caracteres)\n   * @returns Promise con la respuesta\n   *\n   * @example\n   * ```typescript\n   * // En un blog público\n   * const response = await this.feedbackService.createAnonymousReaction(\n   *   { entityType: 'blog', entityId: 'my-post-slug' },\n   *   'positive'\n   * );\n   * ```\n   */\n  async createAnonymousReaction(\n    entityRef: EntityRef,\n    value: ReactionValue,\n    comment?: string\n  ): Promise<CreateFeedbackResponse> {\n    const request = {\n      type: 'reaction' as const,\n      entityRef,\n      reactionValue: value,\n      description: comment || '',\n      deviceContext: this.captureDeviceContext(),\n      appId: this.config.appId,\n    };\n\n    return firstValueFrom(\n      this.http.post<CreateFeedbackResponse>(`${this.baseUrl}/anonymous`, request)\n    );\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"]}
|
|
@@ -493,5 +493,25 @@ export const VALTECH_DEFAULT_CONTENT = {
|
|
|
493
493
|
maintenanceMessage: 'We are making improvements. Come back soon.',
|
|
494
494
|
},
|
|
495
495
|
},
|
|
496
|
+
ContentReaction: {
|
|
497
|
+
es: {
|
|
498
|
+
question: '¿Te resultó útil este contenido?',
|
|
499
|
+
commentPlaceholder: 'Cuéntanos más (opcional)...',
|
|
500
|
+
submit: 'Enviar',
|
|
501
|
+
update: 'Actualizar',
|
|
502
|
+
thankYou: '¡Gracias por tu opinión!',
|
|
503
|
+
submitted: 'Tu opinión ha sido registrada',
|
|
504
|
+
errorSubmitting: 'Error al enviar. Intenta de nuevo.',
|
|
505
|
+
},
|
|
506
|
+
en: {
|
|
507
|
+
question: 'Was this content helpful?',
|
|
508
|
+
commentPlaceholder: 'Tell us more (optional)...',
|
|
509
|
+
submit: 'Submit',
|
|
510
|
+
update: 'Update',
|
|
511
|
+
thankYou: 'Thanks for your feedback!',
|
|
512
|
+
submitted: 'Your feedback has been recorded',
|
|
513
|
+
errorSubmitting: 'Error submitting. Please try again.',
|
|
514
|
+
},
|
|
515
|
+
},
|
|
496
516
|
};
|
|
497
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default-content.js","sourceRoot":"","sources":["../../../../../../src/lib/services/i18n/default-content.ts"],"names":[],"mappings":"AAEA;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,MAAM,CAAC,MAAM,uBAAuB,GAAiB;IACnD,OAAO,EAAE;QACP,EAAE,EAAE;YACF,mBAAmB;YACnB,MAAM,EAAE,QAAQ;YAChB,MAAM,EAAE,UAAU;YAClB,IAAI,EAAE,SAAS;YACf,MAAM,EAAE,UAAU;YAClB,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,QAAQ;YACf,IAAI,EAAE,QAAQ;YACd,IAAI,EAAE,WAAW;YACjB,QAAQ,EAAE,UAAU;YACpB,OAAO,EAAE,aAAa;YACtB,MAAM,EAAE,QAAQ;YAChB,SAAS,EAAE,WAAW;YAEtB,UAAU;YACV,OAAO,EAAE,OAAO;YAChB,KAAK,EAAE,OAAO;YACd,OAAO,EAAE,aAAa;YACtB,IAAI,EAAE,aAAa;YAEnB,iBAAiB;YACjB,aAAa,EAAE,uCAAuC;YACtD,aAAa,EAAE,uCAAuC;YAEtD,mBAAmB;YACnB,SAAS,EAAE,8BAA8B;YACzC,QAAQ,EAAE,yBAAyB;YAEnC,aAAa;YACb,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,eAAe;YACzB,OAAO,EAAE,QAAQ;YACjB,MAAM,EAAE,eAAe;YAEvB,UAAU;YACV,eAAe,EAAE,SAAS;YAC1B,eAAe,EAAE,SAAS;YAE1B,gCAAgC;YAChC,EAAE,EAAE,SAAS;YACb,YAAY,EAAE,oBAAoB;YAClC,iBAAiB,EAAE,gBAAgB;YACnC,aAAa,EAAE,yBAAyB;YACxC,SAAS,EAAE,mBAAmB;YAC9B,KAAK,EAAE,SAAS;YAChB,KAAK,EAAE,SAAS;YAEhB,uBAAuB;YACvB,SAAS,EAAE,cAAc;YACzB,OAAO,EAAE,WAAW;YACpB,GAAG,EAAE,KAAK;YACV,IAAI,EAAE,MAAM;YAEZ,4BAA4B;YAC5B,UAAU,EAAE,eAAe;YAC3B,cAAc,EAAE,6BAA6B;YAC7C,KAAK,EAAE,QAAQ;YACf,WAAW,EAAE,aAAa;YAC1B,YAAY,EAAE,kBAAkB;YAChC,eAAe,EAAE,+BAA+B;YAChD,aAAa,EAAE,6BAA6B;YAC5C,gBAAgB,EAAE,wBAAwB;YAC1C,eAAe,EAAE,+CAA+C;YAChE,sBAAsB,EAAE,6BAA6B;YACrD,qBAAqB,EAAE,sDAAsD;YAE7E,8BAA8B;YAC9B,YAAY,EAAE,kBAAkB;YAChC,QAAQ,EAAE,gBAAgB;YAC1B,GAAG,EAAE,MAAM;YAEX,uBAAuB;YACvB,gBAAgB,EAAE,wBAAwB;YAC1C,MAAM,EAAE,cAAc;YACtB,cAAc,EAAE,2CAA2C;YAE3D,yBAAyB;YACzB,iBAAiB,EAAE,2BAA2B;YAC9C,QAAQ,EAAE,QAAQ;YAClB,cAAc,EAAE,uBAAuB;YACvC,QAAQ,EAAE,WAAW;YACrB,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,WAAW;YAElB,wCAAwC;YACxC,OAAO,EAAE,UAAU;YACnB,OAAO,EAAE,WAAW;YACpB,EAAE,EAAE,IAAI;YACR,OAAO,EAAE,YAAY;YACrB,SAAS,EAAE,gBAAgB;YAC3B,YAAY,EAAE,iBAAiB;YAC/B,QAAQ,EAAE,kBAAkB;YAC5B,QAAQ,EAAE,eAAe;YAEzB,iCAAiC;YACjC,MAAM,EAAE,SAAS;YACjB,MAAM,EAAE,QAAQ;YAChB,OAAO,EAAE,SAAS;YAClB,IAAI,EAAE,KAAK;YACX,KAAK,EAAE,QAAQ;YAEf,SAAS;YACT,iBAAiB,EAAE,gCAAgC;YAEnD,YAAY;YACZ,UAAU,EAAE,kBAAkB;YAE9B,YAAY;YACZ,MAAM,EAAE,aAAa;YACrB,aAAa,EAAE,sBAAsB;YAErC,WAAW;YACX,MAAM,EAAE,QAAQ;YAChB,MAAM,EAAE,WAAW;YACnB,WAAW,EAAE,oBAAoB;YACjC,WAAW,EAAE,oBAAoB;YACjC,eAAe,EAAE,uBAAuB;YACxC,WAAW,EAAE,gBAAgB;YAC7B,QAAQ,EAAE,aAAa;YACvB,aAAa,EAAE,qBAAqB;YACpC,OAAO,EAAE,UAAU;YACnB,MAAM,EAAE,aAAa;YACrB,YAAY,EAAE,0BAA0B;YACxC,gBAAgB,EAAE,yCAAyC;YAE3D,aAAa;YACb,OAAO,EAAE,aAAa;YACtB,UAAU,EAAE,iBAAiB;YAC7B,QAAQ,EAAE,eAAe;YACzB,OAAO,EAAE,cAAc;YAEvB,QAAQ;YACR,QAAQ,EAAE,YAAY;YACtB,WAAW,EAAE,kBAAkB;YAC/B,YAAY,EAAE,mBAAmB;YACjC,cAAc,EAAE,mCAAmC;YACnD,aAAa,EAAE,kBAAkB;YACjC,KAAK,EAAE,YAAY;YAEnB,aAAa;YACb,OAAO,EAAE,SAAS;YAElB,gBAAgB;YAChB,IAAI,EAAE,SAAS;YACf,EAAE,EAAE,OAAO;YAEX,UAAU;YACV,QAAQ,EAAE,WAAW;YACrB,QAAQ,EAAE,UAAU;YAEpB,OAAO;YACP,MAAM,EAAE,WAAW;YACnB,aAAa,EAAE,0BAA0B;YACzC,WAAW,EAAE,gBAAgB;YAC7B,MAAM,EAAE,WAAW;YAEnB,iBAAiB;YACjB,cAAc,EAAE,WAAW;YAC3B,iBAAiB,EAAE,YAAY;YAC/B,WAAW,EAAE,QAAQ;YACrB,eAAe,EAAE,aAAa;YAC9B,aAAa,EAAE,SAAS;SACzB;QACD,EAAE,EAAE;YACF,iBAAiB;YACjB,MAAM,EAAE,QAAQ;YAChB,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,MAAM;YACZ,MAAM,EAAE,QAAQ;YAChB,IAAI,EAAE,MAAM;YACZ,KAAK,EAAE,OAAO;YACd,IAAI,EAAE,MAAM;YACZ,IAAI,EAAE,MAAM;YACZ,QAAQ,EAAE,UAAU;YACpB,OAAO,EAAE,YAAY;YACrB,MAAM,EAAE,QAAQ;YAChB,SAAS,EAAE,YAAY;YAEvB,SAAS;YACT,OAAO,EAAE,SAAS;YAClB,KAAK,EAAE,OAAO;YACd,OAAO,EAAE,SAAS;YAClB,IAAI,EAAE,aAAa;YAEnB,gBAAgB;YAChB,aAAa,EAAE,kCAAkC;YACjD,aAAa,EAAE,kCAAkC;YAEjD,kBAAkB;YAClB,SAAS,EAAE,kBAAkB;YAC7B,QAAQ,EAAE,wBAAwB;YAElC,aAAa;YACb,IAAI,EAAE,MAAM;YACZ,QAAQ,EAAE,UAAU;YACpB,OAAO,EAAE,SAAS;YAClB,MAAM,EAAE,SAAS;YAEjB,YAAY;YACZ,eAAe,EAAE,SAAS;YAC1B,eAAe,EAAE,SAAS;YAE1B,8BAA8B;YAC9B,EAAE,EAAE,IAAI;YACR,YAAY,EAAE,eAAe;YAC7B,iBAAiB,EAAE,WAAW;YAC9B,aAAa,EAAE,gBAAgB;YAC/B,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,OAAO;YACd,KAAK,EAAE,OAAO;YAEd,qBAAqB;YACrB,SAAS,EAAE,YAAY;YACvB,OAAO,EAAE,UAAU;YACnB,GAAG,EAAE,KAAK;YACV,IAAI,EAAE,MAAM;YAEZ,qBAAqB;YACrB,UAAU,EAAE,aAAa;YACzB,cAAc,EAAE,kBAAkB;YAClC,KAAK,EAAE,OAAO;YACd,WAAW,EAAE,aAAa;YAC1B,YAAY,EAAE,eAAe;YAC7B,eAAe,EAAE,4BAA4B;YAC7C,aAAa,EAAE,wBAAwB;YACvC,gBAAgB,EAAE,qBAAqB;YACvC,eAAe,EAAE,4CAA4C;YAC7D,sBAAsB,EAAE,yBAAyB;YACjD,qBAAqB,EAAE,oDAAoD;YAE3E,2BAA2B;YAC3B,YAAY,EAAE,kBAAkB;YAChC,QAAQ,EAAE,UAAU;YACpB,GAAG,EAAE,MAAM;YAEX,qBAAqB;YACrB,gBAAgB,EAAE,oBAAoB;YACtC,MAAM,EAAE,SAAS;YACjB,cAAc,EAAE,8BAA8B;YAE9C,uBAAuB;YACvB,iBAAiB,EAAE,sBAAsB;YACzC,QAAQ,EAAE,UAAU;YACpB,cAAc,EAAE,oBAAoB;YACpC,QAAQ,EAAE,UAAU;YACpB,IAAI,EAAE,MAAM;YACZ,KAAK,EAAE,OAAO;YAEd,uCAAuC;YACvC,OAAO,EAAE,SAAS;YAClB,OAAO,EAAE,SAAS;YAClB,EAAE,EAAE,IAAI;YACR,OAAO,EAAE,UAAU;YACnB,SAAS,EAAE,YAAY;YACvB,YAAY,EAAE,eAAe;YAC7B,QAAQ,EAAE,WAAW;YACrB,QAAQ,EAAE,WAAW;YAErB,gCAAgC;YAChC,MAAM,EAAE,QAAQ;YAChB,MAAM,EAAE,QAAQ;YAChB,OAAO,EAAE,SAAS;YAClB,IAAI,EAAE,MAAM;YACZ,KAAK,EAAE,QAAQ;YAEf,SAAS;YACT,iBAAiB,EAAE,sBAAsB;YAEzC,YAAY;YACZ,UAAU,EAAE,cAAc;YAE1B,YAAY;YACZ,MAAM,EAAE,QAAQ;YAChB,aAAa,EAAE,gBAAgB;YAE/B,WAAW;YACX,MAAM,EAAE,QAAQ;YAChB,MAAM,EAAE,UAAU;YAClB,WAAW,EAAE,cAAc;YAC3B,WAAW,EAAE,cAAc;YAC3B,eAAe,EAAE,mBAAmB;YACpC,WAAW,EAAE,cAAc;YAC3B,QAAQ,EAAE,UAAU;YACpB,aAAa,EAAE,iBAAiB;YAChC,OAAO,EAAE,SAAS;YAClB,MAAM,EAAE,SAAS;YACjB,YAAY,EAAE,oBAAoB;YAClC,gBAAgB,EAAE,sCAAsC;YAExD,aAAa;YACb,OAAO,EAAE,UAAU;YACnB,UAAU,EAAE,gBAAgB;YAC5B,QAAQ,EAAE,cAAc;YACxB,OAAO,EAAE,aAAa;YAEtB,QAAQ;YACR,QAAQ,EAAE,WAAW;YACrB,WAAW,EAAE,eAAe;YAC5B,YAAY,EAAE,kBAAkB;YAChC,cAAc,EAAE,kBAAkB;YAClC,aAAa,EAAE,mBAAmB;YAClC,KAAK,EAAE,OAAO;YAEd,aAAa;YACb,OAAO,EAAE,UAAU;YAEnB,gBAAgB;YAChB,IAAI,EAAE,MAAM;YACZ,EAAE,EAAE,IAAI;YAER,UAAU;YACV,QAAQ,EAAE,UAAU;YACpB,QAAQ,EAAE,UAAU;YAEpB,OAAO;YACP,MAAM,EAAE,QAAQ;YAChB,aAAa,EAAE,iBAAiB;YAChC,WAAW,EAAE,cAAc;YAC3B,MAAM,EAAE,SAAS;YAEjB,iBAAiB;YACjB,cAAc,EAAE,SAAS;YACzB,iBAAiB,EAAE,YAAY;YAC/B,WAAW,EAAE,MAAM;YACnB,eAAe,EAAE,UAAU;YAC3B,aAAa,EAAE,QAAQ;SACxB;KACF;IAED,KAAK,EAAE;QACL,EAAE,EAAE;YACF,QAAQ;YACR,UAAU,EAAE,gBAAgB;YAC5B,KAAK,EAAE,oBAAoB;YAC3B,gBAAgB,EAAE,cAAc;YAChC,aAAa,EAAE,wBAAwB;YACvC,YAAY,EAAE,0BAA0B;YACxC,QAAQ,EAAE,YAAY;YACtB,mBAAmB,EAAE,UAAU;YAC/B,gBAAgB,EAAE,4BAA4B;YAC9C,WAAW,EAAE,gBAAgB;YAE7B,QAAQ;YACR,cAAc,EAAE,gBAAgB;YAChC,kBAAkB,EAAE,sBAAsB;YAC1C,iBAAiB,EAAE,qBAAqB;YACxC,qBAAqB,EAAE,yBAAyB;YAChD,UAAU,EAAE,eAAe;YAE3B,WAAW;YACX,aAAa,EAAE,cAAc;YAC7B,QAAQ,EAAE,iBAAiB;YAC3B,eAAe,EAAE,WAAW;YAC5B,YAAY,EAAE,wBAAwB;YACtC,aAAa,EAAE,qBAAqB;YACpC,iBAAiB,EAAE,qBAAqB;YACxC,cAAc,EAAE,aAAa;YAC7B,SAAS,EAAE,oBAAoB;YAC/B,QAAQ,EAAE,aAAa;YACvB,UAAU,EAAE,oBAAoB;YAChC,MAAM,EAAE,gBAAgB;YAExB,eAAe;YACf,WAAW,EAAE,kBAAkB;YAC/B,iBAAiB,EAAE,wDAAwD;YAC3E,YAAY,EAAE,WAAW;YACzB,YAAY,EAAE,wBAAwB;YACtC,aAAa,EAAE,uBAAuB;YACtC,cAAc,EAAE,6BAA6B;YAC7C,MAAM,EAAE,UAAU;YAClB,QAAQ,EAAE,wBAAwB;YAClC,QAAQ,EAAE,oCAAoC;YAC9C,aAAa,EAAE,gCAAgC;YAE/C,MAAM;YACN,QAAQ,EAAE,kBAAkB;YAC5B,OAAO,EAAE,8CAA8C;YACvD,QAAQ,EAAE,uCAAuC;YACjD,MAAM,EAAE,yCAAyC;YAEjD,wBAAwB;YACxB,WAAW,EAAE,sBAAsB;YACnC,iBAAiB,EAAE,0DAA0D;YAC7E,YAAY,EAAE,eAAe;YAC7B,UAAU,EAAE,2BAA2B;YACvC,eAAe,EAAE,sBAAsB;YACvC,UAAU,EAAE,wBAAwB;YACpC,gBAAgB,EAAE,sDAAsD;YACxE,WAAW,EAAE,kBAAkB;YAC/B,eAAe,EAAE,qBAAqB;YACtC,WAAW,EAAE,oBAAoB;YACjC,eAAe,EAAE,oDAAoD;YAErE,QAAQ;YACR,WAAW,EAAE,6BAA6B;YAC1C,WAAW,EACT,4EAA4E;YAC9E,kBAAkB,EAAE,wBAAwB;YAC5C,GAAG,EAAE,GAAG;YACR,aAAa,EAAE,wBAAwB;YAEvC,SAAS;YACT,OAAO,EAAE,cAAc;YACvB,iBAAiB,EAAE,4BAA4B;YAC/C,UAAU,EAAE,gCAAgC;YAE5C,uCAAuC;YACvC,uBAAuB,EAAE,iCAAiC;YAC1D,qBAAqB,EAAE,uCAAuC;YAC9D,qBAAqB,EAAE,8BAA8B;YACrD,gBAAgB,EAAE,gCAAgC;YAClD,iBAAiB,EAAE,4BAA4B;YAC/C,gBAAgB,EAAE,mBAAmB;YACrC,gBAAgB,EAAE,4CAA4C;YAC9D,oBAAoB,EAAE,wCAAwC;YAC9D,mBAAmB,EAAE,mCAAmC;YACxD,iBAAiB,EAAE,wCAAwC;YAC3D,gBAAgB,EAAE,wBAAwB;YAC1C,YAAY,EAAE,yCAAyC;SACxD;QACD,EAAE,EAAE;YACF,QAAQ;YACR,UAAU,EAAE,SAAS;YACrB,KAAK,EAAE,OAAO;YACd,gBAAgB,EAAE,eAAe;YACjC,aAAa,EAAE,mBAAmB;YAClC,YAAY,EAAE,qBAAqB;YACnC,QAAQ,EAAE,UAAU;YACpB,mBAAmB,EAAE,UAAU;YAC/B,gBAAgB,EAAE,sBAAsB;YACxC,WAAW,EAAE,SAAS;YAEtB,QAAQ;YACR,cAAc,EAAE,kBAAkB;YAClC,kBAAkB,EAAE,sBAAsB;YAC1C,iBAAiB,EAAE,qBAAqB;YACxC,qBAAqB,EAAE,yBAAyB;YAChD,UAAU,EAAE,eAAe;YAE3B,WAAW;YACX,aAAa,EAAE,gBAAgB;YAC/B,QAAQ,EAAE,WAAW;YACrB,eAAe,EAAE,WAAW;YAC5B,YAAY,EAAE,kBAAkB;YAChC,aAAa,EAAE,sBAAsB;YACrC,iBAAiB,EAAE,sBAAsB;YACzC,cAAc,EAAE,SAAS;YACzB,SAAS,EAAE,wBAAwB;YACnC,QAAQ,EAAE,SAAS;YACnB,UAAU,EAAE,0BAA0B;YACtC,MAAM,EAAE,SAAS;YAEjB,eAAe;YACf,WAAW,EAAE,cAAc;YAC3B,iBAAiB,EAAE,iDAAiD;YACpE,YAAY,EAAE,QAAQ;YACtB,YAAY,EAAE,kBAAkB;YAChC,aAAa,EAAE,oBAAoB;YACnC,cAAc,EAAE,2BAA2B;YAC3C,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,sBAAsB;YAChC,QAAQ,EAAE,gCAAgC;YAC1C,aAAa,EAAE,0BAA0B;YAEzC,MAAM;YACN,QAAQ,EAAE,kBAAkB;YAC5B,OAAO,EAAE,4CAA4C;YACrD,QAAQ,EAAE,mCAAmC;YAC7C,MAAM,EAAE,mCAAmC;YAE3C,wBAAwB;YACxB,WAAW,EAAE,kBAAkB;YAC/B,iBAAiB,EAAE,+CAA+C;YAClE,YAAY,EAAE,WAAW;YACzB,UAAU,EAAE,uBAAuB;YACnC,eAAe,EAAE,kBAAkB;YACnC,UAAU,EAAE,gBAAgB;YAC5B,gBAAgB,EAAE,iDAAiD;YACnE,WAAW,EAAE,cAAc;YAC3B,eAAe,EAAE,sBAAsB;YACvC,WAAW,EAAE,iBAAiB;YAC9B,eAAe,EAAE,wCAAwC;YAEzD,QAAQ;YACR,WAAW,EAAE,wBAAwB;YACrC,WAAW,EAAE,iEAAiE;YAC9E,kBAAkB,EAAE,sBAAsB;YAC1C,GAAG,EAAE,KAAK;YACV,aAAa,EAAE,gBAAgB;YAE/B,SAAS;YACT,OAAO,EAAE,UAAU;YACnB,iBAAiB,EAAE,sBAAsB;YACzC,UAAU,EAAE,mBAAmB;YAE/B,SAAS;YACT,uBAAuB,EAAE,6BAA6B;YACtD,qBAAqB,EAAE,4BAA4B;YACnD,qBAAqB,EAAE,iCAAiC;YACxD,gBAAgB,EAAE,kCAAkC;YACpD,iBAAiB,EAAE,sBAAsB;YACzC,gBAAgB,EAAE,gBAAgB;YAClC,gBAAgB,EAAE,sCAAsC;YACxD,oBAAoB,EAAE,oCAAoC;YAC1D,mBAAmB,EAAE,6BAA6B;YAClD,iBAAiB,EAAE,sBAAsB;YACzC,gBAAgB,EAAE,sBAAsB;YACxC,YAAY,EAAE,sCAAsC;SACrD;KACF;IAED,SAAS,EAAE;QACT,EAAE,EAAE;YACF,gBAAgB;YAChB,eAAe,EAAE,0BAA0B;YAC3C,SAAS,EAAE,kBAAkB;YAC7B,aAAa,EAAE,mDAAmD;YAElE,mBAAmB;YACnB,eAAe,EAAE,0BAA0B;YAC3C,kBAAkB,EAAE,4CAA4C;SACjE;QACD,EAAE,EAAE;YACF,gBAAgB;YAChB,eAAe,EAAE,uBAAuB;YACxC,SAAS,EAAE,YAAY;YACvB,aAAa,EAAE,4CAA4C;YAE3D,mBAAmB;YACnB,eAAe,EAAE,0BAA0B;YAC3C,kBAAkB,EAAE,6CAA6C;SAClE;KACF;CACF,CAAC","sourcesContent":["import { ContentStore } from './types';\n\n/**\n * Traducciones por defecto de valtech-components.\n *\n * Estas traducciones se cargan automáticamente cuando se usa provideValtechI18n()\n * sin necesidad de configuración adicional.\n *\n * Las apps pueden sobrescribir cualquier key pasando su propio content:\n *\n * @example Sobrescribir traducciones específicas\n * ```typescript\n * provideValtechI18n({\n *   content: {\n *     _global: {\n *       es: { success: '¡Lo hicimos!' },  // Sobrescribe solo esta key\n *       en: { success: 'We did it!' },\n *     },\n *   },\n * })\n * ```\n *\n * @example Desactivar traducciones por defecto\n * ```typescript\n * provideValtechI18n({\n *   includeDefaultContent: false,\n *   content: MY_CUSTOM_CONTENT,\n * })\n * ```\n */\nexport const VALTECH_DEFAULT_CONTENT: ContentStore = {\n  _global: {\n    es: {\n      // Acciones comunes\n      submit: 'Enviar',\n      cancel: 'Cancelar',\n      save: 'Guardar',\n      delete: 'Eliminar',\n      edit: 'Editar',\n      close: 'Cerrar',\n      back: 'Volver',\n      next: 'Siguiente',\n      previous: 'Anterior',\n      loading: 'Cargando...',\n      search: 'Buscar',\n      learnMore: 'Saber más',\n\n      // Estados\n      success: 'Éxito',\n      error: 'Error',\n      warning: 'Advertencia',\n      info: 'Información',\n\n      // Confirmaciones\n      confirmDelete: '¿Estás seguro de que deseas eliminar?',\n      confirmCancel: '¿Estás seguro de que deseas cancelar?',\n\n      // Mensajes comunes\n      noResults: 'No se encontraron resultados',\n      required: 'Este campo es requerido',\n\n      // Navegación\n      home: 'Inicio',\n      settings: 'Configuración',\n      profile: 'Perfil',\n      logout: 'Cerrar sesión',\n\n      // Idiomas\n      languageName_es: 'Español',\n      languageName_en: 'English',\n\n      // Componentes - Modales/Selects\n      ok: 'Aceptar',\n      selectOption: 'Seleccionar opción',\n      selectPlaceholder: 'Seleccionar...',\n      itemsSelected: 'elementos seleccionados',\n      selectAll: 'Seleccionar todos',\n      clear: 'Limpiar',\n      apply: 'Aplicar',\n\n      // Componentes - Fechas\n      startDate: 'Fecha inicio',\n      endDate: 'Fecha fin',\n      day: 'día',\n      days: 'días',\n\n      // Componentes - Formularios\n      uploadFile: 'Subir archivo',\n      noFileSelected: 'No has seleccionado archivo',\n      title: 'Título',\n      description: 'Descripción',\n      feedbackType: 'Tipo de feedback',\n      feedbackSuccess: 'Feedback enviado exitosamente',\n      feedbackError: 'Error al enviar el feedback',\n      titlePlaceholder: 'Describe brevemente...',\n      titleValidation: 'El título debe tener entre 5 y 200 caracteres',\n      descriptionPlaceholder: 'Proporciona más detalles...',\n      descriptionValidation: 'La descripción debe tener entre 10 y 5000 caracteres',\n\n      // Componentes - Búsqueda/Grid\n      searchNumber: 'Buscar número...',\n      selected: 'seleccionado/s',\n      max: 'máx:',\n\n      // Componentes - Listas\n      loadMoreComments: 'Cargar más comentarios',\n      noData: 'No hay datos',\n      noRecordsFound: 'No se encontraron registros para mostrar.',\n\n      // Componentes - Acciones\n      copiedToClipboard: '¡Copiado al portapapeles!',\n      language: 'Idioma',\n      selectLanguage: 'Seleccionar idioma...',\n      download: 'Descargar',\n      copy: 'Copiar',\n      share: 'Compartir',\n\n      // Componentes - Data Table / Paginación\n      actions: 'Acciones',\n      showing: 'Mostrando',\n      of: 'de',\n      perPage: 'por página',\n      firstPage: 'Primera página',\n      previousPage: 'Página anterior',\n      nextPage: 'Página siguiente',\n      lastPage: 'Última página',\n\n      // Componentes - Participant Card\n      winner: 'Ganador',\n      ticket: 'boleto',\n      tickets: 'boletos',\n      more: 'más',\n      notes: 'Notas:',\n\n      // Footer\n      allRightsReserved: 'Todos los derechos reservados.',\n\n      // Clipboard\n      linkCopied: '¡Enlace copiado!',\n\n      // Selection\n      select: 'Seleccionar',\n      selectOptions: 'Seleccionar opciones',\n\n      // Comments\n      pinned: 'Fijado',\n      edited: '(editado)',\n      showComment: 'Mostrar comentario',\n      hideReplies: 'Ocultar respuestas',\n      loadMoreReplies: 'Cargar más respuestas',\n      viewReplies: 'Ver respuestas',\n      comments: 'Comentarios',\n      noCommentsYet: 'Sin comentarios aún',\n      publish: 'Publicar',\n      sortBy: 'Ordenar por',\n      writeComment: 'Escribe un comentario...',\n      beFirstToComment: '¡Sé el primero en compartir tu opinión!',\n\n      // Timestamps\n      justNow: 'Ahora mismo',\n      minutesAgo: 'hace {minutes}m',\n      hoursAgo: 'hace {hours}h',\n      daysAgo: 'hace {days}d',\n\n      // Lists\n      loadMore: 'Cargar más',\n      noMoreItems: 'No hay más items',\n      loadingItems: 'Cargando items...',\n      allItemsLoaded: 'Todos los items han sido cargados',\n      errorOccurred: 'Ocurrió un error',\n      retry: 'Reintentar',\n\n      // Navigation\n      seeMore: 'Ver más',\n\n      // Forms - Range\n      from: 'Inicial',\n      to: 'Final',\n\n      // Stepper\n      decrease: 'Disminuir',\n      increase: 'Aumentar',\n\n      // Misc\n      random: 'Aleatorio',\n      pullToRefresh: 'Arrastra para actualizar',\n      timeExpired: 'Tiempo agotado',\n      qrCode: 'Código QR',\n\n      // Payment status\n      paymentPending: 'Pendiente',\n      paymentProcessing: 'Procesando',\n      paymentPaid: 'Pagado',\n      paymentRefunded: 'Reembolsado',\n      paymentFailed: 'Fallido',\n    },\n    en: {\n      // Common actions\n      submit: 'Submit',\n      cancel: 'Cancel',\n      save: 'Save',\n      delete: 'Delete',\n      edit: 'Edit',\n      close: 'Close',\n      back: 'Back',\n      next: 'Next',\n      previous: 'Previous',\n      loading: 'Loading...',\n      search: 'Search',\n      learnMore: 'Learn more',\n\n      // States\n      success: 'Success',\n      error: 'Error',\n      warning: 'Warning',\n      info: 'Information',\n\n      // Confirmations\n      confirmDelete: 'Are you sure you want to delete?',\n      confirmCancel: 'Are you sure you want to cancel?',\n\n      // Common messages\n      noResults: 'No results found',\n      required: 'This field is required',\n\n      // Navigation\n      home: 'Home',\n      settings: 'Settings',\n      profile: 'Profile',\n      logout: 'Log out',\n\n      // Languages\n      languageName_es: 'Español',\n      languageName_en: 'English',\n\n      // Components - Modals/Selects\n      ok: 'OK',\n      selectOption: 'Select option',\n      selectPlaceholder: 'Select...',\n      itemsSelected: 'items selected',\n      selectAll: 'Select all',\n      clear: 'Clear',\n      apply: 'Apply',\n\n      // Components - Dates\n      startDate: 'Start date',\n      endDate: 'End date',\n      day: 'day',\n      days: 'days',\n\n      // Components - Forms\n      uploadFile: 'Upload file',\n      noFileSelected: 'No file selected',\n      title: 'Title',\n      description: 'Description',\n      feedbackType: 'Feedback type',\n      feedbackSuccess: 'Feedback sent successfully',\n      feedbackError: 'Error sending feedback',\n      titlePlaceholder: 'Describe briefly...',\n      titleValidation: 'Title must be between 5 and 200 characters',\n      descriptionPlaceholder: 'Provide more details...',\n      descriptionValidation: 'Description must be between 10 and 5000 characters',\n\n      // Components - Search/Grid\n      searchNumber: 'Search number...',\n      selected: 'selected',\n      max: 'max:',\n\n      // Components - Lists\n      loadMoreComments: 'Load more comments',\n      noData: 'No data',\n      noRecordsFound: 'No records found to display.',\n\n      // Components - Actions\n      copiedToClipboard: 'Copied to clipboard!',\n      language: 'Language',\n      selectLanguage: 'Select language...',\n      download: 'Download',\n      copy: 'Copy',\n      share: 'Share',\n\n      // Components - Data Table / Pagination\n      actions: 'Actions',\n      showing: 'Showing',\n      of: 'of',\n      perPage: 'per page',\n      firstPage: 'First page',\n      previousPage: 'Previous page',\n      nextPage: 'Next page',\n      lastPage: 'Last page',\n\n      // Components - Participant Card\n      winner: 'Winner',\n      ticket: 'ticket',\n      tickets: 'tickets',\n      more: 'more',\n      notes: 'Notes:',\n\n      // Footer\n      allRightsReserved: 'All rights reserved.',\n\n      // Clipboard\n      linkCopied: 'Link copied!',\n\n      // Selection\n      select: 'Select',\n      selectOptions: 'Select options',\n\n      // Comments\n      pinned: 'Pinned',\n      edited: '(edited)',\n      showComment: 'Show comment',\n      hideReplies: 'Hide replies',\n      loadMoreReplies: 'Load more replies',\n      viewReplies: 'View replies',\n      comments: 'Comments',\n      noCommentsYet: 'No comments yet',\n      publish: 'Publish',\n      sortBy: 'Sort by',\n      writeComment: 'Write a comment...',\n      beFirstToComment: 'Be the first to share your thoughts!',\n\n      // Timestamps\n      justNow: 'Just now',\n      minutesAgo: '{minutes}m ago',\n      hoursAgo: '{hours}h ago',\n      daysAgo: '{days}d ago',\n\n      // Lists\n      loadMore: 'Load more',\n      noMoreItems: 'No more items',\n      loadingItems: 'Loading items...',\n      allItemsLoaded: 'All items loaded',\n      errorOccurred: 'An error occurred',\n      retry: 'Retry',\n\n      // Navigation\n      seeMore: 'See more',\n\n      // Forms - Range\n      from: 'From',\n      to: 'To',\n\n      // Stepper\n      decrease: 'Decrease',\n      increase: 'Increase',\n\n      // Misc\n      random: 'Random',\n      pullToRefresh: 'Pull to refresh',\n      timeExpired: 'Time expired',\n      qrCode: 'QR Code',\n\n      // Payment status\n      paymentPending: 'Pending',\n      paymentProcessing: 'Processing',\n      paymentPaid: 'Paid',\n      paymentRefunded: 'Refunded',\n      paymentFailed: 'Failed',\n    },\n  },\n\n  _auth: {\n    es: {\n      // Login\n      loginTitle: 'Iniciar sesión',\n      email: 'Correo electrónico',\n      emailPlaceholder: 'tu@email.com',\n      emailRequired: 'El correo es requerido',\n      emailInvalid: 'Ingresa un correo válido',\n      password: 'Contraseña',\n      passwordPlaceholder: '••••••••',\n      passwordRequired: 'La contraseña es requerida',\n      loginSubmit: 'Iniciar sesión',\n\n      // OAuth\n      orContinueWith: 'o continúa con',\n      continueWithGoogle: 'Continuar con Google',\n      continueWithApple: 'Continuar con Apple',\n      continueWithMicrosoft: 'Continuar con Microsoft',\n      connecting: 'Conectando...',\n\n      // Register\n      registerTitle: 'Crear cuenta',\n      fullName: 'Nombre completo',\n      namePlaceholder: 'Tu nombre',\n      nameRequired: 'El nombre es requerido',\n      nameMinLength: 'Mínimo 2 caracteres',\n      passwordMinLength: 'Mínimo 8 caracteres',\n      registerSubmit: 'Registrarse',\n      noAccount: '¿No tienes cuenta?',\n      register: 'Registrarse',\n      hasAccount: '¿Ya tienes cuenta?',\n      signIn: 'Iniciar sesión',\n\n      // Verify email\n      verifyTitle: 'Verificar correo',\n      verifyDescription: 'Ingresa el código de verificación enviado a tu correo.',\n      verifySubmit: 'Verificar',\n      codeRequired: 'El código es requerido',\n      codeMinLength: 'Ingresa los 6 dígitos',\n      noCodeReceived: '¿No has recibido tu código?',\n      resend: 'Reenviar',\n      resendIn: 'Reenviar en {seconds}s',\n      codeSent: 'Código reenviado. Revisa tu email.',\n      emailVerified: '¡Email verificado! Bienvenido.',\n\n      // MFA\n      mfaTitle: 'Verificación MFA',\n      mfaTOTP: 'Ingresa el código de tu app de autenticación',\n      mfaEmail: 'Ingresa el código enviado a tu correo',\n      mfaSMS: 'Ingresa el código enviado a tu teléfono',\n\n      // Forgot/Reset password\n      forgotTitle: 'Recuperar contraseña',\n      forgotDescription: 'Ingresa tu correo electrónico y te enviaremos un código.',\n      forgotSubmit: 'Enviar código',\n      forgotLink: '¿Olvidaste tu contraseña?',\n      recoverPassword: 'Recuperar contraseña',\n      resetTitle: 'Restablecer contraseña',\n      resetDescription: 'Hemos enviado un código de verificación a tu correo.',\n      newPassword: 'Nueva contraseña',\n      newPasswordHint: 'Mínimo 8 caracteres',\n      resetSubmit: 'Cambiar contraseña',\n      passwordUpdated: '¡Contraseña actualizada! Ya puedes iniciar sesión.',\n\n      // Legal\n      legalPrefix: 'Utilizamos los servicios de',\n      legalSuffix:\n        'para ofrecerte una experiencia segura. Al iniciar sesión, aceptas nuestros',\n      termsAndConditions: 'Términos y Condiciones',\n      and: 'y',\n      privacyPolicy: 'Política de Privacidad',\n\n      // Toasts\n      welcome: '¡Bienvenido!',\n      completeAllFields: 'Completa todos los campos.',\n      enterEmail: 'Ingresa tu correo electrónico.',\n\n      // Errors (mapeados de códigos backend)\n      errorInvalidCredentials: 'Correo o contraseña incorrectos',\n      errorEmailNotVerified: 'Debes verificar tu correo electrónico',\n      errorAccountSuspended: 'Tu cuenta ha sido suspendida',\n      errorEmailExists: 'Este correo ya está registrado',\n      errorWeakPassword: 'La contraseña es muy débil',\n      errorInvalidCode: 'Código incorrecto',\n      errorExpiredCode: 'El código ha expirado. Solicita uno nuevo.',\n      errorTooManyAttempts: 'Demasiados intentos, intenta más tarde',\n      errorMFAInvalidCode: 'Código de verificación incorrecto',\n      errorPopupBlocked: 'Por favor, permite ventanas emergentes',\n      errorOAuthFailed: 'Error de autenticación',\n      errorGeneric: 'Ha ocurrido un error. Intenta de nuevo.',\n    },\n    en: {\n      // Login\n      loginTitle: 'Sign in',\n      email: 'Email',\n      emailPlaceholder: 'you@email.com',\n      emailRequired: 'Email is required',\n      emailInvalid: 'Enter a valid email',\n      password: 'Password',\n      passwordPlaceholder: '••••••••',\n      passwordRequired: 'Password is required',\n      loginSubmit: 'Sign in',\n\n      // OAuth\n      orContinueWith: 'or continue with',\n      continueWithGoogle: 'Continue with Google',\n      continueWithApple: 'Continue with Apple',\n      continueWithMicrosoft: 'Continue with Microsoft',\n      connecting: 'Connecting...',\n\n      // Register\n      registerTitle: 'Create account',\n      fullName: 'Full name',\n      namePlaceholder: 'Your name',\n      nameRequired: 'Name is required',\n      nameMinLength: 'Minimum 2 characters',\n      passwordMinLength: 'Minimum 8 characters',\n      registerSubmit: 'Sign up',\n      noAccount: \"Don't have an account?\",\n      register: 'Sign up',\n      hasAccount: 'Already have an account?',\n      signIn: 'Sign in',\n\n      // Verify email\n      verifyTitle: 'Verify email',\n      verifyDescription: 'Enter the verification code sent to your email.',\n      verifySubmit: 'Verify',\n      codeRequired: 'Code is required',\n      codeMinLength: 'Enter all 6 digits',\n      noCodeReceived: \"Didn't receive your code?\",\n      resend: 'Resend',\n      resendIn: 'Resend in {seconds}s',\n      codeSent: 'Code resent. Check your email.',\n      emailVerified: 'Email verified! Welcome.',\n\n      // MFA\n      mfaTitle: 'MFA Verification',\n      mfaTOTP: 'Enter the code from your authenticator app',\n      mfaEmail: 'Enter the code sent to your email',\n      mfaSMS: 'Enter the code sent to your phone',\n\n      // Forgot/Reset password\n      forgotTitle: 'Recover password',\n      forgotDescription: 'Enter your email and we will send you a code.',\n      forgotSubmit: 'Send code',\n      forgotLink: 'Forgot your password?',\n      recoverPassword: 'Recover password',\n      resetTitle: 'Reset password',\n      resetDescription: 'We have sent a verification code to your email.',\n      newPassword: 'New password',\n      newPasswordHint: 'Minimum 8 characters',\n      resetSubmit: 'Change password',\n      passwordUpdated: 'Password updated! You can now sign in.',\n\n      // Legal\n      legalPrefix: 'We use the services of',\n      legalSuffix: 'to offer you a secure experience. By signing in, you accept our',\n      termsAndConditions: 'Terms and Conditions',\n      and: 'and',\n      privacyPolicy: 'Privacy Policy',\n\n      // Toasts\n      welcome: 'Welcome!',\n      completeAllFields: 'Complete all fields.',\n      enterEmail: 'Enter your email.',\n\n      // Errors\n      errorInvalidCredentials: 'Incorrect email or password',\n      errorEmailNotVerified: 'You must verify your email',\n      errorAccountSuspended: 'Your account has been suspended',\n      errorEmailExists: 'This email is already registered',\n      errorWeakPassword: 'Password is too weak',\n      errorInvalidCode: 'Incorrect code',\n      errorExpiredCode: 'Code has expired. Request a new one.',\n      errorTooManyAttempts: 'Too many attempts, try again later',\n      errorMFAInvalidCode: 'Incorrect verification code',\n      errorPopupBlocked: 'Please allow pop-ups',\n      errorOAuthFailed: 'Authentication error',\n      errorGeneric: 'An error occurred. Please try again.',\n    },\n  },\n\n  AppConfig: {\n    es: {\n      // Update banner\n      updateAvailable: 'Nueva versión disponible',\n      updateNow: 'Actualizar ahora',\n      clickToUpdate: 'Haz clic aquí para actualizar a la última versión',\n\n      // Maintenance page\n      maintenanceMode: 'Sistema en mantenimiento',\n      maintenanceMessage: 'Estamos realizando mejoras. Vuelve pronto.',\n    },\n    en: {\n      // Update banner\n      updateAvailable: 'New version available',\n      updateNow: 'Update now',\n      clickToUpdate: 'Click here to update to the latest version',\n\n      // Maintenance page\n      maintenanceMode: 'System under maintenance',\n      maintenanceMessage: 'We are making improvements. Come back soon.',\n    },\n  },\n};\n"]}
|
|
517
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default-content.js","sourceRoot":"","sources":["../../../../../../src/lib/services/i18n/default-content.ts"],"names":[],"mappings":"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","sourcesContent":["import { ContentStore } from './types';\n\n/**\n * Traducciones por defecto de valtech-components.\n *\n * Estas traducciones se cargan automáticamente cuando se usa provideValtechI18n()\n * sin necesidad de configuración adicional.\n *\n * Las apps pueden sobrescribir cualquier key pasando su propio content:\n *\n * @example Sobrescribir traducciones específicas\n * ```typescript\n * provideValtechI18n({\n *   content: {\n *     _global: {\n *       es: { success: '¡Lo hicimos!' },  // Sobrescribe solo esta key\n *       en: { success: 'We did it!' },\n *     },\n *   },\n * })\n * ```\n *\n * @example Desactivar traducciones por defecto\n * ```typescript\n * provideValtechI18n({\n *   includeDefaultContent: false,\n *   content: MY_CUSTOM_CONTENT,\n * })\n * ```\n */\nexport const VALTECH_DEFAULT_CONTENT: ContentStore = {\n  _global: {\n    es: {\n      // Acciones comunes\n      submit: 'Enviar',\n      cancel: 'Cancelar',\n      save: 'Guardar',\n      delete: 'Eliminar',\n      edit: 'Editar',\n      close: 'Cerrar',\n      back: 'Volver',\n      next: 'Siguiente',\n      previous: 'Anterior',\n      loading: 'Cargando...',\n      search: 'Buscar',\n      learnMore: 'Saber más',\n\n      // Estados\n      success: 'Éxito',\n      error: 'Error',\n      warning: 'Advertencia',\n      info: 'Información',\n\n      // Confirmaciones\n      confirmDelete: '¿Estás seguro de que deseas eliminar?',\n      confirmCancel: '¿Estás seguro de que deseas cancelar?',\n\n      // Mensajes comunes\n      noResults: 'No se encontraron resultados',\n      required: 'Este campo es requerido',\n\n      // Navegación\n      home: 'Inicio',\n      settings: 'Configuración',\n      profile: 'Perfil',\n      logout: 'Cerrar sesión',\n\n      // Idiomas\n      languageName_es: 'Español',\n      languageName_en: 'English',\n\n      // Componentes - Modales/Selects\n      ok: 'Aceptar',\n      selectOption: 'Seleccionar opción',\n      selectPlaceholder: 'Seleccionar...',\n      itemsSelected: 'elementos seleccionados',\n      selectAll: 'Seleccionar todos',\n      clear: 'Limpiar',\n      apply: 'Aplicar',\n\n      // Componentes - Fechas\n      startDate: 'Fecha inicio',\n      endDate: 'Fecha fin',\n      day: 'día',\n      days: 'días',\n\n      // Componentes - Formularios\n      uploadFile: 'Subir archivo',\n      noFileSelected: 'No has seleccionado archivo',\n      title: 'Título',\n      description: 'Descripción',\n      feedbackType: 'Tipo de feedback',\n      feedbackSuccess: 'Feedback enviado exitosamente',\n      feedbackError: 'Error al enviar el feedback',\n      titlePlaceholder: 'Describe brevemente...',\n      titleValidation: 'El título debe tener entre 5 y 200 caracteres',\n      descriptionPlaceholder: 'Proporciona más detalles...',\n      descriptionValidation: 'La descripción debe tener entre 10 y 5000 caracteres',\n\n      // Componentes - Búsqueda/Grid\n      searchNumber: 'Buscar número...',\n      selected: 'seleccionado/s',\n      max: 'máx:',\n\n      // Componentes - Listas\n      loadMoreComments: 'Cargar más comentarios',\n      noData: 'No hay datos',\n      noRecordsFound: 'No se encontraron registros para mostrar.',\n\n      // Componentes - Acciones\n      copiedToClipboard: '¡Copiado al portapapeles!',\n      language: 'Idioma',\n      selectLanguage: 'Seleccionar idioma...',\n      download: 'Descargar',\n      copy: 'Copiar',\n      share: 'Compartir',\n\n      // Componentes - Data Table / Paginación\n      actions: 'Acciones',\n      showing: 'Mostrando',\n      of: 'de',\n      perPage: 'por página',\n      firstPage: 'Primera página',\n      previousPage: 'Página anterior',\n      nextPage: 'Página siguiente',\n      lastPage: 'Última página',\n\n      // Componentes - Participant Card\n      winner: 'Ganador',\n      ticket: 'boleto',\n      tickets: 'boletos',\n      more: 'más',\n      notes: 'Notas:',\n\n      // Footer\n      allRightsReserved: 'Todos los derechos reservados.',\n\n      // Clipboard\n      linkCopied: '¡Enlace copiado!',\n\n      // Selection\n      select: 'Seleccionar',\n      selectOptions: 'Seleccionar opciones',\n\n      // Comments\n      pinned: 'Fijado',\n      edited: '(editado)',\n      showComment: 'Mostrar comentario',\n      hideReplies: 'Ocultar respuestas',\n      loadMoreReplies: 'Cargar más respuestas',\n      viewReplies: 'Ver respuestas',\n      comments: 'Comentarios',\n      noCommentsYet: 'Sin comentarios aún',\n      publish: 'Publicar',\n      sortBy: 'Ordenar por',\n      writeComment: 'Escribe un comentario...',\n      beFirstToComment: '¡Sé el primero en compartir tu opinión!',\n\n      // Timestamps\n      justNow: 'Ahora mismo',\n      minutesAgo: 'hace {minutes}m',\n      hoursAgo: 'hace {hours}h',\n      daysAgo: 'hace {days}d',\n\n      // Lists\n      loadMore: 'Cargar más',\n      noMoreItems: 'No hay más items',\n      loadingItems: 'Cargando items...',\n      allItemsLoaded: 'Todos los items han sido cargados',\n      errorOccurred: 'Ocurrió un error',\n      retry: 'Reintentar',\n\n      // Navigation\n      seeMore: 'Ver más',\n\n      // Forms - Range\n      from: 'Inicial',\n      to: 'Final',\n\n      // Stepper\n      decrease: 'Disminuir',\n      increase: 'Aumentar',\n\n      // Misc\n      random: 'Aleatorio',\n      pullToRefresh: 'Arrastra para actualizar',\n      timeExpired: 'Tiempo agotado',\n      qrCode: 'Código QR',\n\n      // Payment status\n      paymentPending: 'Pendiente',\n      paymentProcessing: 'Procesando',\n      paymentPaid: 'Pagado',\n      paymentRefunded: 'Reembolsado',\n      paymentFailed: 'Fallido',\n    },\n    en: {\n      // Common actions\n      submit: 'Submit',\n      cancel: 'Cancel',\n      save: 'Save',\n      delete: 'Delete',\n      edit: 'Edit',\n      close: 'Close',\n      back: 'Back',\n      next: 'Next',\n      previous: 'Previous',\n      loading: 'Loading...',\n      search: 'Search',\n      learnMore: 'Learn more',\n\n      // States\n      success: 'Success',\n      error: 'Error',\n      warning: 'Warning',\n      info: 'Information',\n\n      // Confirmations\n      confirmDelete: 'Are you sure you want to delete?',\n      confirmCancel: 'Are you sure you want to cancel?',\n\n      // Common messages\n      noResults: 'No results found',\n      required: 'This field is required',\n\n      // Navigation\n      home: 'Home',\n      settings: 'Settings',\n      profile: 'Profile',\n      logout: 'Log out',\n\n      // Languages\n      languageName_es: 'Español',\n      languageName_en: 'English',\n\n      // Components - Modals/Selects\n      ok: 'OK',\n      selectOption: 'Select option',\n      selectPlaceholder: 'Select...',\n      itemsSelected: 'items selected',\n      selectAll: 'Select all',\n      clear: 'Clear',\n      apply: 'Apply',\n\n      // Components - Dates\n      startDate: 'Start date',\n      endDate: 'End date',\n      day: 'day',\n      days: 'days',\n\n      // Components - Forms\n      uploadFile: 'Upload file',\n      noFileSelected: 'No file selected',\n      title: 'Title',\n      description: 'Description',\n      feedbackType: 'Feedback type',\n      feedbackSuccess: 'Feedback sent successfully',\n      feedbackError: 'Error sending feedback',\n      titlePlaceholder: 'Describe briefly...',\n      titleValidation: 'Title must be between 5 and 200 characters',\n      descriptionPlaceholder: 'Provide more details...',\n      descriptionValidation: 'Description must be between 10 and 5000 characters',\n\n      // Components - Search/Grid\n      searchNumber: 'Search number...',\n      selected: 'selected',\n      max: 'max:',\n\n      // Components - Lists\n      loadMoreComments: 'Load more comments',\n      noData: 'No data',\n      noRecordsFound: 'No records found to display.',\n\n      // Components - Actions\n      copiedToClipboard: 'Copied to clipboard!',\n      language: 'Language',\n      selectLanguage: 'Select language...',\n      download: 'Download',\n      copy: 'Copy',\n      share: 'Share',\n\n      // Components - Data Table / Pagination\n      actions: 'Actions',\n      showing: 'Showing',\n      of: 'of',\n      perPage: 'per page',\n      firstPage: 'First page',\n      previousPage: 'Previous page',\n      nextPage: 'Next page',\n      lastPage: 'Last page',\n\n      // Components - Participant Card\n      winner: 'Winner',\n      ticket: 'ticket',\n      tickets: 'tickets',\n      more: 'more',\n      notes: 'Notes:',\n\n      // Footer\n      allRightsReserved: 'All rights reserved.',\n\n      // Clipboard\n      linkCopied: 'Link copied!',\n\n      // Selection\n      select: 'Select',\n      selectOptions: 'Select options',\n\n      // Comments\n      pinned: 'Pinned',\n      edited: '(edited)',\n      showComment: 'Show comment',\n      hideReplies: 'Hide replies',\n      loadMoreReplies: 'Load more replies',\n      viewReplies: 'View replies',\n      comments: 'Comments',\n      noCommentsYet: 'No comments yet',\n      publish: 'Publish',\n      sortBy: 'Sort by',\n      writeComment: 'Write a comment...',\n      beFirstToComment: 'Be the first to share your thoughts!',\n\n      // Timestamps\n      justNow: 'Just now',\n      minutesAgo: '{minutes}m ago',\n      hoursAgo: '{hours}h ago',\n      daysAgo: '{days}d ago',\n\n      // Lists\n      loadMore: 'Load more',\n      noMoreItems: 'No more items',\n      loadingItems: 'Loading items...',\n      allItemsLoaded: 'All items loaded',\n      errorOccurred: 'An error occurred',\n      retry: 'Retry',\n\n      // Navigation\n      seeMore: 'See more',\n\n      // Forms - Range\n      from: 'From',\n      to: 'To',\n\n      // Stepper\n      decrease: 'Decrease',\n      increase: 'Increase',\n\n      // Misc\n      random: 'Random',\n      pullToRefresh: 'Pull to refresh',\n      timeExpired: 'Time expired',\n      qrCode: 'QR Code',\n\n      // Payment status\n      paymentPending: 'Pending',\n      paymentProcessing: 'Processing',\n      paymentPaid: 'Paid',\n      paymentRefunded: 'Refunded',\n      paymentFailed: 'Failed',\n    },\n  },\n\n  _auth: {\n    es: {\n      // Login\n      loginTitle: 'Iniciar sesión',\n      email: 'Correo electrónico',\n      emailPlaceholder: 'tu@email.com',\n      emailRequired: 'El correo es requerido',\n      emailInvalid: 'Ingresa un correo válido',\n      password: 'Contraseña',\n      passwordPlaceholder: '••••••••',\n      passwordRequired: 'La contraseña es requerida',\n      loginSubmit: 'Iniciar sesión',\n\n      // OAuth\n      orContinueWith: 'o continúa con',\n      continueWithGoogle: 'Continuar con Google',\n      continueWithApple: 'Continuar con Apple',\n      continueWithMicrosoft: 'Continuar con Microsoft',\n      connecting: 'Conectando...',\n\n      // Register\n      registerTitle: 'Crear cuenta',\n      fullName: 'Nombre completo',\n      namePlaceholder: 'Tu nombre',\n      nameRequired: 'El nombre es requerido',\n      nameMinLength: 'Mínimo 2 caracteres',\n      passwordMinLength: 'Mínimo 8 caracteres',\n      registerSubmit: 'Registrarse',\n      noAccount: '¿No tienes cuenta?',\n      register: 'Registrarse',\n      hasAccount: '¿Ya tienes cuenta?',\n      signIn: 'Iniciar sesión',\n\n      // Verify email\n      verifyTitle: 'Verificar correo',\n      verifyDescription: 'Ingresa el código de verificación enviado a tu correo.',\n      verifySubmit: 'Verificar',\n      codeRequired: 'El código es requerido',\n      codeMinLength: 'Ingresa los 6 dígitos',\n      noCodeReceived: '¿No has recibido tu código?',\n      resend: 'Reenviar',\n      resendIn: 'Reenviar en {seconds}s',\n      codeSent: 'Código reenviado. Revisa tu email.',\n      emailVerified: '¡Email verificado! Bienvenido.',\n\n      // MFA\n      mfaTitle: 'Verificación MFA',\n      mfaTOTP: 'Ingresa el código de tu app de autenticación',\n      mfaEmail: 'Ingresa el código enviado a tu correo',\n      mfaSMS: 'Ingresa el código enviado a tu teléfono',\n\n      // Forgot/Reset password\n      forgotTitle: 'Recuperar contraseña',\n      forgotDescription: 'Ingresa tu correo electrónico y te enviaremos un código.',\n      forgotSubmit: 'Enviar código',\n      forgotLink: '¿Olvidaste tu contraseña?',\n      recoverPassword: 'Recuperar contraseña',\n      resetTitle: 'Restablecer contraseña',\n      resetDescription: 'Hemos enviado un código de verificación a tu correo.',\n      newPassword: 'Nueva contraseña',\n      newPasswordHint: 'Mínimo 8 caracteres',\n      resetSubmit: 'Cambiar contraseña',\n      passwordUpdated: '¡Contraseña actualizada! Ya puedes iniciar sesión.',\n\n      // Legal\n      legalPrefix: 'Utilizamos los servicios de',\n      legalSuffix:\n        'para ofrecerte una experiencia segura. Al iniciar sesión, aceptas nuestros',\n      termsAndConditions: 'Términos y Condiciones',\n      and: 'y',\n      privacyPolicy: 'Política de Privacidad',\n\n      // Toasts\n      welcome: '¡Bienvenido!',\n      completeAllFields: 'Completa todos los campos.',\n      enterEmail: 'Ingresa tu correo electrónico.',\n\n      // Errors (mapeados de códigos backend)\n      errorInvalidCredentials: 'Correo o contraseña incorrectos',\n      errorEmailNotVerified: 'Debes verificar tu correo electrónico',\n      errorAccountSuspended: 'Tu cuenta ha sido suspendida',\n      errorEmailExists: 'Este correo ya está registrado',\n      errorWeakPassword: 'La contraseña es muy débil',\n      errorInvalidCode: 'Código incorrecto',\n      errorExpiredCode: 'El código ha expirado. Solicita uno nuevo.',\n      errorTooManyAttempts: 'Demasiados intentos, intenta más tarde',\n      errorMFAInvalidCode: 'Código de verificación incorrecto',\n      errorPopupBlocked: 'Por favor, permite ventanas emergentes',\n      errorOAuthFailed: 'Error de autenticación',\n      errorGeneric: 'Ha ocurrido un error. Intenta de nuevo.',\n    },\n    en: {\n      // Login\n      loginTitle: 'Sign in',\n      email: 'Email',\n      emailPlaceholder: 'you@email.com',\n      emailRequired: 'Email is required',\n      emailInvalid: 'Enter a valid email',\n      password: 'Password',\n      passwordPlaceholder: '••••••••',\n      passwordRequired: 'Password is required',\n      loginSubmit: 'Sign in',\n\n      // OAuth\n      orContinueWith: 'or continue with',\n      continueWithGoogle: 'Continue with Google',\n      continueWithApple: 'Continue with Apple',\n      continueWithMicrosoft: 'Continue with Microsoft',\n      connecting: 'Connecting...',\n\n      // Register\n      registerTitle: 'Create account',\n      fullName: 'Full name',\n      namePlaceholder: 'Your name',\n      nameRequired: 'Name is required',\n      nameMinLength: 'Minimum 2 characters',\n      passwordMinLength: 'Minimum 8 characters',\n      registerSubmit: 'Sign up',\n      noAccount: \"Don't have an account?\",\n      register: 'Sign up',\n      hasAccount: 'Already have an account?',\n      signIn: 'Sign in',\n\n      // Verify email\n      verifyTitle: 'Verify email',\n      verifyDescription: 'Enter the verification code sent to your email.',\n      verifySubmit: 'Verify',\n      codeRequired: 'Code is required',\n      codeMinLength: 'Enter all 6 digits',\n      noCodeReceived: \"Didn't receive your code?\",\n      resend: 'Resend',\n      resendIn: 'Resend in {seconds}s',\n      codeSent: 'Code resent. Check your email.',\n      emailVerified: 'Email verified! Welcome.',\n\n      // MFA\n      mfaTitle: 'MFA Verification',\n      mfaTOTP: 'Enter the code from your authenticator app',\n      mfaEmail: 'Enter the code sent to your email',\n      mfaSMS: 'Enter the code sent to your phone',\n\n      // Forgot/Reset password\n      forgotTitle: 'Recover password',\n      forgotDescription: 'Enter your email and we will send you a code.',\n      forgotSubmit: 'Send code',\n      forgotLink: 'Forgot your password?',\n      recoverPassword: 'Recover password',\n      resetTitle: 'Reset password',\n      resetDescription: 'We have sent a verification code to your email.',\n      newPassword: 'New password',\n      newPasswordHint: 'Minimum 8 characters',\n      resetSubmit: 'Change password',\n      passwordUpdated: 'Password updated! You can now sign in.',\n\n      // Legal\n      legalPrefix: 'We use the services of',\n      legalSuffix: 'to offer you a secure experience. By signing in, you accept our',\n      termsAndConditions: 'Terms and Conditions',\n      and: 'and',\n      privacyPolicy: 'Privacy Policy',\n\n      // Toasts\n      welcome: 'Welcome!',\n      completeAllFields: 'Complete all fields.',\n      enterEmail: 'Enter your email.',\n\n      // Errors\n      errorInvalidCredentials: 'Incorrect email or password',\n      errorEmailNotVerified: 'You must verify your email',\n      errorAccountSuspended: 'Your account has been suspended',\n      errorEmailExists: 'This email is already registered',\n      errorWeakPassword: 'Password is too weak',\n      errorInvalidCode: 'Incorrect code',\n      errorExpiredCode: 'Code has expired. Request a new one.',\n      errorTooManyAttempts: 'Too many attempts, try again later',\n      errorMFAInvalidCode: 'Incorrect verification code',\n      errorPopupBlocked: 'Please allow pop-ups',\n      errorOAuthFailed: 'Authentication error',\n      errorGeneric: 'An error occurred. Please try again.',\n    },\n  },\n\n  AppConfig: {\n    es: {\n      // Update banner\n      updateAvailable: 'Nueva versión disponible',\n      updateNow: 'Actualizar ahora',\n      clickToUpdate: 'Haz clic aquí para actualizar a la última versión',\n\n      // Maintenance page\n      maintenanceMode: 'Sistema en mantenimiento',\n      maintenanceMessage: 'Estamos realizando mejoras. Vuelve pronto.',\n    },\n    en: {\n      // Update banner\n      updateAvailable: 'New version available',\n      updateNow: 'Update now',\n      clickToUpdate: 'Click here to update to the latest version',\n\n      // Maintenance page\n      maintenanceMode: 'System under maintenance',\n      maintenanceMessage: 'We are making improvements. Come back soon.',\n    },\n  },\n\n  ContentReaction: {\n    es: {\n      question: '¿Te resultó útil este contenido?',\n      commentPlaceholder: 'Cuéntanos más (opcional)...',\n      submit: 'Enviar',\n      update: 'Actualizar',\n      thankYou: '¡Gracias por tu opinión!',\n      submitted: 'Tu opinión ha sido registrada',\n      errorSubmitting: 'Error al enviar. Intenta de nuevo.',\n    },\n    en: {\n      question: 'Was this content helpful?',\n      commentPlaceholder: 'Tell us more (optional)...',\n      submit: 'Submit',\n      update: 'Update',\n      thankYou: 'Thanks for your feedback!',\n      submitted: 'Your feedback has been recorded',\n      errorSubmitting: 'Error submitting. Please try again.',\n    },\n  },\n};\n"]}
|
package/esm2022/lib/version.mjs
CHANGED
|
@@ -2,5 +2,5 @@
|
|
|
2
2
|
* Current version of valtech-components.
|
|
3
3
|
* This is automatically updated during the publish process.
|
|
4
4
|
*/
|
|
5
|
-
export const VERSION = '2.0.
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
export const VERSION = '2.0.684';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC42ODQnO1xuIl19
|
|
@@ -50,7 +50,7 @@ import 'prismjs/components/prism-json';
|
|
|
50
50
|
* Current version of valtech-components.
|
|
51
51
|
* This is automatically updated during the publish process.
|
|
52
52
|
*/
|
|
53
|
-
const VERSION = '2.0.
|
|
53
|
+
const VERSION = '2.0.684';
|
|
54
54
|
|
|
55
55
|
/**
|
|
56
56
|
* Servicio para gestionar presets de componentes.
|
|
@@ -4246,6 +4246,26 @@ const VALTECH_DEFAULT_CONTENT = {
|
|
|
4246
4246
|
maintenanceMessage: 'We are making improvements. Come back soon.',
|
|
4247
4247
|
},
|
|
4248
4248
|
},
|
|
4249
|
+
ContentReaction: {
|
|
4250
|
+
es: {
|
|
4251
|
+
question: '¿Te resultó útil este contenido?',
|
|
4252
|
+
commentPlaceholder: 'Cuéntanos más (opcional)...',
|
|
4253
|
+
submit: 'Enviar',
|
|
4254
|
+
update: 'Actualizar',
|
|
4255
|
+
thankYou: '¡Gracias por tu opinión!',
|
|
4256
|
+
submitted: 'Tu opinión ha sido registrada',
|
|
4257
|
+
errorSubmitting: 'Error al enviar. Intenta de nuevo.',
|
|
4258
|
+
},
|
|
4259
|
+
en: {
|
|
4260
|
+
question: 'Was this content helpful?',
|
|
4261
|
+
commentPlaceholder: 'Tell us more (optional)...',
|
|
4262
|
+
submit: 'Submit',
|
|
4263
|
+
update: 'Update',
|
|
4264
|
+
thankYou: 'Thanks for your feedback!',
|
|
4265
|
+
submitted: 'Your feedback has been recorded',
|
|
4266
|
+
errorSubmitting: 'Error submitting. Please try again.',
|
|
4267
|
+
},
|
|
4268
|
+
},
|
|
4249
4269
|
};
|
|
4250
4270
|
|
|
4251
4271
|
/**
|
|
@@ -38403,35 +38423,38 @@ class FeedbackService {
|
|
|
38403
38423
|
* ```
|
|
38404
38424
|
*/
|
|
38405
38425
|
async checkFeedback(entityType, entityId) {
|
|
38426
|
+
// Si no hay usuario autenticado, no puede haber feedback previo
|
|
38427
|
+
// Retornar inmediatamente sin llamar al API (evita 401 y redirect a login)
|
|
38428
|
+
const userId = this.auth?.user()?.userId;
|
|
38429
|
+
if (!userId) {
|
|
38430
|
+
return { operationId: '', hasFeedback: false };
|
|
38431
|
+
}
|
|
38406
38432
|
// 1. Intentar Firebase primero (si está disponible)
|
|
38407
|
-
if (this.firestore
|
|
38433
|
+
if (this.firestore) {
|
|
38408
38434
|
try {
|
|
38409
|
-
|
|
38410
|
-
|
|
38411
|
-
|
|
38412
|
-
|
|
38413
|
-
|
|
38414
|
-
|
|
38415
|
-
|
|
38416
|
-
|
|
38417
|
-
|
|
38418
|
-
|
|
38419
|
-
|
|
38420
|
-
|
|
38421
|
-
|
|
38422
|
-
createdAt: doc.createdAt?.toISOString(),
|
|
38423
|
-
};
|
|
38424
|
-
}
|
|
38425
|
-
// Doc no existe = no hay feedback
|
|
38426
|
-
return { operationId: '', hasFeedback: false };
|
|
38435
|
+
// Path: feedback/{entityType}/{entityId}/{userId}
|
|
38436
|
+
// FirestoreService agrega automáticamente el prefijo apps/{appId}/
|
|
38437
|
+
const collectionPath = `feedback/${entityType}/${entityId}`;
|
|
38438
|
+
const doc = await this.firestore.getDoc(collectionPath, userId);
|
|
38439
|
+
if (doc) {
|
|
38440
|
+
return {
|
|
38441
|
+
operationId: '',
|
|
38442
|
+
hasFeedback: true,
|
|
38443
|
+
feedbackId: doc.feedbackId,
|
|
38444
|
+
type: doc.type,
|
|
38445
|
+
reactionValue: doc.reactionValue,
|
|
38446
|
+
createdAt: doc.createdAt?.toISOString(),
|
|
38447
|
+
};
|
|
38427
38448
|
}
|
|
38449
|
+
// Doc no existe = no hay feedback
|
|
38450
|
+
return { operationId: '', hasFeedback: false };
|
|
38428
38451
|
}
|
|
38429
38452
|
catch (error) {
|
|
38430
38453
|
console.warn('[FeedbackService] Firebase check failed, falling back to API:', error);
|
|
38431
38454
|
// Fallback a API
|
|
38432
38455
|
}
|
|
38433
38456
|
}
|
|
38434
|
-
// 2. Fallback: llamar API
|
|
38457
|
+
// 2. Fallback: llamar API (solo si hay usuario autenticado)
|
|
38435
38458
|
const params = new URLSearchParams({
|
|
38436
38459
|
appId: this.config.appId,
|
|
38437
38460
|
entityType,
|