tas-uell-sdk 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,304 @@
1
+ # TAS-Uell-SDK
2
+
3
+ Angular library for TAS (Telemedicine Assistance Service) video call functionality using OpenTok/Vonage Video API.
4
+
5
+ ## Features
6
+
7
+ - 📹 Video call integration with OpenTok/Vonage
8
+ - 🔘 Ready-to-use button component to initiate calls
9
+ - 🖼️ Picture-in-Picture (PiP) mode support
10
+ - 🎤 Mute/unmute functionality
11
+ - 📱 Draggable floating call widget
12
+ - 🔌 Pluggable architecture for HTTP and user data providers
13
+
14
+ ## Installation
15
+
16
+ ```bash
17
+ npm install tas-uell-sdk
18
+ ```
19
+
20
+ ### Peer Dependencies
21
+
22
+ Make sure you have these peer dependencies installed:
23
+
24
+ ```bash
25
+ npm install @angular/common @angular/core @ng-bootstrap/ng-bootstrap @opentok/client interactjs rxjs
26
+ ```
27
+
28
+ ## Setup
29
+
30
+ ### 1. Implement Required Adapters
31
+
32
+ The library requires two adapters to integrate with your application:
33
+
34
+ #### HTTP Client Adapter
35
+
36
+ Create a service that implements `TasHttpClient`:
37
+
38
+ ```typescript
39
+ import { Injectable } from "@angular/core";
40
+ import { Observable } from "rxjs";
41
+ import { TasHttpClient } from "tas-uell-sdk";
42
+
43
+ @Injectable({ providedIn: 'root' })
44
+ export class TasHttpAdapter implements TasHttpClient {
45
+ constructor(private http: HttpClient) {}
46
+
47
+ post<T>(url: string, body: any, options?: { headers?: Record<string, string> }): Observable<T> {
48
+ return this.http.post<T>(url, body, { headers: options?.headers });
49
+ }
50
+ }
51
+ ```
52
+
53
+ #### User Data Provider Adapter
54
+
55
+ Create a service that implements `TasUserDataProvider`:
56
+
57
+ ```typescript
58
+ import { Injectable } from "@angular/core";
59
+ import { TasUserDataProvider, TasUserData } from "tas-uell-sdk";
60
+
61
+ @Injectable({ providedIn: 'root' })
62
+ export class TasUserDataAdapter implements TasUserDataProvider {
63
+ getUserData(): TasUserData | null {
64
+ // Return the current user's data
65
+ const user = this.authService.getCurrentUser();
66
+ if (user) {
67
+ return {
68
+ id: user.id,
69
+ name: user.firstName,
70
+ surname: user.lastName,
71
+ };
72
+ }
73
+ return null;
74
+ }
75
+
76
+ getTenantId(): string | null {
77
+ // Return the current tenant ID
78
+ return this.authService.getTenantId();
79
+ }
80
+ }
81
+ ```
82
+
83
+ ### 2. Configure the Module
84
+
85
+ In your root module (e.g., `AppModule`):
86
+
87
+ ```typescript
88
+ import { NgModule } from '@angular/core';
89
+ import { TasModule, TAS_CONFIG, TAS_HTTP_CLIENT, TAS_USER_DATA_PROVIDER } from 'tas-uell-sdk';
90
+ import { TasHttpAdapter } from './adapters/tas-http-adapter.service';
91
+ import { TasUserDataAdapter } from './adapters/tas-user-data-adapter.service';
92
+ import { environment } from '../environments/environment';
93
+
94
+ @NgModule({
95
+ imports: [
96
+ // ... other imports
97
+ TasModule
98
+ ],
99
+ providers: [
100
+ {
101
+ provide: TAS_CONFIG,
102
+ useValue: {
103
+ tokBoxApiKey: environment.tokBoxApiKey,
104
+ apiBaseUrl: environment.apiUrl
105
+ }
106
+ },
107
+ { provide: TAS_HTTP_CLIENT, useExisting: TasHttpAdapter },
108
+ { provide: TAS_USER_DATA_PROVIDER, useExisting: TasUserDataAdapter }
109
+ ]
110
+ })
111
+ export class AppModule { }
112
+ ```
113
+
114
+ ### 3. Import in Feature Modules
115
+
116
+ In any feature module where you want to use the TAS components:
117
+
118
+ ```typescript
119
+ import { TasModule } from 'tas-uell-sdk';
120
+
121
+ @NgModule({
122
+ imports: [
123
+ TasModule
124
+ ]
125
+ })
126
+ export class MyFeatureModule { }
127
+ ```
128
+
129
+ ## Usage
130
+
131
+ ### TAS Button Component
132
+
133
+ Add the video call button to your template:
134
+
135
+ ```html
136
+ <tas-btn *ngIf="showVideoCall"></tas-btn>
137
+ ```
138
+
139
+ When clicked, this button will:
140
+ 1. Create a video room via your API
141
+ 2. Generate a session token
142
+ 3. Open the video call modal
143
+
144
+ ### Floating Call Component
145
+
146
+ Add the floating PiP widget to your root component template:
147
+
148
+ ```html
149
+ <router-outlet></router-outlet>
150
+ <tas-floating-call></tas-floating-call>
151
+ ```
152
+
153
+ This component shows a draggable mini video player when the user switches to PiP mode during a call.
154
+
155
+ ## Components
156
+
157
+ | Component | Selector | Description |
158
+ |-----------|----------|-------------|
159
+ | `TasButtonComponent` | `<tas-btn>` | Button to initiate a video call |
160
+ | `TasVideocallComponent` | `<tas-videocall>` | Full video call modal (opened automatically) |
161
+ | `TasFloatingCallComponent` | `<tas-floating-call>` | Floating PiP widget for minimized calls |
162
+
163
+ ## Services
164
+
165
+ ### TasService
166
+
167
+ The main service for managing video calls.
168
+
169
+ ```typescript
170
+ import { TasService } from 'tas-uell-sdk';
171
+
172
+ @Component({...})
173
+ export class MyComponent {
174
+ constructor(private tasService: TasService) {}
175
+
176
+ // Subscribe to call state changes
177
+ ngOnInit() {
178
+ this.tasService.callState$.subscribe(state => {
179
+ console.log('Call state:', state); // IDLE, CONNECTING, CONNECTED, DISCONNECTED, ERROR
180
+ });
181
+
182
+ this.tasService.viewMode$.subscribe(mode => {
183
+ console.log('View mode:', mode); // FULLSCREEN, PIP
184
+ });
185
+
186
+ this.tasService.isMuted$.subscribe(muted => {
187
+ console.log('Is muted:', muted);
188
+ });
189
+ }
190
+ }
191
+ ```
192
+
193
+ ## Interfaces
194
+
195
+ ### TasConfig
196
+
197
+ ```typescript
198
+ interface TasConfig {
199
+ tokBoxApiKey: string; // Your OpenTok/Vonage API Key
200
+ apiBaseUrl: string; // Base URL for TAS API endpoints
201
+ }
202
+ ```
203
+
204
+ ### TasHttpClient
205
+
206
+ ```typescript
207
+ interface TasHttpClient {
208
+ post<T>(url: string, body: any, options?: { headers?: Record<string, string> }): Observable<T>;
209
+ }
210
+ ```
211
+
212
+ ### TasUserDataProvider
213
+
214
+ ```typescript
215
+ interface TasUserDataProvider {
216
+ getUserData(): TasUserData | null;
217
+ getTenantId(): string | null;
218
+ }
219
+ ```
220
+
221
+ ### TasUserData
222
+
223
+ ```typescript
224
+ interface TasUserData {
225
+ id: string | number;
226
+ name: string;
227
+ surname: string;
228
+ }
229
+ ```
230
+
231
+ ## API Endpoints
232
+
233
+ The library expects the following API endpoints to be available:
234
+
235
+ | Method | Endpoint | Description |
236
+ |--------|----------|-------------|
237
+ | POST | `{apiBaseUrl}/v2/room` | Create a video room |
238
+ | POST | `{apiBaseUrl}/v2/room/token` | Generate session token |
239
+
240
+ ### Create Room Request
241
+
242
+ ```typescript
243
+ {
244
+ tenant: string;
245
+ userId: string;
246
+ product: string;
247
+ record: boolean;
248
+ roomType: string;
249
+ type: string;
250
+ }
251
+ ```
252
+
253
+ ### Generate Token Request
254
+
255
+ ```typescript
256
+ {
257
+ sessionId: string;
258
+ name: string;
259
+ lastname: string;
260
+ }
261
+ ```
262
+
263
+ ## Styling
264
+
265
+ The library includes default styles for all components. You may need to add global styles for the modal:
266
+
267
+ ```scss
268
+ // styles.scss
269
+ ngb-modal-window.tas-video-modal {
270
+ .modal-dialog {
271
+ max-width: 100vw;
272
+ height: 100vh;
273
+ margin: 0;
274
+ }
275
+ .modal-content {
276
+ background-color: transparent;
277
+ border: none;
278
+ height: 100%;
279
+ }
280
+ }
281
+ ```
282
+
283
+ ## Building the Library
284
+
285
+ ```bash
286
+ cd TAS-Uell-SDK
287
+ npm install
288
+ npm run build
289
+ ```
290
+
291
+ The built library will be in the `dist/` folder.
292
+
293
+ ## Publishing
294
+
295
+ ```bash
296
+ cd dist/tas-uell-sdk
297
+ npm publish
298
+ ```
299
+
300
+ ## License
301
+
302
+ MIT
303
+
304
+
@@ -0,0 +1,123 @@
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";
5
+ import { Subscription } from "rxjs";
6
+ import { switchMap } from "rxjs/operators";
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "../../services/tas.service";
9
+ import * as i2 from "@ng-bootstrap/ng-bootstrap";
10
+ import * as i3 from "@angular/common";
11
+ export class TasButtonComponent {
12
+ constructor(tasService, modalService, userDataProvider) {
13
+ this.tasService = tasService;
14
+ this.modalService = modalService;
15
+ this.userDataProvider = userDataProvider;
16
+ this.isLoading = false;
17
+ this.userData = null;
18
+ this.tenantId = null;
19
+ this.subscriptions = new Subscription();
20
+ this.currentModalRef = null;
21
+ }
22
+ 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;
77
+ }
78
+ this.userData = this.userDataProvider.getUserData();
79
+ this.tenantId = this.userDataProvider.getTenantId();
80
+ }
81
+ setupViewModeSubscription() {
82
+ this.subscriptions.add(this.tasService.viewMode$.subscribe((mode) => {
83
+ if (mode === ViewMode.FULLSCREEN &&
84
+ this.tasService.isCallActive() &&
85
+ !this.currentModalRef) {
86
+ const sessionId = this.tasService.sessionId;
87
+ const token = this.tasService.token;
88
+ if (sessionId && token) {
89
+ this.openVideoCallModal(sessionId, token, true);
90
+ }
91
+ }
92
+ }));
93
+ }
94
+ openVideoCallModal(sessionId, token, isReturningFromPip = false) {
95
+ this.currentModalRef = this.modalService.open(TasVideocallComponent, {
96
+ size: "xl",
97
+ windowClass: "tas-video-modal",
98
+ backdrop: "static",
99
+ keyboard: false,
100
+ });
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;
109
+ });
110
+ }
111
+ }
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"] }] });
114
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasButtonComponent, decorators: [{
115
+ 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"]}
@@ -0,0 +1,79 @@
1
+ import { Component } from "@angular/core";
2
+ import { CallState, ViewMode } from "../../services/tas.service";
3
+ import { Subscription } from "rxjs";
4
+ import interact from "interactjs";
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "../../services/tas.service";
7
+ export class TasFloatingCallComponent {
8
+ constructor(tasService) {
9
+ this.tasService = tasService;
10
+ this.isVisible = false;
11
+ this.isMuted = false;
12
+ this.subscriptions = new Subscription();
13
+ }
14
+ ngOnInit() {
15
+ this.setupSubscriptions();
16
+ }
17
+ ngOnDestroy() {
18
+ this.subscriptions.unsubscribe();
19
+ interact(".tas-floating-container").unset();
20
+ }
21
+ // Public Methods
22
+ onExpand() {
23
+ this.tasService.exitPipMode();
24
+ }
25
+ onHangUp() {
26
+ this.tasService.disconnectSession();
27
+ }
28
+ toggleMute() {
29
+ this.tasService.toggleMute();
30
+ }
31
+ // Private Methods
32
+ setupSubscriptions() {
33
+ this.subscriptions.add(this.tasService.callState$.subscribe((state) => {
34
+ if (state === CallState.DISCONNECTED) {
35
+ this.isVisible = false;
36
+ }
37
+ }));
38
+ this.subscriptions.add(this.tasService.viewMode$.subscribe((mode) => {
39
+ this.isVisible =
40
+ mode === ViewMode.PIP && this.tasService.isCallActive();
41
+ if (this.isVisible) {
42
+ setTimeout(() => this.initInteract(), 100);
43
+ }
44
+ }));
45
+ this.subscriptions.add(this.tasService.isMuted$.subscribe((muted) => {
46
+ this.isMuted = muted;
47
+ }));
48
+ }
49
+ initInteract() {
50
+ interact(".tas-floating-container").unset();
51
+ interact(".tas-floating-container").draggable({
52
+ inertia: true,
53
+ modifiers: [
54
+ interact.modifiers.restrictRect({
55
+ restriction: "body",
56
+ endOnly: true,
57
+ }),
58
+ ],
59
+ autoScroll: false,
60
+ listeners: {
61
+ move: (event) => {
62
+ 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;
65
+ target.style.transform = `translate(${x}px, ${y}px)`;
66
+ target.setAttribute("data-x", String(x));
67
+ target.setAttribute("data-y", String(y));
68
+ },
69
+ },
70
+ });
71
+ }
72
+ }
73
+ 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"] });
75
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasFloatingCallComponent, decorators: [{
76
+ 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"] }]
78
+ }], 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"]}