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.
Files changed (31) hide show
  1. package/INSTALL_AND_TEST.md +427 -0
  2. package/README.md +122 -229
  3. package/esm2020/lib/components/tas-btn/tas-btn.component.mjs +79 -95
  4. package/esm2020/lib/components/tas-floating-call/tas-floating-call.component.mjs +74 -24
  5. package/esm2020/lib/components/tas-videocall/tas-videocall.component.mjs +50 -45
  6. package/esm2020/lib/components/tas-waiting-room/tas-waiting-room.component.mjs +179 -0
  7. package/esm2020/lib/config/tas.config.mjs +10 -0
  8. package/esm2020/lib/interfaces/tas.interfaces.mjs +32 -2
  9. package/esm2020/lib/services/tas.service.mjs +27 -48
  10. package/esm2020/lib/tas-uell-sdk.module.mjs +82 -0
  11. package/esm2020/public-api.mjs +13 -12
  12. package/esm2020/tas-uell-sdk.mjs +1 -1
  13. package/fesm2015/tas-uell-sdk.mjs +440 -218
  14. package/fesm2015/tas-uell-sdk.mjs.map +1 -1
  15. package/fesm2020/tas-uell-sdk.mjs +437 -216
  16. package/fesm2020/tas-uell-sdk.mjs.map +1 -1
  17. package/lib/components/tas-btn/tas-btn.component.d.ts +15 -11
  18. package/lib/components/tas-floating-call/tas-floating-call.component.d.ts +3 -2
  19. package/lib/components/tas-videocall/tas-videocall.component.d.ts +4 -3
  20. package/lib/components/tas-waiting-room/tas-waiting-room.component.d.ts +56 -0
  21. package/lib/config/tas.config.d.ts +28 -0
  22. package/lib/interfaces/tas.interfaces.d.ts +41 -6
  23. package/lib/services/tas.service.d.ts +6 -16
  24. package/lib/tas-uell-sdk.module.d.ts +46 -0
  25. package/package.json +36 -23
  26. package/public-api.d.ts +8 -7
  27. package/src/lib/styles/tas-global.scss +37 -0
  28. package/esm2020/lib/tas.config.mjs +0 -14
  29. package/esm2020/lib/tas.module.mjs +0 -73
  30. package/lib/tas.config.d.ts +0 -51
  31. package/lib/tas.module.d.ts +0 -34
@@ -1,123 +1,107 @@
1
- import { Component, Inject, Optional } from "@angular/core";
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 { switchMap } from "rxjs/operators";
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 "../../services/tas.service";
9
- import * as i2 from "@ng-bootstrap/ng-bootstrap";
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(tasService, modalService, userDataProvider) {
13
- this.tasService = tasService;
11
+ constructor(modalService, tasService) {
14
12
  this.modalService = modalService;
15
- this.userDataProvider = userDataProvider;
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.loadUserData();
24
- this.setupViewModeSubscription();
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
- this.userData = this.userDataProvider.getUserData();
79
- this.tenantId = this.userDataProvider.getTenantId();
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.currentModalRef) {
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(sessionId, token, true);
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
- openVideoCallModal(sessionId, token, isReturningFromPip = false) {
95
- this.currentModalRef = this.modalService.open(TasVideocallComponent, {
96
- size: "xl",
97
- windowClass: "tas-video-modal",
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
- this.currentModalRef.componentInstance.sessionId = sessionId;
102
- this.currentModalRef.componentInstance.token = token;
103
- this.currentModalRef.componentInstance.isReturningFromPip =
104
- isReturningFromPip;
105
- this.currentModalRef.result.then(() => {
106
- this.currentModalRef = null;
107
- }, () => {
108
- this.currentModalRef = null;
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.TasService }, { token: i2.NgbModal }, { token: TAS_USER_DATA_PROVIDER, optional: true }], target: i0.ɵɵFactoryTarget.Component });
113
- TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TasButtonComponent, selector: "tas-btn", ngImport: i0, template: "<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", styles: [":host{display:inline-block}.boton{background-color:#ee316b!important;color:#fff!important;margin-right:24px}.boton:disabled{background-color:#ccc!important;border-color:#ccc!important;cursor:not-allowed}.boton i{margin-right:5px}\n"], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
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 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", styles: [":host{display:inline-block}.boton{background-color:#ee316b!important;color:#fff!important;margin-right:24px}.boton:disabled{background-color:#ccc!important;border-color:#ccc!important;cursor:not-allowed}.boton i{margin-right:5px}\n"] }]
117
- }], ctorParameters: function () { return [{ type: i1.TasService }, { type: i2.NgbModal }, { type: undefined, decorators: [{
118
- type: Optional
119
- }, {
120
- type: Inject,
121
- args: [TAS_USER_DATA_PROVIDER]
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 "@angular/core";
2
- import { CallState, ViewMode } from "../../services/tas.service";
3
- import { Subscription } from "rxjs";
4
- import interact from "interactjs";
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(".tas-floating-container").unset();
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
- this.subscriptions.add(this.tasService.callState$.subscribe((state) => {
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
- this.subscriptions.add(this.tasService.viewMode$.subscribe((mode) => {
39
- this.isVisible =
40
- mode === ViewMode.PIP && this.tasService.isCallActive();
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
- this.subscriptions.add(this.tasService.isMuted$.subscribe((muted) => {
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(".tas-floating-container").unset();
51
- interact(".tas-floating-container").draggable({
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.restrictRect({
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("data-x")) || 0) + event.dx;
64
- const y = (parseFloat(target.getAttribute("data-y")) || 0) + event.dy;
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("data-x", String(x));
67
- target.setAttribute("data-y", String(y));
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 <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\u00F3fono' : 'Silenciar micr\u00F3fono'\"\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", 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;opacity:0;visibility:hidden;pointer-events:none}.tas-floating-container.visible{opacity:1;visibility:visible;pointer-events:auto}.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)}.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"] });
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: "tas-floating-call", template: "<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\u00F3fono' : 'Silenciar micr\u00F3fono'\"\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", 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;opacity:0;visibility:hidden;pointer-events:none}.tas-floating-container.visible{opacity:1;visibility:visible;pointer-events:auto}.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)}.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"] }]
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"]}