tas-uell-sdk 0.0.5 → 0.0.6

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 (28) hide show
  1. package/README.md +94 -53
  2. package/esm2020/lib/components/tas-avatar/tas-avatar.component.mjs +75 -0
  3. package/esm2020/lib/components/tas-btn/tas-btn.component.mjs +156 -63
  4. package/esm2020/lib/components/tas-floating-call/tas-floating-call.component.mjs +48 -23
  5. package/esm2020/lib/components/tas-videocall/tas-videocall.component.mjs +109 -18
  6. package/esm2020/lib/components/tas-waiting-room/tas-waiting-room.component.mjs +158 -150
  7. package/esm2020/lib/config/tas.config.mjs +1 -1
  8. package/esm2020/lib/interfaces/tas.interfaces.mjs +39 -2
  9. package/esm2020/lib/services/tas.service.mjs +363 -34
  10. package/esm2020/lib/tas-uell-sdk.module.mjs +19 -21
  11. package/esm2020/public-api.mjs +2 -1
  12. package/fesm2015/tas-uell-sdk.mjs +945 -292
  13. package/fesm2015/tas-uell-sdk.mjs.map +1 -1
  14. package/fesm2020/tas-uell-sdk.mjs +940 -290
  15. package/fesm2020/tas-uell-sdk.mjs.map +1 -1
  16. package/lib/components/tas-avatar/tas-avatar.component.d.ts +9 -0
  17. package/lib/components/tas-btn/tas-btn.component.d.ts +33 -15
  18. package/lib/components/tas-floating-call/tas-floating-call.component.d.ts +5 -1
  19. package/lib/components/tas-videocall/tas-videocall.component.d.ts +23 -2
  20. package/lib/components/tas-waiting-room/tas-waiting-room.component.d.ts +28 -34
  21. package/lib/config/tas.config.d.ts +4 -0
  22. package/lib/interfaces/tas.interfaces.d.ts +103 -35
  23. package/lib/services/tas.service.d.ts +86 -9
  24. package/lib/tas-uell-sdk.module.d.ts +4 -3
  25. package/package.json +1 -1
  26. package/public-api.d.ts +1 -0
  27. package/src/lib/styles/tas-global.scss +27 -28
  28. package/INSTALL_AND_TEST.md +0 -427
package/README.md CHANGED
@@ -29,7 +29,7 @@ npm install @ng-bootstrap/ng-bootstrap @opentok/client interactjs
29
29
 
30
30
  ### 1. Create an HTTP Adapter
31
31
 
32
- The library requires an HTTP adapter that implements `TasHttpClient` interface:
32
+ The library requires an HTTP adapter that implements the `TasHttpClient` interface (which includes `post` and `patch` methods):
33
33
 
34
34
  ```typescript
35
35
  import { Injectable } from '@angular/core';
@@ -42,8 +42,14 @@ export class TasHttpAdapterService implements TasHttpClient {
42
42
  constructor(private http: HttpClient) {}
43
43
 
44
44
  post<T>(url: string, options: { body: any; headers?: Record<string, string> }): Observable<T> {
45
- return this.http.post<T>(`https://your-api.com/${url}`, options.body, {
46
- headers: options.headers
45
+ return this.http.post<T>(`https://your-api.com/${url}`, options.body, {
46
+ headers: options.headers,
47
+ });
48
+ }
49
+
50
+ patch<T>(url: string, options: { body: any; headers?: Record<string, string> }): Observable<T> {
51
+ return this.http.patch<T>(`https://your-api.com/${url}`, options.body, {
52
+ headers: options.headers,
47
53
  });
48
54
  }
49
55
  }
@@ -60,22 +66,26 @@ import { TasHttpAdapterService } from './adapters/tas-http-adapter.service';
60
66
  @NgModule({
61
67
  imports: [
62
68
  TasUellSdkModule.forRoot({
63
- config: {
64
- tokBoxApiKey: 'YOUR_TOKBOX_API_KEY'
69
+ config: {
70
+ tokBoxApiKey: 'YOUR_TOKBOX_API_KEY',
71
+ // apiBaseUrl: 'https://your-api.com/v2', // Optional
65
72
  },
66
- httpClient: TasHttpAdapterService
67
- })
68
- ]
73
+ httpClient: TasHttpAdapterService,
74
+ }),
75
+ ],
69
76
  })
70
- export class AppModule { }
77
+ export class AppModule {}
71
78
  ```
72
79
 
73
80
  ### 3. Add Global Styles
74
81
 
75
- Add the TAS modal styles to your global `styles.scss`:
82
+ Add the TAS modal styles to your global `styles.scss`. You can either import the library's global styles or copy the CSS below:
76
83
 
77
84
  ```scss
78
- /* TAS Video Modal */
85
+ /* Option 1: Import from library */
86
+ @import 'tas-uell-sdk/src/lib/styles/tas-global.scss';
87
+
88
+ /* Option 2: Manual styles */
79
89
  .tas-video-modal {
80
90
  .modal-dialog {
81
91
  max-width: 100vw;
@@ -93,7 +103,6 @@ Add the TAS modal styles to your global `styles.scss`:
93
103
  }
94
104
  }
95
105
 
96
- /* TAS Waiting Room Modal */
97
106
  .tas-waiting-room-modal {
98
107
  .modal-dialog {
99
108
  max-width: 480px;
@@ -122,75 +131,107 @@ Add `<tas-floating-call>` to your root component template (e.g., `app.component.
122
131
 
123
132
  ## Usage
124
133
 
134
+ > **Note:** The appointment/room must be pre-created in your backend before using this library.
135
+
125
136
  ### TAS Button Component
126
137
 
127
138
  Use the `<tas-btn>` component to initiate a video call:
128
139
 
129
140
  ```html
130
141
  <tas-btn
131
- [appointmentId]="appointment.id"
132
- [product]="'your-product'"
133
- [tenantId]="tenantId"
142
+ [roomType]="'TAS'"
143
+ [entityId]="appointment.entityId"
144
+ [tenant]="tenantId"
145
+ [businessRole]="'BACKOFFICE'"
134
146
  [currentUser]="currentUser"
135
- [ownerUserIds]="[ownerId]"
136
- [regularUserIds]="regularUserIds"
137
- [moderatorUserIds]="moderatorUserIds"
138
147
  ></tas-btn>
139
148
  ```
140
149
 
141
150
  ### Input Properties
142
151
 
143
- | Property | Type | Required | Description |
144
- |----------|------|----------|-------------|
145
- | `appointmentId` | `number` | Yes | The appointment/session ID |
146
- | `product` | `string` | Yes | Product identifier |
147
- | `tenantId` | `string` | Yes | Tenant identifier |
148
- | `currentUser` | `TasCurrentUser` | Yes | Current user info (name, lastname, role) |
149
- | `ownerUserIds` | `number[]` | Yes | Array with exactly one owner user ID |
150
- | `regularUserIds` | `number[]` | No | Array of regular participant user IDs |
151
- | `moderatorUserIds` | `number[]` | No | Array of moderator user IDs |
152
+ | Property | Type | Required | Default | Description |
153
+ | :--- | :--- | :--- | :--- | :--- |
154
+ | `entityId` | `number` | Yes | - | The entity/license ID |
155
+ | `tenant` | `string` | Yes | - | Tenant identifier |
156
+ | `roomType` | `TasRoomType` | No | `'TAS'` | Room type (TAS, JM, WEBINAR, WELLNESS_MANAGER) |
157
+ | `businessRole` | `TasBusinessRole` | No | `'USER'` | Role (ADMIN_MANAGER, MANAGER, BACKOFFICE, USER) |
158
+ | `currentUser` | `TasCurrentUser` | Yes | - | Current user info (name, lastname, role) |
159
+
160
+ ### Waiting Room Behavior
161
+
162
+ When clicking the TAS button, a waiting room modal opens:
163
+
164
+ 1. **Status Check**: The library calls `/v2/proxy/video/status` to get session info
165
+ 2. **Token Acquisition**: Calls `/v2/proxy/video/start` to get the video call token
166
+
167
+ **Owner/Backoffice Users** (`businessRole: 'BACKOFFICE'` or user with `role: 'OWNER'`):
168
+ - Token is obtained immediately after status check
169
+ - Once token is received, the "Join" button appears
170
+ - If `/start` fails, an error screen is shown with a "Retry" button
171
+
172
+ **Non-Owner Users** (regular attendees):
173
+ - See "Medicina laboral te va a admitir en unos instantes..."
174
+ - Wait until the backend marks the session as `joinable: true`
175
+ - Once joinable, the token is obtained and the "Join" button appears
176
+
177
+ ### Error Handling
178
+
179
+ The library handles errors gracefully:
180
+ - If `/status` fails: Shows error with retry option
181
+ - If `/start` fails: Shows error message from backend, stops polling, shows retry option
182
+ - Network errors: Displays user-friendly error message
152
183
 
153
- ### TasCurrentUser Interface
184
+ ## Interfaces & Enums
154
185
 
186
+ ### TasCurrentUser
155
187
  ```typescript
156
188
  interface TasCurrentUser {
157
189
  name: string;
158
190
  lastname: string;
159
- role: TasUserRole; // 'OWNER' | 'USER' | 'MODERATOR'
191
+ role: TasUserRole;
160
192
  }
161
193
  ```
162
194
 
163
- ## Exported Components
195
+ ### TasCallConfig
196
+ ```typescript
197
+ interface TasCallConfig {
198
+ roomType: TasRoomType;
199
+ entityId: number;
200
+ tenant: string;
201
+ businessRole: TasBusinessRole;
202
+ currentUser: TasCurrentUser;
203
+ }
204
+ ```
164
205
 
165
- - `TasButtonComponent` - Button to initiate video calls
166
- - `TasVideocallComponent` - Full-screen video call interface
167
- - `TasWaitingRoomComponent` - Pre-call waiting room
168
- - `TasFloatingCallComponent` - Picture-in-Picture floating window
206
+ ### Enums
207
+ - **TasUserRole**: `OWNER`, `USER`, `MODERATOR`
208
+ - **TasBusinessRole**: `ADMIN_MANAGER`, `MANAGER`, `BACKOFFICE`, `USER`
209
+ - **TasRoomType**: `TAS`, `JM`, `WEBINAR`, `WELLNESS_MANAGER`
210
+ - **CallState**: `IDLE`, `CONNECTING`, `CONNECTED`, `DISCONNECTED`, `ERROR`
211
+ - **ViewMode**: `FULLSCREEN`, `PIP`
169
212
 
170
213
  ## Exported Services
171
214
 
172
- - `TasService` - Core service for managing video sessions
215
+ ### TasService
216
+ Core service for managing video sessions and state.
173
217
 
174
- ## Exported Interfaces & Types
218
+ **Public API:**
219
+ - `callState$: Observable<CallState>` - Current connection status
220
+ - `viewMode$: Observable<ViewMode>` - Current view mode
221
+ - `isMuted$: Observable<boolean>` - Local audio mute state
222
+ - `joinable$: Observable<boolean>` - Whether the call is joinable
223
+ - `ownerHasJoined$: Observable<boolean>` - Whether an owner has joined
224
+ - `toggleMute()` - Toggles local audio
225
+ - `disconnectSession()` - Ends the current session
226
+ - `getProxyVideoStatus(params)` - Gets session status
175
227
 
176
- - `TasConfig`
177
- - `TasHttpClient`
178
- - `TasCurrentUser`
179
- - `CreateRoomRequest`
180
- - `CreateRoomResponse`
181
- - `GenerateTokenRequest`
182
- - `GenerateTokenResponse`
183
- - `TasRoomType`
184
- - `TasSessionType`
185
- - `TasUserRole`
186
- - `CallState`
187
- - `ViewMode`
188
-
189
- ## Building the Library
228
+ ## Exported Components
190
229
 
191
- ```bash
192
- ng build tas-uell-sdk --configuration=production
193
- ```
230
+ - `TasButtonComponent` - `<tas-btn>`
231
+ - `TasVideocallComponent` - Full-screen video call interface
232
+ - `TasWaitingRoomComponent` - Pre-call waiting room
233
+ - `TasFloatingCallComponent` - `<tas-floating-call>`
234
+ - `TasAvatarComponent` - `<tas-avatar>`
194
235
 
195
236
  ## License
196
237
 
@@ -0,0 +1,75 @@
1
+ import { Component, Input, ChangeDetectionStrategy } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class TasAvatarComponent {
4
+ constructor() {
5
+ this.name = '';
6
+ this.size = 80;
7
+ }
8
+ get initials() {
9
+ if (!this.name)
10
+ return '';
11
+ return this.name
12
+ .split(' ')
13
+ .filter((n) => n.length > 0)
14
+ .map((n) => n[0])
15
+ .join('')
16
+ .toUpperCase()
17
+ .substring(0, 2);
18
+ }
19
+ get fontSize() {
20
+ return Math.round(this.size * 0.4);
21
+ }
22
+ }
23
+ TasAvatarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasAvatarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
24
+ TasAvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasAvatarComponent, selector: "tas-avatar", inputs: { name: "name", size: "size" }, ngImport: i0, template: `
25
+ <div
26
+ class="avatar"
27
+ [style.width.px]="size"
28
+ [style.height.px]="size"
29
+ [style.fontSize.px]="fontSize"
30
+ >
31
+ <span class="initials">{{ initials }}</span>
32
+ </div>
33
+ `, isInline: true, styles: [".avatar{display:flex;align-items:center;justify-content:center;border-radius:50%;background-color:#fff;color:#0072ac;font-weight:600;font-family:inherit;box-shadow:0 4px 12px #00000026}.initials{text-transform:uppercase;-webkit-user-select:none;user-select:none}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush });
34
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasAvatarComponent, decorators: [{
35
+ type: Component,
36
+ args: [{
37
+ selector: 'tas-avatar',
38
+ template: `
39
+ <div
40
+ class="avatar"
41
+ [style.width.px]="size"
42
+ [style.height.px]="size"
43
+ [style.fontSize.px]="fontSize"
44
+ >
45
+ <span class="initials">{{ initials }}</span>
46
+ </div>
47
+ `,
48
+ styles: [
49
+ `
50
+ .avatar {
51
+ display: flex;
52
+ align-items: center;
53
+ justify-content: center;
54
+ border-radius: 50%;
55
+ background-color: #ffffff;
56
+ color: #0072ac;
57
+ font-weight: 600;
58
+ font-family: inherit;
59
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
60
+ }
61
+
62
+ .initials {
63
+ text-transform: uppercase;
64
+ user-select: none;
65
+ }
66
+ `,
67
+ ],
68
+ changeDetection: ChangeDetectionStrategy.OnPush,
69
+ }]
70
+ }], propDecorators: { name: [{
71
+ type: Input
72
+ }], size: [{
73
+ type: Input
74
+ }] } });
75
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFzLWF2YXRhci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90YXMtdWVsbC1zZGsvc3JjL2xpYi9jb21wb25lbnRzL3Rhcy1hdmF0YXIvdGFzLWF2YXRhci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBb0MxRSxNQUFNLE9BQU8sa0JBQWtCO0lBbEMvQjtRQW1DVyxTQUFJLEdBQVcsRUFBRSxDQUFDO1FBQ2xCLFNBQUksR0FBVyxFQUFFLENBQUM7S0FnQjVCO0lBZEMsSUFBSSxRQUFRO1FBQ1YsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJO1lBQUUsT0FBTyxFQUFFLENBQUM7UUFDMUIsT0FBTyxJQUFJLENBQUMsSUFBSTthQUNiLEtBQUssQ0FBQyxHQUFHLENBQUM7YUFDVixNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO2FBQzNCLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO2FBQ2hCLElBQUksQ0FBQyxFQUFFLENBQUM7YUFDUixXQUFXLEVBQUU7YUFDYixTQUFTLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQ3JCLENBQUM7SUFFRCxJQUFJLFFBQVE7UUFDVixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLENBQUMsQ0FBQztJQUNyQyxDQUFDOztnSEFqQlUsa0JBQWtCO29HQUFsQixrQkFBa0IsMEZBaENuQjs7Ozs7Ozs7O0dBU1Q7NEZBdUJVLGtCQUFrQjtrQkFsQzlCLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLFlBQVk7b0JBQ3RCLFFBQVEsRUFBRTs7Ozs7Ozs7O0dBU1Q7b0JBQ0QsTUFBTSxFQUFFO3dCQUNOOzs7Ozs7Ozs7Ozs7Ozs7OztLQWlCQztxQkFDRjtvQkFDRCxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtpQkFDaEQ7OEJBRVUsSUFBSTtzQkFBWixLQUFLO2dCQUNHLElBQUk7c0JBQVosS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIENoYW5nZURldGVjdGlvblN0cmF0ZWd5IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3Rhcy1hdmF0YXInLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXZcbiAgICAgIGNsYXNzPVwiYXZhdGFyXCJcbiAgICAgIFtzdHlsZS53aWR0aC5weF09XCJzaXplXCJcbiAgICAgIFtzdHlsZS5oZWlnaHQucHhdPVwic2l6ZVwiXG4gICAgICBbc3R5bGUuZm9udFNpemUucHhdPVwiZm9udFNpemVcIlxuICAgID5cbiAgICAgIDxzcGFuIGNsYXNzPVwiaW5pdGlhbHNcIj57eyBpbml0aWFscyB9fTwvc3Bhbj5cbiAgICA8L2Rpdj5cbiAgYCxcbiAgc3R5bGVzOiBbXG4gICAgYFxuICAgICAgLmF2YXRhciB7XG4gICAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgICAgICBib3JkZXItcmFkaXVzOiA1MCU7XG4gICAgICAgIGJhY2tncm91bmQtY29sb3I6ICNmZmZmZmY7XG4gICAgICAgIGNvbG9yOiAjMDA3MmFjO1xuICAgICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgICBmb250LWZhbWlseTogaW5oZXJpdDtcbiAgICAgICAgYm94LXNoYWRvdzogMCA0cHggMTJweCByZ2JhKDAsIDAsIDAsIDAuMTUpO1xuICAgICAgfVxuXG4gICAgICAuaW5pdGlhbHMge1xuICAgICAgICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICAgICAgICB1c2VyLXNlbGVjdDogbm9uZTtcbiAgICAgIH1cbiAgICBgLFxuICBdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgVGFzQXZhdGFyQ29tcG9uZW50IHtcbiAgQElucHV0KCkgbmFtZTogc3RyaW5nID0gJyc7XG4gIEBJbnB1dCgpIHNpemU6IG51bWJlciA9IDgwO1xuXG4gIGdldCBpbml0aWFscygpOiBzdHJpbmcge1xuICAgIGlmICghdGhpcy5uYW1lKSByZXR1cm4gJyc7XG4gICAgcmV0dXJuIHRoaXMubmFtZVxuICAgICAgLnNwbGl0KCcgJylcbiAgICAgIC5maWx0ZXIoKG4pID0+IG4ubGVuZ3RoID4gMClcbiAgICAgIC5tYXAoKG4pID0+IG5bMF0pXG4gICAgICAuam9pbignJylcbiAgICAgIC50b1VwcGVyQ2FzZSgpXG4gICAgICAuc3Vic3RyaW5nKDAsIDIpO1xuICB9XG5cbiAgZ2V0IGZvbnRTaXplKCk6IG51bWJlciB7XG4gICAgcmV0dXJuIE1hdGgucm91bmQodGhpcy5zaXplICogMC40KTtcbiAgfVxufVxuIl19
@@ -1,8 +1,8 @@
1
- import { Component, Input } from "@angular/core";
2
- import { Subscription } from "rxjs";
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";
1
+ import { Component, Input } from '@angular/core';
2
+ import { Subscription } from 'rxjs';
3
+ import { TasRoomType, ViewMode, TasBusinessRole, } from '../../interfaces/tas.interfaces';
4
+ import { TasWaitingRoomComponent } from '../tas-waiting-room/tas-waiting-room.component';
5
+ import { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';
6
6
  import * as i0 from "@angular/core";
7
7
  import * as i1 from "@ng-bootstrap/ng-bootstrap";
8
8
  import * as i2 from "../../services/tas.service";
@@ -11,107 +11,200 @@ export class TasButtonComponent {
11
11
  constructor(modalService, tasService) {
12
12
  this.modalService = modalService;
13
13
  this.tasService = tasService;
14
- this.appointmentId = 1;
15
- this.product = "uell";
16
- this.tenantId = "";
17
- this.regularUserIds = [];
18
- this.moderatorUserIds = [];
19
- /** Optional: If provided, skips room creation and goes directly to getting a token */
20
- this.existingSessionId = "";
21
- /** Optional: Custom button text */
22
- this.buttonText = "Iniciar TAS";
14
+ // Status endpoint params
15
+ this.roomType = TasRoomType.TAS;
16
+ this.businessRole = TasBusinessRole.USER;
23
17
  this.isLoading = false;
18
+ this.buttonText = 'Iniciar TAS';
19
+ // Status check state
20
+ this.isCheckingStatus = false;
21
+ this.isStatusError = false;
22
+ this.statusErrorMessage = '';
24
23
  this.subscriptions = new Subscription();
25
24
  this.currentModalRef = null;
26
25
  this.videoCallModalRef = null;
26
+ this.statusPollingInterval = null;
27
+ this.STATUS_POLL_INTERVAL_MS = 30000; // 30 seconds
27
28
  }
28
- ngOnInit() {
29
- if (!this.ownerUserIds || this.ownerUserIds.length !== 1) {
30
- throw new Error('tas-btn: ownerUserIds input is required and must contain exactly one user');
29
+ /** Whether user is backoffice (or admin/manager) */
30
+ get isBackoffice() {
31
+ return (this.businessRole === TasBusinessRole.BACKOFFICE ||
32
+ this.businessRole === TasBusinessRole.ADMIN_MANAGER ||
33
+ this.businessRole === TasBusinessRole.MANAGER);
34
+ }
35
+ /** Whether the button should be visible */
36
+ get isVisible() {
37
+ // Backoffice: always show (disabled if error)
38
+ // Other roles: hide if status error
39
+ if (this.isBackoffice) {
40
+ return true;
31
41
  }
42
+ return !this.isStatusError;
43
+ }
44
+ /** Whether the button should be disabled */
45
+ get isDisabled() {
46
+ return this.isLoading || this.isCheckingStatus || this.isStatusError;
47
+ }
48
+ ngOnInit() {
32
49
  // Subscribe to viewMode to handle PiP return
33
- this.subscriptions.add(this.tasService.viewMode$.subscribe(mode => {
34
- // Reopen video call modal when returning from PiP
35
- if (mode === ViewMode.FULLSCREEN &&
36
- this.tasService.isCallActive() &&
37
- !this.videoCallModalRef) {
38
- const sessionId = this.tasService.sessionId;
39
- const token = this.tasService.token;
40
- if (sessionId && token) {
41
- this.openVideoCallModal(true);
42
- }
43
- }
50
+ this.subscriptions.add(this.tasService.viewMode$.subscribe((mode) => {
44
51
  // When entering PiP, clear the videoCallModalRef since modal will close
45
52
  if (mode === ViewMode.PIP) {
46
53
  this.videoCallModalRef = null;
47
54
  }
48
55
  }));
56
+ // Start status checking
57
+ this.startStatusPolling();
49
58
  }
50
59
  ngOnDestroy() {
51
60
  this.subscriptions.unsubscribe();
61
+ this.stopStatusPolling();
62
+ }
63
+ /**
64
+ * Start polling status every 30 seconds
65
+ */
66
+ startStatusPolling() {
67
+ // Initial status check
68
+ this.checkStatus();
69
+ // Set up periodic polling
70
+ this.statusPollingInterval = setInterval(() => {
71
+ this.checkStatus();
72
+ }, this.STATUS_POLL_INTERVAL_MS);
73
+ }
74
+ /**
75
+ * Stop status polling
76
+ */
77
+ stopStatusPolling() {
78
+ if (this.statusPollingInterval) {
79
+ clearInterval(this.statusPollingInterval);
80
+ this.statusPollingInterval = null;
81
+ }
82
+ }
83
+ /**
84
+ * Check status endpoint to determine if button should be enabled
85
+ */
86
+ checkStatus() {
87
+ // Skip if required inputs are not available
88
+ if (!this.tenant || !this.entityId) {
89
+ console.log('[TAS DEBUG] checkStatus skipped - missing required inputs');
90
+ return;
91
+ }
92
+ this.isCheckingStatus = true;
93
+ this.statusErrorMessage = '';
94
+ console.log('[TAS DEBUG] checkStatus called with:', {
95
+ roomType: this.roomType,
96
+ entityId: this.entityId,
97
+ tenant: this.tenant,
98
+ businessRole: this.businessRole,
99
+ });
100
+ this.subscriptions.add(this.tasService.getProxyVideoStatus({
101
+ roomType: this.roomType,
102
+ entityId: this.entityId,
103
+ tenant: this.tenant,
104
+ businessRole: this.businessRole,
105
+ }).subscribe({
106
+ next: (response) => {
107
+ // Check if response is actually an error (some HTTP adapters return errors in next)
108
+ // Also check for undefined/null or missing content
109
+ const isErrorResponse = !response ||
110
+ !response.content ||
111
+ response?.ok === false ||
112
+ response?.status >= 400 ||
113
+ response?.error ||
114
+ response?.name === 'HttpErrorResponse';
115
+ if (isErrorResponse) {
116
+ console.error('[TAS DEBUG] Status check returned error in response:', response);
117
+ this.isCheckingStatus = false;
118
+ this.isStatusError = true;
119
+ this.statusErrorMessage = response?.error?.message || response?.message || 'Error checking status';
120
+ }
121
+ else {
122
+ console.log('[TAS DEBUG] Status check successful:', response);
123
+ this.isCheckingStatus = false;
124
+ this.isStatusError = false;
125
+ this.statusErrorMessage = '';
126
+ }
127
+ },
128
+ error: (err) => {
129
+ console.error('[TAS DEBUG] Status check failed:', err);
130
+ this.isCheckingStatus = false;
131
+ this.isStatusError = true;
132
+ this.statusErrorMessage = err?.error?.message || err?.message || 'Error checking status';
133
+ },
134
+ }));
52
135
  }
53
136
  onClick() {
54
- if (!this.tenantId || !this.currentUser?.name) {
55
- console.error("Tenant ID or current user not available");
137
+ console.log('[TAS DEBUG] onClick called');
138
+ console.log('[TAS DEBUG] Inputs:', {
139
+ tenant: this.tenant,
140
+ entityId: this.entityId,
141
+ roomType: this.roomType,
142
+ businessRole: this.businessRole,
143
+ currentUser: this.currentUser,
144
+ });
145
+ if (!this.tenant || !this.currentUser?.name) {
146
+ console.error('[TAS DEBUG] Tenant or current user not available');
147
+ return;
148
+ }
149
+ if (!this.entityId) {
150
+ console.error('[TAS DEBUG] entityId is required');
56
151
  return;
57
152
  }
153
+ console.log('[TAS DEBUG] Validation passed, opening waiting room modal');
58
154
  this.openWaitingRoomModal();
59
155
  }
60
156
  openWaitingRoomModal() {
61
157
  this.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {
62
- size: "lg",
63
- windowClass: "tas-waiting-room-modal",
64
- backdrop: "static",
158
+ size: 'lg',
159
+ windowClass: 'tas-waiting-room-modal',
160
+ backdrop: 'static',
65
161
  keyboard: false,
66
- centered: true
162
+ centered: true,
67
163
  });
68
164
  // Pass all necessary inputs to the waiting room component
69
- this.currentModalRef.componentInstance.appointmentId = this.appointmentId;
70
- this.currentModalRef.componentInstance.product = this.product;
71
- this.currentModalRef.componentInstance.tenantId = this.tenantId;
165
+ this.currentModalRef.componentInstance.roomType = this.roomType;
166
+ this.currentModalRef.componentInstance.entityId = this.entityId;
167
+ this.currentModalRef.componentInstance.tenant = this.tenant;
168
+ this.currentModalRef.componentInstance.businessRole = this.businessRole;
72
169
  this.currentModalRef.componentInstance.currentUser = this.currentUser;
73
- this.currentModalRef.componentInstance.ownerUserIds = this.ownerUserIds;
74
- this.currentModalRef.componentInstance.regularUserIds = this.regularUserIds;
75
- this.currentModalRef.componentInstance.moderatorUserIds = this.moderatorUserIds;
76
- // Pass existing session ID if provided
77
- this.currentModalRef.componentInstance.existingSessionId = this.existingSessionId;
78
- this.currentModalRef.result.then(() => { this.currentModalRef = null; }, () => { this.currentModalRef = null; });
170
+ this.currentModalRef.result.then(() => {
171
+ this.currentModalRef = null;
172
+ }, () => {
173
+ this.currentModalRef = null;
174
+ });
79
175
  }
80
176
  openVideoCallModal(isReturningFromPip = false) {
81
177
  this.videoCallModalRef = this.modalService.open(TasVideocallComponent, {
82
178
  size: 'xl',
83
179
  windowClass: 'tas-video-modal',
84
180
  backdrop: 'static',
85
- keyboard: false
181
+ keyboard: false,
86
182
  });
87
183
  this.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;
88
184
  this.videoCallModalRef.componentInstance.token = this.tasService.token;
185
+ this.videoCallModalRef.componentInstance.businessRole = this.businessRole;
89
186
  this.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;
90
- this.videoCallModalRef.result.then(() => { this.videoCallModalRef = null; }, () => { this.videoCallModalRef = null; });
187
+ this.videoCallModalRef.result.then(() => {
188
+ this.videoCallModalRef = null;
189
+ }, () => {
190
+ this.videoCallModalRef = null;
191
+ });
91
192
  }
92
193
  }
93
- 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 });
94
- 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", existingSessionId: "existingSessionId", buttonText: "buttonText" }, 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\"> {{ buttonText }}</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\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"] }] });
95
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TasButtonComponent, decorators: [{
194
+ TasButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, deps: [{ token: i1.NgbModal }, { token: i2.TasService }], target: i0.ɵɵFactoryTarget.Component });
195
+ TasButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: TasButtonComponent, selector: "tas-btn", inputs: { roomType: "roomType", entityId: "entityId", tenant: "tenant", businessRole: "businessRole", currentUser: "currentUser" }, ngImport: i0, template: "<button\n *ngIf=\"isVisible\"\n type=\"button\"\n class=\"btn btn-primary tas-btn\"\n (click)=\"onClick()\"\n [disabled]=\"isDisabled\"\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\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"] }] });
196
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: TasButtonComponent, decorators: [{
96
197
  type: Component,
97
- 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\"> {{ buttonText }}</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\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"] }]
98
- }], ctorParameters: function () { return [{ type: i1.NgbModal }, { type: i2.TasService }]; }, propDecorators: { appointmentId: [{
99
- type: Input
100
- }], product: [{
198
+ args: [{ selector: 'tas-btn', template: "<button\n *ngIf=\"isVisible\"\n type=\"button\"\n class=\"btn btn-primary tas-btn\"\n (click)=\"onClick()\"\n [disabled]=\"isDisabled\"\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\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"] }]
199
+ }], ctorParameters: function () { return [{ type: i1.NgbModal }, { type: i2.TasService }]; }, propDecorators: { roomType: [{
101
200
  type: Input
102
- }], tenantId: [{
103
- type: Input
104
- }], currentUser: [{
201
+ }], entityId: [{
105
202
  type: Input
106
- }], ownerUserIds: [{
203
+ }], tenant: [{
107
204
  type: Input
108
- }], regularUserIds: [{
205
+ }], businessRole: [{
109
206
  type: Input
110
- }], moderatorUserIds: [{
111
- type: Input
112
- }], existingSessionId: [{
113
- type: Input
114
- }], buttonText: [{
207
+ }], currentUser: [{
115
208
  type: Input
116
209
  }] } });
117
- //# 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":"AAAA,OAAO,EAAE,SAAS,EAAqB,KAAK,EAAE,MAAM,eAAe,CAAC;AAEpE,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AACpC,OAAO,EAAkB,QAAQ,EAAE,MAAM,iCAAiC,CAAC;AAC3E,OAAO,EAAE,uBAAuB,EAAE,MAAM,gDAAgD,CAAC;AACzF,OAAO,EAAE,qBAAqB,EAAE,MAAM,0CAA0C,CAAC;;;;;AAQjF,MAAM,OAAO,kBAAkB;IAqB9B,YACS,YAAsB,EACtB,UAAsB;QADtB,iBAAY,GAAZ,YAAY,CAAU;QACtB,eAAU,GAAV,UAAU,CAAY;QAtBtB,kBAAa,GAAW,CAAC,CAAC;QAC1B,YAAO,GAAW,MAAM,CAAC;QACzB,aAAQ,GAAW,EAAE,CAAC;QAGtB,mBAAc,GAAa,EAAE,CAAC;QAC9B,qBAAgB,GAAa,EAAE,CAAC;QAEzC,sFAAsF;QAC7E,sBAAiB,GAAW,EAAE,CAAC;QAExC,mCAAmC;QAC1B,eAAU,GAAW,aAAa,CAAC;QAErC,cAAS,GAAG,KAAK,CAAC;QAEjB,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QACnC,oBAAe,GAAuB,IAAI,CAAC;QAC3C,sBAAiB,GAAuB,IAAI,CAAC;IAKlD,CAAC;IAEJ,QAAQ;QACP,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;YACzD,MAAM,IAAI,KAAK,CAAC,2EAA2E,CAAC,CAAC;SAC7F;QAED,6CAA6C;QAC7C,IAAI,CAAC,aAAa,CAAC,GAAG,CACrB,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;YAC1C,kDAAkD;YAClD,IAAI,IAAI,KAAK,QAAQ,CAAC,UAAU;gBAC/B,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE;gBAC9B,CAAC,IAAI,CAAC,iBAAiB,EAAE;gBACzB,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;gBAC5C,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;gBACpC,IAAI,SAAS,IAAI,KAAK,EAAE;oBACvB,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC;iBAC9B;aACD;YAED,wEAAwE;YACxE,IAAI,IAAI,KAAK,QAAQ,CAAC,GAAG,EAAE;gBAC1B,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;aAC9B;QACF,CAAC,CAAC,CACF,CAAC;IACH,CAAC;IAED,WAAW;QACV,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;IAED,OAAO;QACN,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,EAAE;YAC9C,OAAO,CAAC,KAAK,CAAC,yCAAyC,CAAC,CAAC;YACzD,OAAO;SACP;QAED,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAC7B,CAAC;IAEO,oBAAoB;QAC3B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,uBAAuB,EAAE;YACtE,IAAI,EAAE,IAAI;YACV,WAAW,EAAE,wBAAwB;YACrC,QAAQ,EAAE,QAAQ;YAClB,QAAQ,EAAE,KAAK;YACf,QAAQ,EAAE,IAAI;SACd,CAAC,CAAC;QAEH,0DAA0D;QAC1D,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;QAC1E,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC;QAC9D,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;QAChE,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACtE,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACxE,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;QAC5E,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,gBAAgB,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAEhF,uCAAuC;QACvC,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC;QAElF,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAC/B,GAAG,EAAE,GAAG,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,CAAC,CAAC,EACtC,GAAG,EAAE,GAAG,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,CAAC,CAAC,CACtC,CAAC;IACH,CAAC;IAEO,kBAAkB,CAAC,qBAA8B,KAAK;QAC7D,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,qBAAqB,EAAE;YACtE,IAAI,EAAE,IAAI;YACV,WAAW,EAAE,iBAAiB;YAC9B,QAAQ,EAAE,QAAQ;YAClB,QAAQ,EAAE,KAAK;SACf,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;QAC/E,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;QACvE,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,kBAAkB,GAAG,kBAAkB,CAAC;QAEjF,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,IAAI,CACjC,GAAG,EAAE,GAAG,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAAC,CAAC,EACxC,GAAG,EAAE,GAAG,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAAC,CAAC,CACxC,CAAC;IACH,CAAC;;+GA7GW,kBAAkB;mGAAlB,kBAAkB,yUCb/B,kUAUA;2FDGa,kBAAkB;kBAL9B,SAAS;+BACC,SAAS;wHAKV,aAAa;sBAArB,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,gBAAgB;sBAAxB,KAAK;gBAGG,iBAAiB;sBAAzB,KAAK;gBAGG,UAAU;sBAAlB,KAAK","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\t/** Optional: If provided, skips room creation and goes directly to getting a token */\n\t@Input() existingSessionId: string = \"\";\n\t\n\t/** Optional: Custom button text */\n\t@Input() buttonText: string = \"Iniciar TAS\";\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\t\t\n\t\t// Pass existing session ID if provided\n\t\tthis.currentModalRef.componentInstance.existingSessionId = this.existingSessionId;\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","<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\"> {{ buttonText }}</span>\n\t<span *ngIf=\"isLoading\"> Processing...</span>\n</button>\n"]}
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tas-btn.component.js","sourceRoot":"","sources":["../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.ts","../../../../../../projects/tas-uell-sdk/src/lib/components/tas-btn/tas-btn.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Input } from '@angular/core';\nimport { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap';\nimport { Subscription } from 'rxjs';\nimport {\n  TasCurrentUser,\n  TasRoomType,\n  ViewMode,\n  TasBusinessRole,\n} from '../../interfaces/tas.interfaces';\nimport { TasWaitingRoomComponent } from '../tas-waiting-room/tas-waiting-room.component';\nimport { TasVideocallComponent } from '../tas-videocall/tas-videocall.component';\nimport { TasService } from '../../services/tas.service';\n\n@Component({\n  selector: 'tas-btn',\n  templateUrl: './tas-btn.component.html',\n  styleUrls: ['./tas-btn.component.scss'],\n})\nexport class TasButtonComponent implements OnInit, OnDestroy {\n  // Status endpoint params\n  @Input() roomType: TasRoomType = TasRoomType.TAS;\n  @Input() entityId!: number;\n  @Input() tenant!: string;\n  @Input() businessRole: TasBusinessRole = TasBusinessRole.USER;\n\n  // User info for token request\n  @Input() currentUser!: TasCurrentUser;\n\n  public isLoading = false;\n  public readonly buttonText = 'Iniciar TAS';\n\n  // Status check state\n  public isCheckingStatus = false;\n  public isStatusError = false;\n  public statusErrorMessage = '';\n\n  private subscriptions = new Subscription();\n  private currentModalRef: NgbModalRef | null = null;\n  private videoCallModalRef: NgbModalRef | null = null;\n  private statusPollingInterval: ReturnType<typeof setInterval> | null = null;\n  private readonly STATUS_POLL_INTERVAL_MS = 30000; // 30 seconds\n\n  /** Whether user is backoffice (or admin/manager) */\n  public get isBackoffice(): boolean {\n    return (\n      this.businessRole === TasBusinessRole.BACKOFFICE ||\n      this.businessRole === TasBusinessRole.ADMIN_MANAGER ||\n      this.businessRole === TasBusinessRole.MANAGER\n    );\n  }\n\n  /** Whether the button should be visible */\n  public get isVisible(): boolean {\n    // Backoffice: always show (disabled if error)\n    // Other roles: hide if status error\n    if (this.isBackoffice) {\n      return true;\n    }\n    return !this.isStatusError;\n  }\n\n  /** Whether the button should be disabled */\n  public get isDisabled(): boolean {\n    return this.isLoading || this.isCheckingStatus || this.isStatusError;\n  }\n\n  constructor(\n    private modalService: NgbModal,\n    private tasService: TasService\n  ) {}\n\n  ngOnInit(): void {\n    // Subscribe to viewMode to handle PiP return\n    this.subscriptions.add(\n      this.tasService.viewMode$.subscribe((mode) => {\n        // When entering PiP, clear the videoCallModalRef since modal will close\n        if (mode === ViewMode.PIP) {\n          this.videoCallModalRef = null;\n        }\n      })\n    );\n\n    // Start status checking\n    this.startStatusPolling();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n    this.stopStatusPolling();\n  }\n\n  /**\n   * Start polling status every 30 seconds\n   */\n  private startStatusPolling(): void {\n    // Initial status check\n    this.checkStatus();\n\n    // Set up periodic polling\n    this.statusPollingInterval = setInterval(() => {\n      this.checkStatus();\n    }, this.STATUS_POLL_INTERVAL_MS);\n  }\n\n  /**\n   * Stop status polling\n   */\n  private stopStatusPolling(): void {\n    if (this.statusPollingInterval) {\n      clearInterval(this.statusPollingInterval);\n      this.statusPollingInterval = null;\n    }\n  }\n\n  /**\n   * Check status endpoint to determine if button should be enabled\n   */\n  private checkStatus(): void {\n    // Skip if required inputs are not available\n    if (!this.tenant || !this.entityId) {\n      console.log('[TAS DEBUG] checkStatus skipped - missing required inputs');\n      return;\n    }\n\n    this.isCheckingStatus = true;\n    this.statusErrorMessage = '';\n\n    console.log('[TAS DEBUG] checkStatus called with:', {\n      roomType: this.roomType,\n      entityId: this.entityId,\n      tenant: this.tenant,\n      businessRole: this.businessRole,\n    });\n\n    this.subscriptions.add(\n      this.tasService.getProxyVideoStatus({\n        roomType: this.roomType,\n        entityId: this.entityId,\n        tenant: this.tenant,\n        businessRole: this.businessRole,\n      }).subscribe({\n        next: (response: any) => {\n          // Check if response is actually an error (some HTTP adapters return errors in next)\n          // Also check for undefined/null or missing content\n          const isErrorResponse = !response || \n                                  !response.content ||\n                                  response?.ok === false || \n                                  response?.status >= 400 || \n                                  response?.error ||\n                                  response?.name === 'HttpErrorResponse';\n          \n          if (isErrorResponse) {\n            console.error('[TAS DEBUG] Status check returned error in response:', response);\n            this.isCheckingStatus = false;\n            this.isStatusError = true;\n            this.statusErrorMessage = response?.error?.message || response?.message || 'Error checking status';\n          } else {\n            console.log('[TAS DEBUG] Status check successful:', response);\n            this.isCheckingStatus = false;\n            this.isStatusError = false;\n            this.statusErrorMessage = '';\n          }\n        },\n        error: (err) => {\n          console.error('[TAS DEBUG] Status check failed:', err);\n          this.isCheckingStatus = false;\n          this.isStatusError = true;\n          this.statusErrorMessage = err?.error?.message || err?.message || 'Error checking status';\n        },\n      })\n    );\n  }\n\n  onClick(): void {\n    console.log('[TAS DEBUG] onClick called');\n    console.log('[TAS DEBUG] Inputs:', {\n      tenant: this.tenant,\n      entityId: this.entityId,\n      roomType: this.roomType,\n      businessRole: this.businessRole,\n      currentUser: this.currentUser,\n    });\n\n    if (!this.tenant || !this.currentUser?.name) {\n      console.error('[TAS DEBUG] Tenant or current user not available');\n      return;\n    }\n\n    if (!this.entityId) {\n      console.error('[TAS DEBUG] entityId is required');\n      return;\n    }\n\n    console.log('[TAS DEBUG] Validation passed, opening waiting room modal');\n    this.openWaitingRoomModal();\n  }\n\n  private openWaitingRoomModal(): void {\n    this.currentModalRef = this.modalService.open(TasWaitingRoomComponent, {\n      size: 'lg',\n      windowClass: 'tas-waiting-room-modal',\n      backdrop: 'static',\n      keyboard: false,\n      centered: true,\n    });\n\n    // Pass all necessary inputs to the waiting room component\n    this.currentModalRef.componentInstance.roomType = this.roomType;\n    this.currentModalRef.componentInstance.entityId = this.entityId;\n    this.currentModalRef.componentInstance.tenant = this.tenant;\n    this.currentModalRef.componentInstance.businessRole = this.businessRole;\n    this.currentModalRef.componentInstance.currentUser = this.currentUser;\n\n    this.currentModalRef.result.then(\n      () => {\n        this.currentModalRef = null;\n      },\n      () => {\n        this.currentModalRef = null;\n      }\n    );\n  }\n\n  private openVideoCallModal(isReturningFromPip: boolean = false): void {\n    this.videoCallModalRef = this.modalService.open(TasVideocallComponent, {\n      size: 'xl',\n      windowClass: 'tas-video-modal',\n      backdrop: 'static',\n      keyboard: false,\n    });\n\n    this.videoCallModalRef.componentInstance.sessionId = this.tasService.sessionId;\n    this.videoCallModalRef.componentInstance.token = this.tasService.token;\n    this.videoCallModalRef.componentInstance.businessRole = this.businessRole;\n    this.videoCallModalRef.componentInstance.isReturningFromPip = isReturningFromPip;\n\n    this.videoCallModalRef.result.then(\n      () => {\n        this.videoCallModalRef = null;\n      },\n      () => {\n        this.videoCallModalRef = null;\n      }\n    );\n  }\n}\n","<button\n  *ngIf=\"isVisible\"\n  type=\"button\"\n  class=\"btn btn-primary tas-btn\"\n  (click)=\"onClick()\"\n  [disabled]=\"isDisabled\"\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\n"]}