tas-uell-sdk 0.2.0 → 0.3.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 +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
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { Component } from '@angular/core';
|
|
2
2
|
import { CallState, ViewMode } from '../../interfaces/tas.interfaces';
|
|
3
3
|
import { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';
|
|
4
|
+
import { TasFeedbackModalComponent } from '../tas-feedback-modal/tas-feedback-modal.component';
|
|
4
5
|
import { Subscription } from 'rxjs';
|
|
5
6
|
import interact from 'interactjs';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
@@ -14,6 +15,7 @@ export class TasFloatingCallComponent {
|
|
|
14
15
|
this.isMuted = false;
|
|
15
16
|
this.subscriptions = new Subscription();
|
|
16
17
|
this.videoCallModalRef = null;
|
|
18
|
+
this.feedbackShown = false; // Track if feedback modal has been shown
|
|
17
19
|
// Margin from screen edges (in pixels)
|
|
18
20
|
this.PIP_MARGIN = 20;
|
|
19
21
|
}
|
|
@@ -35,12 +37,45 @@ export class TasFloatingCallComponent {
|
|
|
35
37
|
toggleMute() {
|
|
36
38
|
this.tasService.toggleMute();
|
|
37
39
|
}
|
|
40
|
+
/**
|
|
41
|
+
* Open feedback modal when call ends from PiP mode
|
|
42
|
+
*/
|
|
43
|
+
openFeedbackModal() {
|
|
44
|
+
// Prevent opening feedback modal multiple times
|
|
45
|
+
if (this.feedbackShown) {
|
|
46
|
+
this.isVisible = false;
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
// Get videoCallId from service
|
|
50
|
+
const videoCallId = this.tasService.videoCallId;
|
|
51
|
+
// If no videoCallId, skip feedback and hide directly
|
|
52
|
+
if (!videoCallId) {
|
|
53
|
+
this.isVisible = false;
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
this.feedbackShown = true;
|
|
57
|
+
const modalRef = this.modalService.open(TasFeedbackModalComponent, {
|
|
58
|
+
centered: true,
|
|
59
|
+
backdrop: true,
|
|
60
|
+
keyboard: true,
|
|
61
|
+
windowClass: 'tas-feedback-modal-wrapper',
|
|
62
|
+
});
|
|
63
|
+
modalRef.componentInstance.videoCallId = videoCallId;
|
|
64
|
+
modalRef.componentInstance.tenant = this.tasService.tenant ?? '';
|
|
65
|
+
modalRef.componentInstance.businessRole = this.tasService.businessRole;
|
|
66
|
+
// Hide floating call after feedback modal is closed/dismissed
|
|
67
|
+
modalRef.result.finally(() => {
|
|
68
|
+
this.isVisible = false;
|
|
69
|
+
this.feedbackShown = false; // Reset for potential future calls
|
|
70
|
+
});
|
|
71
|
+
}
|
|
38
72
|
// Private Methods
|
|
39
73
|
setupSubscriptions() {
|
|
40
74
|
// Call state subscription
|
|
41
75
|
this.subscriptions.add(this.tasService.callState$.subscribe((state) => {
|
|
42
|
-
if
|
|
43
|
-
|
|
76
|
+
// Only handle disconnect feedback if we're in PiP mode (isVisible)
|
|
77
|
+
if (state === CallState.DISCONNECTED && this.isVisible) {
|
|
78
|
+
this.openFeedbackModal();
|
|
44
79
|
}
|
|
45
80
|
}));
|
|
46
81
|
// View mode subscription
|
|
@@ -151,4 +186,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImpo
|
|
|
151
186
|
type: Component,
|
|
152
187
|
args: [{ selector: 'tas-floating-call', template: "<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n <!-- Video content area - shows main video only -->\n <div class=\"floating-content\">\n <!-- Main video container (subscriber if available, otherwise publisher) -->\n <div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n <!-- Bottom controls -->\n <div class=\"floating-controls\">\n <button\n class=\"action-btn expand-btn\"\n (click)=\"onExpand()\"\n title=\"Expand to fullscreen\"\n >\n <i class=\"fa fa-expand\"></i>\n </button>\n <button\n class=\"action-btn mute-btn\"\n [class.muted]=\"isMuted\"\n (click)=\"toggleMute()\"\n [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\"\n >\n <i\n class=\"fa\"\n [class.fa-microphone]=\"!isMuted\"\n [class.fa-microphone-slash]=\"isMuted\"\n ></i>\n </button>\n <button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n <i class=\"fa fa-phone\" style=\"transform: rotate(135deg)\"></i>\n </button>\n </div>\n </div>\n</div>\n", styles: [".tas-floating-container{position:fixed;bottom:20px;right:20px;width:280px;height:180px;background:#000;border-radius:12px;box-shadow:0 8px 32px #00000080;z-index:9999;overflow:hidden;touch-action:none;-webkit-user-select:none;user-select:none;transition:opacity .3s ease,visibility .3s ease,box-shadow .2s ease;opacity:0;visibility:hidden;pointer-events:none}.tas-floating-container.visible{opacity:1;visibility:visible;pointer-events:auto}.tas-floating-container:hover{box-shadow:0 8px 32px #00000080,0 0 0 2px #ffffff4d}.floating-content{position:relative;width:100%;height:100%;overflow:hidden}.pip-main-video{position:absolute;top:0;left:0;width:100%;height:100%;background:#000}.pip-main-video ::ng-deep video{width:100%;height:100%;object-fit:cover}.pip-main-video ::ng-deep .OT_subscriber,.pip-main-video ::ng-deep .OT_publisher{width:100%!important;height:100%!important}.pip-main-video ::ng-deep .OT_edge-bar-item,.pip-main-video ::ng-deep .OT_mute,.pip-main-video ::ng-deep .OT_audio-level-meter,.pip-main-video ::ng-deep .OT_bar,.pip-main-video ::ng-deep .OT_name{display:none!important}.floating-controls{position:absolute;bottom:10px;left:50%;transform:translate(-50%);display:flex;gap:12px;padding:6px 14px;background:rgba(0,0,0,.7);border-radius:24px;backdrop-filter:blur(8px);opacity:0;visibility:hidden;transition:opacity .3s ease,visibility .3s ease}.tas-floating-container:hover .floating-controls{opacity:1;visibility:visible}.action-btn{width:32px;height:32px;border:none;border-radius:50%;cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:13px;transition:all .2s ease}.action-btn.expand-btn{background:rgba(255,255,255,.2);color:#fff}.action-btn.expand-btn:hover{background:rgba(255,255,255,.35);transform:scale(1.1)}.action-btn.mute-btn{background:rgba(255,255,255,.2);color:#fff}.action-btn.mute-btn:hover{background:rgba(255,255,255,.35);transform:scale(1.1)}.action-btn.mute-btn.muted{background:#f39c12;color:#fff}.action-btn.mute-btn.muted:hover{background:#e67e22}.action-btn.hangup-btn{background:#dc3545;color:#fff}.action-btn.hangup-btn:hover{background:#c82333;transform:scale(1.1)}\n"] }]
|
|
153
188
|
}], ctorParameters: function () { return [{ type: i1.TasService }, { type: i2.NgbModal }]; } });
|
|
154
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-floating-call.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-floating-call/tas-floating-call.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-floating-call/tas-floating-call.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy } from '@angular/core';\nimport { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';\nimport { TasService } from '../../services/tas.service';\nimport { CallState, ViewMode } from '../../interfaces/tas.interfaces';\nimport { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';\nimport { Subscription } from 'rxjs';\nimport interact from 'interactjs';\n\n@Component({\n  selector: 'tas-floating-call',\n  templateUrl: './tas-floating-call.component.html',\n  styleUrls: ['./tas-floating-call.component.scss'],\n})\nexport class TasFloatingCallComponent implements OnInit, OnDestroy {\n  public isVisible = false;\n  public isMuted = false;\n\n  private subscriptions = new Subscription();\n  private videoCallModalRef: NgbModalRef | null = null;\n\n  constructor(\n    private tasService: TasService,\n    private modalService: NgbModal\n  ) {}\n\n  ngOnInit(): void {\n    this.setupSubscriptions();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    interact('.tas-floating-container').unset();\n  }\n\n  // Public Methods\n  onExpand(): void {\n    this.openVideoCallModal(true);\n    this.tasService.exitPipMode();\n  }\n\n  onHangUp(): void {\n    this.tasService.disconnectSession();\n  }\n\n  toggleMute(): void {\n    this.tasService.toggleMute();\n  }\n\n  // Private Methods\n  private setupSubscriptions(): void {\n    // Call state subscription\n    this.subscriptions.add(\n      this.tasService.callState$.subscribe((state) => {\n        if (state === CallState.DISCONNECTED) {\n          this.isVisible = false;\n        }\n      })\n    );\n\n    // View mode subscription\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        this.isVisible = mode === ViewMode.PIP && this.tasService.isCallActive();\n        if (this.isVisible) {\n          setTimeout(() => this.initInteract(), 100);\n          // Clear modal ref if we enter PiP mode (modal closes itself)\n          this.videoCallModalRef = null;\n        }\n      })\n    );\n\n    // Mute state subscription\n    this.subscriptions.add(\n      this.tasService.isMuted$.subscribe((muted) => {\n        this.isMuted = muted;\n      })\n    );\n  }\n\n  private openVideoCallModal(isReturningFromPip: boolean = false): void {\n    if (this.videoCallModalRef) {\n      return;\n    }\n\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.tasService.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  // Margin from screen edges (in pixels)\n  private readonly PIP_MARGIN = 20;\n\n  private initInteract(): void {\n    interact('.tas-floating-container').unset();\n\n    // Create restriction area with margin\n    const margin = this.PIP_MARGIN;\n    const restrictToBodyWithMargin = {\n      restriction: () => {\n        return {\n          left: margin,\n          top: margin,\n          right: window.innerWidth - margin,\n          bottom: window.innerHeight - margin,\n        };\n      },\n      elementRect: { left: 0, right: 1, top: 0, bottom: 1 },\n    };\n\n    interact('.tas-floating-container')\n      .draggable({\n        inertia: true,\n        modifiers: [interact.modifiers.restrict(restrictToBodyWithMargin)],\n        autoScroll: false,\n        listeners: {\n          move: (event) => {\n            const target = event.target;\n            const x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;\n            const y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;\n\n            target.style.transform = `translate(${x}px, ${y}px)`;\n            target.setAttribute('data-x', String(x));\n            target.setAttribute('data-y', String(y));\n          },\n        },\n      })\n      .resizable({\n        edges: { left: false, right: true, bottom: true, top: false },\n        listeners: {\n          move: (event) => {\n            const target = event.target;\n            let x = parseFloat(target.getAttribute('data-x')) || 0;\n            let y = parseFloat(target.getAttribute('data-y')) || 0;\n\n            // Update element size\n            target.style.width = `${event.rect.width}px`;\n            target.style.height = `${event.rect.height}px`;\n\n            // Translate when resizing from top or left edges\n            x += event.deltaRect.left;\n            y += event.deltaRect.top;\n\n            target.style.transform = `translate(${x}px, ${y}px)`;\n            target.setAttribute('data-x', String(x));\n            target.setAttribute('data-y', String(y));\n          },\n        },\n        modifiers: [\n          interact.modifiers.restrictEdges({\n            outer: {\n              left: margin,\n              top: margin,\n              right: window.innerWidth - margin,\n              bottom: window.innerHeight - margin,\n            },\n          }),\n          interact.modifiers.restrictSize({\n            min: { width: 200, height: 130 },\n            max: { width: 500, height: 350 },\n          }),\n          interact.modifiers.aspectRatio({ ratio: 'preserve' }),\n        ],\n        inertia: true,\n      });\n  }\n}\n","<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n  <!-- Video content area - shows main video only -->\n  <div class=\"floating-content\">\n    <!-- Main video container (subscriber if available, otherwise publisher) -->\n    <div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n    <!-- Bottom controls -->\n    <div class=\"floating-controls\">\n      <button\n        class=\"action-btn expand-btn\"\n        (click)=\"onExpand()\"\n        title=\"Expand to fullscreen\"\n      >\n        <i class=\"fa fa-expand\"></i>\n      </button>\n      <button\n        class=\"action-btn mute-btn\"\n        [class.muted]=\"isMuted\"\n        (click)=\"toggleMute()\"\n        [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\"\n      >\n        <i\n          class=\"fa\"\n          [class.fa-microphone]=\"!isMuted\"\n          [class.fa-microphone-slash]=\"isMuted\"\n        ></i>\n      </button>\n      <button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n        <i class=\"fa fa-phone\" style=\"transform: rotate(135deg)\"></i>\n      </button>\n    </div>\n  </div>\n</div>\n"]}
|
|
189
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-floating-call.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-floating-call/tas-floating-call.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-floating-call/tas-floating-call.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy } from '@angular/core';\nimport { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';\nimport { TasService } from '../../services/tas.service';\nimport { CallState, ViewMode } from '../../interfaces/tas.interfaces';\nimport { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';\nimport { TasFeedbackModalComponent } from '../tas-feedback-modal/tas-feedback-modal.component';\nimport { Subscription } from 'rxjs';\nimport interact from 'interactjs';\n\n@Component({\n  selector: 'tas-floating-call',\n  templateUrl: './tas-floating-call.component.html',\n  styleUrls: ['./tas-floating-call.component.scss'],\n})\nexport class TasFloatingCallComponent implements OnInit, OnDestroy {\n  public isVisible = false;\n  public isMuted = false;\n\n  private subscriptions = new Subscription();\n  private videoCallModalRef: NgbModalRef | null = null;\n  private feedbackShown = false; // Track if feedback modal has been shown\n\n  constructor(\n    private tasService: TasService,\n    private modalService: NgbModal\n  ) {}\n\n  ngOnInit(): void {\n    this.setupSubscriptions();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    interact('.tas-floating-container').unset();\n  }\n\n  // Public Methods\n  onExpand(): void {\n    this.openVideoCallModal(true);\n    this.tasService.exitPipMode();\n  }\n\n  onHangUp(): void {\n    this.tasService.disconnectSession();\n  }\n\n  toggleMute(): void {\n    this.tasService.toggleMute();\n  }\n\n  /**\n   * Open feedback modal when call ends from PiP mode\n   */\n  private openFeedbackModal(): void {\n    // Prevent opening feedback modal multiple times\n    if (this.feedbackShown) {\n      this.isVisible = false;\n      return;\n    }\n\n    // Get videoCallId from service\n    const videoCallId = this.tasService.videoCallId;\n\n    // If no videoCallId, skip feedback and hide directly\n    if (!videoCallId) {\n      this.isVisible = false;\n      return;\n    }\n\n    this.feedbackShown = true;\n\n    const modalRef = this.modalService.open(TasFeedbackModalComponent, {\n      centered: true,\n      backdrop: true,\n      keyboard: true,\n      windowClass: 'tas-feedback-modal-wrapper',\n    });\n\n    modalRef.componentInstance.videoCallId = videoCallId;\n    modalRef.componentInstance.tenant = this.tasService.tenant ?? '';\n    modalRef.componentInstance.businessRole = this.tasService.businessRole;\n\n    // Hide floating call after feedback modal is closed/dismissed\n    modalRef.result.finally(() => {\n      this.isVisible = false;\n      this.feedbackShown = false; // Reset for potential future calls\n    });\n  }\n\n  // Private Methods\n  private setupSubscriptions(): void {\n    // Call state subscription\n    this.subscriptions.add(\n      this.tasService.callState$.subscribe((state) => {\n        // Only handle disconnect feedback if we're in PiP mode (isVisible)\n        if (state === CallState.DISCONNECTED && this.isVisible) {\n          this.openFeedbackModal();\n        }\n      })\n    );\n\n    // View mode subscription\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        this.isVisible = mode === ViewMode.PIP && this.tasService.isCallActive();\n        if (this.isVisible) {\n          setTimeout(() => this.initInteract(), 100);\n          // Clear modal ref if we enter PiP mode (modal closes itself)\n          this.videoCallModalRef = null;\n        }\n      })\n    );\n\n    // Mute state subscription\n    this.subscriptions.add(\n      this.tasService.isMuted$.subscribe((muted) => {\n        this.isMuted = muted;\n      })\n    );\n  }\n\n  private openVideoCallModal(isReturningFromPip: boolean = false): void {\n    if (this.videoCallModalRef) {\n      return;\n    }\n\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.tasService.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  // Margin from screen edges (in pixels)\n  private readonly PIP_MARGIN = 20;\n\n  private initInteract(): void {\n    interact('.tas-floating-container').unset();\n\n    // Create restriction area with margin\n    const margin = this.PIP_MARGIN;\n    const restrictToBodyWithMargin = {\n      restriction: () => {\n        return {\n          left: margin,\n          top: margin,\n          right: window.innerWidth - margin,\n          bottom: window.innerHeight - margin,\n        };\n      },\n      elementRect: { left: 0, right: 1, top: 0, bottom: 1 },\n    };\n\n    interact('.tas-floating-container')\n      .draggable({\n        inertia: true,\n        modifiers: [interact.modifiers.restrict(restrictToBodyWithMargin)],\n        autoScroll: false,\n        listeners: {\n          move: (event) => {\n            const target = event.target;\n            const x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;\n            const y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;\n\n            target.style.transform = `translate(${x}px, ${y}px)`;\n            target.setAttribute('data-x', String(x));\n            target.setAttribute('data-y', String(y));\n          },\n        },\n      })\n      .resizable({\n        edges: { left: false, right: true, bottom: true, top: false },\n        listeners: {\n          move: (event) => {\n            const target = event.target;\n            let x = parseFloat(target.getAttribute('data-x')) || 0;\n            let y = parseFloat(target.getAttribute('data-y')) || 0;\n\n            // Update element size\n            target.style.width = `${event.rect.width}px`;\n            target.style.height = `${event.rect.height}px`;\n\n            // Translate when resizing from top or left edges\n            x += event.deltaRect.left;\n            y += event.deltaRect.top;\n\n            target.style.transform = `translate(${x}px, ${y}px)`;\n            target.setAttribute('data-x', String(x));\n            target.setAttribute('data-y', String(y));\n          },\n        },\n        modifiers: [\n          interact.modifiers.restrictEdges({\n            outer: {\n              left: margin,\n              top: margin,\n              right: window.innerWidth - margin,\n              bottom: window.innerHeight - margin,\n            },\n          }),\n          interact.modifiers.restrictSize({\n            min: { width: 200, height: 130 },\n            max: { width: 500, height: 350 },\n          }),\n          interact.modifiers.aspectRatio({ ratio: 'preserve' }),\n        ],\n        inertia: true,\n      });\n  }\n}\n","<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n  <!-- Video content area - shows main video only -->\n  <div class=\"floating-content\">\n    <!-- Main video container (subscriber if available, otherwise publisher) -->\n    <div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n    <!-- Bottom controls -->\n    <div class=\"floating-controls\">\n      <button\n        class=\"action-btn expand-btn\"\n        (click)=\"onExpand()\"\n        title=\"Expand to fullscreen\"\n      >\n        <i class=\"fa fa-expand\"></i>\n      </button>\n      <button\n        class=\"action-btn mute-btn\"\n        [class.muted]=\"isMuted\"\n        (click)=\"toggleMute()\"\n        [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\"\n      >\n        <i\n          class=\"fa\"\n          [class.fa-microphone]=\"!isMuted\"\n          [class.fa-microphone-slash]=\"isMuted\"\n        ></i>\n      </button>\n      <button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n        <i class=\"fa fa-phone\" style=\"transform: rotate(135deg)\"></i>\n      </button>\n    </div>\n  </div>\n</div>\n"]}
|
package/esm2020/lib/components/tas-incoming-appointment/tas-incoming-appointment.component.mjs
CHANGED
|
@@ -16,10 +16,13 @@ export class TasIncomingAppointmentComponent {
|
|
|
16
16
|
this.appointments = [];
|
|
17
17
|
this.isLoading = true;
|
|
18
18
|
this.hasError = false;
|
|
19
|
+
// The appointmentId from status API - only this appointment shows tas-btn
|
|
20
|
+
this.activeAppointmentId = null;
|
|
19
21
|
this.subscriptions = new Subscription();
|
|
20
22
|
}
|
|
21
23
|
ngOnInit() {
|
|
22
24
|
this.loadAppointments();
|
|
25
|
+
this.checkStatus();
|
|
23
26
|
}
|
|
24
27
|
ngOnDestroy() {
|
|
25
28
|
this.subscriptions.unsubscribe();
|
|
@@ -37,8 +40,9 @@ export class TasIncomingAppointmentComponent {
|
|
|
37
40
|
const appointments = Array.isArray(response)
|
|
38
41
|
? response
|
|
39
42
|
: response?.content || [];
|
|
40
|
-
//
|
|
41
|
-
|
|
43
|
+
// Deduplicate by id and sort by date and startTime ascending (earliest first)
|
|
44
|
+
const uniqueAppointments = appointments.filter((appt, index, self) => index === self.findIndex((a) => a.id === appt.id));
|
|
45
|
+
this.appointments = uniqueAppointments.sort((a, b) => {
|
|
42
46
|
const dateTimeA = `${a.date}T${a.startTime}`;
|
|
43
47
|
const dateTimeB = `${b.date}T${b.startTime}`;
|
|
44
48
|
return dateTimeB.localeCompare(dateTimeA);
|
|
@@ -51,15 +55,49 @@ export class TasIncomingAppointmentComponent {
|
|
|
51
55
|
},
|
|
52
56
|
}));
|
|
53
57
|
}
|
|
58
|
+
/**
|
|
59
|
+
* Check status endpoint to get the active appointmentId
|
|
60
|
+
*/
|
|
61
|
+
checkStatus() {
|
|
62
|
+
if (!this.tenant || !this.entityId) {
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
this.subscriptions.add(this.tasService
|
|
66
|
+
.getProxyVideoStatus({
|
|
67
|
+
roomType: this.roomType,
|
|
68
|
+
entityId: this.entityId,
|
|
69
|
+
tenant: this.tenant,
|
|
70
|
+
businessRole: this.businessRole,
|
|
71
|
+
})
|
|
72
|
+
.subscribe({
|
|
73
|
+
next: (response) => {
|
|
74
|
+
// Store the appointmentId from status - tas-btn only shows for this one
|
|
75
|
+
this.activeAppointmentId = response?.content?.appointmentId ?? null;
|
|
76
|
+
},
|
|
77
|
+
error: () => {
|
|
78
|
+
this.activeAppointmentId = null;
|
|
79
|
+
},
|
|
80
|
+
}));
|
|
81
|
+
}
|
|
54
82
|
onEnterCall(appointment) {
|
|
55
83
|
this.enterCall.emit(appointment);
|
|
56
84
|
}
|
|
57
85
|
/**
|
|
58
|
-
* Check if tas-btn should be shown for an appointment
|
|
86
|
+
* Check if tas-btn should be shown for an appointment.
|
|
87
|
+
* Only shows when appointment.id matches the activeAppointmentId from status API.
|
|
88
|
+
* tas-btn handles its own polling for joinable state.
|
|
59
89
|
*/
|
|
60
90
|
shouldShowTasBtn(appointment) {
|
|
61
|
-
|
|
91
|
+
const hasValidStatus = appointment.status === AppointmentStatus.CONFIRMED ||
|
|
62
92
|
appointment.status === AppointmentStatus.ACTIVE;
|
|
93
|
+
// Only show for the appointment that matches status API response
|
|
94
|
+
return hasValidStatus && appointment.id === this.activeAppointmentId;
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* TrackBy function for ngFor
|
|
98
|
+
*/
|
|
99
|
+
trackByAppointmentId(index, appointment) {
|
|
100
|
+
return appointment.id;
|
|
63
101
|
}
|
|
64
102
|
/**
|
|
65
103
|
* Format date to Spanish format: "Lunes 8 de diciembre"
|
|
@@ -91,17 +129,17 @@ export class TasIncomingAppointmentComponent {
|
|
|
91
129
|
*/
|
|
92
130
|
getOtherParticipant(appointment) {
|
|
93
131
|
if (!appointment.participants || appointment.participants.length === 0) {
|
|
94
|
-
return appointment.title;
|
|
132
|
+
return appointment.title;
|
|
95
133
|
}
|
|
96
134
|
const otherParticipant = appointment.participants.find(p => p.userId !== this.currentUser?.id);
|
|
97
135
|
return otherParticipant?.name || appointment.title;
|
|
98
136
|
}
|
|
99
137
|
}
|
|
100
138
|
TasIncomingAppointmentComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasIncomingAppointmentComponent, deps: [{ token: i1.TasService }], target: i0.ɵɵFactoryTarget.Component });
|
|
101
|
-
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", fromDate: "fromDate", toDate: "toDate" }, 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 && appointments.length === 0\">\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 <!-- Appointments list -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n <div class=\"appointment-card\" *ngFor=\"let appt of appointments\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n </div>\n <tas-btn\n *ngIf=\"shouldShowTasBtn(appt)\"\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appt.roomType\"\n [entityId]=\"appt.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n\n", styles: ["
|
|
139
|
+
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", fromDate: "fromDate", toDate: "toDate" }, 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 && appointments.length === 0\">\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 <!-- Appointments list -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n <div class=\"appointment-card\" [ngClass]=\"{ 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\" *ngFor=\"let appt of appointments; trackBy: trackByAppointmentId\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\" [ngClass]=\"{ 'compact': !shouldShowTasBtn(appt), 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\">\n <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n </div>\n <tas-btn\n *ngIf=\"shouldShowTasBtn(appt)\"\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appt.roomType\"\n [entityId]=\"appt.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n\n", styles: ["@charset \"UTF-8\";: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;gap:16px}.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}.date-time.compact{flex-direction:row;align-items:center;gap:8px}.date-time.compact .date:after{content:\"\\b7\";margin-left:8px}.date-time.cancelled{color:#8a95ab}.date-time.cancelled .date{color:#8a95ab;text-decoration:line-through;font-size:14px}.date-time.cancelled .time{color:#8a95ab;text-decoration:line-through;font-size:12px}.appointment-card.cancelled{background:#f3f4f6;border:none}.appointment-card.cancelled .doctor-name{color:#8a95ab;text-decoration:line-through}.appointment-card.cancelled tas-avatar ::ng-deep .avatar{box-shadow:none}\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"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] });
|
|
102
140
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasIncomingAppointmentComponent, decorators: [{
|
|
103
141
|
type: Component,
|
|
104
|
-
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 && appointments.length === 0\">\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 <!-- Appointments list -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n <div class=\"appointment-card\" *ngFor=\"let appt of appointments\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\">\n <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n </div>\n <tas-btn\n *ngIf=\"shouldShowTasBtn(appt)\"\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appt.roomType\"\n [entityId]=\"appt.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n\n", styles: ["
|
|
142
|
+
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 && appointments.length === 0\">\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 <!-- Appointments list -->\n <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n <div class=\"appointment-card\" [ngClass]=\"{ 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\" *ngFor=\"let appt of appointments; trackBy: trackByAppointmentId\">\n <div class=\"appointment-header\">\n <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n </div>\n <div class=\"appointment-details\">\n <div class=\"date-time\" [ngClass]=\"{ 'compact': !shouldShowTasBtn(appt), 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\">\n <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n </div>\n <tas-btn\n *ngIf=\"shouldShowTasBtn(appt)\"\n variant=\"teal\"\n buttonLabel=\"Ingresar\"\n [roomType]=\"appt.roomType\"\n [entityId]=\"appt.entityId\"\n [tenant]=\"tenant\"\n [businessRole]=\"businessRole\"\n [currentUser]=\"currentUser\"\n ></tas-btn>\n </div>\n </div>\n </div>\n</div>\n\n", styles: ["@charset \"UTF-8\";: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;gap:16px}.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}.date-time.compact{flex-direction:row;align-items:center;gap:8px}.date-time.compact .date:after{content:\"\\b7\";margin-left:8px}.date-time.cancelled{color:#8a95ab}.date-time.cancelled .date{color:#8a95ab;text-decoration:line-through;font-size:14px}.date-time.cancelled .time{color:#8a95ab;text-decoration:line-through;font-size:12px}.appointment-card.cancelled{background:#f3f4f6;border:none}.appointment-card.cancelled .doctor-name{color:#8a95ab;text-decoration:line-through}.appointment-card.cancelled tas-avatar ::ng-deep .avatar{box-shadow:none}\n"] }]
|
|
105
143
|
}], ctorParameters: function () { return [{ type: i1.TasService }]; }, propDecorators: { roomType: [{
|
|
106
144
|
type: Input
|
|
107
145
|
}], entityId: [{
|
|
@@ -119,4 +157,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImpo
|
|
|
119
157
|
}], enterCall: [{
|
|
120
158
|
type: Output
|
|
121
159
|
}] } });
|
|
122
|
-
//# 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":"AAAA,OAAO,EACL,SAAS,EAGT,KAAK,EACL,MAAM,EACN,YAAY,GACb,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAEpC,OAAO,EAEL,iBAAiB,EACjB,WAAW,EACX,eAAe,GAEhB,MAAM,iCAAiC,CAAC;;;;;;AAOzC,MAAM,OAAO,+BAA+B;IAoB1C,YAAoB,UAAsB;QAAtB,eAAU,GAAV,UAAU,CAAY;QAnB1C,iCAAiC;QACxB,aAAQ,GAAgB,WAAW,CAAC,GAAG,CAAC;QAGxC,iBAAY,GAAoB,eAAe,CAAC,IAAI,CAAC;QAOpD,cAAS,GAAG,IAAI,YAAY,EAAkB,CAAC;QAElD,iBAAY,GAAqB,EAAE,CAAC;QACpC,cAAS,GAAG,IAAI,CAAC;QACjB,aAAQ,GAAG,KAAK,CAAC;QAEhB,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;IAEE,CAAC;IAE9C,QAAQ;QACN,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;IACnC,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,UAAU;aACZ,eAAe,CAAC;YACf,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,IAAI,CAAC,QAAQ;SACxB,CAAC;aACD,SAAS,CAAC;YACT,IAAI,EAAE,CAAC,QAAa,EAAE,EAAE;gBACtB,6EAA6E;gBAC7E,MAAM,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC;oBAC1C,CAAC,CAAC,QAAQ;oBACV,CAAC,CAAC,QAAQ,EAAE,OAAO,IAAI,EAAE,CAAC;gBAE5B,4DAA4D;gBAC5D,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,CAAiB,EAAE,CAAiB,EAAE,EAAE;oBAC7E,MAAM,SAAS,GAAG,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,SAAS,EAAE,CAAC;oBAC7C,MAAM,SAAS,GAAG,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,SAAS,EAAE,CAAC;oBAC7C,OAAO,SAAS,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;gBAC5C,CAAC,CAAC,CAAC;gBACH,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACzB,CAAC;YACD,KAAK,EAAE,GAAG,EAAE;gBACV,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;gBACrB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACzB,CAAC;SACF,CAAC,CACL,CAAC;IACJ,CAAC;IAEM,WAAW,CAAC,WAA2B;QAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IACnC,CAAC;IAED;;OAEG;IACI,gBAAgB,CAAC,WAA2B;QACjD,OAAO,WAAW,CAAC,MAAM,KAAK,iBAAiB,CAAC,SAAS;YAClD,WAAW,CAAC,MAAM,KAAK,iBAAiB,CAAC,MAAM,CAAC;IACzD,CAAC;IAED;;OAEG;IACI,qBAAqB,CAAC,WAA2B;QACtD,MAAM,CAAC,IAAI,EAAE,KAAK,EAAE,GAAG,CAAC,GAAG,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACnE,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC;QAE5C,MAAM,QAAQ,GAAG;YACf,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,WAAW;YACzC,QAAQ,EAAE,SAAS,EAAE,QAAQ;SAC9B,CAAC;QACF,MAAM,UAAU,GAAG;YACjB,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,OAAO;YACrD,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,SAAS,EAAE,WAAW,EAAE,WAAW;SACrE,CAAC;QAEF,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QAC9B,MAAM,SAAS,GAAG,UAAU,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;QAE9C,OAAO,GAAG,OAAO,IAAI,MAAM,OAAO,SAAS,EAAE,CAAC;IAChD,CAAC;IAED;;OAEG;IACI,eAAe,CAAC,WAA2B;QAChD,OAAO,GAAG,WAAW,CAAC,SAAS,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;IAC7D,CAAC;IAED;;OAEG;IACI,mBAAmB,CAAC,WAA2B;QACpD,IAAI,CAAC,WAAW,CAAC,YAAY,IAAI,WAAW,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;YACtE,OAAO,WAAW,CAAC,KAAK,CAAC,CAAC,uCAAuC;SAClE;QAED,MAAM,gBAAgB,GAAG,WAAW,CAAC,YAAY,CAAC,IAAI,CACpD,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,WAAW,EAAE,EAAE,CACvC,CAAC;QAEF,OAAO,gBAAgB,EAAE,IAAI,IAAI,WAAW,CAAC,KAAK,CAAC;IACrD,CAAC;;6HApHU,+BAA+B;iHAA/B,+BAA+B,iRCvB5C,s9DAoDA;4FD7Ba,+BAA+B;kBAL3C,SAAS;+BACE,0BAA0B;iGAM3B,QAAQ;sBAAhB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBAEI,SAAS;sBAAlB,MAAM","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  // Date range inputs for fetching appointments\n  @Input() fromDate!: string; // YYYY-MM-DD format\n  @Input() toDate!: string;   // YYYY-MM-DD format\n\n  @Output() enterCall = new EventEmitter<TasAppointment>();\n\n  public appointments: TasAppointment[] = [];\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    this.subscriptions.add(\n      this.tasService\n        .getAppointments({ \n          fromDate: this.fromDate, \n          toDate: this.toDate, \n          entityId: this.entityId \n        })\n        .subscribe({\n          next: (response: any) => {\n            // Handle both array response and wrapped response (e.g., { content: [...] })\n            const appointments = Array.isArray(response)\n              ? response\n              : response?.content || [];\n            \n            // Sort by date and startTime descending (most recent first)\n            this.appointments = appointments.sort((a: TasAppointment, b: TasAppointment) => {\n              const dateTimeA = `${a.date}T${a.startTime}`;\n              const dateTimeB = `${b.date}T${b.startTime}`;\n              return dateTimeB.localeCompare(dateTimeA);\n            });\n            this.isLoading = false;\n          },\n          error: () => {\n            this.hasError = true;\n            this.isLoading = false;\n          },\n        })\n    );\n  }\n\n  public onEnterCall(appointment: TasAppointment): void {\n    this.enterCall.emit(appointment);\n  }\n\n  /**\n   * Check if tas-btn should be shown for an appointment (CONFIRMED or ACTIVE status)\n   */\n  public shouldShowTasBtn(appointment: TasAppointment): boolean {\n    return appointment.status === AppointmentStatus.CONFIRMED || \n           appointment.status === AppointmentStatus.ACTIVE;\n  }\n\n  /**\n   * Format date to Spanish format: \"Lunes 8 de diciembre\"\n   */\n  public formatAppointmentDate(appointment: TasAppointment): string {\n    const [year, month, day] = 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 formatTimeRange(appointment: TasAppointment): string {\n    return `${appointment.startTime} - ${appointment.endTime}`;\n  }\n\n  /**\n   * Get the other participant in the call (not the current user)\n   */\n  public getOtherParticipant(appointment: TasAppointment): string {\n    if (!appointment.participants || appointment.participants.length === 0) {\n      return appointment.title; // Fallback to title if no participants\n    }\n    \n    const otherParticipant = appointment.participants.find(\n      p => p.userId !== this.currentUser?.id\n    );\n    \n    return otherParticipant?.name || appointment.title;\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 && appointments.length === 0\">\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  <!-- Appointments list -->\n  <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n    <div class=\"appointment-card\" *ngFor=\"let appt of appointments\">\n      <div class=\"appointment-header\">\n        <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n        <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n      </div>\n      <div class=\"appointment-details\">\n        <div class=\"date-time\">\n          <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n          <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n        </div>\n        <tas-btn\n          *ngIf=\"shouldShowTasBtn(appt)\"\n          variant=\"teal\"\n          buttonLabel=\"Ingresar\"\n          [roomType]=\"appt.roomType\"\n          [entityId]=\"appt.entityId\"\n          [tenant]=\"tenant\"\n          [businessRole]=\"businessRole\"\n          [currentUser]=\"currentUser\"\n        ></tas-btn>\n      </div>\n    </div>\n  </div>\n</div>\n\n"]}
|
|
160
|
+
//# 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  // Date range inputs for fetching appointments\n  @Input() fromDate!: string; // YYYY-MM-DD format\n  @Input() toDate!: string;   // YYYY-MM-DD format\n\n  @Output() enterCall = new EventEmitter<TasAppointment>();\n\n  public appointments: TasAppointment[] = [];\n  public isLoading = true;\n  public hasError = false;\n\n  // The appointmentId from status API - only this appointment shows tas-btn\n  public activeAppointmentId: number | null = null;\n\n  private subscriptions = new Subscription();\n\n  constructor(private tasService: TasService) {}\n\n  ngOnInit(): void {\n    this.loadAppointments();\n    this.checkStatus();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  private loadAppointments(): void {\n    this.subscriptions.add(\n      this.tasService\n        .getAppointments({ \n          fromDate: this.fromDate, \n          toDate: this.toDate, \n          entityId: this.entityId \n        })\n        .subscribe({\n          next: (response: any) => {\n            // Handle both array response and wrapped response (e.g., { content: [...] })\n            const appointments = Array.isArray(response)\n              ? response\n              : response?.content || [];\n            \n            // Deduplicate by id and sort by date and startTime ascending (earliest first)\n            const uniqueAppointments = appointments.filter(\n              (appt: TasAppointment, index: number, self: TasAppointment[]) =>\n                index === self.findIndex((a: TasAppointment) => a.id === appt.id)\n            );\n            this.appointments = uniqueAppointments.sort((a: TasAppointment, b: TasAppointment) => {\n              const dateTimeA = `${a.date}T${a.startTime}`;\n              const dateTimeB = `${b.date}T${b.startTime}`;\n              return dateTimeB.localeCompare(dateTimeA);\n            });\n            this.isLoading = false;\n          },\n          error: () => {\n            this.hasError = true;\n            this.isLoading = false;\n          },\n        })\n    );\n  }\n\n  /**\n   * Check status endpoint to get the active appointmentId\n   */\n  private checkStatus(): void {\n    if (!this.tenant || !this.entityId) {\n      return;\n    }\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            // Store the appointmentId from status - tas-btn only shows for this one\n            this.activeAppointmentId = response?.content?.appointmentId ?? null;\n          },\n          error: () => {\n            this.activeAppointmentId = null;\n          },\n        })\n    );\n  }\n\n  public onEnterCall(appointment: TasAppointment): void {\n    this.enterCall.emit(appointment);\n  }\n\n  /**\n   * Check if tas-btn should be shown for an appointment.\n   * Only shows when appointment.id matches the activeAppointmentId from status API.\n   * tas-btn handles its own polling for joinable state.\n   */\n  public shouldShowTasBtn(appointment: TasAppointment): boolean {\n    const hasValidStatus = appointment.status === AppointmentStatus.CONFIRMED ||\n                          appointment.status === AppointmentStatus.ACTIVE;\n    \n    // Only show for the appointment that matches status API response\n    return hasValidStatus && appointment.id === this.activeAppointmentId;\n  }\n\n  /**\n   * TrackBy function for ngFor\n   */\n  public trackByAppointmentId(index: number, appointment: TasAppointment): number {\n    return appointment.id;\n  }\n\n  /**\n   * Format date to Spanish format: \"Lunes 8 de diciembre\"\n   */\n  public formatAppointmentDate(appointment: TasAppointment): string {\n    const [year, month, day] = 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 formatTimeRange(appointment: TasAppointment): string {\n    return `${appointment.startTime} - ${appointment.endTime}`;\n  }\n\n  /**\n   * Get the other participant in the call (not the current user)\n   */\n  public getOtherParticipant(appointment: TasAppointment): string {\n    if (!appointment.participants || appointment.participants.length === 0) {\n      return appointment.title;\n    }\n    \n    const otherParticipant = appointment.participants.find(\n      p => p.userId !== this.currentUser?.id\n    );\n    \n    return otherParticipant?.name || appointment.title;\n  }\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 && appointments.length === 0\">\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  <!-- Appointments list -->\n  <div class=\"card-content appointment-state\" *ngIf=\"!isLoading && appointments.length > 0\">\n    <div class=\"appointment-card\" [ngClass]=\"{ 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\" *ngFor=\"let appt of appointments; trackBy: trackByAppointmentId\">\n      <div class=\"appointment-header\">\n        <tas-avatar [name]=\"getOtherParticipant(appt)\" [size]=\"48\"></tas-avatar>\n        <span class=\"doctor-name\">{{ getOtherParticipant(appt) }}</span>\n      </div>\n      <div class=\"appointment-details\">\n        <div class=\"date-time\" [ngClass]=\"{ 'compact': !shouldShowTasBtn(appt), 'cancelled': appt.status === 'CANCELLED' || appt.status === 'RESCHEDULED' }\">\n          <span class=\"date\">{{ formatAppointmentDate(appt) }}</span>\n          <span class=\"time\">{{ formatTimeRange(appt) }}</span>\n        </div>\n        <tas-btn\n          *ngIf=\"shouldShowTasBtn(appt)\"\n          variant=\"teal\"\n          buttonLabel=\"Ingresar\"\n          [roomType]=\"appt.roomType\"\n          [entityId]=\"appt.entityId\"\n          [tenant]=\"tenant\"\n          [businessRole]=\"businessRole\"\n          [currentUser]=\"currentUser\"\n        ></tas-btn>\n      </div>\n    </div>\n  </div>\n</div>\n\n"]}
|