tas-uell-sdk 0.1.0 → 0.1.1

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 CHANGED
@@ -222,6 +222,8 @@ interface TasAppointment {
222
222
  status: AppointmentStatus;
223
223
  title: string;
224
224
  notes: string;
225
+ entityId: number; // Entity/ausencia ID
226
+ roomType: TasRoomType; // Room type (always TAS)
225
227
  }
226
228
  ```
227
229
 
@@ -231,7 +233,7 @@ interface TasAppointment {
231
233
  - **TasRoomType**: `TAS`, `JM`, `WEBINAR`, `WELLNESS_MANAGER`
232
234
  - **CallState**: `IDLE`, `CONNECTING`, `CONNECTED`, `DISCONNECTED`, `ERROR`
233
235
  - **ViewMode**: `FULLSCREEN`, `PIP`
234
- - **AppointmentStatus**: `CONFIRMED`, `CANCELLED`
236
+ - **AppointmentStatus**: `CONFIRMED`, `CANCELLED`, `RESCHEDULED`
235
237
 
236
238
  ## Exported Services
237
239
 
@@ -6,11 +6,13 @@ import { TasVideocallComponent } from '../tas-videocall/tas-videocall.component'
6
6
  import * as i0 from "@angular/core";
7
7
  import * as i1 from "@ng-bootstrap/ng-bootstrap";
8
8
  import * as i2 from "../../services/tas.service";
9
- import * as i3 from "@angular/common";
9
+ import * as i3 from "../../services/tas-utility.service";
10
+ import * as i4 from "@angular/common";
10
11
  export class TasButtonComponent {
11
- constructor(modalService, tasService) {
12
+ constructor(modalService, tasService, tasUtilityService) {
12
13
  this.modalService = modalService;
13
14
  this.tasService = tasService;
15
+ this.tasUtilityService = tasUtilityService;
14
16
  // Status endpoint params
15
17
  this.roomType = TasRoomType.TAS;
16
18
  this.businessRole = TasBusinessRole.USER;
@@ -23,6 +25,7 @@ export class TasButtonComponent {
23
25
  this.isStatusError = false;
24
26
  this.statusErrorMessage = '';
25
27
  this.isJoinable = false; // Tracks joinable field from status response
28
+ this.shouldShowButton = true; // Controls button visibility
26
29
  this.subscriptions = new Subscription();
27
30
  this.currentModalRef = null;
28
31
  this.videoCallModalRef = null;
@@ -97,38 +100,37 @@ export class TasButtonComponent {
97
100
  }
98
101
  this.isCheckingStatus = true;
99
102
  this.statusErrorMessage = '';
100
- this.subscriptions.add(this.tasService.getProxyVideoStatus({
103
+ this.subscriptions.add(this.tasService
104
+ .getProxyVideoStatus({
101
105
  roomType: this.roomType,
102
106
  entityId: this.entityId,
103
107
  tenant: this.tenant,
104
108
  businessRole: this.businessRole,
105
- }).subscribe({
109
+ })
110
+ .subscribe({
106
111
  next: (response) => {
107
- // Check if response is actually an error (some HTTP adapters return errors in next)
108
- // Also check for undefined/null or missing content
109
- const isErrorResponse = !response ||
110
- !response.content ||
111
- response?.ok === false ||
112
- response?.status >= 400 ||
113
- response?.error ||
114
- response?.name === 'HttpErrorResponse';
115
- if (isErrorResponse) {
116
- this.isCheckingStatus = false;
117
- this.isStatusError = true;
118
- this.statusErrorMessage = response?.error?.message || response?.message || 'Error checking status';
119
- }
120
- else {
121
- this.isCheckingStatus = false;
122
- this.isStatusError = false;
123
- this.statusErrorMessage = '';
124
- // Update joinable state from response
125
- this.isJoinable = response.content?.joinable ?? false;
126
- }
112
+ this.isCheckingStatus = false;
113
+ this.isStatusError = false;
114
+ this.statusErrorMessage = '';
115
+ this.shouldShowButton = true;
116
+ // Update joinable state from response
117
+ this.isJoinable = response.content?.joinable ?? false;
127
118
  },
128
119
  error: (err) => {
129
120
  this.isCheckingStatus = false;
130
- this.isStatusError = true;
131
- this.statusErrorMessage = err?.error?.message || err?.message || 'Error checking status';
121
+ const errorMessage = this.tasUtilityService.extractErrorMessage(err, 'Error checking status');
122
+ this.statusErrorMessage = errorMessage;
123
+ // Use utility service to determine if button should be shown
124
+ this.shouldShowButton = this.tasUtilityService.shouldShowButton(errorMessage);
125
+ // Stop polling on error
126
+ this.stopStatusPolling();
127
+ // If button should be hidden, don't treat as error
128
+ if (!this.shouldShowButton) {
129
+ this.isStatusError = false;
130
+ }
131
+ else {
132
+ this.isStatusError = true;
133
+ }
132
134
  },
133
135
  }));
134
136
  }
@@ -179,12 +181,12 @@ export class TasButtonComponent {
179
181
  });
180
182
  }
181
183
  }
182
- TasButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, deps: [{ token: i1.NgbModal }, { token: i2.TasService }], target: i0.ɵɵFactoryTarget.Component });
183
- 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 [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;padding:10px 24px;font-weight:500;margin-right:0}.tas-btn--teal:hover:not(:disabled){background-color:#00838f!important;border-color:#00838f!important}\n"], directives: [{ type: i1.NgbTooltip, selector: "[ngbTooltip]", inputs: ["animation", "autoClose", "placement", "triggers", "container", "disableTooltip", "tooltipClass", "openDelay", "closeDelay", "ngbTooltip"], outputs: ["shown", "hidden"], exportAs: ["ngbTooltip"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
184
+ 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 });
185
+ 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"] }] });
184
186
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, decorators: [{
185
187
  type: Component,
186
- args: [{ selector: 'tas-btn', template: "<span\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;padding:10px 24px;font-weight:500;margin-right:0}.tas-btn--teal:hover:not(:disabled){background-color:#00838f!important;border-color:#00838f!important}\n"] }]
187
- }], ctorParameters: function () { return [{ type: i1.NgbModal }, { type: i2.TasService }]; }, propDecorators: { roomType: [{
188
+ 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"] }]
189
+ }], ctorParameters: function () { return [{ type: i1.NgbModal }, { type: i2.TasService }, { type: i3.TasUtilityService }]; }, propDecorators: { roomType: [{
188
190
  type: Input
189
191
  }], entityId: [{
190
192
  type: Input
@@ -199,4 +201,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImpo
199
201
  }], buttonLabel: [{
200
202
  type: Input
201
203
  }] } });
202
- //# 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  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';\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\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  ) {}\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.getProxyVideoStatus({\n        roomType: this.roomType,\n        entityId: this.entityId,\n        tenant: this.tenant,\n        businessRole: this.businessRole,\n      }).subscribe({\n        next: (response: any) => {\n          // Check if response is actually an error (some HTTP adapters return errors in next)\n          // Also check for undefined/null or missing content\n          const isErrorResponse = !response || \n                                  !response.content ||\n                                  response?.ok === false || \n                                  response?.status >= 400 || \n                                  response?.error ||\n                                  response?.name === 'HttpErrorResponse';\n          \n          if (isErrorResponse) {\n            this.isCheckingStatus = false;\n            this.isStatusError = true;\n            this.statusErrorMessage = response?.error?.message || response?.message || 'Error checking status';\n          } else {\n            this.isCheckingStatus = false;\n            this.isStatusError = false;\n            this.statusErrorMessage = '';\n            // Update joinable state from response\n            this.isJoinable = response.content?.joinable ?? false;\n          }\n        },\n        error: (err) => {\n          this.isCheckingStatus = false;\n          this.isStatusError = true;\n          this.statusErrorMessage = err?.error?.message || err?.message || 'Error checking status';\n        },\n      })\n    );\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  [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"]}
204
+ //# 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 = true; // Controls button 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            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            this.isCheckingStatus = false;\n            const errorMessage = this.tasUtilityService.extractErrorMessage(err, 'Error checking status');\n            this.statusErrorMessage = errorMessage;\n\n            // Use utility service to determine if button should be shown\n            this.shouldShowButton = this.tasUtilityService.shouldShowButton(errorMessage);\n\n            // Stop polling on error\n            this.stopStatusPolling();\n\n            // If button should be hidden, don't treat as error\n            if (!this.shouldShowButton) {\n              this.isStatusError = false;\n            } else {\n              this.isStatusError = true;\n            }\n          },\n        })\n    );\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"]}
@@ -29,7 +29,7 @@ export class TasIncomingAppointmentComponent {
29
29
  const in7Days = new Date(today);
30
30
  in7Days.setDate(today.getDate() + 7);
31
31
  this.subscriptions.add(this.tasService
32
- .getAppointments({ fromDate: this.formatDate(today), toDate: this.formatDate(in7Days) })
32
+ .getAppointments({ fromDate: this.formatDate(today), toDate: this.formatDate(in7Days), entityId: this.entityId })
33
33
  .subscribe({
34
34
  next: (response) => {
35
35
  // Handle both array response and wrapped response (e.g., { content: [...] })
@@ -89,10 +89,10 @@ export class TasIncomingAppointmentComponent {
89
89
  }
90
90
  }
91
91
  TasIncomingAppointmentComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasIncomingAppointmentComponent, deps: [{ token: i1.TasService }], target: i0.ɵɵFactoryTarget.Component });
92
- TasIncomingAppointmentComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasIncomingAppointmentComponent, selector: "tas-incoming-appointment", inputs: { roomType: "roomType", entityId: "entityId", tenant: "tenant", businessRole: "businessRole", currentUser: "currentUser" }, outputs: { enterCall: "enterCall" }, ngImport: i0, template: "<div class=\"incoming-appointment-card\">\n <h3 class=\"card-title\">Pr\u00F3ximo turno</h3>\n\n <!-- Loading state -->\n <div class=\"card-content\" *ngIf=\"isLoading\">\n <div class=\"loading-spinner\"></div>\n </div>\n\n <!-- Empty state -->\n <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n <div class=\"icon-container\">\n <div class=\"icon-circle\">\n <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n </div>\n <span class=\"sparkle sparkle-1\">\u2726</span>\n <span class=\"sparkle sparkle-2\">\u2726</span>\n <span class=\"sparkle sparkle-3\">\u2726</span>\n <span class=\"sparkle sparkle-4\">\u2726</span>\n </div>\n <h4 class=\"empty-title\">Todav\u00EDa no ten\u00E9s turnos agendados</h4>\n <p class=\"empty-subtitle\">\n En caso de que Medicina Laboral requiera una consulta, lo ver\u00E1s en esta secci\u00F3n.\n </p>\n </div>\n\n <!-- Appointment state -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n \n <div class=\"appointment-card\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n \n <span class=\"doctor-name\">{{ appointment.title }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formattedDate }}</span>\n <span class=\"time\">{{ formattedTimeRange }}</span>\n </div>\n <tas-btn\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"roomType\"\n [entityId]=\"entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block}.incoming-appointment-card{background:#ffffff;border-radius:12px;box-shadow:0 2px 8px #00000014;padding:24px;min-width:360px}.card-title{font-size:16px;font-weight:400;color:#6b7280;margin:0 0 24px}.card-content{display:flex;flex-direction:column;align-items:center}.loading-spinner{width:40px;height:40px;border:3px solid #e0f7fa;border-top-color:#0097a7;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.empty-state{text-align:center;padding:20px 0}.icon-container{position:relative;width:120px;height:120px;margin-bottom:24px}.icon-circle{width:100%;height:100%;background:#e0f7fa;border-radius:50%;display:flex;align-items:center;justify-content:center}.icon-circle i{font-size:40px;color:#0097a7}.sparkle{position:absolute;color:#0097a7;font-size:12px}.sparkle-1{top:10px;right:5px}.sparkle-2{top:0;right:30px}.sparkle-3{top:25px;left:0}.sparkle-4{bottom:20px;right:0}.empty-title{font-size:18px;font-weight:600;color:#1f2937;margin:0 0 12px}.empty-subtitle{font-size:14px;color:#6b7280;margin:0;max-width:320px;line-height:1.5}.appointment-state{align-items:stretch}.appointment-card{border-radius:12px;border:1px solid var(--Primary-White-Uell50, #8ED1D8);background:#F1FAFA;padding:1.5rem}.appointment-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.doctor-name{overflow:hidden;color:var(--Neutral-GreyDark, #383E52);text-overflow:ellipsis;font-size:16px;font-style:normal;font-weight:600;line-height:24px;letter-spacing:.016px}.appointment-details{display:flex;justify-content:space-between;align-items:flex-end}.date-time{display:flex;flex-direction:column;gap:4px}.date{color:var(--Neutral-GreyDark, #383E52);font-size:12px;font-style:normal;font-weight:600;line-height:16px;letter-spacing:.06px}.time{font-size:14px;color:var(--Neutral-GreyDark, #383E52);font-family:Inter;font-size:10px;font-style:normal;font-weight:400;line-height:14px;letter-spacing:.04px}\n"], components: [{ type: i2.TasAvatarComponent, selector: "tas-avatar", inputs: ["name", "size"] }, { type: i3.TasButtonComponent, selector: "tas-btn", inputs: ["roomType", "entityId", "tenant", "businessRole", "currentUser", "variant", "buttonLabel"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
92
+ TasIncomingAppointmentComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasIncomingAppointmentComponent, selector: "tas-incoming-appointment", inputs: { roomType: "roomType", entityId: "entityId", tenant: "tenant", businessRole: "businessRole", currentUser: "currentUser" }, outputs: { enterCall: "enterCall" }, ngImport: i0, template: "<div class=\"incoming-appointment-card\">\n <h3 class=\"card-title\">Pr\u00F3ximo turno</h3>\n\n <!-- Loading state -->\n <div class=\"card-content\" *ngIf=\"isLoading\">\n <div class=\"loading-spinner\"></div>\n </div>\n\n <!-- Empty state -->\n <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n <div class=\"icon-container\">\n <div class=\"icon-circle\">\n <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n </div>\n <span class=\"sparkle sparkle-1\">\u2726</span>\n <span class=\"sparkle sparkle-2\">\u2726</span>\n <span class=\"sparkle sparkle-3\">\u2726</span>\n <span class=\"sparkle sparkle-4\">\u2726</span>\n </div>\n <h4 class=\"empty-title\">Todav\u00EDa no ten\u00E9s turnos agendados</h4>\n <p class=\"empty-subtitle\">\n En caso de que Medicina Laboral requiera una consulta, lo ver\u00E1s en esta secci\u00F3n.\n </p>\n </div>\n\n <!-- Appointment state -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n \n <div class=\"appointment-card\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n \n <span class=\"doctor-name\">{{ appointment.title }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formattedDate }}</span>\n <span class=\"time\">{{ formattedTimeRange }}</span>\n </div>\n <tas-btn\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appointment.roomType\"\n [entityId]=\"appointment.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block}.incoming-appointment-card{background:#ffffff;border-radius:12px;box-shadow:0 2px 8px #00000014;padding:24px;min-width:360px}.card-title{font-size:16px;font-weight:400;color:#6b7280;margin:0 0 24px}.card-content{display:flex;flex-direction:column;align-items:center}.loading-spinner{width:40px;height:40px;border:3px solid #e0f7fa;border-top-color:#0097a7;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.empty-state{text-align:center;padding:20px 0}.icon-container{position:relative;width:120px;height:120px;margin-bottom:24px}.icon-circle{width:100%;height:100%;background:#e0f7fa;border-radius:50%;display:flex;align-items:center;justify-content:center}.icon-circle i{font-size:40px;color:#0097a7}.sparkle{position:absolute;color:#0097a7;font-size:12px}.sparkle-1{top:10px;right:5px}.sparkle-2{top:0;right:30px}.sparkle-3{top:25px;left:0}.sparkle-4{bottom:20px;right:0}.empty-title{font-size:18px;font-weight:600;color:#1f2937;margin:0 0 12px}.empty-subtitle{font-size:14px;color:#6b7280;margin:0;max-width:320px;line-height:1.5}.appointment-state{align-items:stretch}.appointment-card{border-radius:12px;border:1px solid var(--Primary-White-Uell50, #8ED1D8);background:#F1FAFA;padding:1.5rem}.appointment-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.doctor-name{overflow:hidden;color:var(--Neutral-GreyDark, #383E52);text-overflow:ellipsis;font-size:16px;font-style:normal;font-weight:600;line-height:24px;letter-spacing:.016px}.appointment-details{display:flex;justify-content:space-between;align-items:flex-end}.date-time{display:flex;flex-direction:column;gap:4px}.date{color:var(--Neutral-GreyDark, #383E52);font-size:12px;font-style:normal;font-weight:600;line-height:16px;letter-spacing:.06px}.time{font-size:14px;color:var(--Neutral-GreyDark, #383E52);font-family:Inter;font-size:10px;font-style:normal;font-weight:400;line-height:14px;letter-spacing:.04px}\n"], components: [{ type: i2.TasAvatarComponent, selector: "tas-avatar", inputs: ["name", "size"] }, { type: i3.TasButtonComponent, selector: "tas-btn", inputs: ["roomType", "entityId", "tenant", "businessRole", "currentUser", "variant", "buttonLabel"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
93
93
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasIncomingAppointmentComponent, decorators: [{
94
94
  type: Component,
95
- args: [{ selector: 'tas-incoming-appointment', template: "<div class=\"incoming-appointment-card\">\n <h3 class=\"card-title\">Pr\u00F3ximo turno</h3>\n\n <!-- Loading state -->\n <div class=\"card-content\" *ngIf=\"isLoading\">\n <div class=\"loading-spinner\"></div>\n </div>\n\n <!-- Empty state -->\n <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n <div class=\"icon-container\">\n <div class=\"icon-circle\">\n <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n </div>\n <span class=\"sparkle sparkle-1\">\u2726</span>\n <span class=\"sparkle sparkle-2\">\u2726</span>\n <span class=\"sparkle sparkle-3\">\u2726</span>\n <span class=\"sparkle sparkle-4\">\u2726</span>\n </div>\n <h4 class=\"empty-title\">Todav\u00EDa no ten\u00E9s turnos agendados</h4>\n <p class=\"empty-subtitle\">\n En caso de que Medicina Laboral requiera una consulta, lo ver\u00E1s en esta secci\u00F3n.\n </p>\n </div>\n\n <!-- Appointment state -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n \n <div class=\"appointment-card\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n \n <span class=\"doctor-name\">{{ appointment.title }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formattedDate }}</span>\n <span class=\"time\">{{ formattedTimeRange }}</span>\n </div>\n <tas-btn\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"roomType\"\n [entityId]=\"entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block}.incoming-appointment-card{background:#ffffff;border-radius:12px;box-shadow:0 2px 8px #00000014;padding:24px;min-width:360px}.card-title{font-size:16px;font-weight:400;color:#6b7280;margin:0 0 24px}.card-content{display:flex;flex-direction:column;align-items:center}.loading-spinner{width:40px;height:40px;border:3px solid #e0f7fa;border-top-color:#0097a7;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.empty-state{text-align:center;padding:20px 0}.icon-container{position:relative;width:120px;height:120px;margin-bottom:24px}.icon-circle{width:100%;height:100%;background:#e0f7fa;border-radius:50%;display:flex;align-items:center;justify-content:center}.icon-circle i{font-size:40px;color:#0097a7}.sparkle{position:absolute;color:#0097a7;font-size:12px}.sparkle-1{top:10px;right:5px}.sparkle-2{top:0;right:30px}.sparkle-3{top:25px;left:0}.sparkle-4{bottom:20px;right:0}.empty-title{font-size:18px;font-weight:600;color:#1f2937;margin:0 0 12px}.empty-subtitle{font-size:14px;color:#6b7280;margin:0;max-width:320px;line-height:1.5}.appointment-state{align-items:stretch}.appointment-card{border-radius:12px;border:1px solid var(--Primary-White-Uell50, #8ED1D8);background:#F1FAFA;padding:1.5rem}.appointment-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.doctor-name{overflow:hidden;color:var(--Neutral-GreyDark, #383E52);text-overflow:ellipsis;font-size:16px;font-style:normal;font-weight:600;line-height:24px;letter-spacing:.016px}.appointment-details{display:flex;justify-content:space-between;align-items:flex-end}.date-time{display:flex;flex-direction:column;gap:4px}.date{color:var(--Neutral-GreyDark, #383E52);font-size:12px;font-style:normal;font-weight:600;line-height:16px;letter-spacing:.06px}.time{font-size:14px;color:var(--Neutral-GreyDark, #383E52);font-family:Inter;font-size:10px;font-style:normal;font-weight:400;line-height:14px;letter-spacing:.04px}\n"] }]
95
+ args: [{ selector: 'tas-incoming-appointment', template: "<div class=\"incoming-appointment-card\">\n <h3 class=\"card-title\">Pr\u00F3ximo turno</h3>\n\n <!-- Loading state -->\n <div class=\"card-content\" *ngIf=\"isLoading\">\n <div class=\"loading-spinner\"></div>\n </div>\n\n <!-- Empty state -->\n <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n <div class=\"icon-container\">\n <div class=\"icon-circle\">\n <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n </div>\n <span class=\"sparkle sparkle-1\">\u2726</span>\n <span class=\"sparkle sparkle-2\">\u2726</span>\n <span class=\"sparkle sparkle-3\">\u2726</span>\n <span class=\"sparkle sparkle-4\">\u2726</span>\n </div>\n <h4 class=\"empty-title\">Todav\u00EDa no ten\u00E9s turnos agendados</h4>\n <p class=\"empty-subtitle\">\n En caso de que Medicina Laboral requiera una consulta, lo ver\u00E1s en esta secci\u00F3n.\n </p>\n </div>\n\n <!-- Appointment state -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n \n <div class=\"appointment-card\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n \n <span class=\"doctor-name\">{{ appointment.title }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formattedDate }}</span>\n <span class=\"time\">{{ formattedTimeRange }}</span>\n </div>\n <tas-btn\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appointment.roomType\"\n [entityId]=\"appointment.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block}.incoming-appointment-card{background:#ffffff;border-radius:12px;box-shadow:0 2px 8px #00000014;padding:24px;min-width:360px}.card-title{font-size:16px;font-weight:400;color:#6b7280;margin:0 0 24px}.card-content{display:flex;flex-direction:column;align-items:center}.loading-spinner{width:40px;height:40px;border:3px solid #e0f7fa;border-top-color:#0097a7;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.empty-state{text-align:center;padding:20px 0}.icon-container{position:relative;width:120px;height:120px;margin-bottom:24px}.icon-circle{width:100%;height:100%;background:#e0f7fa;border-radius:50%;display:flex;align-items:center;justify-content:center}.icon-circle i{font-size:40px;color:#0097a7}.sparkle{position:absolute;color:#0097a7;font-size:12px}.sparkle-1{top:10px;right:5px}.sparkle-2{top:0;right:30px}.sparkle-3{top:25px;left:0}.sparkle-4{bottom:20px;right:0}.empty-title{font-size:18px;font-weight:600;color:#1f2937;margin:0 0 12px}.empty-subtitle{font-size:14px;color:#6b7280;margin:0;max-width:320px;line-height:1.5}.appointment-state{align-items:stretch}.appointment-card{border-radius:12px;border:1px solid var(--Primary-White-Uell50, #8ED1D8);background:#F1FAFA;padding:1.5rem}.appointment-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.doctor-name{overflow:hidden;color:var(--Neutral-GreyDark, #383E52);text-overflow:ellipsis;font-size:16px;font-style:normal;font-weight:600;line-height:24px;letter-spacing:.016px}.appointment-details{display:flex;justify-content:space-between;align-items:flex-end}.date-time{display:flex;flex-direction:column;gap:4px}.date{color:var(--Neutral-GreyDark, #383E52);font-size:12px;font-style:normal;font-weight:600;line-height:16px;letter-spacing:.06px}.time{font-size:14px;color:var(--Neutral-GreyDark, #383E52);font-family:Inter;font-size:10px;font-style:normal;font-weight:400;line-height:14px;letter-spacing:.04px}\n"] }]
96
96
  }], ctorParameters: function () { return [{ type: i1.TasService }]; }, propDecorators: { roomType: [{
97
97
  type: Input
98
98
  }], entityId: [{
@@ -106,4 +106,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImpo
106
106
  }], enterCall: [{
107
107
  type: Output
108
108
  }] } });
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-incoming-appointment.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  OnDestroy,\n  Input,\n  Output,\n  EventEmitter,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { TasService } from '../../services/tas.service';\nimport {\n  TasAppointment,\n  AppointmentStatus,\n  TasRoomType,\n  TasBusinessRole,\n  TasCurrentUser,\n} from '../../interfaces/tas.interfaces';\n\n@Component({\n  selector: 'tas-incoming-appointment',\n  templateUrl: './tas-incoming-appointment.component.html',\n  styleUrls: ['./tas-incoming-appointment.component.scss'],\n})\nexport class TasIncomingAppointmentComponent implements OnInit, OnDestroy {\n  // Passthrough inputs for tas-btn\n  @Input() roomType: TasRoomType = TasRoomType.TAS;\n  @Input() entityId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n  @Input() currentUser!: TasCurrentUser;\n\n  @Output() enterCall = new EventEmitter<TasAppointment>();\n\n  public appointment: TasAppointment | null = null;\n  public isLoading = true;\n  public hasError = false;\n\n  private subscriptions = new Subscription();\n\n  constructor(private tasService: TasService) {}\n\n  ngOnInit(): void {\n    this.loadAppointments();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  private loadAppointments(): void {\n    const today = new Date();\n    const in7Days = new Date(today);\n    in7Days.setDate(today.getDate() + 7);\n\n    this.subscriptions.add(\n      this.tasService\n        .getAppointments({ fromDate: this.formatDate(today), toDate: this.formatDate(in7Days) })\n        .subscribe({\n          next: (response: any) => {\n            // Handle both array response and wrapped response (e.g., { content: [...] })\n            const appointments: TasAppointment[] = Array.isArray(response)\n              ? response\n              : response?.content || [];\n\n            // Filter for confirmed appointments and get the first one\n            const confirmed = appointments.filter(\n              (a) => a.status === AppointmentStatus.CONFIRMED\n            );\n            this.appointment = confirmed.length > 0 ? confirmed[0] : null;\n            this.isLoading = false;\n          },\n          error: () => {\n            this.hasError = true;\n            this.isLoading = false;\n          },\n        })\n    );\n  }\n\n  public onEnterCall(): void {\n    if (this.appointment) {\n      this.enterCall.emit(this.appointment);\n    }\n  }\n\n  /**\n   * Format date to Spanish format: \"Lunes 8 de diciembre\"\n   */\n  public get formattedDate(): string {\n    if (!this.appointment) return '';\n\n    const [year, month, day] = this.appointment.date.split('-').map(Number);\n    const date = new Date(year, month - 1, day);\n\n    const dayNames = [\n      'Domingo', 'Lunes', 'Martes', 'Miércoles',\n      'Jueves', 'Viernes', 'Sábado'\n    ];\n    const monthNames = [\n      'enero', 'febrero', 'marzo', 'abril', 'mayo', 'junio',\n      'julio', 'agosto', 'septiembre', 'octubre', 'noviembre', 'diciembre'\n    ];\n\n    const dayName = dayNames[date.getDay()];\n    const dayNum = date.getDate();\n    const monthName = monthNames[date.getMonth()];\n\n    return `${dayName} ${dayNum} de ${monthName}`;\n  }\n\n  /**\n   * Format time range: \"9:00 - 9:30\"\n   */\n  public get formattedTimeRange(): string {\n    if (!this.appointment) return '';\n    return `${this.appointment.startTime} - ${this.appointment.endTime}`;\n  }\n\n  private formatDate(date: Date): string {\n    const year = date.getFullYear();\n    const month = String(date.getMonth() + 1).padStart(2, '0');\n    const day = String(date.getDate()).padStart(2, '0');\n    return `${year}-${month}-${day}`;\n  }\n}\n","<div class=\"incoming-appointment-card\">\n  <h3 class=\"card-title\">Próximo turno</h3>\n\n  <!-- Loading state -->\n  <div class=\"card-content\" *ngIf=\"isLoading\">\n    <div class=\"loading-spinner\"></div>\n  </div>\n\n  <!-- Empty state -->\n  <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n    <div class=\"icon-container\">\n      <div class=\"icon-circle\">\n        <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n      </div>\n      <span class=\"sparkle sparkle-1\">✦</span>\n      <span class=\"sparkle sparkle-2\">✦</span>\n      <span class=\"sparkle sparkle-3\">✦</span>\n      <span class=\"sparkle sparkle-4\">✦</span>\n    </div>\n    <h4 class=\"empty-title\">Todavía no tenés turnos agendados</h4>\n    <p class=\"empty-subtitle\">\n      En caso de que Medicina Laboral requiera una consulta, lo verás en esta sección.\n    </p>\n  </div>\n\n  <!-- Appointment state -->\n  <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n    \n    <div class=\"appointment-card\">\n      <div class=\"appointment-header\">\n        <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n        \n        <span class=\"doctor-name\">{{ appointment.title }}</span>\n      </div>\n      <div class=\"appointment-details\">\n        <div class=\"date-time\">\n          <span class=\"date\">{{ formattedDate }}</span>\n          <span class=\"time\">{{ formattedTimeRange }}</span>\n        </div>\n        <tas-btn\n          variant=\"teal\"\n          buttonLabel=\"Ingresar\"\n          [roomType]=\"roomType\"\n          [entityId]=\"entityId\"\n          [tenant]=\"tenant\"\n          [businessRole]=\"businessRole\"\n          [currentUser]=\"currentUser\"\n        ></tas-btn>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-incoming-appointment.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  OnDestroy,\n  Input,\n  Output,\n  EventEmitter,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { TasService } from '../../services/tas.service';\nimport {\n  TasAppointment,\n  AppointmentStatus,\n  TasRoomType,\n  TasBusinessRole,\n  TasCurrentUser,\n} from '../../interfaces/tas.interfaces';\n\n@Component({\n  selector: 'tas-incoming-appointment',\n  templateUrl: './tas-incoming-appointment.component.html',\n  styleUrls: ['./tas-incoming-appointment.component.scss'],\n})\nexport class TasIncomingAppointmentComponent implements OnInit, OnDestroy {\n  // Passthrough inputs for tas-btn\n  @Input() roomType: TasRoomType = TasRoomType.TAS;\n  @Input() entityId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n  @Input() currentUser!: TasCurrentUser;\n\n  @Output() enterCall = new EventEmitter<TasAppointment>();\n\n  public appointment: TasAppointment | null = null;\n  public isLoading = true;\n  public hasError = false;\n\n  private subscriptions = new Subscription();\n\n  constructor(private tasService: TasService) {}\n\n  ngOnInit(): void {\n    this.loadAppointments();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  private loadAppointments(): void {\n    const today = new Date();\n    const in7Days = new Date(today);\n    in7Days.setDate(today.getDate() + 7);\n\n    this.subscriptions.add(\n      this.tasService\n        .getAppointments({ fromDate: this.formatDate(today), toDate: this.formatDate(in7Days), entityId: this.entityId })\n        .subscribe({\n          next: (response: any) => {\n            // Handle both array response and wrapped response (e.g., { content: [...] })\n            const appointments: TasAppointment[] = Array.isArray(response)\n              ? response\n              : response?.content || [];\n\n            // Filter for confirmed appointments and get the first one\n            const confirmed = appointments.filter(\n              (a) => a.status === AppointmentStatus.CONFIRMED\n            );\n            this.appointment = confirmed.length > 0 ? confirmed[0] : null;\n            this.isLoading = false;\n          },\n          error: () => {\n            this.hasError = true;\n            this.isLoading = false;\n          },\n        })\n    );\n  }\n\n  public onEnterCall(): void {\n    if (this.appointment) {\n      this.enterCall.emit(this.appointment);\n    }\n  }\n\n  /**\n   * Format date to Spanish format: \"Lunes 8 de diciembre\"\n   */\n  public get formattedDate(): string {\n    if (!this.appointment) return '';\n\n    const [year, month, day] = this.appointment.date.split('-').map(Number);\n    const date = new Date(year, month - 1, day);\n\n    const dayNames = [\n      'Domingo', 'Lunes', 'Martes', 'Miércoles',\n      'Jueves', 'Viernes', 'Sábado'\n    ];\n    const monthNames = [\n      'enero', 'febrero', 'marzo', 'abril', 'mayo', 'junio',\n      'julio', 'agosto', 'septiembre', 'octubre', 'noviembre', 'diciembre'\n    ];\n\n    const dayName = dayNames[date.getDay()];\n    const dayNum = date.getDate();\n    const monthName = monthNames[date.getMonth()];\n\n    return `${dayName} ${dayNum} de ${monthName}`;\n  }\n\n  /**\n   * Format time range: \"9:00 - 9:30\"\n   */\n  public get formattedTimeRange(): string {\n    if (!this.appointment) return '';\n    return `${this.appointment.startTime} - ${this.appointment.endTime}`;\n  }\n\n  private formatDate(date: Date): string {\n    const year = date.getFullYear();\n    const month = String(date.getMonth() + 1).padStart(2, '0');\n    const day = String(date.getDate()).padStart(2, '0');\n    return `${year}-${month}-${day}`;\n  }\n}\n","<div class=\"incoming-appointment-card\">\n  <h3 class=\"card-title\">Próximo turno</h3>\n\n  <!-- Loading state -->\n  <div class=\"card-content\" *ngIf=\"isLoading\">\n    <div class=\"loading-spinner\"></div>\n  </div>\n\n  <!-- Empty state -->\n  <div class=\"card-content empty-state\" *ngIf=\"!isLoading && !appointment\">\n    <div class=\"icon-container\">\n      <div class=\"icon-circle\">\n        <i class=\"fa fa-calendar\" aria-hidden=\"true\"></i>\n      </div>\n      <span class=\"sparkle sparkle-1\">✦</span>\n      <span class=\"sparkle sparkle-2\">✦</span>\n      <span class=\"sparkle sparkle-3\">✦</span>\n      <span class=\"sparkle sparkle-4\">✦</span>\n    </div>\n    <h4 class=\"empty-title\">Todavía no tenés turnos agendados</h4>\n    <p class=\"empty-subtitle\">\n      En caso de que Medicina Laboral requiera una consulta, lo verás en esta sección.\n    </p>\n  </div>\n\n  <!-- Appointment state -->\n  <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointment\">\n    \n    <div class=\"appointment-card\">\n      <div class=\"appointment-header\">\n        <tas-avatar [name]=\"appointment.title\" [size]=\"48\"></tas-avatar>\n        \n        <span class=\"doctor-name\">{{ appointment.title }}</span>\n      </div>\n      <div class=\"appointment-details\">\n        <div class=\"date-time\">\n          <span class=\"date\">{{ formattedDate }}</span>\n          <span class=\"time\">{{ formattedTimeRange }}</span>\n        </div>\n        <tas-btn\n          variant=\"teal\"\n          buttonLabel=\"Ingresar\"\n          [roomType]=\"appointment.roomType\"\n          [entityId]=\"appointment.entityId\"\n          [tenant]=\"tenant\"\n          [businessRole]=\"businessRole\"\n          [currentUser]=\"currentUser\"\n        ></tas-btn>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
@@ -71,5 +71,6 @@ export var AppointmentStatus;
71
71
  (function (AppointmentStatus) {
72
72
  AppointmentStatus["CONFIRMED"] = "CONFIRMED";
73
73
  AppointmentStatus["CANCELLED"] = "CANCELLED";
74
+ AppointmentStatus["RESCHEDULED"] = "RESCHEDULED";
74
75
  })(AppointmentStatus || (AppointmentStatus = {}));
75
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFzLmludGVyZmFjZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90YXMtdWVsbC1zZGsvc3JjL2xpYi9pbnRlcmZhY2VzL3Rhcy5pbnRlcmZhY2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFFBQVE7QUFDUixNQUFNLENBQU4sSUFBWSxXQUtYO0FBTEQsV0FBWSxXQUFXO0lBQ3JCLDBCQUFXLENBQUE7SUFDWCx3QkFBUyxDQUFBO0lBQ1Qsa0NBQW1CLENBQUE7SUFDbkIsb0RBQXFDLENBQUE7QUFDdkMsQ0FBQyxFQUxXLFdBQVcsS0FBWCxXQUFXLFFBS3RCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FHWDtBQUhELFdBQVksY0FBYztJQUN4Qiw2Q0FBMkIsQ0FBQTtJQUMzQix5Q0FBdUIsQ0FBQTtBQUN6QixDQUFDLEVBSFcsY0FBYyxLQUFkLGNBQWMsUUFHekI7QUFFRCxNQUFNLENBQU4sSUFBWSxXQUlYO0FBSkQsV0FBWSxXQUFXO0lBQ3JCLDhCQUFlLENBQUE7SUFDZiw0QkFBYSxDQUFBO0lBQ2Isc0NBQXVCLENBQUE7QUFDekIsQ0FBQyxFQUpXLFdBQVcsS0FBWCxXQUFXLFFBSXRCO0FBRUQsTUFBTSxDQUFOLElBQVksZUFLWDtBQUxELFdBQVksZUFBZTtJQUN6QixrREFBK0IsQ0FBQTtJQUMvQixzQ0FBbUIsQ0FBQTtJQUNuQiw0Q0FBeUIsQ0FBQTtJQUN6QixnQ0FBYSxDQUFBO0FBQ2YsQ0FBQyxFQUxXLGVBQWUsS0FBZixlQUFlLFFBSzFCO0FBRUQsTUFBTSxDQUFOLElBQVksa0JBS1g7QUFMRCxXQUFZLGtCQUFrQjtJQUM1Qix1Q0FBaUIsQ0FBQTtJQUNqQiwyQ0FBcUIsQ0FBQTtJQUNyQix5Q0FBbUIsQ0FBQTtJQUNuQiw2Q0FBdUIsQ0FBQTtBQUN6QixDQUFDLEVBTFcsa0JBQWtCLEtBQWxCLGtCQUFrQixRQUs3QjtBQUVELE1BQU0sQ0FBTixJQUFZLFVBSVg7QUFKRCxXQUFZLFVBQVU7SUFDcEIsK0JBQWlCLENBQUE7SUFDakIsbUNBQXFCLENBQUE7SUFDckIsbUNBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQUpXLFVBQVUsS0FBVixVQUFVLFFBSXJCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FPWDtBQVBELFdBQVksY0FBYztJQUN4QiwyREFBeUMsQ0FBQTtJQUN6QywyREFBeUMsQ0FBQTtJQUN6Qyw2QkFBVyxDQUFBO0lBQ1gsaURBQStCLENBQUE7SUFDL0IscUVBQW1ELENBQUE7SUFDbkQsK0RBQTZDLENBQUE7QUFDL0MsQ0FBQyxFQVBXLGNBQWMsS0FBZCxjQUFjLFFBT3pCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FLWDtBQUxELFdBQVksY0FBYztJQUN4Qix1Q0FBcUIsQ0FBQTtJQUNyQixxQ0FBbUIsQ0FBQTtJQUNuQixtQ0FBaUIsQ0FBQTtJQUNqQix1Q0FBcUIsQ0FBQTtBQUN2QixDQUFDLEVBTFcsY0FBYyxLQUFkLGNBQWMsUUFLekI7QUFFRCxNQUFNLENBQU4sSUFBWSxTQU1YO0FBTkQsV0FBWSxTQUFTO0lBQ25CLDBCQUFhLENBQUE7SUFDYixzQ0FBeUIsQ0FBQTtJQUN6QixvQ0FBdUIsQ0FBQTtJQUN2QiwwQ0FBNkIsQ0FBQTtJQUM3Qiw0QkFBZSxDQUFBO0FBQ2pCLENBQUMsRUFOVyxTQUFTLEtBQVQsU0FBUyxRQU1wQjtBQUVELE1BQU0sQ0FBTixJQUFZLFFBR1g7QUFIRCxXQUFZLFFBQVE7SUFDbEIscUNBQXlCLENBQUE7SUFDekIsdUJBQVcsQ0FBQTtBQUNiLENBQUMsRUFIVyxRQUFRLEtBQVIsUUFBUSxRQUduQjtBQW9HRCxvQkFBb0I7QUFDcEIsTUFBTSxDQUFOLElBQVksaUJBR1g7QUFIRCxXQUFZLGlCQUFpQjtJQUMzQiw0Q0FBdUIsQ0FBQTtJQUN2Qiw0Q0FBdUIsQ0FBQTtBQUN6QixDQUFDLEVBSFcsaUJBQWlCLEtBQWpCLGlCQUFpQixRQUc1QiIsInNvdXJjZXNDb250ZW50IjpbIi8vIEVudW1zXG5leHBvcnQgZW51bSBUYXNSb29tVHlwZSB7XG4gIFRBUyA9ICdUQVMnLFxuICBKTSA9ICdKTScsXG4gIFdFQklOQVIgPSAnV0VCSU5BUicsXG4gIFdFTExORVNTX01BTkFHRVIgPSAnV0VMTE5FU1NfTUFOQUdFUicsXG59XG5cbmV4cG9ydCBlbnVtIFRhc1Nlc3Npb25UeXBlIHtcbiAgU1BPTlRBTkVPVVMgPSAnU1BPTlRBTkVPVVMnLFxuICBTQ0hFRFVMRUQgPSAnU0NIRURVTEVEJyxcbn1cblxuZXhwb3J0IGVudW0gVGFzVXNlclJvbGUge1xuICBPV05FUiA9ICdPV05FUicsXG4gIFVTRVIgPSAnVVNFUicsXG4gIE1PREVSQVRPUiA9ICdNT0RFUkFUT1InLFxufVxuXG5leHBvcnQgZW51bSBUYXNCdXNpbmVzc1JvbGUge1xuICBBRE1JTl9NQU5BR0VSID0gJ0FETUlOX01BTkFHRVInLFxuICBNQU5BR0VSID0gJ01BTkFHRVInLFxuICBCQUNLT0ZGSUNFID0gJ0JBQ0tPRkZJQ0UnLFxuICBVU0VSID0gJ1VTRVInLFxufVxuXG5leHBvcnQgZW51bSBWaWRlb1Nlc3Npb25TdGF0dXMge1xuICBBQ1RJVkUgPSAnQUNUSVZFJyxcbiAgSU5BQ1RJVkUgPSAnSU5BQ1RJVkUnLFxuICBQRU5ESU5HID0gJ1BFTkRJTkcnLFxuICBTQ0hFRFVMRUQgPSAnU0NIRURVTEVEJyxcbn1cblxuZXhwb3J0IGVudW0gVXNlclN0YXR1cyB7XG4gIEFDVElWRSA9ICdBQ1RJVkUnLFxuICBJTkFDVElWRSA9ICdJTkFDVElWRScsXG4gIEFTU0lHTkVEID0gJ0FTU0lHTkVEJyxcbn1cblxuZXhwb3J0IGVudW0gVXNlckNhbGxBY3Rpb24ge1xuICBXQUlUSU5HX1JPT01fRU5URVIgPSAnV0FJVElOR19ST09NX0VOVEVSJyxcbiAgV0FJVElOR19ST09NX0xFQVZFID0gJ1dBSVRJTkdfUk9PTV9MRUFWRScsXG4gIEJBTiA9ICdCQU4nLFxuICBDSEFOR0VfU1RBVFVTID0gJ0NIQU5HRV9TVEFUVVMnLFxuICBSRVFVRVNUX0dFT0xPQ0FMSVpBVElPTiA9ICdSRVFVRVNUX0dFT0xPQ0FMSVpBVElPTicsXG4gIEFDVElWQVRFX0dFT0xPQ0FUSU9OID0gJ0FDVElWQVRFX0dFT0xPQ0FUSU9OJyxcbn1cblxuZXhwb3J0IGVudW0gUm9vbVVzZXJTdGF0dXMge1xuICBBU1NJR05FRCA9ICdBU1NJR05FRCcsXG4gIFdBSVRJTkcgPSAnV0FJVElORycsXG4gIEpPSU5FRCA9ICdKT0lORUQnLFxuICBGSU5JU0hFRCA9ICdGSU5JU0hFRCcsXG59XG5cbmV4cG9ydCBlbnVtIENhbGxTdGF0ZSB7XG4gIElETEUgPSAnSURMRScsXG4gIENPTk5FQ1RJTkcgPSAnQ09OTkVDVElORycsXG4gIENPTk5FQ1RFRCA9ICdDT05ORUNURUQnLFxuICBESVNDT05ORUNURUQgPSAnRElTQ09OTkVDVEVEJyxcbiAgRVJST1IgPSAnRVJST1InLFxufVxuXG5leHBvcnQgZW51bSBWaWV3TW9kZSB7XG4gIEZVTExTQ1JFRU4gPSAnRlVMTFNDUkVFTicsXG4gIFBJUCA9ICdQSVAnLFxufVxuXG4vLyBVc2VyIGludGVyZmFjZXNcbmV4cG9ydCBpbnRlcmZhY2UgVGFzQ3VycmVudFVzZXIge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGxhc3RuYW1lOiBzdHJpbmc7XG4gIHJvbGU6IFRhc1VzZXJSb2xlO1xufVxuXG4vKiogSW5wdXQgY29uZmlndXJhdGlvbiBmb3IgdGhlIFRBUyB2aWRlbyBjYWxsIGxpYnJhcnkgKi9cbmV4cG9ydCBpbnRlcmZhY2UgVGFzQ2FsbENvbmZpZyB7XG4gIC8vIFN0YXR1cyBlbmRwb2ludCBwYXJhbXNcbiAgYXBwb2ludG1lbnRJZD86IG51bWJlcjtcbiAgc2Vzc2lvbklkPzogc3RyaW5nO1xuICByb29tVHlwZTogVGFzUm9vbVR5cGU7XG4gIGVudGl0eUlkOiBudW1iZXI7XG4gIHRlbmFudDogc3RyaW5nO1xuICBidXNpbmVzc1JvbGU6IFRhc0J1c2luZXNzUm9sZTtcbiAgLy8gQ3VycmVudCB1c2VyIGluZm8gKGZvciB0b2tlbiByZXF1ZXN0KVxuICBjdXJyZW50VXNlcjogVGFzQ3VycmVudFVzZXI7XG59XG5cbi8vIEFQSSBSZXF1ZXN0L1Jlc3BvbnNlIGludGVyZmFjZXNcbmV4cG9ydCBpbnRlcmZhY2UgU3RhcnRTZXNzaW9uUmVxdWVzdCB7XG4gIHNlc3Npb25JZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGxhc3RuYW1lOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgR2VuZXJhdGVUb2tlblJlcXVlc3Qge1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBsYXN0bmFtZTogc3RyaW5nO1xuICByb2xlVkM6IFRhc1VzZXJSb2xlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEdlbmVyYXRlVG9rZW5SZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICB0b2tlbjogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9TdGF0dXNSZXF1ZXN0IHtcbiAgYXBwb2ludG1lbnRJZD86IG51bWJlcjtcbiAgYnVzaW5lc3NSb2xlPzogVGFzQnVzaW5lc3NSb2xlO1xuICB0ZW5hbnQ/OiBzdHJpbmc7XG4gIHNlc3Npb25JZD86IHN0cmluZztcbiAgcm9vbVR5cGU/OiBUYXNSb29tVHlwZTtcbiAgZW50aXR5SWQ/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJveHlWaWRlb1N0YXR1c1VzZXIge1xuICB1c2VySWQ6IG51bWJlcjtcbiAgcm9sOiBUYXNVc2VyUm9sZTtcbiAgc3RhdHVzOiBVc2VyU3RhdHVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9TdGF0dXNSZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICB2aWRlb0NhbGxJZDogbnVtYmVyO1xuICAgIHNlc3Npb25JZDogc3RyaW5nO1xuICAgIHN0YXR1c1ZMOiBWaWRlb1Nlc3Npb25TdGF0dXM7XG4gICAgam9pbmFibGU6IGJvb2xlYW47XG4gICAgd2FpdGluZ1Jvb21BdmFpbGFibGU6IGJvb2xlYW47XG4gICAgYXBwb2ludG1lbnRJZDogbnVtYmVyO1xuICAgIGFjdGl2YXRlR2VvOiBib29sZWFuO1xuICAgIGdlb1JlcXVlc3RBY3RpdmU/OiBib29sZWFuOyAvLyBPd25lciBzZW50IGdlbyByZXF1ZXN0LCB3YWl0aW5nIGZvciB1c2VyIHJlc3BvbnNlXG4gICAgYWxsR2VvR3JhbnRlZD86IGJvb2xlYW47IC8vIEFsbCB1c2VycyBoYXZlIHJlc3BvbmRlZCB3aXRoIGdlb1xuICAgIHVzZXJzOiBQcm94eVZpZGVvU3RhdHVzVXNlcltdO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9GaW5pc2hSZXF1ZXN0IHtcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gIGJ1c2luZXNzUm9sZTogVGFzQnVzaW5lc3NSb2xlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEZpbmlzaFNlc3Npb25SZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICBtZXNzYWdlOiBzdHJpbmc7XG4gICAgY2FsbFN0YXR1czogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9Vc2VyTW9kaWZ5UmVxdWVzdCB7XG4gIHVzZXJJZD86IG51bWJlcjtcbiAgdmlkZW9DYWxsSWQ6IG51bWJlcjtcbiAgYWN0aW9uPzogVXNlckNhbGxBY3Rpb247XG4gIG5ld1N0YXR1cz86IFJvb21Vc2VyU3RhdHVzO1xuICBsYXRpdHVkZT86IG51bWJlcjtcbiAgbG9uZ2l0dWRlPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFdhaXRpbmdSb29tVXNlciB7XG4gIHVzZXJJZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIHN0YXR1czogUm9vbVVzZXJTdGF0dXM7XG59XG5cbi8vIEFwcG9pbnRtZW50IHR5cGVzXG5leHBvcnQgZW51bSBBcHBvaW50bWVudFN0YXR1cyB7XG4gIENPTkZJUk1FRCA9ICdDT05GSVJNRUQnLFxuICBDQU5DRUxMRUQgPSAnQ0FOQ0VMTEVEJyxcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUYXNBcHBvaW50bWVudCB7XG4gIGlkOiBudW1iZXI7XG4gIGFnZW5kYUlkOiBudW1iZXI7XG4gIGRhdGU6IHN0cmluZzsgICAgICAgLy8gXCJZWVlZLU1NLUREXCJcbiAgc3RhcnRUaW1lOiBzdHJpbmc7ICAvLyBcIkhIOm1tXCJcbiAgZW5kVGltZTogc3RyaW5nOyAgICAvLyBcIkhIOm1tXCJcbiAgYm9va2luZ1R5cGU6IHN0cmluZztcbiAgc3RhdHVzOiBBcHBvaW50bWVudFN0YXR1cztcbiAgdGl0bGU6IHN0cmluZztcbiAgbm90ZXM6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBHZXRBcHBvaW50bWVudHNSZXF1ZXN0IHtcbiAgZnJvbURhdGU6IHN0cmluZzsgIC8vIFlZWVktTU0tRERcbiAgdG9EYXRlOiBzdHJpbmc7ICAgIC8vIFlZWVktTU0tRERcbn1cbiJdfQ==
76
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFzLmludGVyZmFjZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90YXMtdWVsbC1zZGsvc3JjL2xpYi9pbnRlcmZhY2VzL3Rhcy5pbnRlcmZhY2VzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFFBQVE7QUFDUixNQUFNLENBQU4sSUFBWSxXQUtYO0FBTEQsV0FBWSxXQUFXO0lBQ3JCLDBCQUFXLENBQUE7SUFDWCx3QkFBUyxDQUFBO0lBQ1Qsa0NBQW1CLENBQUE7SUFDbkIsb0RBQXFDLENBQUE7QUFDdkMsQ0FBQyxFQUxXLFdBQVcsS0FBWCxXQUFXLFFBS3RCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FHWDtBQUhELFdBQVksY0FBYztJQUN4Qiw2Q0FBMkIsQ0FBQTtJQUMzQix5Q0FBdUIsQ0FBQTtBQUN6QixDQUFDLEVBSFcsY0FBYyxLQUFkLGNBQWMsUUFHekI7QUFFRCxNQUFNLENBQU4sSUFBWSxXQUlYO0FBSkQsV0FBWSxXQUFXO0lBQ3JCLDhCQUFlLENBQUE7SUFDZiw0QkFBYSxDQUFBO0lBQ2Isc0NBQXVCLENBQUE7QUFDekIsQ0FBQyxFQUpXLFdBQVcsS0FBWCxXQUFXLFFBSXRCO0FBRUQsTUFBTSxDQUFOLElBQVksZUFLWDtBQUxELFdBQVksZUFBZTtJQUN6QixrREFBK0IsQ0FBQTtJQUMvQixzQ0FBbUIsQ0FBQTtJQUNuQiw0Q0FBeUIsQ0FBQTtJQUN6QixnQ0FBYSxDQUFBO0FBQ2YsQ0FBQyxFQUxXLGVBQWUsS0FBZixlQUFlLFFBSzFCO0FBRUQsTUFBTSxDQUFOLElBQVksa0JBS1g7QUFMRCxXQUFZLGtCQUFrQjtJQUM1Qix1Q0FBaUIsQ0FBQTtJQUNqQiwyQ0FBcUIsQ0FBQTtJQUNyQix5Q0FBbUIsQ0FBQTtJQUNuQiw2Q0FBdUIsQ0FBQTtBQUN6QixDQUFDLEVBTFcsa0JBQWtCLEtBQWxCLGtCQUFrQixRQUs3QjtBQUVELE1BQU0sQ0FBTixJQUFZLFVBSVg7QUFKRCxXQUFZLFVBQVU7SUFDcEIsK0JBQWlCLENBQUE7SUFDakIsbUNBQXFCLENBQUE7SUFDckIsbUNBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQUpXLFVBQVUsS0FBVixVQUFVLFFBSXJCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FPWDtBQVBELFdBQVksY0FBYztJQUN4QiwyREFBeUMsQ0FBQTtJQUN6QywyREFBeUMsQ0FBQTtJQUN6Qyw2QkFBVyxDQUFBO0lBQ1gsaURBQStCLENBQUE7SUFDL0IscUVBQW1ELENBQUE7SUFDbkQsK0RBQTZDLENBQUE7QUFDL0MsQ0FBQyxFQVBXLGNBQWMsS0FBZCxjQUFjLFFBT3pCO0FBRUQsTUFBTSxDQUFOLElBQVksY0FLWDtBQUxELFdBQVksY0FBYztJQUN4Qix1Q0FBcUIsQ0FBQTtJQUNyQixxQ0FBbUIsQ0FBQTtJQUNuQixtQ0FBaUIsQ0FBQTtJQUNqQix1Q0FBcUIsQ0FBQTtBQUN2QixDQUFDLEVBTFcsY0FBYyxLQUFkLGNBQWMsUUFLekI7QUFFRCxNQUFNLENBQU4sSUFBWSxTQU1YO0FBTkQsV0FBWSxTQUFTO0lBQ25CLDBCQUFhLENBQUE7SUFDYixzQ0FBeUIsQ0FBQTtJQUN6QixvQ0FBdUIsQ0FBQTtJQUN2QiwwQ0FBNkIsQ0FBQTtJQUM3Qiw0QkFBZSxDQUFBO0FBQ2pCLENBQUMsRUFOVyxTQUFTLEtBQVQsU0FBUyxRQU1wQjtBQUVELE1BQU0sQ0FBTixJQUFZLFFBR1g7QUFIRCxXQUFZLFFBQVE7SUFDbEIscUNBQXlCLENBQUE7SUFDekIsdUJBQVcsQ0FBQTtBQUNiLENBQUMsRUFIVyxRQUFRLEtBQVIsUUFBUSxRQUduQjtBQW9HRCxvQkFBb0I7QUFDcEIsTUFBTSxDQUFOLElBQVksaUJBSVg7QUFKRCxXQUFZLGlCQUFpQjtJQUMzQiw0Q0FBdUIsQ0FBQTtJQUN2Qiw0Q0FBdUIsQ0FBQTtJQUN2QixnREFBMkIsQ0FBQTtBQUM3QixDQUFDLEVBSlcsaUJBQWlCLEtBQWpCLGlCQUFpQixRQUk1QiIsInNvdXJjZXNDb250ZW50IjpbIi8vIEVudW1zXG5leHBvcnQgZW51bSBUYXNSb29tVHlwZSB7XG4gIFRBUyA9ICdUQVMnLFxuICBKTSA9ICdKTScsXG4gIFdFQklOQVIgPSAnV0VCSU5BUicsXG4gIFdFTExORVNTX01BTkFHRVIgPSAnV0VMTE5FU1NfTUFOQUdFUicsXG59XG5cbmV4cG9ydCBlbnVtIFRhc1Nlc3Npb25UeXBlIHtcbiAgU1BPTlRBTkVPVVMgPSAnU1BPTlRBTkVPVVMnLFxuICBTQ0hFRFVMRUQgPSAnU0NIRURVTEVEJyxcbn1cblxuZXhwb3J0IGVudW0gVGFzVXNlclJvbGUge1xuICBPV05FUiA9ICdPV05FUicsXG4gIFVTRVIgPSAnVVNFUicsXG4gIE1PREVSQVRPUiA9ICdNT0RFUkFUT1InLFxufVxuXG5leHBvcnQgZW51bSBUYXNCdXNpbmVzc1JvbGUge1xuICBBRE1JTl9NQU5BR0VSID0gJ0FETUlOX01BTkFHRVInLFxuICBNQU5BR0VSID0gJ01BTkFHRVInLFxuICBCQUNLT0ZGSUNFID0gJ0JBQ0tPRkZJQ0UnLFxuICBVU0VSID0gJ1VTRVInLFxufVxuXG5leHBvcnQgZW51bSBWaWRlb1Nlc3Npb25TdGF0dXMge1xuICBBQ1RJVkUgPSAnQUNUSVZFJyxcbiAgSU5BQ1RJVkUgPSAnSU5BQ1RJVkUnLFxuICBQRU5ESU5HID0gJ1BFTkRJTkcnLFxuICBTQ0hFRFVMRUQgPSAnU0NIRURVTEVEJyxcbn1cblxuZXhwb3J0IGVudW0gVXNlclN0YXR1cyB7XG4gIEFDVElWRSA9ICdBQ1RJVkUnLFxuICBJTkFDVElWRSA9ICdJTkFDVElWRScsXG4gIEFTU0lHTkVEID0gJ0FTU0lHTkVEJyxcbn1cblxuZXhwb3J0IGVudW0gVXNlckNhbGxBY3Rpb24ge1xuICBXQUlUSU5HX1JPT01fRU5URVIgPSAnV0FJVElOR19ST09NX0VOVEVSJyxcbiAgV0FJVElOR19ST09NX0xFQVZFID0gJ1dBSVRJTkdfUk9PTV9MRUFWRScsXG4gIEJBTiA9ICdCQU4nLFxuICBDSEFOR0VfU1RBVFVTID0gJ0NIQU5HRV9TVEFUVVMnLFxuICBSRVFVRVNUX0dFT0xPQ0FMSVpBVElPTiA9ICdSRVFVRVNUX0dFT0xPQ0FMSVpBVElPTicsXG4gIEFDVElWQVRFX0dFT0xPQ0FUSU9OID0gJ0FDVElWQVRFX0dFT0xPQ0FUSU9OJyxcbn1cblxuZXhwb3J0IGVudW0gUm9vbVVzZXJTdGF0dXMge1xuICBBU1NJR05FRCA9ICdBU1NJR05FRCcsXG4gIFdBSVRJTkcgPSAnV0FJVElORycsXG4gIEpPSU5FRCA9ICdKT0lORUQnLFxuICBGSU5JU0hFRCA9ICdGSU5JU0hFRCcsXG59XG5cbmV4cG9ydCBlbnVtIENhbGxTdGF0ZSB7XG4gIElETEUgPSAnSURMRScsXG4gIENPTk5FQ1RJTkcgPSAnQ09OTkVDVElORycsXG4gIENPTk5FQ1RFRCA9ICdDT05ORUNURUQnLFxuICBESVNDT05ORUNURUQgPSAnRElTQ09OTkVDVEVEJyxcbiAgRVJST1IgPSAnRVJST1InLFxufVxuXG5leHBvcnQgZW51bSBWaWV3TW9kZSB7XG4gIEZVTExTQ1JFRU4gPSAnRlVMTFNDUkVFTicsXG4gIFBJUCA9ICdQSVAnLFxufVxuXG4vLyBVc2VyIGludGVyZmFjZXNcbmV4cG9ydCBpbnRlcmZhY2UgVGFzQ3VycmVudFVzZXIge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGxhc3RuYW1lOiBzdHJpbmc7XG4gIHJvbGU6IFRhc1VzZXJSb2xlO1xufVxuXG4vKiogSW5wdXQgY29uZmlndXJhdGlvbiBmb3IgdGhlIFRBUyB2aWRlbyBjYWxsIGxpYnJhcnkgKi9cbmV4cG9ydCBpbnRlcmZhY2UgVGFzQ2FsbENvbmZpZyB7XG4gIC8vIFN0YXR1cyBlbmRwb2ludCBwYXJhbXNcbiAgYXBwb2ludG1lbnRJZD86IG51bWJlcjtcbiAgc2Vzc2lvbklkPzogc3RyaW5nO1xuICByb29tVHlwZTogVGFzUm9vbVR5cGU7XG4gIGVudGl0eUlkOiBudW1iZXI7XG4gIHRlbmFudDogc3RyaW5nO1xuICBidXNpbmVzc1JvbGU6IFRhc0J1c2luZXNzUm9sZTtcbiAgLy8gQ3VycmVudCB1c2VyIGluZm8gKGZvciB0b2tlbiByZXF1ZXN0KVxuICBjdXJyZW50VXNlcjogVGFzQ3VycmVudFVzZXI7XG59XG5cbi8vIEFQSSBSZXF1ZXN0L1Jlc3BvbnNlIGludGVyZmFjZXNcbmV4cG9ydCBpbnRlcmZhY2UgU3RhcnRTZXNzaW9uUmVxdWVzdCB7XG4gIHNlc3Npb25JZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGxhc3RuYW1lOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgR2VuZXJhdGVUb2tlblJlcXVlc3Qge1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBsYXN0bmFtZTogc3RyaW5nO1xuICByb2xlVkM6IFRhc1VzZXJSb2xlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEdlbmVyYXRlVG9rZW5SZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICB0b2tlbjogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9TdGF0dXNSZXF1ZXN0IHtcbiAgYXBwb2ludG1lbnRJZD86IG51bWJlcjtcbiAgYnVzaW5lc3NSb2xlPzogVGFzQnVzaW5lc3NSb2xlO1xuICB0ZW5hbnQ/OiBzdHJpbmc7XG4gIHNlc3Npb25JZD86IHN0cmluZztcbiAgcm9vbVR5cGU/OiBUYXNSb29tVHlwZTtcbiAgZW50aXR5SWQ/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJveHlWaWRlb1N0YXR1c1VzZXIge1xuICB1c2VySWQ6IG51bWJlcjtcbiAgcm9sOiBUYXNVc2VyUm9sZTtcbiAgc3RhdHVzOiBVc2VyU3RhdHVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9TdGF0dXNSZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICB2aWRlb0NhbGxJZDogbnVtYmVyO1xuICAgIHNlc3Npb25JZDogc3RyaW5nO1xuICAgIHN0YXR1c1ZMOiBWaWRlb1Nlc3Npb25TdGF0dXM7XG4gICAgam9pbmFibGU6IGJvb2xlYW47XG4gICAgd2FpdGluZ1Jvb21BdmFpbGFibGU6IGJvb2xlYW47XG4gICAgYXBwb2ludG1lbnRJZDogbnVtYmVyO1xuICAgIGFjdGl2YXRlR2VvOiBib29sZWFuO1xuICAgIGdlb1JlcXVlc3RBY3RpdmU/OiBib29sZWFuOyAvLyBPd25lciBzZW50IGdlbyByZXF1ZXN0LCB3YWl0aW5nIGZvciB1c2VyIHJlc3BvbnNlXG4gICAgYWxsR2VvR3JhbnRlZD86IGJvb2xlYW47IC8vIEFsbCB1c2VycyBoYXZlIHJlc3BvbmRlZCB3aXRoIGdlb1xuICAgIHVzZXJzOiBQcm94eVZpZGVvU3RhdHVzVXNlcltdO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9GaW5pc2hSZXF1ZXN0IHtcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gIGJ1c2luZXNzUm9sZTogVGFzQnVzaW5lc3NSb2xlO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEZpbmlzaFNlc3Npb25SZXNwb25zZSB7XG4gIGNvbnRlbnQ6IHtcbiAgICBtZXNzYWdlOiBzdHJpbmc7XG4gICAgY2FsbFN0YXR1czogc3RyaW5nO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFByb3h5VmlkZW9Vc2VyTW9kaWZ5UmVxdWVzdCB7XG4gIHVzZXJJZD86IG51bWJlcjtcbiAgdmlkZW9DYWxsSWQ6IG51bWJlcjtcbiAgYWN0aW9uPzogVXNlckNhbGxBY3Rpb247XG4gIG5ld1N0YXR1cz86IFJvb21Vc2VyU3RhdHVzO1xuICBsYXRpdHVkZT86IG51bWJlcjtcbiAgbG9uZ2l0dWRlPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFdhaXRpbmdSb29tVXNlciB7XG4gIHVzZXJJZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIHN0YXR1czogUm9vbVVzZXJTdGF0dXM7XG59XG5cbi8vIEFwcG9pbnRtZW50IHR5cGVzXG5leHBvcnQgZW51bSBBcHBvaW50bWVudFN0YXR1cyB7XG4gIENPTkZJUk1FRCA9ICdDT05GSVJNRUQnLFxuICBDQU5DRUxMRUQgPSAnQ0FOQ0VMTEVEJyxcbiAgUkVTQ0hFRFVMRUQgPSAnUkVTQ0hFRFVMRUQnLFxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRhc0FwcG9pbnRtZW50IHtcbiAgaWQ6IG51bWJlcjtcbiAgYWdlbmRhSWQ6IG51bWJlcjtcbiAgZGF0ZTogc3RyaW5nOyAgICAgICAvLyBcIllZWVktTU0tRERcIlxuICBzdGFydFRpbWU6IHN0cmluZzsgIC8vIFwiSEg6bW1cIlxuICBlbmRUaW1lOiBzdHJpbmc7ICAgIC8vIFwiSEg6bW1cIlxuICBib29raW5nVHlwZTogc3RyaW5nO1xuICBzdGF0dXM6IEFwcG9pbnRtZW50U3RhdHVzO1xuICB0aXRsZTogc3RyaW5nO1xuICBub3Rlczogc3RyaW5nO1xuICBlbnRpdHlJZDogbnVtYmVyO1xuICByb29tVHlwZTogVGFzUm9vbVR5cGU7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgR2V0QXBwb2ludG1lbnRzUmVxdWVzdCB7XG4gIGZyb21EYXRlOiBzdHJpbmc7ICAvLyBZWVlZLU1NLUREXG4gIHRvRGF0ZTogc3RyaW5nOyAgICAvLyBZWVlZLU1NLUREXG4gIGVudGl0eUlkPzogbnVtYmVyOyAvLyBPcHRpb25hbCBlbnRpdHkgSUQgZmlsdGVyXG59XG4iXX0=
@@ -0,0 +1,60 @@
1
+ import { Injectable } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class TasUtilityService {
4
+ /**
5
+ * Determines if the button should be shown based on the error message.
6
+ * Returns false if the error message matches any of the patterns that indicate
7
+ * the button should be hidden (but polling should continue).
8
+ *
9
+ * @param errorMessage The error message from the status call
10
+ * @returns true if button should be shown, false if it should be hidden
11
+ */
12
+ shouldShowButton(errorMessage) {
13
+ if (!errorMessage) {
14
+ return true;
15
+ }
16
+ const message = errorMessage.trim();
17
+ // List of error messages that should hide the button
18
+ const hideButtonMessages = [
19
+ 'Debe enviarse sessionId o roomType + entityId',
20
+ 'No existe videollamada con ese sessionId',
21
+ 'Usuario no pertenece a la llamada',
22
+ ];
23
+ // Check exact matches
24
+ if (hideButtonMessages.some((msg) => message === msg)) {
25
+ return false;
26
+ }
27
+ // Check pattern match for "No existe videocall vigente para roomType=x y entityId=x"
28
+ // Using startsWith as primary check to handle any variations
29
+ if (message.startsWith('No existe videocall vigente para roomType=')) {
30
+ return false;
31
+ }
32
+ // Also check regex pattern as backup
33
+ const vigentePattern = /^No existe videocall vigente para roomType=.+ y entityId=\s*\d+$/;
34
+ if (vigentePattern.test(message)) {
35
+ return false;
36
+ }
37
+ return true;
38
+ }
39
+ /**
40
+ * Check if at least one OWNER has joined the call.
41
+ */
42
+ checkIfOwnerJoined(users) {
43
+ return users.some((u) => u.rol === 'OWNER' && u.status === 'JOINED');
44
+ }
45
+ /**
46
+ * Extract error message from various error structures.
47
+ */
48
+ extractErrorMessage(err, fallback = 'Error desconocido') {
49
+ return err?.error?.message || err?.message || fallback;
50
+ }
51
+ }
52
+ TasUtilityService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasUtilityService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
53
+ TasUtilityService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasUtilityService, providedIn: 'root' });
54
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasUtilityService, decorators: [{
55
+ type: Injectable,
56
+ args: [{
57
+ providedIn: 'root',
58
+ }]
59
+ }] });
60
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFzLXV0aWxpdHkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Rhcy11ZWxsLXNkay9zcmMvbGliL3NlcnZpY2VzL3Rhcy11dGlsaXR5LnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFLM0MsTUFBTSxPQUFPLGlCQUFpQjtJQUM1Qjs7Ozs7OztPQU9HO0lBQ0gsZ0JBQWdCLENBQUMsWUFBdUM7UUFDdEQsSUFBSSxDQUFDLFlBQVksRUFBRTtZQUNqQixPQUFPLElBQUksQ0FBQztTQUNiO1FBRUQsTUFBTSxPQUFPLEdBQUcsWUFBWSxDQUFDLElBQUksRUFBRSxDQUFDO1FBRXBDLHFEQUFxRDtRQUNyRCxNQUFNLGtCQUFrQixHQUFHO1lBQ3pCLCtDQUErQztZQUMvQywwQ0FBMEM7WUFDMUMsbUNBQW1DO1NBQ3BDLENBQUM7UUFFRixzQkFBc0I7UUFDdEIsSUFBSSxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLE9BQU8sS0FBSyxHQUFHLENBQUMsRUFBRTtZQUNyRCxPQUFPLEtBQUssQ0FBQztTQUNkO1FBRUQscUZBQXFGO1FBQ3JGLDZEQUE2RDtRQUM3RCxJQUFJLE9BQU8sQ0FBQyxVQUFVLENBQUMsNENBQTRDLENBQUMsRUFBRTtZQUNwRSxPQUFPLEtBQUssQ0FBQztTQUNkO1FBRUQscUNBQXFDO1FBQ3JDLE1BQU0sY0FBYyxHQUFHLGtFQUFrRSxDQUFDO1FBQzFGLElBQUksY0FBYyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsRUFBRTtZQUNoQyxPQUFPLEtBQUssQ0FBQztTQUNkO1FBRUQsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxrQkFBa0IsQ0FDaEIsS0FBd0Q7UUFFeEQsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLE9BQU8sSUFBSSxDQUFDLENBQUMsTUFBTSxLQUFLLFFBQVEsQ0FBQyxDQUFDO0lBQ3ZFLENBQUM7SUFFRDs7T0FFRztJQUNILG1CQUFtQixDQUFDLEdBQVEsRUFBRSxRQUFRLEdBQUcsbUJBQW1CO1FBQzFELE9BQU8sR0FBRyxFQUFFLEtBQUssRUFBRSxPQUFPLElBQUksR0FBRyxFQUFFLE9BQU8sSUFBSSxRQUFRLENBQUM7SUFDekQsQ0FBQzs7K0dBekRVLGlCQUFpQjttSEFBakIsaUJBQWlCLGNBRmhCLE1BQU07NEZBRVAsaUJBQWlCO2tCQUg3QixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuZXhwb3J0IGNsYXNzIFRhc1V0aWxpdHlTZXJ2aWNlIHtcbiAgLyoqXG4gICAqIERldGVybWluZXMgaWYgdGhlIGJ1dHRvbiBzaG91bGQgYmUgc2hvd24gYmFzZWQgb24gdGhlIGVycm9yIG1lc3NhZ2UuXG4gICAqIFJldHVybnMgZmFsc2UgaWYgdGhlIGVycm9yIG1lc3NhZ2UgbWF0Y2hlcyBhbnkgb2YgdGhlIHBhdHRlcm5zIHRoYXQgaW5kaWNhdGVcbiAgICogdGhlIGJ1dHRvbiBzaG91bGQgYmUgaGlkZGVuIChidXQgcG9sbGluZyBzaG91bGQgY29udGludWUpLlxuICAgKlxuICAgKiBAcGFyYW0gZXJyb3JNZXNzYWdlIFRoZSBlcnJvciBtZXNzYWdlIGZyb20gdGhlIHN0YXR1cyBjYWxsXG4gICAqIEByZXR1cm5zIHRydWUgaWYgYnV0dG9uIHNob3VsZCBiZSBzaG93biwgZmFsc2UgaWYgaXQgc2hvdWxkIGJlIGhpZGRlblxuICAgKi9cbiAgc2hvdWxkU2hvd0J1dHRvbihlcnJvck1lc3NhZ2U6IHN0cmluZyB8IG51bGwgfCB1bmRlZmluZWQpOiBib29sZWFuIHtcbiAgICBpZiAoIWVycm9yTWVzc2FnZSkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgY29uc3QgbWVzc2FnZSA9IGVycm9yTWVzc2FnZS50cmltKCk7XG5cbiAgICAvLyBMaXN0IG9mIGVycm9yIG1lc3NhZ2VzIHRoYXQgc2hvdWxkIGhpZGUgdGhlIGJ1dHRvblxuICAgIGNvbnN0IGhpZGVCdXR0b25NZXNzYWdlcyA9IFtcbiAgICAgICdEZWJlIGVudmlhcnNlIHNlc3Npb25JZCBvIHJvb21UeXBlICsgZW50aXR5SWQnLFxuICAgICAgJ05vIGV4aXN0ZSB2aWRlb2xsYW1hZGEgY29uIGVzZSBzZXNzaW9uSWQnLFxuICAgICAgJ1VzdWFyaW8gbm8gcGVydGVuZWNlIGEgbGEgbGxhbWFkYScsXG4gICAgXTtcblxuICAgIC8vIENoZWNrIGV4YWN0IG1hdGNoZXNcbiAgICBpZiAoaGlkZUJ1dHRvbk1lc3NhZ2VzLnNvbWUoKG1zZykgPT4gbWVzc2FnZSA9PT0gbXNnKSkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cblxuICAgIC8vIENoZWNrIHBhdHRlcm4gbWF0Y2ggZm9yIFwiTm8gZXhpc3RlIHZpZGVvY2FsbCB2aWdlbnRlIHBhcmEgcm9vbVR5cGU9eCB5IGVudGl0eUlkPXhcIlxuICAgIC8vIFVzaW5nIHN0YXJ0c1dpdGggYXMgcHJpbWFyeSBjaGVjayB0byBoYW5kbGUgYW55IHZhcmlhdGlvbnNcbiAgICBpZiAobWVzc2FnZS5zdGFydHNXaXRoKCdObyBleGlzdGUgdmlkZW9jYWxsIHZpZ2VudGUgcGFyYSByb29tVHlwZT0nKSkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cblxuICAgIC8vIEFsc28gY2hlY2sgcmVnZXggcGF0dGVybiBhcyBiYWNrdXBcbiAgICBjb25zdCB2aWdlbnRlUGF0dGVybiA9IC9eTm8gZXhpc3RlIHZpZGVvY2FsbCB2aWdlbnRlIHBhcmEgcm9vbVR5cGU9LisgeSBlbnRpdHlJZD1cXHMqXFxkKyQvO1xuICAgIGlmICh2aWdlbnRlUGF0dGVybi50ZXN0KG1lc3NhZ2UpKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgYXQgbGVhc3Qgb25lIE9XTkVSIGhhcyBqb2luZWQgdGhlIGNhbGwuXG4gICAqL1xuICBjaGVja0lmT3duZXJKb2luZWQoXG4gICAgdXNlcnM6IHsgdXNlcklkOiBudW1iZXI7IHJvbDogc3RyaW5nOyBzdGF0dXM6IHN0cmluZyB9W11cbiAgKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHVzZXJzLnNvbWUoKHUpID0+IHUucm9sID09PSAnT1dORVInICYmIHUuc3RhdHVzID09PSAnSk9JTkVEJyk7XG4gIH1cblxuICAvKipcbiAgICogRXh0cmFjdCBlcnJvciBtZXNzYWdlIGZyb20gdmFyaW91cyBlcnJvciBzdHJ1Y3R1cmVzLlxuICAgKi9cbiAgZXh0cmFjdEVycm9yTWVzc2FnZShlcnI6IGFueSwgZmFsbGJhY2sgPSAnRXJyb3IgZGVzY29ub2NpZG8nKTogc3RyaW5nIHtcbiAgICByZXR1cm4gZXJyPy5lcnJvcj8ubWVzc2FnZSB8fCBlcnI/Lm1lc3NhZ2UgfHwgZmFsbGJhY2s7XG4gIH1cbn1cblxuIl19