tas-uell-sdk 0.2.0 → 0.3.0
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/README.md +2 -0
- package/esm2020/lib/components/tas-btn/tas-btn.component.mjs +2 -13
- package/esm2020/lib/components/tas-feedback-modal/tas-feedback-modal.component.mjs +229 -0
- package/esm2020/lib/components/tas-floating-call/tas-floating-call.component.mjs +38 -3
- package/esm2020/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.mjs +46 -8
- package/esm2020/lib/components/tas-videocall/tas-videocall.component.mjs +41 -6
- package/esm2020/lib/components/tas-waiting-room/tas-waiting-room.component.mjs +18 -3
- package/esm2020/lib/interfaces/tas.interfaces.mjs +6 -1
- package/esm2020/lib/services/tas.service.mjs +40 -1
- package/esm2020/lib/tas-uell-sdk.module.mjs +8 -3
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/tas-uell-sdk.mjs +421 -36
- package/fesm2015/tas-uell-sdk.mjs.map +1 -1
- package/fesm2020/tas-uell-sdk.mjs +416 -36
- package/fesm2020/tas-uell-sdk.mjs.map +1 -1
- package/lib/components/tas-btn/tas-btn.component.d.ts +1 -2
- package/lib/components/tas-feedback-modal/tas-feedback-modal.component.d.ts +101 -0
- package/lib/components/tas-floating-call/tas-floating-call.component.d.ts +5 -0
- package/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.d.ts +12 -1
- package/lib/components/tas-videocall/tas-videocall.component.d.ts +8 -2
- package/lib/components/tas-waiting-room/tas-waiting-room.component.d.ts +9 -0
- package/lib/interfaces/tas.interfaces.d.ts +17 -0
- package/lib/services/tas.service.d.ts +15 -1
- package/lib/tas-uell-sdk.module.d.ts +5 -4
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/src/lib/styles/tas-global.scss +17 -0
package/README.md
CHANGED
|
@@ -270,6 +270,8 @@ Initiates video calls. Supports style variants:
|
|
|
270
270
|
| :--- | :--- | :--- | :--- |
|
|
271
271
|
| `variant` | `'default' \| 'teal'` | `'default'` | Button style variant |
|
|
272
272
|
| `buttonLabel` | `string` | `'Iniciar TAS'` | Button text |
|
|
273
|
+
| `skipStatusCheck` | `boolean` | `false` | When true, shows button immediately without API status check |
|
|
274
|
+
| `devOpenFeedbackModal` | `boolean` | `false` | DEV ONLY: Clicking button opens feedback modal directly |
|
|
273
275
|
|
|
274
276
|
### TasIncomingAppointmentComponent - `<tas-incoming-appointment>`
|
|
275
277
|
|
|
@@ -19,9 +19,6 @@ export class TasButtonComponent {
|
|
|
19
19
|
// Style customization
|
|
20
20
|
this.variant = 'default';
|
|
21
21
|
this.buttonLabel = 'Iniciar TAS';
|
|
22
|
-
// Skip status check - when true, button shows immediately without API call
|
|
23
|
-
// Useful when parent component already knows the appointment is valid
|
|
24
|
-
this.skipStatusCheck = false;
|
|
25
22
|
this.isLoading = false;
|
|
26
23
|
// Status check state
|
|
27
24
|
this.isCheckingStatus = false;
|
|
@@ -66,12 +63,6 @@ export class TasButtonComponent {
|
|
|
66
63
|
this.videoCallModalRef = null;
|
|
67
64
|
}
|
|
68
65
|
}));
|
|
69
|
-
// If skipStatusCheck is true, show button immediately without polling
|
|
70
|
-
if (this.skipStatusCheck) {
|
|
71
|
-
this.shouldShowButton = true;
|
|
72
|
-
this.isJoinable = true;
|
|
73
|
-
return;
|
|
74
|
-
}
|
|
75
66
|
// Start status checking
|
|
76
67
|
this.startStatusPolling();
|
|
77
68
|
}
|
|
@@ -193,7 +184,7 @@ export class TasButtonComponent {
|
|
|
193
184
|
}
|
|
194
185
|
}
|
|
195
186
|
TasButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, deps: [{ token: i1.NgbModal }, { token: i2.TasService }, { token: i3.TasUtilityService }], target: i0.ɵɵFactoryTarget.Component });
|
|
196
|
-
TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasButtonComponent, selector: "tas-btn", inputs: { roomType: "roomType", entityId: "entityId", tenant: "tenant", businessRole: "businessRole", currentUser: "currentUser", variant: "variant", buttonLabel: "buttonLabel"
|
|
187
|
+
TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasButtonComponent, selector: "tas-btn", inputs: { roomType: "roomType", entityId: "entityId", tenant: "tenant", businessRole: "businessRole", currentUser: "currentUser", variant: "variant", buttonLabel: "buttonLabel" }, ngImport: i0, template: "<span\n *ngIf=\"shouldShowButton\"\n [ngbTooltip]=\"isDisabled && disabledReason ? disabledReason : null\"\n container=\"body\"\n placement=\"top\"\n tooltipClass=\"tas-btn-tooltip\"\n>\n <button\n type=\"button\"\n class=\"btn btn-primary tas-btn\"\n [class.tas-btn--teal]=\"variant === 'teal'\"\n (click)=\"onClick()\"\n [disabled]=\"isDisabled\"\n >\n <i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n <span *ngIf=\"!isLoading\">{{ buttonLabel }}</span>\n <span *ngIf=\"isLoading\">Processing...</span>\n </button>\n</span>\n", styles: [":host{display:inline-block}.tas-btn{background-color:#ee316b!important;color:#fff!important;border-color:#ee316b!important;margin-right:24px;display:flex;padding:6px 14px;justify-content:center;align-items:center;gap:7px;flex-shrink:0;flex-grow:0}.tas-btn:disabled{background-color:#ccc!important;border-color:#ccc!important;cursor:not-allowed}.tas-btn:hover:not(:disabled){background-color:#d62a5f!important;border-color:#d62a5f!important}.tas-btn i{margin-right:5px}.tas-btn--teal{background-color:#0097a7!important;border-color:#0097a7!important;border-radius:24px;font-weight:500;margin-right:0;padding:6px 14px}.tas-btn--teal:hover:not(:disabled){background-color:#00838f!important;border-color:#00838f!important}\n"], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "triggers", "container", "disableTooltip", "tooltipClass", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }] });
|
|
197
188
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, decorators: [{
|
|
198
189
|
type: Component,
|
|
199
190
|
args: [{ selector: 'tas-btn', template: "<span\n *ngIf=\"shouldShowButton\"\n [ngbTooltip]=\"isDisabled && disabledReason ? disabledReason : null\"\n container=\"body\"\n placement=\"top\"\n tooltipClass=\"tas-btn-tooltip\"\n>\n <button\n type=\"button\"\n class=\"btn btn-primary tas-btn\"\n [class.tas-btn--teal]=\"variant === 'teal'\"\n (click)=\"onClick()\"\n [disabled]=\"isDisabled\"\n >\n <i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n <span *ngIf=\"!isLoading\">{{ buttonLabel }}</span>\n <span *ngIf=\"isLoading\">Processing...</span>\n </button>\n</span>\n", styles: [":host{display:inline-block}.tas-btn{background-color:#ee316b!important;color:#fff!important;border-color:#ee316b!important;margin-right:24px;display:flex;padding:6px 14px;justify-content:center;align-items:center;gap:7px;flex-shrink:0;flex-grow:0}.tas-btn:disabled{background-color:#ccc!important;border-color:#ccc!important;cursor:not-allowed}.tas-btn:hover:not(:disabled){background-color:#d62a5f!important;border-color:#d62a5f!important}.tas-btn i{margin-right:5px}.tas-btn--teal{background-color:#0097a7!important;border-color:#0097a7!important;border-radius:24px;font-weight:500;margin-right:0;padding:6px 14px}.tas-btn--teal:hover:not(:disabled){background-color:#00838f!important;border-color:#00838f!important}\n"] }]
|
|
@@ -211,7 +202,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImpo
|
|
|
211
202
|
type: Input
|
|
212
203
|
}], buttonLabel: [{
|
|
213
204
|
type: Input
|
|
214
|
-
}], skipStatusCheck: [{
|
|
215
|
-
type: Input
|
|
216
205
|
}] } });
|
|
217
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-btn.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Input } from '@angular/core';\nimport { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';\nimport { Subscription } from 'rxjs';\nimport {\n  TasCurrentUser,\n  \n  TasRoomType,\n  ViewMode,\n  TasBusinessRole,\n} from '../../interfaces/tas.interfaces';\nimport { TasWaitingRoomComponent } from '../tas-waiting-room/tas-waiting-room.component';\nimport { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';\nimport { TasService } from '../../services/tas.service';\nimport { TasUtilityService } from '../../services/tas-utility.service';\n\n@Component({\n  selector: 'tas-btn',\n  templateUrl: './tas-btn.component.html',\n  styleUrls: ['./tas-btn.component.scss'],\n})\nexport class TasButtonComponent implements OnInit, OnDestroy {\n  // Status endpoint params\n  @Input() roomType: TasRoomType = TasRoomType.TAS;\n  @Input() entityId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n\n  // User info for token request\n  @Input() currentUser!: TasCurrentUser;\n\n  // Style customization\n  @Input() variant: 'default' | 'teal' = 'default';\n  @Input() buttonLabel: string = 'Iniciar TAS';\n\n  // Skip status check - when true, button shows immediately without API call\n  // Useful when parent component already knows the appointment is valid\n  @Input() skipStatusCheck = false;\n\n  public isLoading = false;\n\n  // Status check state\n  public isCheckingStatus = false;\n  public isStatusError = false;\n  public statusErrorMessage = '';\n  public isJoinable = false; // Tracks joinable field from status response\n  public shouldShowButton = false; // Hidden by default, shown after status check confirms visibility\n\n  private subscriptions = new Subscription();\n  private currentModalRef: NgbModalRef | null = null;\n  private videoCallModalRef: NgbModalRef | null = null;\n  private statusPollingInterval: ReturnType<typeof setInterval> | null = null;\n  private readonly STATUS_POLL_INTERVAL_MS = 30000; // 30 seconds\n\n  /** Whether user is backoffice (or admin/manager) */\n  public get isBackoffice(): boolean {\n    return (\n      this.businessRole === TasBusinessRole.BACKOFFICE ||\n      this.businessRole === TasBusinessRole.ADMIN_MANAGER ||\n      this.businessRole === TasBusinessRole.MANAGER\n    );\n  }\n\n  /** Whether the button should be disabled */\n  public get isDisabled(): boolean {\n    return this.isLoading || this.isStatusError || !this.isJoinable;\n  }\n\n  /** Reason why the button is disabled (for tooltip) */\n  public get disabledReason(): string {\n    if (this.isLoading) {\n      return '';\n    }\n    if (this.isStatusError) {\n      return this.statusErrorMessage || 'Error al verificar el estado';\n    }\n    if (!this.isJoinable) {\n      return 'Todavía no es el horario de la llamada';\n    }\n    return '';\n  }\n\n  constructor(\n    private modalService: NgbModal,\n    private tasService: TasService,\n    private tasUtilityService: TasUtilityService\n  ) {}\n\n  ngOnInit(): void {\n    // Subscribe to viewMode to handle PiP return\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        // When entering PiP, clear the videoCallModalRef since modal will close\n        if (mode === ViewMode.PIP) {\n          this.videoCallModalRef = null;\n        }\n      })\n    );\n\n    // If skipStatusCheck is true, show button immediately without polling\n    if (this.skipStatusCheck) {\n      this.shouldShowButton = true;\n      this.isJoinable = true;\n      return;\n    }\n\n    // Start status checking\n    this.startStatusPolling();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    this.stopStatusPolling();\n  }\n\n  /**\n   * Start polling status every 30 seconds\n   */\n  private startStatusPolling(): void {\n    // Initial status check\n    this.checkStatus();\n\n    // Set up periodic polling\n    this.statusPollingInterval = setInterval(() => {\n      this.checkStatus();\n    }, this.STATUS_POLL_INTERVAL_MS);\n  }\n\n  /**\n   * Stop status polling\n   */\n  private stopStatusPolling(): void {\n    if (this.statusPollingInterval) {\n      clearInterval(this.statusPollingInterval);\n      this.statusPollingInterval = null;\n    }\n  }\n\n  /**\n   * Check status endpoint to determine if button should be enabled\n   */\n  private checkStatus(): void {\n    // Skip if required inputs are not available\n    if (!this.tenant || !this.entityId) {\n      return;\n    }\n\n    this.isCheckingStatus = true;\n    this.statusErrorMessage = '';\n\n    this.subscriptions.add(\n      this.tasService\n        .getProxyVideoStatus({\n          roomType: this.roomType,\n          entityId: this.entityId,\n          tenant: this.tenant,\n          businessRole: this.businessRole,\n        })\n        .subscribe({\n          next: (response) => {\n            // Validate response structure\n            if (!response || !response.content) {\n              this.handleStatusError('Invalid response');\n              return;\n            }\n\n            this.isCheckingStatus = false;\n            this.isStatusError = false;\n            this.statusErrorMessage = '';\n            this.shouldShowButton = true;\n            // Update joinable state from response\n            this.isJoinable = response.content?.joinable ?? false;\n          },\n          error: (err) => {\n\n            this.handleStatusError(err);\n          },\n        })\n    );\n  }\n\n  private handleStatusError(err: any): void {\n    this.isCheckingStatus = false;\n    const errorMessage = this.tasUtilityService.extractErrorMessage(err, 'Error checking status');\n    this.statusErrorMessage = errorMessage;\n\n    // On any status error, hide the button\n    this.shouldShowButton = false;\n    this.isStatusError = false; // We don't show error UI, just hide the button\n\n    // Stop polling on error\n    this.stopStatusPolling();\n  }\n\n  onClick(): void {\n    if (!this.tenant || !this.currentUser?.name) {\n      return;\n    }\n\n    if (!this.entityId) {\n      return;\n    }\n\n    this.openWaitingRoomModal();\n  }\n\n  private openWaitingRoomModal(): void {\n    this.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {\n      size: 'lg',\n      windowClass: 'tas-waiting-room-modal',\n      backdrop: 'static',\n      keyboard: false,\n      centered: true,\n    });\n\n    // Pass all necessary inputs to the waiting room component\n    this.currentModalRef.componentInstance.roomType = this.roomType;\n    this.currentModalRef.componentInstance.entityId = this.entityId;\n    this.currentModalRef.componentInstance.tenant = this.tenant;\n    this.currentModalRef.componentInstance.businessRole = this.businessRole;\n    this.currentModalRef.componentInstance.currentUser = this.currentUser;\n\n    this.currentModalRef.result.then(\n      () => {\n        this.currentModalRef = null;\n      },\n      () => {\n        this.currentModalRef = null;\n      }\n    );\n  }\n\n  private openVideoCallModal(isReturningFromPip: boolean = false): void {\n    this.videoCallModalRef = this.modalService.open(TasVideocallComponent, {\n      size: 'xl',\n      windowClass: 'tas-video-modal',\n      backdrop: 'static',\n      keyboard: false,\n    });\n\n    this.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;\n    this.videoCallModalRef.componentInstance.token = this.tasService.token;\n    this.videoCallModalRef.componentInstance.businessRole = this.businessRole;\n    this.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;\n\n    this.videoCallModalRef.result.then(\n      () => {\n        this.videoCallModalRef = null;\n      },\n      () => {\n        this.videoCallModalRef = null;\n      }\n    );\n  }\n}\n","<span\n  *ngIf=\"shouldShowButton\"\n  [ngbTooltip]=\"isDisabled && disabledReason ? disabledReason : null\"\n  container=\"body\"\n  placement=\"top\"\n  tooltipClass=\"tas-btn-tooltip\"\n>\n  <button\n    type=\"button\"\n    class=\"btn btn-primary tas-btn\"\n    [class.tas-btn--teal]=\"variant === 'teal'\"\n    (click)=\"onClick()\"\n    [disabled]=\"isDisabled\"\n  >\n    <i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n    <span *ngIf=\"!isLoading\">{{ buttonLabel }}</span>\n    <span *ngIf=\"isLoading\">Processing...</span>\n  </button>\n</span>\n"]}
|
|
206
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-btn.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Input } from '@angular/core';\nimport { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';\nimport { Subscription } from 'rxjs';\nimport {\n  TasCurrentUser,\n  \n  TasRoomType,\n  ViewMode,\n  TasBusinessRole,\n} from '../../interfaces/tas.interfaces';\nimport { TasWaitingRoomComponent } from '../tas-waiting-room/tas-waiting-room.component';\nimport { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';\nimport { TasService } from '../../services/tas.service';\nimport { TasUtilityService } from '../../services/tas-utility.service';\n\n@Component({\n  selector: 'tas-btn',\n  templateUrl: './tas-btn.component.html',\n  styleUrls: ['./tas-btn.component.scss'],\n})\nexport class TasButtonComponent implements OnInit, OnDestroy {\n  // Status endpoint params\n  @Input() roomType: TasRoomType = TasRoomType.TAS;\n  @Input() entityId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n\n  // User info for token request\n  @Input() currentUser!: TasCurrentUser;\n\n  // Style customization\n  @Input() variant: 'default' | 'teal' = 'default';\n  @Input() buttonLabel: string = 'Iniciar TAS';\n\n  public isLoading = false;\n\n  // Status check state\n  public isCheckingStatus = false;\n  public isStatusError = false;\n  public statusErrorMessage = '';\n  public isJoinable = false; // Tracks joinable field from status response\n  public shouldShowButton = false; // Hidden by default, shown after status check confirms visibility\n\n  private subscriptions = new Subscription();\n  private currentModalRef: NgbModalRef | null = null;\n  private videoCallModalRef: NgbModalRef | null = null;\n  private statusPollingInterval: ReturnType<typeof setInterval> | null = null;\n  private readonly STATUS_POLL_INTERVAL_MS = 30000; // 30 seconds\n\n  /** Whether user is backoffice (or admin/manager) */\n  public get isBackoffice(): boolean {\n    return (\n      this.businessRole === TasBusinessRole.BACKOFFICE ||\n      this.businessRole === TasBusinessRole.ADMIN_MANAGER ||\n      this.businessRole === TasBusinessRole.MANAGER\n    );\n  }\n\n  /** Whether the button should be disabled */\n  public get isDisabled(): boolean {\n    return this.isLoading || this.isStatusError || !this.isJoinable;\n  }\n\n  /** Reason why the button is disabled (for tooltip) */\n  public get disabledReason(): string {\n    if (this.isLoading) {\n      return '';\n    }\n    if (this.isStatusError) {\n      return this.statusErrorMessage || 'Error al verificar el estado';\n    }\n    if (!this.isJoinable) {\n      return 'Todavía no es el horario de la llamada';\n    }\n    return '';\n  }\n\n  constructor(\n    private modalService: NgbModal,\n    private tasService: TasService,\n    private tasUtilityService: TasUtilityService\n  ) {}\n\n  ngOnInit(): void {\n    // Subscribe to viewMode to handle PiP return\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        // When entering PiP, clear the videoCallModalRef since modal will close\n        if (mode === ViewMode.PIP) {\n          this.videoCallModalRef = null;\n        }\n      })\n    );\n\n    // Start status checking\n    this.startStatusPolling();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    this.stopStatusPolling();\n  }\n\n  /**\n   * Start polling status every 30 seconds\n   */\n  private startStatusPolling(): void {\n    // Initial status check\n    this.checkStatus();\n\n    // Set up periodic polling\n    this.statusPollingInterval = setInterval(() => {\n      this.checkStatus();\n    }, this.STATUS_POLL_INTERVAL_MS);\n  }\n\n  /**\n   * Stop status polling\n   */\n  private stopStatusPolling(): void {\n    if (this.statusPollingInterval) {\n      clearInterval(this.statusPollingInterval);\n      this.statusPollingInterval = null;\n    }\n  }\n\n  /**\n   * Check status endpoint to determine if button should be enabled\n   */\n  private checkStatus(): void {\n    // Skip if required inputs are not available\n    if (!this.tenant || !this.entityId) {\n      return;\n    }\n\n    this.isCheckingStatus = true;\n    this.statusErrorMessage = '';\n\n    this.subscriptions.add(\n      this.tasService\n        .getProxyVideoStatus({\n          roomType: this.roomType,\n          entityId: this.entityId,\n          tenant: this.tenant,\n          businessRole: this.businessRole,\n        })\n        .subscribe({\n          next: (response) => {\n            // Validate response structure\n            if (!response || !response.content) {\n              this.handleStatusError('Invalid response');\n              return;\n            }\n\n            this.isCheckingStatus = false;\n            this.isStatusError = false;\n            this.statusErrorMessage = '';\n            this.shouldShowButton = true;\n            // Update joinable state from response\n            this.isJoinable = response.content?.joinable ?? false;\n          },\n          error: (err) => {\n\n            this.handleStatusError(err);\n          },\n        })\n    );\n  }\n\n  private handleStatusError(err: any): void {\n    this.isCheckingStatus = false;\n    const errorMessage = this.tasUtilityService.extractErrorMessage(err, 'Error checking status');\n    this.statusErrorMessage = errorMessage;\n\n    // On any status error, hide the button\n    this.shouldShowButton = false;\n    this.isStatusError = false; // We don't show error UI, just hide the button\n\n    // Stop polling on error\n    this.stopStatusPolling();\n  }\n\n  onClick(): void {\n    if (!this.tenant || !this.currentUser?.name) {\n      return;\n    }\n\n    if (!this.entityId) {\n      return;\n    }\n\n    this.openWaitingRoomModal();\n  }\n\n  private openWaitingRoomModal(): void {\n    this.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {\n      size: 'lg',\n      windowClass: 'tas-waiting-room-modal',\n      backdrop: 'static',\n      keyboard: false,\n      centered: true,\n    });\n\n    // Pass all necessary inputs to the waiting room component\n    this.currentModalRef.componentInstance.roomType = this.roomType;\n    this.currentModalRef.componentInstance.entityId = this.entityId;\n    this.currentModalRef.componentInstance.tenant = this.tenant;\n    this.currentModalRef.componentInstance.businessRole = this.businessRole;\n    this.currentModalRef.componentInstance.currentUser = this.currentUser;\n\n    this.currentModalRef.result.then(\n      () => {\n        this.currentModalRef = null;\n      },\n      () => {\n        this.currentModalRef = null;\n      }\n    );\n  }\n\n  private openVideoCallModal(isReturningFromPip: boolean = false): void {\n    this.videoCallModalRef = this.modalService.open(TasVideocallComponent, {\n      size: 'xl',\n      windowClass: 'tas-video-modal',\n      backdrop: 'static',\n      keyboard: false,\n    });\n\n    this.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;\n    this.videoCallModalRef.componentInstance.token = this.tasService.token;\n    this.videoCallModalRef.componentInstance.businessRole = this.businessRole;\n    this.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;\n\n    this.videoCallModalRef.result.then(\n      () => {\n        this.videoCallModalRef = null;\n      },\n      () => {\n        this.videoCallModalRef = null;\n      }\n    );\n  }\n}\n","<span\n  *ngIf=\"shouldShowButton\"\n  [ngbTooltip]=\"isDisabled && disabledReason ? disabledReason : null\"\n  container=\"body\"\n  placement=\"top\"\n  tooltipClass=\"tas-btn-tooltip\"\n>\n  <button\n    type=\"button\"\n    class=\"btn btn-primary tas-btn\"\n    [class.tas-btn--teal]=\"variant === 'teal'\"\n    (click)=\"onClick()\"\n    [disabled]=\"isDisabled\"\n  >\n    <i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n    <span *ngIf=\"!isLoading\">{{ buttonLabel }}</span>\n    <span *ngIf=\"isLoading\">Processing...</span>\n  </button>\n</span>\n"]}
|
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { Subscription } from 'rxjs';
|
|
3
|
+
import { TasBusinessRole, FeedbackMotiveType, } from '../../interfaces/tas.interfaces';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@ng-bootstrap/ng-bootstrap";
|
|
6
|
+
import * as i2 from "../../services/tas.service";
|
|
7
|
+
import * as i3 from "@angular/common";
|
|
8
|
+
import * as i4 from "@angular/forms";
|
|
9
|
+
export class TasFeedbackModalComponent {
|
|
10
|
+
constructor(activeModal, tasService) {
|
|
11
|
+
this.activeModal = activeModal;
|
|
12
|
+
this.tasService = tasService;
|
|
13
|
+
this.businessRole = TasBusinessRole.USER;
|
|
14
|
+
this.motives = [];
|
|
15
|
+
this.filteredMotives = [];
|
|
16
|
+
this.selectedMotive = null;
|
|
17
|
+
this.rating = 0;
|
|
18
|
+
this.hoverRating = 0;
|
|
19
|
+
this.observation = '';
|
|
20
|
+
this.isSubmitting = false;
|
|
21
|
+
this.showToast = false;
|
|
22
|
+
this.isDropdownOpen = false;
|
|
23
|
+
this.subscriptions = new Subscription();
|
|
24
|
+
this.toastTimeout = null;
|
|
25
|
+
}
|
|
26
|
+
ngOnInit() {
|
|
27
|
+
this.loadMotives();
|
|
28
|
+
}
|
|
29
|
+
ngOnDestroy() {
|
|
30
|
+
this.subscriptions.unsubscribe();
|
|
31
|
+
if (this.toastTimeout) {
|
|
32
|
+
clearTimeout(this.toastTimeout);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Check if current user can see all motives (BUSINESS + TECHNICAL)
|
|
37
|
+
* Only BACKOFFICE, ADMIN_MANAGER, MANAGER roles see BUSINESS motives
|
|
38
|
+
*/
|
|
39
|
+
get canSeeBusinessMotives() {
|
|
40
|
+
return (this.businessRole === TasBusinessRole.BACKOFFICE ||
|
|
41
|
+
this.businessRole === TasBusinessRole.ADMIN_MANAGER ||
|
|
42
|
+
this.businessRole === TasBusinessRole.MANAGER);
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Observation is required when "Otro problema tecnico" is selected
|
|
46
|
+
*/
|
|
47
|
+
get isObservationRequired() {
|
|
48
|
+
return this.selectedMotive?.description === 'Otro problema tecnico';
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Can submit if rating > 0 OR motive is selected
|
|
52
|
+
* If observation is required, it must not be empty
|
|
53
|
+
*/
|
|
54
|
+
get canSubmit() {
|
|
55
|
+
const hasRatingOrMotive = this.rating > 0 || this.selectedMotive !== null;
|
|
56
|
+
if (!hasRatingOrMotive)
|
|
57
|
+
return false;
|
|
58
|
+
if (this.isObservationRequired && !this.observation.trim())
|
|
59
|
+
return false;
|
|
60
|
+
return true;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Progress percentage for the divider line
|
|
64
|
+
* 0 = nothing selected, 50 = one selected, 100 = both selected
|
|
65
|
+
*/
|
|
66
|
+
get feedbackProgress() {
|
|
67
|
+
const hasRating = this.rating > 0;
|
|
68
|
+
const hasMotive = this.selectedMotive !== null;
|
|
69
|
+
if (hasRating && hasMotive)
|
|
70
|
+
return 100;
|
|
71
|
+
if (hasRating || hasMotive)
|
|
72
|
+
return 50;
|
|
73
|
+
return 0;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Set star rating
|
|
77
|
+
*/
|
|
78
|
+
setRating(value) {
|
|
79
|
+
// Toggle off if clicking the same rating
|
|
80
|
+
if (this.rating === value) {
|
|
81
|
+
this.rating = 0;
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
this.rating = value;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Set hover preview rating
|
|
89
|
+
*/
|
|
90
|
+
setHoverRating(value) {
|
|
91
|
+
this.hoverRating = value;
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Clear hover preview
|
|
95
|
+
*/
|
|
96
|
+
clearHoverRating() {
|
|
97
|
+
this.hoverRating = 0;
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Get the display rating (hover preview or actual)
|
|
101
|
+
*/
|
|
102
|
+
getDisplayRating() {
|
|
103
|
+
return this.hoverRating > 0 ? this.hoverRating : this.rating;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Toggle dropdown open/close
|
|
107
|
+
*/
|
|
108
|
+
toggleDropdown() {
|
|
109
|
+
this.isDropdownOpen = !this.isDropdownOpen;
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Close dropdown
|
|
113
|
+
*/
|
|
114
|
+
closeDropdown() {
|
|
115
|
+
this.isDropdownOpen = false;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Select a motive from dropdown
|
|
119
|
+
*/
|
|
120
|
+
selectMotive(motive) {
|
|
121
|
+
this.selectedMotive = motive;
|
|
122
|
+
this.closeDropdown();
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Clear selected motive
|
|
126
|
+
*/
|
|
127
|
+
clearMotive() {
|
|
128
|
+
this.selectedMotive = null;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Submit feedback
|
|
132
|
+
*/
|
|
133
|
+
submit() {
|
|
134
|
+
if (!this.canSubmit || this.isSubmitting)
|
|
135
|
+
return;
|
|
136
|
+
this.isSubmitting = true;
|
|
137
|
+
const payload = {
|
|
138
|
+
videoCallId: this.videoCallId,
|
|
139
|
+
motiveId: this.selectedMotive?.id ?? 0,
|
|
140
|
+
observation: this.observation.trim(),
|
|
141
|
+
rating: this.rating || 1,
|
|
142
|
+
tenant: this.tenant,
|
|
143
|
+
};
|
|
144
|
+
if (this.selectedMotive) {
|
|
145
|
+
payload.motiveType = this.selectedMotive.motiveType;
|
|
146
|
+
}
|
|
147
|
+
this.subscriptions.add(this.tasService.saveFeedback(payload).subscribe({
|
|
148
|
+
next: () => {
|
|
149
|
+
this.isSubmitting = false;
|
|
150
|
+
this.showToastNotification();
|
|
151
|
+
},
|
|
152
|
+
error: (err) => {
|
|
153
|
+
console.error('Error saving feedback:', err);
|
|
154
|
+
this.isSubmitting = false;
|
|
155
|
+
// Still close modal on error, just don't show toast
|
|
156
|
+
this.activeModal.close('submitted');
|
|
157
|
+
},
|
|
158
|
+
}));
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Close modal without saving
|
|
162
|
+
*/
|
|
163
|
+
dismiss() {
|
|
164
|
+
this.activeModal.dismiss('dismissed');
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Load motives from API
|
|
168
|
+
*/
|
|
169
|
+
loadMotives() {
|
|
170
|
+
this.subscriptions.add(this.tasService.getMotives().subscribe({
|
|
171
|
+
next: (motives) => {
|
|
172
|
+
this.motives = motives;
|
|
173
|
+
this.filterMotives();
|
|
174
|
+
},
|
|
175
|
+
error: (err) => {
|
|
176
|
+
console.error('Error loading motives:', err);
|
|
177
|
+
// Fallback mock motives for development/testing
|
|
178
|
+
this.motives = [
|
|
179
|
+
{ id: 1, description: 'Problemas de audio', motiveType: FeedbackMotiveType.TECHNICAL },
|
|
180
|
+
{ id: 2, description: 'Problemas de video', motiveType: FeedbackMotiveType.TECHNICAL },
|
|
181
|
+
{ id: 3, description: 'Conexión inestable', motiveType: FeedbackMotiveType.TECHNICAL },
|
|
182
|
+
{ id: 4, description: 'Otro problema tecnico', motiveType: FeedbackMotiveType.TECHNICAL },
|
|
183
|
+
{ id: 5, description: 'Problema de negocio', motiveType: FeedbackMotiveType.BUSINESS },
|
|
184
|
+
];
|
|
185
|
+
this.filterMotives();
|
|
186
|
+
},
|
|
187
|
+
}));
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* Filter motives based on user role
|
|
191
|
+
* USERs see only TECHNICAL motives
|
|
192
|
+
* Owners (BACKOFFICE, ADMIN_MANAGER, MANAGER) see all motives
|
|
193
|
+
*/
|
|
194
|
+
filterMotives() {
|
|
195
|
+
if (this.canSeeBusinessMotives) {
|
|
196
|
+
this.filteredMotives = this.motives;
|
|
197
|
+
}
|
|
198
|
+
else {
|
|
199
|
+
this.filteredMotives = this.motives.filter((m) => !m.motiveType || m.motiveType === FeedbackMotiveType.TECHNICAL);
|
|
200
|
+
}
|
|
201
|
+
// If still empty after filtering, and we have motives, just show all as fallback
|
|
202
|
+
if (this.filteredMotives.length === 0 && this.motives.length > 0) {
|
|
203
|
+
this.filteredMotives = this.motives;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
/**
|
|
207
|
+
* Show toast notification and auto-dismiss after 3s
|
|
208
|
+
*/
|
|
209
|
+
showToastNotification() {
|
|
210
|
+
this.showToast = true;
|
|
211
|
+
this.toastTimeout = setTimeout(() => {
|
|
212
|
+
this.showToast = false;
|
|
213
|
+
this.activeModal.close('submitted');
|
|
214
|
+
}, 3000);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
TasFeedbackModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasFeedbackModalComponent, deps: [{ token: i1.NgbActiveModal }, { token: i2.TasService }], target: i0.ɵɵFactoryTarget.Component });
|
|
218
|
+
TasFeedbackModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasFeedbackModalComponent, selector: "tas-feedback-modal", inputs: { videoCallId: "videoCallId", tenant: "tenant", businessRole: "businessRole" }, ngImport: i0, template: "<div class=\"tas-feedback-modal\">\n <!-- Header with subtitle -->\n <div class=\"feedback-header\">\n <div class=\"header-text\">\n <span class=\"subtitle\">Cierre de la consulta</span>\n <h2>Calidad de la videollamada</h2>\n </div>\n <button\n type=\"button\"\n class=\"close-btn\"\n (click)=\"dismiss()\"\n aria-label=\"Cerrar\"\n >\n <i class=\"fa fa-times\"></i>\n </button>\n </div>\n\n <!-- Progress Divider -->\n <div class=\"progress-divider\">\n <div class=\"progress-fill\" [style.width.%]=\"feedbackProgress\"></div>\n </div>\n <div class=\"accent-line\"></div>\n\n <div class=\"feedback-content\">\n <!-- Star Rating with question -->\n <div class=\"rating-section\">\n <p class=\"rating-question\">\u00BFC\u00F3mo fue la calidad de la llamada?</p>\n <div\n class=\"stars-container\"\n (mouseleave)=\"clearHoverRating()\"\n >\n <button\n *ngFor=\"let star of [1, 2, 3, 4, 5]\"\n type=\"button\"\n class=\"star-btn\"\n [class.filled]=\"star <= getDisplayRating()\"\n (click)=\"setRating(star)\"\n (mouseenter)=\"setHoverRating(star)\"\n [attr.aria-label]=\"'Calificar ' + star + ' estrellas'\"\n >\n <i class=\"fa\" [class.fa-star]=\"star <= getDisplayRating()\" [class.fa-star-o]=\"star > getDisplayRating()\"></i>\n </button>\n </div>\n </div>\n\n <!-- Motive Dropdown -->\n <div class=\"motive-section\">\n <label class=\"motive-label\">Motivo (opcional)</label>\n <div class=\"custom-dropdown\" [class.open]=\"isDropdownOpen\">\n <button\n type=\"button\"\n class=\"dropdown-trigger\"\n (click)=\"toggleDropdown()\"\n [attr.aria-expanded]=\"isDropdownOpen\"\n aria-haspopup=\"listbox\"\n >\n <span class=\"dropdown-text\">\n {{ selectedMotive?.description || 'Seleccionar motivo' }}\n </span>\n <i class=\"fa fa-chevron-down dropdown-arrow\"></i>\n </button>\n <div\n class=\"dropdown-menu\"\n *ngIf=\"isDropdownOpen\"\n role=\"listbox\"\n >\n <button\n *ngFor=\"let motive of filteredMotives\"\n type=\"button\"\n class=\"dropdown-item\"\n [class.selected]=\"selectedMotive?.id === motive.id\"\n (click)=\"selectMotive(motive)\"\n role=\"option\"\n [attr.aria-selected]=\"selectedMotive?.id === motive.id\"\n >\n {{ motive.description }}\n </button>\n </div>\n </div>\n </div>\n\n <!-- Observation Textarea -->\n <div class=\"observation-section\">\n <label class=\"observation-label\" [class.required]=\"isObservationRequired\">\n {{ isObservationRequired ? 'Observacion (requerida)' : 'Observacion (opcional)' }}\n </label>\n <textarea\n class=\"observation-textarea\"\n [(ngModel)]=\"observation\"\n [placeholder]=\"isObservationRequired ? 'Por favor, describe el problema...' : 'Escribe tu comentario...'\"\n rows=\"3\"\n [attr.aria-required]=\"isObservationRequired\"\n ></textarea>\n </div>\n </div>\n\n <div class=\"feedback-footer\">\n <button\n type=\"button\"\n class=\"submit-btn\"\n [disabled]=\"!canSubmit || isSubmitting\"\n (click)=\"submit()\"\n >\n <span *ngIf=\"!isSubmitting\">Enviar</span>\n <span *ngIf=\"isSubmitting\">\n <i class=\"fa fa-spinner fa-spin\"></i>\n Enviando...\n </span>\n </button>\n </div>\n\n <!-- Toast Notification -->\n <div class=\"toast-notification\" *ngIf=\"showToast\" role=\"alert\" aria-live=\"polite\">\n <i class=\"fa fa-check-circle\"></i>\n <span>Gracias por tu feedback</span>\n </div>\n</div>\n\n<!-- Backdrop to close dropdown when clicking outside -->\n<div\n class=\"dropdown-backdrop\"\n *ngIf=\"isDropdownOpen\"\n (click)=\"closeDropdown()\"\n></div>\n", styles: [":host{display:block;width:100%}::ng-deep .tas-feedback-modal-wrapper .modal-content{overflow:visible!important;border:none;background:transparent}::ng-deep .tas-feedback-modal-wrapper .modal-dialog{max-width:450px}.tas-feedback-modal{width:100%;background:#fff;border-radius:16px;overflow:visible;position:relative;padding:.5rem;box-shadow:0 20px 25px -5px #0000001a,0 8px 10px -6px #0000001a}.feedback-header{display:flex;justify-content:space-between;align-items:flex-start;padding:1.25rem 1.5rem 1rem;background:#fff;border-radius:16px 16px 0 0}.feedback-header .header-text{display:flex;flex-direction:column;gap:.25rem}.feedback-header .header-text .subtitle{font-size:.8125rem;color:#6b7280;font-weight:400}.feedback-header .header-text h2{margin:0;font-size:1.25rem;font-weight:600;color:#1f2937;letter-spacing:-.01em}.feedback-header .close-btn{background:transparent;border:none;padding:8px;cursor:pointer;color:#9ca3af;font-size:1.25rem;line-height:1;transition:all .2s ease;margin-top:-4px}.feedback-header .close-btn:hover{color:#4b5563}.feedback-header .close-btn:focus{outline:none;color:#4b5563}.progress-divider{height:4px;background:#e5e7eb;margin:0;position:relative;overflow:hidden}.progress-divider .progress-fill{height:100%;background:linear-gradient(90deg,var(--Primary-Uell, #1da4b1) 0%,#4fd1c5 100%);transition:width .4s cubic-bezier(.4,0,.2,1);border-radius:0 2px 2px 0}.feedback-content{padding:1.5rem;display:flex;flex-direction:column;gap:1.5rem}.rating-section{display:flex;flex-direction:column;align-items:flex-start;gap:1rem}.rating-section .rating-question{margin:0;font-size:.9375rem;color:#4b5563;font-weight:400;align-self:center}.rating-section .stars-container{display:flex;gap:8px;width:100%;justify-content:center}.rating-section .star-btn{background:transparent;border:none;padding:4px;cursor:pointer;font-size:2rem;color:#d1d5db;transition:all .15s ease;line-height:1}.rating-section .star-btn:hover{transform:scale(1.15)}.rating-section .star-btn.filled{color:#f5a623}.rating-section .star-btn.filled i.fa-star{text-shadow:0 2px 4px rgba(245,166,35,.3)}.rating-section .star-btn i.fa-star-o{color:#d1d5db}.rating-section .star-btn:focus{outline:none}.motive-section{display:flex;flex-direction:column;gap:.5rem;position:relative;overflow:visible}.motive-section .motive-label{font-size:.875rem;font-weight:500;color:#4b5563}.motive-section .custom-dropdown{position:relative;overflow:visible}.motive-section .custom-dropdown .dropdown-trigger{width:100%;display:flex;justify-content:space-between;align-items:center;padding:1rem;background:#fff;border:1px solid #d1d5db;border-radius:8px;cursor:pointer;font-size:.9375rem;color:#4b5563;transition:all .2s ease;min-height:52px}.motive-section .custom-dropdown .dropdown-trigger:hover{border-color:#9ca3af}.motive-section .custom-dropdown .dropdown-trigger:focus{outline:none;border-color:var(--Primary-Uell, #1da4b1)}.motive-section .custom-dropdown .dropdown-trigger .dropdown-text{flex:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.motive-section .custom-dropdown .dropdown-trigger .dropdown-arrow{font-size:.875rem;color:#6b7280;transition:transform .2s ease;margin-left:.5rem}.motive-section .custom-dropdown.open .dropdown-trigger{border-color:var(--Primary-Uell, #1da4b1)}.motive-section .custom-dropdown.open .dropdown-trigger .dropdown-arrow{transform:rotate(180deg)}.motive-section .custom-dropdown .dropdown-menu{position:absolute;top:calc(100% + 4px);left:0;right:0;display:block;min-height:40px;min-width:100%;background:#fff;border:1px solid #e5e7eb;border-radius:8px;box-shadow:0 4px 16px #0000001f;max-height:250px;overflow:hidden;overflow-y:auto;z-index:1050;visibility:visible;opacity:1;padding:0}.motive-section .custom-dropdown .dropdown-menu .dropdown-item{width:100%;padding:1rem 1.25rem;background:transparent;border:none;border-bottom:1px dashed #e5e7eb;text-align:left;font-size:.9375rem;color:#4b5563;cursor:pointer;transition:all .15s ease;position:relative}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:last-child{border-bottom:none}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:hover{background:rgba(29,164,177,.05);color:#374151}.motive-section .custom-dropdown .dropdown-menu .dropdown-item.selected{background:rgba(29,164,177,.08);color:var(--Primary-Uell, #1da4b1);border-left:3px solid var(--Primary-Uell, #1da4b1);padding-left:calc(1.25rem - 3px)}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:focus{outline:none;background:rgba(29,164,177,.05)}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:first-child{border-radius:8px 8px 0 0}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:last-child{border-radius:0 0 8px 8px}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:only-child{border-radius:8px}.motive-section .clear-motive-btn{position:absolute;right:40px;top:32px;background:#f3f4f6;border:none;padding:4px 6px;border-radius:4px;cursor:pointer;color:#6b7280;font-size:.625rem;transition:all .2s ease}.motive-section .clear-motive-btn:hover{background:#e5e7eb;color:#374151}.motive-section .clear-motive-btn:focus{outline:none}.observation-section{display:flex;flex-direction:column;gap:.5rem}.observation-section .observation-label{font-size:.875rem;font-weight:500;color:#4b5563}.observation-section .observation-label.required{color:#dc2626}.observation-section .observation-label.required:after{content:\" *\";font-weight:600}.observation-section .observation-textarea{width:100%;padding:.75rem 1rem;border:1px solid #d1d5db;border-radius:12px;font-size:.875rem;font-family:inherit;resize:vertical;min-height:80px;transition:all .2s ease;background:#fff}.observation-section .observation-textarea::placeholder{color:#9ca3af}.observation-section .observation-textarea:hover{border-color:#9ca3af}.observation-section .observation-textarea:focus{outline:none;border-color:var(--Primary-Uell, #1da4b1);box-shadow:0 0 0 3px #1da4b126}.feedback-footer{padding:1rem 1.5rem 1.5rem;display:flex;justify-content:flex-end}.feedback-footer .submit-btn{padding:.75rem 2rem;background:var(--Primary-Uell, #1da4b1);color:#fff;border:none;border-radius:24px;font-size:.9375rem;font-weight:600;cursor:pointer;transition:all .2s ease;min-width:120px}.feedback-footer .submit-btn:hover:not(:disabled){background:#178e99}.feedback-footer .submit-btn:disabled{background:rgba(29,164,177,.5);cursor:not-allowed}.feedback-footer .submit-btn:focus{outline:none;box-shadow:0 0 0 3px #1da4b126}.feedback-footer .submit-btn i.fa-spinner{margin-right:6px}.toast-notification{position:fixed;bottom:24px;left:24px;display:flex;align-items:center;gap:10px;padding:14px 20px;background:#383e52;color:#fff;border-radius:8px;box-shadow:0 4px 12px #00000040;font-size:.875rem;font-weight:500;z-index:1060;animation:slideIn .3s ease}.toast-notification i.fa-check-circle{color:#4ade80;font-size:1.125rem}@keyframes slideIn{0%{transform:translate(-100%);opacity:0}to{transform:translate(0);opacity:1}}.dropdown-backdrop{position:fixed;inset:0;z-index:99}\n"], directives: [{ type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
|
|
219
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasFeedbackModalComponent, decorators: [{
|
|
220
|
+
type: Component,
|
|
221
|
+
args: [{ selector: 'tas-feedback-modal', template: "<div class=\"tas-feedback-modal\">\n <!-- Header with subtitle -->\n <div class=\"feedback-header\">\n <div class=\"header-text\">\n <span class=\"subtitle\">Cierre de la consulta</span>\n <h2>Calidad de la videollamada</h2>\n </div>\n <button\n type=\"button\"\n class=\"close-btn\"\n (click)=\"dismiss()\"\n aria-label=\"Cerrar\"\n >\n <i class=\"fa fa-times\"></i>\n </button>\n </div>\n\n <!-- Progress Divider -->\n <div class=\"progress-divider\">\n <div class=\"progress-fill\" [style.width.%]=\"feedbackProgress\"></div>\n </div>\n <div class=\"accent-line\"></div>\n\n <div class=\"feedback-content\">\n <!-- Star Rating with question -->\n <div class=\"rating-section\">\n <p class=\"rating-question\">\u00BFC\u00F3mo fue la calidad de la llamada?</p>\n <div\n class=\"stars-container\"\n (mouseleave)=\"clearHoverRating()\"\n >\n <button\n *ngFor=\"let star of [1, 2, 3, 4, 5]\"\n type=\"button\"\n class=\"star-btn\"\n [class.filled]=\"star <= getDisplayRating()\"\n (click)=\"setRating(star)\"\n (mouseenter)=\"setHoverRating(star)\"\n [attr.aria-label]=\"'Calificar ' + star + ' estrellas'\"\n >\n <i class=\"fa\" [class.fa-star]=\"star <= getDisplayRating()\" [class.fa-star-o]=\"star > getDisplayRating()\"></i>\n </button>\n </div>\n </div>\n\n <!-- Motive Dropdown -->\n <div class=\"motive-section\">\n <label class=\"motive-label\">Motivo (opcional)</label>\n <div class=\"custom-dropdown\" [class.open]=\"isDropdownOpen\">\n <button\n type=\"button\"\n class=\"dropdown-trigger\"\n (click)=\"toggleDropdown()\"\n [attr.aria-expanded]=\"isDropdownOpen\"\n aria-haspopup=\"listbox\"\n >\n <span class=\"dropdown-text\">\n {{ selectedMotive?.description || 'Seleccionar motivo' }}\n </span>\n <i class=\"fa fa-chevron-down dropdown-arrow\"></i>\n </button>\n <div\n class=\"dropdown-menu\"\n *ngIf=\"isDropdownOpen\"\n role=\"listbox\"\n >\n <button\n *ngFor=\"let motive of filteredMotives\"\n type=\"button\"\n class=\"dropdown-item\"\n [class.selected]=\"selectedMotive?.id === motive.id\"\n (click)=\"selectMotive(motive)\"\n role=\"option\"\n [attr.aria-selected]=\"selectedMotive?.id === motive.id\"\n >\n {{ motive.description }}\n </button>\n </div>\n </div>\n </div>\n\n <!-- Observation Textarea -->\n <div class=\"observation-section\">\n <label class=\"observation-label\" [class.required]=\"isObservationRequired\">\n {{ isObservationRequired ? 'Observacion (requerida)' : 'Observacion (opcional)' }}\n </label>\n <textarea\n class=\"observation-textarea\"\n [(ngModel)]=\"observation\"\n [placeholder]=\"isObservationRequired ? 'Por favor, describe el problema...' : 'Escribe tu comentario...'\"\n rows=\"3\"\n [attr.aria-required]=\"isObservationRequired\"\n ></textarea>\n </div>\n </div>\n\n <div class=\"feedback-footer\">\n <button\n type=\"button\"\n class=\"submit-btn\"\n [disabled]=\"!canSubmit || isSubmitting\"\n (click)=\"submit()\"\n >\n <span *ngIf=\"!isSubmitting\">Enviar</span>\n <span *ngIf=\"isSubmitting\">\n <i class=\"fa fa-spinner fa-spin\"></i>\n Enviando...\n </span>\n </button>\n </div>\n\n <!-- Toast Notification -->\n <div class=\"toast-notification\" *ngIf=\"showToast\" role=\"alert\" aria-live=\"polite\">\n <i class=\"fa fa-check-circle\"></i>\n <span>Gracias por tu feedback</span>\n </div>\n</div>\n\n<!-- Backdrop to close dropdown when clicking outside -->\n<div\n class=\"dropdown-backdrop\"\n *ngIf=\"isDropdownOpen\"\n (click)=\"closeDropdown()\"\n></div>\n", styles: [":host{display:block;width:100%}::ng-deep .tas-feedback-modal-wrapper .modal-content{overflow:visible!important;border:none;background:transparent}::ng-deep .tas-feedback-modal-wrapper .modal-dialog{max-width:450px}.tas-feedback-modal{width:100%;background:#fff;border-radius:16px;overflow:visible;position:relative;padding:.5rem;box-shadow:0 20px 25px -5px #0000001a,0 8px 10px -6px #0000001a}.feedback-header{display:flex;justify-content:space-between;align-items:flex-start;padding:1.25rem 1.5rem 1rem;background:#fff;border-radius:16px 16px 0 0}.feedback-header .header-text{display:flex;flex-direction:column;gap:.25rem}.feedback-header .header-text .subtitle{font-size:.8125rem;color:#6b7280;font-weight:400}.feedback-header .header-text h2{margin:0;font-size:1.25rem;font-weight:600;color:#1f2937;letter-spacing:-.01em}.feedback-header .close-btn{background:transparent;border:none;padding:8px;cursor:pointer;color:#9ca3af;font-size:1.25rem;line-height:1;transition:all .2s ease;margin-top:-4px}.feedback-header .close-btn:hover{color:#4b5563}.feedback-header .close-btn:focus{outline:none;color:#4b5563}.progress-divider{height:4px;background:#e5e7eb;margin:0;position:relative;overflow:hidden}.progress-divider .progress-fill{height:100%;background:linear-gradient(90deg,var(--Primary-Uell, #1da4b1) 0%,#4fd1c5 100%);transition:width .4s cubic-bezier(.4,0,.2,1);border-radius:0 2px 2px 0}.feedback-content{padding:1.5rem;display:flex;flex-direction:column;gap:1.5rem}.rating-section{display:flex;flex-direction:column;align-items:flex-start;gap:1rem}.rating-section .rating-question{margin:0;font-size:.9375rem;color:#4b5563;font-weight:400;align-self:center}.rating-section .stars-container{display:flex;gap:8px;width:100%;justify-content:center}.rating-section .star-btn{background:transparent;border:none;padding:4px;cursor:pointer;font-size:2rem;color:#d1d5db;transition:all .15s ease;line-height:1}.rating-section .star-btn:hover{transform:scale(1.15)}.rating-section .star-btn.filled{color:#f5a623}.rating-section .star-btn.filled i.fa-star{text-shadow:0 2px 4px rgba(245,166,35,.3)}.rating-section .star-btn i.fa-star-o{color:#d1d5db}.rating-section .star-btn:focus{outline:none}.motive-section{display:flex;flex-direction:column;gap:.5rem;position:relative;overflow:visible}.motive-section .motive-label{font-size:.875rem;font-weight:500;color:#4b5563}.motive-section .custom-dropdown{position:relative;overflow:visible}.motive-section .custom-dropdown .dropdown-trigger{width:100%;display:flex;justify-content:space-between;align-items:center;padding:1rem;background:#fff;border:1px solid #d1d5db;border-radius:8px;cursor:pointer;font-size:.9375rem;color:#4b5563;transition:all .2s ease;min-height:52px}.motive-section .custom-dropdown .dropdown-trigger:hover{border-color:#9ca3af}.motive-section .custom-dropdown .dropdown-trigger:focus{outline:none;border-color:var(--Primary-Uell, #1da4b1)}.motive-section .custom-dropdown .dropdown-trigger .dropdown-text{flex:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.motive-section .custom-dropdown .dropdown-trigger .dropdown-arrow{font-size:.875rem;color:#6b7280;transition:transform .2s ease;margin-left:.5rem}.motive-section .custom-dropdown.open .dropdown-trigger{border-color:var(--Primary-Uell, #1da4b1)}.motive-section .custom-dropdown.open .dropdown-trigger .dropdown-arrow{transform:rotate(180deg)}.motive-section .custom-dropdown .dropdown-menu{position:absolute;top:calc(100% + 4px);left:0;right:0;display:block;min-height:40px;min-width:100%;background:#fff;border:1px solid #e5e7eb;border-radius:8px;box-shadow:0 4px 16px #0000001f;max-height:250px;overflow:hidden;overflow-y:auto;z-index:1050;visibility:visible;opacity:1;padding:0}.motive-section .custom-dropdown .dropdown-menu .dropdown-item{width:100%;padding:1rem 1.25rem;background:transparent;border:none;border-bottom:1px dashed #e5e7eb;text-align:left;font-size:.9375rem;color:#4b5563;cursor:pointer;transition:all .15s ease;position:relative}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:last-child{border-bottom:none}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:hover{background:rgba(29,164,177,.05);color:#374151}.motive-section .custom-dropdown .dropdown-menu .dropdown-item.selected{background:rgba(29,164,177,.08);color:var(--Primary-Uell, #1da4b1);border-left:3px solid var(--Primary-Uell, #1da4b1);padding-left:calc(1.25rem - 3px)}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:focus{outline:none;background:rgba(29,164,177,.05)}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:first-child{border-radius:8px 8px 0 0}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:last-child{border-radius:0 0 8px 8px}.motive-section .custom-dropdown .dropdown-menu .dropdown-item:only-child{border-radius:8px}.motive-section .clear-motive-btn{position:absolute;right:40px;top:32px;background:#f3f4f6;border:none;padding:4px 6px;border-radius:4px;cursor:pointer;color:#6b7280;font-size:.625rem;transition:all .2s ease}.motive-section .clear-motive-btn:hover{background:#e5e7eb;color:#374151}.motive-section .clear-motive-btn:focus{outline:none}.observation-section{display:flex;flex-direction:column;gap:.5rem}.observation-section .observation-label{font-size:.875rem;font-weight:500;color:#4b5563}.observation-section .observation-label.required{color:#dc2626}.observation-section .observation-label.required:after{content:\" *\";font-weight:600}.observation-section .observation-textarea{width:100%;padding:.75rem 1rem;border:1px solid #d1d5db;border-radius:12px;font-size:.875rem;font-family:inherit;resize:vertical;min-height:80px;transition:all .2s ease;background:#fff}.observation-section .observation-textarea::placeholder{color:#9ca3af}.observation-section .observation-textarea:hover{border-color:#9ca3af}.observation-section .observation-textarea:focus{outline:none;border-color:var(--Primary-Uell, #1da4b1);box-shadow:0 0 0 3px #1da4b126}.feedback-footer{padding:1rem 1.5rem 1.5rem;display:flex;justify-content:flex-end}.feedback-footer .submit-btn{padding:.75rem 2rem;background:var(--Primary-Uell, #1da4b1);color:#fff;border:none;border-radius:24px;font-size:.9375rem;font-weight:600;cursor:pointer;transition:all .2s ease;min-width:120px}.feedback-footer .submit-btn:hover:not(:disabled){background:#178e99}.feedback-footer .submit-btn:disabled{background:rgba(29,164,177,.5);cursor:not-allowed}.feedback-footer .submit-btn:focus{outline:none;box-shadow:0 0 0 3px #1da4b126}.feedback-footer .submit-btn i.fa-spinner{margin-right:6px}.toast-notification{position:fixed;bottom:24px;left:24px;display:flex;align-items:center;gap:10px;padding:14px 20px;background:#383e52;color:#fff;border-radius:8px;box-shadow:0 4px 12px #00000040;font-size:.875rem;font-weight:500;z-index:1060;animation:slideIn .3s ease}.toast-notification i.fa-check-circle{color:#4ade80;font-size:1.125rem}@keyframes slideIn{0%{transform:translate(-100%);opacity:0}to{transform:translate(0);opacity:1}}.dropdown-backdrop{position:fixed;inset:0;z-index:99}\n"] }]
|
|
222
|
+
}], ctorParameters: function () { return [{ type: i1.NgbActiveModal }, { type: i2.TasService }]; }, propDecorators: { videoCallId: [{
|
|
223
|
+
type: Input
|
|
224
|
+
}], tenant: [{
|
|
225
|
+
type: Input
|
|
226
|
+
}], businessRole: [{
|
|
227
|
+
type: Input
|
|
228
|
+
}] } });
|
|
229
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-feedback-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-feedback-modal/tas-feedback-modal.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-feedback-modal/tas-feedback-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, OnDestroy } from '@angular/core';\nimport { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap';\nimport { Subscription } from 'rxjs';\nimport { TasService } from '../../services/tas.service';\nimport {\n  TasBusinessRole,\n  FeedbackMotive,\n  FeedbackMotiveType,\n  SaveFeedbackRequest,\n} from '../../interfaces/tas.interfaces';\n\n@Component({\n  selector: 'tas-feedback-modal',\n  templateUrl: './tas-feedback-modal.component.html',\n  styleUrls: ['./tas-feedback-modal.component.scss'],\n})\nexport class TasFeedbackModalComponent implements OnInit, OnDestroy {\n  @Input() videoCallId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n\n  motives: FeedbackMotive[] = [];\n  filteredMotives: FeedbackMotive[] = [];\n  selectedMotive: FeedbackMotive | null = null;\n  rating: number = 0;\n  hoverRating: number = 0;\n  observation: string = '';\n  isSubmitting = false;\n  showToast = false;\n  isDropdownOpen = false;\n\n  private subscriptions = new Subscription();\n  private toastTimeout: ReturnType<typeof setTimeout> | null = null;\n\n  constructor(\n    public activeModal: NgbActiveModal,\n    private tasService: TasService\n  ) {}\n\n  ngOnInit(): void {\n    this.loadMotives();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    if (this.toastTimeout) {\n      clearTimeout(this.toastTimeout);\n    }\n  }\n\n  /**\n   * Check if current user can see all motives (BUSINESS + TECHNICAL)\n   * Only BACKOFFICE, ADMIN_MANAGER, MANAGER roles see BUSINESS motives\n   */\n  get canSeeBusinessMotives(): boolean {\n    return (\n      this.businessRole === TasBusinessRole.BACKOFFICE ||\n      this.businessRole === TasBusinessRole.ADMIN_MANAGER ||\n      this.businessRole === TasBusinessRole.MANAGER\n    );\n  }\n\n  /**\n   * Observation is required when \"Otro problema tecnico\" is selected\n   */\n  get isObservationRequired(): boolean {\n    return this.selectedMotive?.description === 'Otro problema tecnico';\n  }\n\n  /**\n   * Can submit if rating > 0 OR motive is selected\n   * If observation is required, it must not be empty\n   */\n  get canSubmit(): boolean {\n    const hasRatingOrMotive = this.rating > 0 || this.selectedMotive !== null;\n    if (!hasRatingOrMotive) return false;\n    if (this.isObservationRequired && !this.observation.trim()) return false;\n    return true;\n  }\n\n  /**\n   * Progress percentage for the divider line\n   * 0 = nothing selected, 50 = one selected, 100 = both selected\n   */\n  get feedbackProgress(): number {\n    const hasRating = this.rating > 0;\n    const hasMotive = this.selectedMotive !== null;\n    if (hasRating && hasMotive) return 100;\n    if (hasRating || hasMotive) return 50;\n    return 0;\n  }\n\n  /**\n   * Set star rating\n   */\n  setRating(value: number): void {\n    // Toggle off if clicking the same rating\n    if (this.rating === value) {\n      this.rating = 0;\n    } else {\n      this.rating = value;\n    }\n  }\n\n  /**\n   * Set hover preview rating\n   */\n  setHoverRating(value: number): void {\n    this.hoverRating = value;\n  }\n\n  /**\n   * Clear hover preview\n   */\n  clearHoverRating(): void {\n    this.hoverRating = 0;\n  }\n\n  /**\n   * Get the display rating (hover preview or actual)\n   */\n  getDisplayRating(): number {\n    return this.hoverRating > 0 ? this.hoverRating : this.rating;\n  }\n\n  /**\n   * Toggle dropdown open/close\n   */\n  toggleDropdown(): void {\n    this.isDropdownOpen = !this.isDropdownOpen;\n  }\n\n  /**\n   * Close dropdown\n   */\n  closeDropdown(): void {\n    this.isDropdownOpen = false;\n  }\n\n  /**\n   * Select a motive from dropdown\n   */\n  selectMotive(motive: FeedbackMotive): void {\n    this.selectedMotive = motive;\n    this.closeDropdown();\n  }\n\n  /**\n   * Clear selected motive\n   */\n  clearMotive(): void {\n    this.selectedMotive = null;\n  }\n\n  /**\n   * Submit feedback\n   */\n  submit(): void {\n    if (!this.canSubmit || this.isSubmitting) return;\n\n    this.isSubmitting = true;\n\n    const payload: SaveFeedbackRequest = {\n      videoCallId: this.videoCallId,\n      motiveId: this.selectedMotive?.id ?? 0,\n      observation: this.observation.trim(),\n      rating: this.rating || 1, // API requires 1-5, default to 1 if no rating\n      tenant: this.tenant,\n    };\n\n    if (this.selectedMotive) {\n      payload.motiveType = this.selectedMotive.motiveType;\n    }\n\n    this.subscriptions.add(\n      this.tasService.saveFeedback(payload).subscribe({\n        next: () => {\n          this.isSubmitting = false;\n          this.showToastNotification();\n        },\n        error: (err) => {\n          console.error('Error saving feedback:', err);\n          this.isSubmitting = false;\n          // Still close modal on error, just don't show toast\n          this.activeModal.close('submitted');\n        },\n      })\n    );\n  }\n\n  /**\n   * Close modal without saving\n   */\n  dismiss(): void {\n    this.activeModal.dismiss('dismissed');\n  }\n\n  /**\n   * Load motives from API\n   */\n  private loadMotives(): void {\n    this.subscriptions.add(\n      this.tasService.getMotives().subscribe({\n        next: (motives) => {\n          this.motives = motives;\n          this.filterMotives();\n        },\n        error: (err) => {\n          console.error('Error loading motives:', err);\n          // Fallback mock motives for development/testing\n          this.motives = [\n            { id: 1, description: 'Problemas de audio', motiveType: FeedbackMotiveType.TECHNICAL },\n            { id: 2, description: 'Problemas de video', motiveType: FeedbackMotiveType.TECHNICAL },\n            { id: 3, description: 'Conexión inestable', motiveType: FeedbackMotiveType.TECHNICAL },\n            { id: 4, description: 'Otro problema tecnico', motiveType: FeedbackMotiveType.TECHNICAL },\n            { id: 5, description: 'Problema de negocio', motiveType: FeedbackMotiveType.BUSINESS },\n          ];\n          this.filterMotives();\n        },\n      })\n    );\n  }\n\n  /**\n   * Filter motives based on user role\n   * USERs see only TECHNICAL motives\n   * Owners (BACKOFFICE, ADMIN_MANAGER, MANAGER) see all motives\n   */\n  private filterMotives(): void {\n    if (this.canSeeBusinessMotives) {\n      this.filteredMotives = this.motives;\n    } else {\n      this.filteredMotives = this.motives.filter(\n        (m) => !m.motiveType || m.motiveType === FeedbackMotiveType.TECHNICAL\n      );\n    }\n\n    // If still empty after filtering, and we have motives, just show all as fallback\n    if (this.filteredMotives.length === 0 && this.motives.length > 0) {\n      this.filteredMotives = this.motives;\n    }\n  }\n\n  /**\n   * Show toast notification and auto-dismiss after 3s\n   */\n  private showToastNotification(): void {\n    this.showToast = true;\n    this.toastTimeout = setTimeout(() => {\n      this.showToast = false;\n      this.activeModal.close('submitted');\n    }, 3000);\n  }\n}\n","<div class=\"tas-feedback-modal\">\n  <!-- Header with subtitle -->\n  <div class=\"feedback-header\">\n    <div class=\"header-text\">\n      <span class=\"subtitle\">Cierre de la consulta</span>\n      <h2>Calidad de la videollamada</h2>\n    </div>\n    <button\n      type=\"button\"\n      class=\"close-btn\"\n      (click)=\"dismiss()\"\n      aria-label=\"Cerrar\"\n    >\n      <i class=\"fa fa-times\"></i>\n    </button>\n  </div>\n\n  <!-- Progress Divider -->\n  <div class=\"progress-divider\">\n    <div class=\"progress-fill\" [style.width.%]=\"feedbackProgress\"></div>\n  </div>\n  <div class=\"accent-line\"></div>\n\n  <div class=\"feedback-content\">\n    <!-- Star Rating with question -->\n    <div class=\"rating-section\">\n      <p class=\"rating-question\">¿Cómo fue la calidad de la llamada?</p>\n      <div\n        class=\"stars-container\"\n        (mouseleave)=\"clearHoverRating()\"\n      >\n        <button\n          *ngFor=\"let star of [1, 2, 3, 4, 5]\"\n          type=\"button\"\n          class=\"star-btn\"\n          [class.filled]=\"star <= getDisplayRating()\"\n          (click)=\"setRating(star)\"\n          (mouseenter)=\"setHoverRating(star)\"\n          [attr.aria-label]=\"'Calificar ' + star + ' estrellas'\"\n        >\n          <i class=\"fa\" [class.fa-star]=\"star <= getDisplayRating()\" [class.fa-star-o]=\"star > getDisplayRating()\"></i>\n        </button>\n      </div>\n    </div>\n\n    <!-- Motive Dropdown -->\n    <div class=\"motive-section\">\n      <label class=\"motive-label\">Motivo (opcional)</label>\n      <div class=\"custom-dropdown\" [class.open]=\"isDropdownOpen\">\n        <button\n          type=\"button\"\n          class=\"dropdown-trigger\"\n          (click)=\"toggleDropdown()\"\n          [attr.aria-expanded]=\"isDropdownOpen\"\n          aria-haspopup=\"listbox\"\n        >\n          <span class=\"dropdown-text\">\n            {{ selectedMotive?.description || 'Seleccionar motivo' }}\n          </span>\n          <i class=\"fa fa-chevron-down dropdown-arrow\"></i>\n        </button>\n        <div\n          class=\"dropdown-menu\"\n          *ngIf=\"isDropdownOpen\"\n          role=\"listbox\"\n        >\n          <button\n            *ngFor=\"let motive of filteredMotives\"\n            type=\"button\"\n            class=\"dropdown-item\"\n            [class.selected]=\"selectedMotive?.id === motive.id\"\n            (click)=\"selectMotive(motive)\"\n            role=\"option\"\n            [attr.aria-selected]=\"selectedMotive?.id === motive.id\"\n          >\n            {{ motive.description }}\n          </button>\n        </div>\n      </div>\n    </div>\n\n    <!-- Observation Textarea -->\n    <div class=\"observation-section\">\n      <label class=\"observation-label\" [class.required]=\"isObservationRequired\">\n        {{ isObservationRequired ? 'Observacion (requerida)' : 'Observacion (opcional)' }}\n      </label>\n      <textarea\n        class=\"observation-textarea\"\n        [(ngModel)]=\"observation\"\n        [placeholder]=\"isObservationRequired ? 'Por favor, describe el problema...' : 'Escribe tu comentario...'\"\n        rows=\"3\"\n        [attr.aria-required]=\"isObservationRequired\"\n      ></textarea>\n    </div>\n  </div>\n\n  <div class=\"feedback-footer\">\n    <button\n      type=\"button\"\n      class=\"submit-btn\"\n      [disabled]=\"!canSubmit || isSubmitting\"\n      (click)=\"submit()\"\n    >\n      <span *ngIf=\"!isSubmitting\">Enviar</span>\n      <span *ngIf=\"isSubmitting\">\n        <i class=\"fa fa-spinner fa-spin\"></i>\n        Enviando...\n      </span>\n    </button>\n  </div>\n\n  <!-- Toast Notification -->\n  <div class=\"toast-notification\" *ngIf=\"showToast\" role=\"alert\" aria-live=\"polite\">\n    <i class=\"fa fa-check-circle\"></i>\n    <span>Gracias por tu feedback</span>\n  </div>\n</div>\n\n<!-- Backdrop to close dropdown when clicking outside -->\n<div\n  class=\"dropdown-backdrop\"\n  *ngIf=\"isDropdownOpen\"\n  (click)=\"closeDropdown()\"\n></div>\n"]}
|