tas-uell-sdk 0.0.1 → 0.0.3
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/INSTALL_AND_TEST.md +427 -0
- package/README.md +122 -229
- package/esm2020/lib/components/tas-btn/tas-btn.component.mjs +79 -95
- package/esm2020/lib/components/tas-floating-call/tas-floating-call.component.mjs +74 -24
- package/esm2020/lib/components/tas-videocall/tas-videocall.component.mjs +50 -45
- package/esm2020/lib/components/tas-waiting-room/tas-waiting-room.component.mjs +179 -0
- package/esm2020/lib/config/tas.config.mjs +10 -0
- package/esm2020/lib/interfaces/tas.interfaces.mjs +32 -2
- package/esm2020/lib/services/tas.service.mjs +27 -48
- package/esm2020/lib/tas-uell-sdk.module.mjs +82 -0
- package/esm2020/public-api.mjs +13 -12
- package/esm2020/tas-uell-sdk.mjs +1 -1
- package/fesm2015/tas-uell-sdk.mjs +440 -218
- package/fesm2015/tas-uell-sdk.mjs.map +1 -1
- package/fesm2020/tas-uell-sdk.mjs +437 -216
- package/fesm2020/tas-uell-sdk.mjs.map +1 -1
- package/lib/components/tas-btn/tas-btn.component.d.ts +15 -11
- package/lib/components/tas-floating-call/tas-floating-call.component.d.ts +3 -2
- package/lib/components/tas-videocall/tas-videocall.component.d.ts +4 -3
- package/lib/components/tas-waiting-room/tas-waiting-room.component.d.ts +56 -0
- package/lib/config/tas.config.d.ts +28 -0
- package/lib/interfaces/tas.interfaces.d.ts +41 -6
- package/lib/services/tas.service.d.ts +6 -16
- package/lib/tas-uell-sdk.module.d.ts +46 -0
- package/package.json +36 -23
- package/public-api.d.ts +8 -7
- package/src/lib/styles/tas-global.scss +37 -0
- package/esm2020/lib/tas.config.mjs +0 -14
- package/esm2020/lib/tas.module.mjs +0 -73
- package/lib/tas.config.d.ts +0 -51
- package/lib/tas.module.d.ts +0 -34
|
@@ -1,123 +1,107 @@
|
|
|
1
|
-
import { Component,
|
|
2
|
-
import { ViewMode } from "../../services/tas.service";
|
|
3
|
-
import { TAS_USER_DATA_PROVIDER, } from "../../tas.config";
|
|
4
|
-
import { TasVideocallComponent } from "../tas-videocall/tas-videocall.component";
|
|
1
|
+
import { Component, Input } from "@angular/core";
|
|
5
2
|
import { Subscription } from "rxjs";
|
|
6
|
-
import {
|
|
3
|
+
import { ViewMode } from "../../interfaces/tas.interfaces";
|
|
4
|
+
import { TasWaitingRoomComponent } from "../tas-waiting-room/tas-waiting-room.component";
|
|
5
|
+
import { TasVideocallComponent } from "../tas-videocall/tas-videocall.component";
|
|
7
6
|
import * as i0 from "@angular/core";
|
|
8
|
-
import * as i1 from "
|
|
9
|
-
import * as i2 from "
|
|
7
|
+
import * as i1 from "@ng-bootstrap/ng-bootstrap";
|
|
8
|
+
import * as i2 from "../../services/tas.service";
|
|
10
9
|
import * as i3 from "@angular/common";
|
|
11
10
|
export class TasButtonComponent {
|
|
12
|
-
constructor(
|
|
13
|
-
this.tasService = tasService;
|
|
11
|
+
constructor(modalService, tasService) {
|
|
14
12
|
this.modalService = modalService;
|
|
15
|
-
this.
|
|
13
|
+
this.tasService = tasService;
|
|
14
|
+
this.appointmentId = 1;
|
|
15
|
+
this.product = "uell";
|
|
16
|
+
this.tenantId = "";
|
|
17
|
+
this.regularUserIds = [];
|
|
18
|
+
this.moderatorUserIds = [];
|
|
16
19
|
this.isLoading = false;
|
|
17
|
-
this.userData = null;
|
|
18
|
-
this.tenantId = null;
|
|
19
20
|
this.subscriptions = new Subscription();
|
|
20
21
|
this.currentModalRef = null;
|
|
22
|
+
this.videoCallModalRef = null;
|
|
21
23
|
}
|
|
22
24
|
ngOnInit() {
|
|
23
|
-
this.
|
|
24
|
-
|
|
25
|
-
}
|
|
26
|
-
ngOnDestroy() {
|
|
27
|
-
this.subscriptions.unsubscribe();
|
|
28
|
-
}
|
|
29
|
-
onClick() {
|
|
30
|
-
if (!this.userData || !this.tenantId) {
|
|
31
|
-
console.error("User data or tenant ID not available");
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
this.isLoading = true;
|
|
35
|
-
this.subscriptions.add(this.tasService
|
|
36
|
-
.createRoom({
|
|
37
|
-
tenant: this.tenantId,
|
|
38
|
-
userId: this.userData.id.toString(),
|
|
39
|
-
product: "Uell",
|
|
40
|
-
record: true,
|
|
41
|
-
roomType: "TAS",
|
|
42
|
-
type: "SPONTANEOUS",
|
|
43
|
-
})
|
|
44
|
-
.pipe(switchMap((response) => {
|
|
45
|
-
const sessionId = response.content.sessionId;
|
|
46
|
-
return this.tasService
|
|
47
|
-
.generateToken({
|
|
48
|
-
sessionId: sessionId,
|
|
49
|
-
name: this.userData.name,
|
|
50
|
-
lastname: this.userData.surname,
|
|
51
|
-
})
|
|
52
|
-
.pipe(switchMap((tokenResponse) => {
|
|
53
|
-
return [
|
|
54
|
-
{
|
|
55
|
-
sessionId,
|
|
56
|
-
token: tokenResponse.content.token,
|
|
57
|
-
},
|
|
58
|
-
];
|
|
59
|
-
}));
|
|
60
|
-
}))
|
|
61
|
-
.subscribe({
|
|
62
|
-
next: ({ sessionId, token }) => {
|
|
63
|
-
this.isLoading = false;
|
|
64
|
-
this.openVideoCallModal(sessionId, token);
|
|
65
|
-
},
|
|
66
|
-
error: (err) => {
|
|
67
|
-
console.error("Error starting video call:", err);
|
|
68
|
-
this.isLoading = false;
|
|
69
|
-
},
|
|
70
|
-
}));
|
|
71
|
-
}
|
|
72
|
-
// Private Methods
|
|
73
|
-
loadUserData() {
|
|
74
|
-
if (!this.userDataProvider) {
|
|
75
|
-
console.warn("TasButtonComponent: UserDataProvider not configured");
|
|
76
|
-
return;
|
|
25
|
+
if (!this.ownerUserIds || this.ownerUserIds.length !== 1) {
|
|
26
|
+
throw new Error('tas-btn: ownerUserIds input is required and must contain exactly one user');
|
|
77
27
|
}
|
|
78
|
-
|
|
79
|
-
this.
|
|
80
|
-
|
|
81
|
-
setupViewModeSubscription() {
|
|
82
|
-
this.subscriptions.add(this.tasService.viewMode$.subscribe((mode) => {
|
|
28
|
+
// Subscribe to viewMode to handle PiP return
|
|
29
|
+
this.subscriptions.add(this.tasService.viewMode$.subscribe(mode => {
|
|
30
|
+
// Reopen video call modal when returning from PiP
|
|
83
31
|
if (mode === ViewMode.FULLSCREEN &&
|
|
84
32
|
this.tasService.isCallActive() &&
|
|
85
|
-
!this.
|
|
33
|
+
!this.videoCallModalRef) {
|
|
86
34
|
const sessionId = this.tasService.sessionId;
|
|
87
35
|
const token = this.tasService.token;
|
|
88
36
|
if (sessionId && token) {
|
|
89
|
-
this.openVideoCallModal(
|
|
37
|
+
this.openVideoCallModal(true);
|
|
90
38
|
}
|
|
91
39
|
}
|
|
40
|
+
// When entering PiP, clear the videoCallModalRef since modal will close
|
|
41
|
+
if (mode === ViewMode.PIP) {
|
|
42
|
+
this.videoCallModalRef = null;
|
|
43
|
+
}
|
|
92
44
|
}));
|
|
93
45
|
}
|
|
94
|
-
|
|
95
|
-
this.
|
|
96
|
-
|
|
97
|
-
|
|
46
|
+
ngOnDestroy() {
|
|
47
|
+
this.subscriptions.unsubscribe();
|
|
48
|
+
}
|
|
49
|
+
onClick() {
|
|
50
|
+
if (!this.tenantId || !this.currentUser?.name) {
|
|
51
|
+
console.error("Tenant ID or current user not available");
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
this.openWaitingRoomModal();
|
|
55
|
+
}
|
|
56
|
+
openWaitingRoomModal() {
|
|
57
|
+
this.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {
|
|
58
|
+
size: "lg",
|
|
59
|
+
windowClass: "tas-waiting-room-modal",
|
|
98
60
|
backdrop: "static",
|
|
99
61
|
keyboard: false,
|
|
62
|
+
centered: true
|
|
100
63
|
});
|
|
101
|
-
|
|
102
|
-
this.currentModalRef.componentInstance.
|
|
103
|
-
this.currentModalRef.componentInstance.
|
|
104
|
-
|
|
105
|
-
this.currentModalRef.
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
64
|
+
// Pass all necessary inputs to the waiting room component
|
|
65
|
+
this.currentModalRef.componentInstance.appointmentId = this.appointmentId;
|
|
66
|
+
this.currentModalRef.componentInstance.product = this.product;
|
|
67
|
+
this.currentModalRef.componentInstance.tenantId = this.tenantId;
|
|
68
|
+
this.currentModalRef.componentInstance.currentUser = this.currentUser;
|
|
69
|
+
this.currentModalRef.componentInstance.ownerUserIds = this.ownerUserIds;
|
|
70
|
+
this.currentModalRef.componentInstance.regularUserIds = this.regularUserIds;
|
|
71
|
+
this.currentModalRef.componentInstance.moderatorUserIds = this.moderatorUserIds;
|
|
72
|
+
this.currentModalRef.result.then(() => { this.currentModalRef = null; }, () => { this.currentModalRef = null; });
|
|
73
|
+
}
|
|
74
|
+
openVideoCallModal(isReturningFromPip = false) {
|
|
75
|
+
this.videoCallModalRef = this.modalService.open(TasVideocallComponent, {
|
|
76
|
+
size: 'xl',
|
|
77
|
+
windowClass: 'tas-video-modal',
|
|
78
|
+
backdrop: 'static',
|
|
79
|
+
keyboard: false
|
|
109
80
|
});
|
|
81
|
+
this.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;
|
|
82
|
+
this.videoCallModalRef.componentInstance.token = this.tasService.token;
|
|
83
|
+
this.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;
|
|
84
|
+
this.videoCallModalRef.result.then(() => { this.videoCallModalRef = null; }, () => { this.videoCallModalRef = null; });
|
|
110
85
|
}
|
|
111
86
|
}
|
|
112
|
-
TasButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasButtonComponent, deps: [{ token: i1.
|
|
113
|
-
TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TasButtonComponent, selector: "tas-btn", ngImport: i0, template: "<button\n
|
|
87
|
+
TasButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasButtonComponent, deps: [{ token: i1.NgbModal }, { token: i2.TasService }], target: i0.ɵɵFactoryTarget.Component });
|
|
88
|
+
TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TasButtonComponent, selector: "tas-btn", inputs: { appointmentId: "appointmentId", product: "product", tenantId: "tenantId", currentUser: "currentUser", ownerUserIds: "ownerUserIds", regularUserIds: "regularUserIds", moderatorUserIds: "moderatorUserIds" }, ngImport: i0, template: "<button\n\ttype=\"button\"\n\tclass=\"btn btn-primary tas-btn\"\n\t(click)=\"onClick()\"\n\t[disabled]=\"isLoading\"\n>\n\t<i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n\t<span *ngIf=\"!isLoading\"> Iniciar TAS</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\n\n", styles: [":host{display:inline-block}.tas-btn{background-color:#ee316b!important;color:#fff!important;border-color:#ee316b!important;margin-right:24px}.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}\n"], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
114
89
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasButtonComponent, decorators: [{
|
|
115
90
|
type: Component,
|
|
116
|
-
args: [{ selector: "tas-btn", template: "<button\n
|
|
117
|
-
}], ctorParameters: function () { return [{ type: i1.
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-btn.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/tas-btn/tas-btn.component.ts","../../../../../src/lib/components/tas-btn/tas-btn.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Inject, Optional } from \"@angular/core\";\nimport { TasService, ViewMode } from \"../../services/tas.service\";\nimport {\n  TAS_USER_DATA_PROVIDER,\n  TasUserDataProvider,\n  TasUserData,\n} from \"../../tas.config\";\nimport { NgbModal, NgbModalRef } from \"@ng-bootstrap/ng-bootstrap\";\nimport { TasVideocallComponent } from \"../tas-videocall/tas-videocall.component\";\nimport { Subscription } from \"rxjs\";\nimport { switchMap } from \"rxjs/operators\";\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  isLoading = false;\n\n  private userData: TasUserData | null = null;\n  private tenantId: string | null = null;\n  private subscriptions = new Subscription();\n  private currentModalRef: NgbModalRef | null = null;\n\n  constructor(\n    private tasService: TasService,\n    private modalService: NgbModal,\n    @Optional()\n    @Inject(TAS_USER_DATA_PROVIDER)\n    private userDataProvider: TasUserDataProvider | null\n  ) {}\n\n  ngOnInit(): void {\n    this.loadUserData();\n    this.setupViewModeSubscription();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  onClick(): void {\n    if (!this.userData || !this.tenantId) {\n      console.error(\"User data or tenant ID not available\");\n      return;\n    }\n\n    this.isLoading = true;\n\n    this.subscriptions.add(\n      this.tasService\n        .createRoom({\n          tenant: this.tenantId,\n          userId: this.userData.id.toString(),\n          product: \"Uell\",\n          record: true,\n          roomType: \"TAS\",\n          type: \"SPONTANEOUS\",\n        })\n        .pipe(\n          switchMap((response) => {\n            const sessionId = response.content.sessionId;\n            return this.tasService\n              .generateToken({\n                sessionId: sessionId,\n                name: this.userData!.name,\n                lastname: this.userData!.surname,\n              })\n              .pipe(\n                switchMap((tokenResponse) => {\n                  return [\n                    {\n                      sessionId,\n                      token: tokenResponse.content.token,\n                    },\n                  ];\n                })\n              );\n          })\n        )\n        .subscribe({\n          next: ({ sessionId, token }) => {\n            this.isLoading = false;\n            this.openVideoCallModal(sessionId, token);\n          },\n          error: (err) => {\n            console.error(\"Error starting video call:\", err);\n            this.isLoading = false;\n          },\n        })\n    );\n  }\n\n  // Private Methods\n  private loadUserData(): void {\n    if (!this.userDataProvider) {\n      console.warn(\"TasButtonComponent: UserDataProvider not configured\");\n      return;\n    }\n    this.userData = this.userDataProvider.getUserData();\n    this.tenantId = this.userDataProvider.getTenantId();\n  }\n\n  private setupViewModeSubscription(): void {\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        if (\n          mode === ViewMode.FULLSCREEN &&\n          this.tasService.isCallActive() &&\n          !this.currentModalRef\n        ) {\n          const sessionId = this.tasService.sessionId;\n          const token = this.tasService.token;\n          if (sessionId && token) {\n            this.openVideoCallModal(sessionId, token, true);\n          }\n        }\n      })\n    );\n  }\n\n  private openVideoCallModal(\n    sessionId: string,\n    token: string,\n    isReturningFromPip: boolean = false\n  ): void {\n    this.currentModalRef = this.modalService.open(TasVideocallComponent, {\n      size: \"xl\",\n      windowClass: \"tas-video-modal\",\n      backdrop: \"static\",\n      keyboard: false,\n    });\n\n    this.currentModalRef.componentInstance.sessionId = sessionId;\n    this.currentModalRef.componentInstance.token = token;\n    this.currentModalRef.componentInstance.isReturningFromPip =\n      isReturningFromPip;\n\n    this.currentModalRef.result.then(\n      () => {\n        this.currentModalRef = null;\n      },\n      () => {\n        this.currentModalRef = null;\n      }\n    );\n  }\n}\n","<button\n  type=\"button\"\n  class=\"btn btn-primary boton\"\n  (click)=\"onClick()\"\n  [disabled]=\"isLoading\"\n>\n  <i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n  <span *ngIf=\"!isLoading\"> Iniciar TAS</span>\n  <span *ngIf=\"isLoading\"> Processing...</span>\n</button>\n"]}
|
|
91
|
+
args: [{ selector: "tas-btn", template: "<button\n\ttype=\"button\"\n\tclass=\"btn btn-primary tas-btn\"\n\t(click)=\"onClick()\"\n\t[disabled]=\"isLoading\"\n>\n\t<i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n\t<span *ngIf=\"!isLoading\"> Iniciar TAS</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\n\n", styles: [":host{display:inline-block}.tas-btn{background-color:#ee316b!important;color:#fff!important;border-color:#ee316b!important;margin-right:24px}.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}\n"] }]
|
|
92
|
+
}], ctorParameters: function () { return [{ type: i1.NgbModal }, { type: i2.TasService }]; }, propDecorators: { appointmentId: [{
|
|
93
|
+
type: Input
|
|
94
|
+
}], product: [{
|
|
95
|
+
type: Input
|
|
96
|
+
}], tenantId: [{
|
|
97
|
+
type: Input
|
|
98
|
+
}], currentUser: [{
|
|
99
|
+
type: Input
|
|
100
|
+
}], ownerUserIds: [{
|
|
101
|
+
type: Input
|
|
102
|
+
}], regularUserIds: [{
|
|
103
|
+
type: Input
|
|
104
|
+
}], moderatorUserIds: [{
|
|
105
|
+
type: Input
|
|
106
|
+
}] } });
|
|
107
|
+
//# 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 { TasCurrentUser, ViewMode } 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\tselector: \"tas-btn\",\n\ttemplateUrl: \"./tas-btn.component.html\",\n\tstyleUrls: [\"./tas-btn.component.scss\"],\n})\nexport class TasButtonComponent implements OnInit, OnDestroy {\n\t@Input() appointmentId: number = 1;\n\t@Input() product: string = \"uell\";\n\t@Input() tenantId: string = \"\";\n\t@Input() currentUser!: TasCurrentUser;\n\t@Input() ownerUserIds!: number[];\n\t@Input() regularUserIds: number[] = [];\n\t@Input() moderatorUserIds: number[] = [];\n\t\n\tpublic isLoading = false;\n\t\n\tprivate subscriptions = new Subscription();\n\tprivate currentModalRef: NgbModalRef | null = null;\n\tprivate videoCallModalRef: NgbModalRef | null = null;\n\n\tconstructor(\n\t\tprivate modalService: NgbModal,\n\t\tprivate tasService: TasService\n\t) {}\n\n\tngOnInit(): void {\n\t\tif (!this.ownerUserIds || this.ownerUserIds.length !== 1) {\n\t\t\tthrow new Error('tas-btn: ownerUserIds input is required and must contain exactly one user');\n\t\t}\n\t\t\n\t\t// Subscribe to viewMode to handle PiP return\n\t\tthis.subscriptions.add(\n\t\t\tthis.tasService.viewMode$.subscribe(mode => {\n\t\t\t\t// Reopen video call modal when returning from PiP\n\t\t\t\tif (mode === ViewMode.FULLSCREEN && \n\t\t\t\t\tthis.tasService.isCallActive() && \n\t\t\t\t\t!this.videoCallModalRef) {\n\t\t\t\t\tconst sessionId = this.tasService.sessionId;\n\t\t\t\t\tconst token = this.tasService.token;\n\t\t\t\t\tif (sessionId && token) {\n\t\t\t\t\t\tthis.openVideoCallModal(true);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\t// When entering PiP, clear the videoCallModalRef since modal will close\n\t\t\t\tif (mode === ViewMode.PIP) {\n\t\t\t\t\tthis.videoCallModalRef = null;\n\t\t\t\t}\n\t\t\t})\n\t\t);\n\t}\n\n\tngOnDestroy(): void {\n\t\tthis.subscriptions.unsubscribe();\n\t}\n\n\tonClick(): void {\n\t\tif (!this.tenantId || !this.currentUser?.name) {\n\t\t\tconsole.error(\"Tenant ID or current user not available\");\n\t\t\treturn;\n\t\t}\n\n\t\tthis.openWaitingRoomModal();\n\t}\n\n\tprivate openWaitingRoomModal(): void {\n\t\tthis.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {\n\t\t\tsize: \"lg\",\n\t\t\twindowClass: \"tas-waiting-room-modal\",\n\t\t\tbackdrop: \"static\",\n\t\t\tkeyboard: false,\n\t\t\tcentered: true\n\t\t});\n\n\t\t// Pass all necessary inputs to the waiting room component\n\t\tthis.currentModalRef.componentInstance.appointmentId = this.appointmentId;\n\t\tthis.currentModalRef.componentInstance.product = this.product;\n\t\tthis.currentModalRef.componentInstance.tenantId = this.tenantId;\n\t\tthis.currentModalRef.componentInstance.currentUser = this.currentUser;\n\t\tthis.currentModalRef.componentInstance.ownerUserIds = this.ownerUserIds;\n\t\tthis.currentModalRef.componentInstance.regularUserIds = this.regularUserIds;\n\t\tthis.currentModalRef.componentInstance.moderatorUserIds = this.moderatorUserIds;\n\n\t\tthis.currentModalRef.result.then(\n\t\t\t() => { this.currentModalRef = null; },\n\t\t\t() => { this.currentModalRef = null; }\n\t\t);\n\t}\n\n\tprivate openVideoCallModal(isReturningFromPip: boolean = false): void {\n\t\tthis.videoCallModalRef = this.modalService.open(TasVideocallComponent, {\n\t\t\tsize: 'xl',\n\t\t\twindowClass: 'tas-video-modal',\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false\n\t\t});\n\n\t\tthis.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;\n\t\tthis.videoCallModalRef.componentInstance.token = this.tasService.token;\n\t\tthis.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;\n\n\t\tthis.videoCallModalRef.result.then(\n\t\t\t() => { this.videoCallModalRef = null; },\n\t\t\t() => { this.videoCallModalRef = null; }\n\t\t);\n\t}\n}\n\n","<button\n\ttype=\"button\"\n\tclass=\"btn btn-primary tas-btn\"\n\t(click)=\"onClick()\"\n\t[disabled]=\"isLoading\"\n>\n\t<i class=\"fa fa-video-camera\" aria-hidden=\"true\" *ngIf=\"!isLoading\"></i>\n\t<span *ngIf=\"!isLoading\"> Iniciar TAS</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\n\n"]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import { Component } from
|
|
2
|
-
import { CallState, ViewMode } from
|
|
3
|
-
import { Subscription } from
|
|
4
|
-
import interact from
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { CallState, ViewMode } from '../../interfaces/tas.interfaces';
|
|
3
|
+
import { Subscription } from 'rxjs';
|
|
4
|
+
import interact from 'interactjs';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "../../services/tas.service";
|
|
7
7
|
export class TasFloatingCallComponent {
|
|
@@ -10,13 +10,15 @@ export class TasFloatingCallComponent {
|
|
|
10
10
|
this.isVisible = false;
|
|
11
11
|
this.isMuted = false;
|
|
12
12
|
this.subscriptions = new Subscription();
|
|
13
|
+
// Margin from screen edges (in pixels)
|
|
14
|
+
this.PIP_MARGIN = 20;
|
|
13
15
|
}
|
|
14
16
|
ngOnInit() {
|
|
15
17
|
this.setupSubscriptions();
|
|
16
18
|
}
|
|
17
19
|
ngOnDestroy() {
|
|
18
20
|
this.subscriptions.unsubscribe();
|
|
19
|
-
interact(
|
|
21
|
+
interact('.tas-floating-container').unset();
|
|
20
22
|
}
|
|
21
23
|
// Public Methods
|
|
22
24
|
onExpand() {
|
|
@@ -30,50 +32,98 @@ export class TasFloatingCallComponent {
|
|
|
30
32
|
}
|
|
31
33
|
// Private Methods
|
|
32
34
|
setupSubscriptions() {
|
|
33
|
-
|
|
35
|
+
// Call state subscription
|
|
36
|
+
this.subscriptions.add(this.tasService.callState$.subscribe(state => {
|
|
34
37
|
if (state === CallState.DISCONNECTED) {
|
|
35
38
|
this.isVisible = false;
|
|
36
39
|
}
|
|
37
40
|
}));
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
+
// View mode subscription
|
|
42
|
+
this.subscriptions.add(this.tasService.viewMode$.subscribe(mode => {
|
|
43
|
+
this.isVisible = mode === ViewMode.PIP && this.tasService.isCallActive();
|
|
41
44
|
if (this.isVisible) {
|
|
42
45
|
setTimeout(() => this.initInteract(), 100);
|
|
43
46
|
}
|
|
44
47
|
}));
|
|
45
|
-
|
|
48
|
+
// Mute state subscription
|
|
49
|
+
this.subscriptions.add(this.tasService.isMuted$.subscribe(muted => {
|
|
46
50
|
this.isMuted = muted;
|
|
47
51
|
}));
|
|
48
52
|
}
|
|
49
53
|
initInteract() {
|
|
50
|
-
interact(
|
|
51
|
-
|
|
54
|
+
interact('.tas-floating-container').unset();
|
|
55
|
+
// Create restriction area with margin
|
|
56
|
+
const margin = this.PIP_MARGIN;
|
|
57
|
+
const restrictToBodyWithMargin = {
|
|
58
|
+
restriction: () => {
|
|
59
|
+
return {
|
|
60
|
+
left: margin,
|
|
61
|
+
top: margin,
|
|
62
|
+
right: window.innerWidth - margin,
|
|
63
|
+
bottom: window.innerHeight - margin
|
|
64
|
+
};
|
|
65
|
+
},
|
|
66
|
+
elementRect: { left: 0, right: 1, top: 0, bottom: 1 }
|
|
67
|
+
};
|
|
68
|
+
interact('.tas-floating-container')
|
|
69
|
+
.draggable({
|
|
52
70
|
inertia: true,
|
|
53
71
|
modifiers: [
|
|
54
|
-
interact.modifiers.
|
|
55
|
-
restriction: "body",
|
|
56
|
-
endOnly: true,
|
|
57
|
-
}),
|
|
72
|
+
interact.modifiers.restrict(restrictToBodyWithMargin)
|
|
58
73
|
],
|
|
59
74
|
autoScroll: false,
|
|
60
75
|
listeners: {
|
|
61
76
|
move: (event) => {
|
|
62
77
|
const target = event.target;
|
|
63
|
-
const x = (parseFloat(target.getAttribute(
|
|
64
|
-
const y = (parseFloat(target.getAttribute(
|
|
78
|
+
const x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;
|
|
79
|
+
const y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;
|
|
65
80
|
target.style.transform = `translate(${x}px, ${y}px)`;
|
|
66
|
-
target.setAttribute(
|
|
67
|
-
target.setAttribute(
|
|
68
|
-
}
|
|
81
|
+
target.setAttribute('data-x', String(x));
|
|
82
|
+
target.setAttribute('data-y', String(y));
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
})
|
|
86
|
+
.resizable({
|
|
87
|
+
edges: { left: false, right: true, bottom: true, top: false },
|
|
88
|
+
listeners: {
|
|
89
|
+
move: (event) => {
|
|
90
|
+
const target = event.target;
|
|
91
|
+
let x = parseFloat(target.getAttribute('data-x')) || 0;
|
|
92
|
+
let y = parseFloat(target.getAttribute('data-y')) || 0;
|
|
93
|
+
// Update element size
|
|
94
|
+
target.style.width = `${event.rect.width}px`;
|
|
95
|
+
target.style.height = `${event.rect.height}px`;
|
|
96
|
+
// Translate when resizing from top or left edges
|
|
97
|
+
x += event.deltaRect.left;
|
|
98
|
+
y += event.deltaRect.top;
|
|
99
|
+
target.style.transform = `translate(${x}px, ${y}px)`;
|
|
100
|
+
target.setAttribute('data-x', String(x));
|
|
101
|
+
target.setAttribute('data-y', String(y));
|
|
102
|
+
}
|
|
69
103
|
},
|
|
104
|
+
modifiers: [
|
|
105
|
+
interact.modifiers.restrictEdges({
|
|
106
|
+
outer: {
|
|
107
|
+
left: margin,
|
|
108
|
+
top: margin,
|
|
109
|
+
right: window.innerWidth - margin,
|
|
110
|
+
bottom: window.innerHeight - margin
|
|
111
|
+
}
|
|
112
|
+
}),
|
|
113
|
+
interact.modifiers.restrictSize({
|
|
114
|
+
min: { width: 200, height: 130 },
|
|
115
|
+
max: { width: 500, height: 350 }
|
|
116
|
+
}),
|
|
117
|
+
interact.modifiers.aspectRatio({ ratio: 'preserve' })
|
|
118
|
+
],
|
|
119
|
+
inertia: true
|
|
70
120
|
});
|
|
71
121
|
}
|
|
72
122
|
}
|
|
73
123
|
TasFloatingCallComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasFloatingCallComponent, deps: [{ token: i1.TasService }], target: i0.ɵɵFactoryTarget.Component });
|
|
74
|
-
TasFloatingCallComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TasFloatingCallComponent, selector: "tas-floating-call", ngImport: i0, template: "<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n
|
|
124
|
+
TasFloatingCallComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TasFloatingCallComponent, selector: "tas-floating-call", ngImport: i0, template: "<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n\t<!-- Video content area - shows main video only -->\n\t<div class=\"floating-content\">\n\t\t<!-- Main video container (subscriber if available, otherwise publisher) -->\n\t\t<div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n\t\t<!-- Bottom controls -->\n\t\t<div class=\"floating-controls\">\n\t\t\t<button class=\"action-btn expand-btn\" (click)=\"onExpand()\" title=\"Expand to fullscreen\">\n\t\t\t\t<i class=\"fa fa-expand\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn mute-btn\" [class.muted]=\"isMuted\" (click)=\"toggleMute()\" [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\">\n\t\t\t\t<i class=\"fa\" [class.fa-microphone]=\"!isMuted\" [class.fa-microphone-slash]=\"isMuted\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n\t\t\t\t<i class=\"fa fa-phone\" style=\"transform: rotate(135deg);\"></i>\n\t\t\t</button>\n\t\t</div>\n\t</div>\n</div>\n\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;-webkit-backdrop-filter:blur(8px);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"] });
|
|
75
125
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasFloatingCallComponent, decorators: [{
|
|
76
126
|
type: Component,
|
|
77
|
-
args: [{ selector:
|
|
127
|
+
args: [{ selector: 'tas-floating-call', template: "<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n\t<!-- Video content area - shows main video only -->\n\t<div class=\"floating-content\">\n\t\t<!-- Main video container (subscriber if available, otherwise publisher) -->\n\t\t<div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n\t\t<!-- Bottom controls -->\n\t\t<div class=\"floating-controls\">\n\t\t\t<button class=\"action-btn expand-btn\" (click)=\"onExpand()\" title=\"Expand to fullscreen\">\n\t\t\t\t<i class=\"fa fa-expand\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn mute-btn\" [class.muted]=\"isMuted\" (click)=\"toggleMute()\" [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\">\n\t\t\t\t<i class=\"fa\" [class.fa-microphone]=\"!isMuted\" [class.fa-microphone-slash]=\"isMuted\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n\t\t\t\t<i class=\"fa fa-phone\" style=\"transform: rotate(135deg);\"></i>\n\t\t\t</button>\n\t\t</div>\n\t</div>\n</div>\n\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;-webkit-backdrop-filter:blur(8px);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"] }]
|
|
78
128
|
}], ctorParameters: function () { return [{ type: i1.TasService }]; } });
|
|
79
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-floating-call.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/tas-floating-call/tas-floating-call.component.ts","../../../../../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 { TasService, CallState, ViewMode } from \"../../services/tas.service\";\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\n  constructor(private tasService: TasService) {}\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.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    this.subscriptions.add(\n      this.tasService.callState$.subscribe((state) => {\n        if (state === CallState.DISCONNECTED) {\n          this.isVisible = false;\n        }\n      })\n    );\n\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        this.isVisible =\n          mode === ViewMode.PIP && this.tasService.isCallActive();\n        if (this.isVisible) {\n          setTimeout(() => this.initInteract(), 100);\n        }\n      })\n    );\n\n    this.subscriptions.add(\n      this.tasService.isMuted$.subscribe((muted) => {\n        this.isMuted = muted;\n      })\n    );\n  }\n\n  private initInteract(): void {\n    interact(\".tas-floating-container\").unset();\n\n    interact(\".tas-floating-container\").draggable({\n      inertia: true,\n      modifiers: [\n        interact.modifiers.restrictRect({\n          restriction: \"body\",\n          endOnly: true,\n        }),\n      ],\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  }\n}\n","<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n  <div class=\"floating-content\">\n    <div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n    <div class=\"floating-controls\">\n      <button\n        class=\"action-btn expand-btn\"\n        (click)=\"onExpand()\"\n        title=\"Expandir a pantalla completa\"\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 ? 'Activar micrófono' : 'Silenciar micrófono'\"\n      >\n        <i\n          class=\"fa\"\n          [class.fa-microphone]=\"!isMuted\"\n          [class.fa-microphone-slash]=\"isMuted\"\n        ></i>\n      </button>\n      <button\n        class=\"action-btn hangup-btn\"\n        (click)=\"onHangUp()\"\n        title=\"Colgar llamada\"\n      >\n        <i class=\"fa fa-phone\" style=\"transform: rotate(135deg)\"></i>\n      </button>\n    </div>\n  </div>\n</div>\n"]}
|
|
129
|
+
//# 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 { TasService } from '../../services/tas.service';\nimport { CallState, ViewMode } from '../../interfaces/tas.interfaces';\nimport { Subscription } from 'rxjs';\nimport interact from 'interactjs';\n\n@Component({\n\tselector: 'tas-floating-call',\n\ttemplateUrl: './tas-floating-call.component.html',\n\tstyleUrls: ['./tas-floating-call.component.scss']\n})\nexport class TasFloatingCallComponent implements OnInit, OnDestroy {\n\n\tpublic isVisible = false;\n\tpublic isMuted = false;\n\n\tprivate subscriptions = new Subscription();\n\n\tconstructor(private tasService: TasService) { }\n\n\tngOnInit(): void {\n\t\tthis.setupSubscriptions();\n\t}\n\n\tngOnDestroy(): void {\n\t\tthis.subscriptions.unsubscribe();\n\t\tinteract('.tas-floating-container').unset();\n\t}\n\n\t// Public Methods\n\tonExpand(): void {\n\t\tthis.tasService.exitPipMode();\n\t}\n\n\tonHangUp(): void {\n\t\tthis.tasService.disconnectSession();\n\t}\n\n\ttoggleMute(): void {\n\t\tthis.tasService.toggleMute();\n\t}\n\n\t// Private Methods\n\tprivate setupSubscriptions(): void {\n\t\t// Call state subscription\n\t\tthis.subscriptions.add(\n\t\t\tthis.tasService.callState$.subscribe(state => {\n\t\t\t\tif (state === CallState.DISCONNECTED) {\n\t\t\t\t\tthis.isVisible = false;\n\t\t\t\t}\n\t\t\t})\n\t\t);\n\n\t\t// View mode subscription\n\t\tthis.subscriptions.add(\n\t\t\tthis.tasService.viewMode$.subscribe(mode => {\n\t\t\t\tthis.isVisible = mode === ViewMode.PIP && this.tasService.isCallActive();\n\t\t\t\tif (this.isVisible) {\n\t\t\t\t\tsetTimeout(() => this.initInteract(), 100);\n\t\t\t\t}\n\t\t\t})\n\t\t);\n\n\t\t// Mute state subscription\n\t\tthis.subscriptions.add(\n\t\t\tthis.tasService.isMuted$.subscribe(muted => {\n\t\t\t\tthis.isMuted = muted;\n\t\t\t})\n\t\t);\n\t}\n\n\t// Margin from screen edges (in pixels)\n\tprivate readonly PIP_MARGIN = 20;\n\n\tprivate initInteract(): void {\n\t\tinteract('.tas-floating-container').unset();\n\n\t\t// Create restriction area with margin\n\t\tconst margin = this.PIP_MARGIN;\n\t\tconst restrictToBodyWithMargin = {\n\t\t\trestriction: () => {\n\t\t\t\treturn {\n\t\t\t\t\tleft: margin,\n\t\t\t\t\ttop: margin,\n\t\t\t\t\tright: window.innerWidth - margin,\n\t\t\t\t\tbottom: window.innerHeight - margin\n\t\t\t\t};\n\t\t\t},\n\t\t\telementRect: { left: 0, right: 1, top: 0, bottom: 1 }\n\t\t};\n\n\t\tinteract('.tas-floating-container')\n\t\t\t.draggable({\n\t\t\t\tinertia: true,\n\t\t\t\tmodifiers: [\n\t\t\t\t\tinteract.modifiers.restrict(restrictToBodyWithMargin)\n\t\t\t\t],\n\t\t\t\tautoScroll: false,\n\t\t\t\tlisteners: {\n\t\t\t\t\tmove: (event) => {\n\t\t\t\t\t\tconst target = event.target;\n\t\t\t\t\t\tconst x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;\n\t\t\t\t\t\tconst y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;\n\n\t\t\t\t\t\ttarget.style.transform = `translate(${x}px, ${y}px)`;\n\t\t\t\t\t\ttarget.setAttribute('data-x', String(x));\n\t\t\t\t\t\ttarget.setAttribute('data-y', String(y));\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t})\n\t\t\t.resizable({\n\t\t\t\tedges: { left: false, right: true, bottom: true, top: false },\n\t\t\t\tlisteners: {\n\t\t\t\t\tmove: (event) => {\n\t\t\t\t\t\tconst target = event.target;\n\t\t\t\t\t\tlet x = parseFloat(target.getAttribute('data-x')) || 0;\n\t\t\t\t\t\tlet y = parseFloat(target.getAttribute('data-y')) || 0;\n\n\t\t\t\t\t\t// Update element size\n\t\t\t\t\t\ttarget.style.width = `${event.rect.width}px`;\n\t\t\t\t\t\ttarget.style.height = `${event.rect.height}px`;\n\n\t\t\t\t\t\t// Translate when resizing from top or left edges\n\t\t\t\t\t\tx += event.deltaRect.left;\n\t\t\t\t\t\ty += event.deltaRect.top;\n\n\t\t\t\t\t\ttarget.style.transform = `translate(${x}px, ${y}px)`;\n\t\t\t\t\t\ttarget.setAttribute('data-x', String(x));\n\t\t\t\t\t\ttarget.setAttribute('data-y', String(y));\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\tmodifiers: [\n\t\t\t\t\tinteract.modifiers.restrictEdges({\n\t\t\t\t\t\touter: {\n\t\t\t\t\t\t\tleft: margin,\n\t\t\t\t\t\t\ttop: margin,\n\t\t\t\t\t\t\tright: window.innerWidth - margin,\n\t\t\t\t\t\t\tbottom: window.innerHeight - margin\n\t\t\t\t\t\t}\n\t\t\t\t\t}),\n\t\t\t\t\tinteract.modifiers.restrictSize({\n\t\t\t\t\t\tmin: { width: 200, height: 130 },\n\t\t\t\t\t\tmax: { width: 500, height: 350 }\n\t\t\t\t\t}),\n\t\t\t\t\tinteract.modifiers.aspectRatio({ ratio: 'preserve' })\n\t\t\t\t],\n\t\t\t\tinertia: true\n\t\t\t});\n\t}\n}\n\n","<div class=\"tas-floating-container\" [class.visible]=\"isVisible\">\n\t<!-- Video content area - shows main video only -->\n\t<div class=\"floating-content\">\n\t\t<!-- Main video container (subscriber if available, otherwise publisher) -->\n\t\t<div id=\"pip-main-video\" class=\"pip-main-video\"></div>\n\n\t\t<!-- Bottom controls -->\n\t\t<div class=\"floating-controls\">\n\t\t\t<button class=\"action-btn expand-btn\" (click)=\"onExpand()\" title=\"Expand to fullscreen\">\n\t\t\t\t<i class=\"fa fa-expand\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn mute-btn\" [class.muted]=\"isMuted\" (click)=\"toggleMute()\" [title]=\"isMuted ? 'Unmute microphone' : 'Mute microphone'\">\n\t\t\t\t<i class=\"fa\" [class.fa-microphone]=\"!isMuted\" [class.fa-microphone-slash]=\"isMuted\"></i>\n\t\t\t</button>\n\t\t\t<button class=\"action-btn hangup-btn\" (click)=\"onHangUp()\" title=\"Hang up call\">\n\t\t\t\t<i class=\"fa fa-phone\" style=\"transform: rotate(135deg);\"></i>\n\t\t\t</button>\n\t\t</div>\n\t</div>\n</div>\n\n"]}
|