@snugdesk/avaya-ipo-widget 0.0.0-watch
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.
Potentially problematic release.
This version of @snugdesk/avaya-ipo-widget might be problematic. Click here for more details.
- package/README.md +138 -0
- package/fesm2022/snugdesk-avaya-ipo-widget.mjs +2892 -0
- package/fesm2022/snugdesk-avaya-ipo-widget.mjs.map +1 -0
- package/index.d.ts +445 -0
- package/package.json +37 -0
- package/src/assets/css/intl-tel-input-dropdown.css +0 -0
- package/src/assets/images/bg-app_color_line.gif +0 -0
- package/src/assets/images/icons/sd-backspace.png +0 -0
- package/src/assets/images/icons/sd-call_failed.gif +0 -0
- package/src/assets/images/icons/sd-call_history_not_found.gif +0 -0
- package/src/assets/images/incomingCall_avatar.gif +0 -0
- package/src/assets/images/logo-avaya.png +0 -0
- package/src/assets/images/logo-avaya_small_color.svg +20 -0
- package/src/assets/images/logo-avaya_small_gray.svg +16 -0
- package/src/assets/sounds/ringback_tone.mp3 +0 -0
|
@@ -0,0 +1,2892 @@
|
|
|
1
|
+
import * as i0 from '@angular/core';
|
|
2
|
+
import { Injectable, EventEmitter, Output, Input, Component, ViewChild, NgModule } from '@angular/core';
|
|
3
|
+
import { trigger, transition, style, animate } from '@angular/animations';
|
|
4
|
+
import * as i1$1 from '@angular/forms';
|
|
5
|
+
import { Validators, FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
6
|
+
import { BehaviorSubject, Subject, lastValueFrom, filter } from 'rxjs';
|
|
7
|
+
import * as i1 from '@angular/common/http';
|
|
8
|
+
import { HttpParams, HttpHeaders } from '@angular/common/http';
|
|
9
|
+
import * as i8 from '@snugdesk/core';
|
|
10
|
+
import * as i3 from '@angular/common';
|
|
11
|
+
import { CommonModule } from '@angular/common';
|
|
12
|
+
import * as i10 from '@angular/material/form-field';
|
|
13
|
+
import { MatFormFieldModule, MatError, MatFormField, MatLabel } from '@angular/material/form-field';
|
|
14
|
+
import * as i11 from '@angular/material/input';
|
|
15
|
+
import { MatInputModule } from '@angular/material/input';
|
|
16
|
+
import * as i5 from 'ngx-intl-tel-input';
|
|
17
|
+
import { CountryISO, SearchCountryField, NgxIntlTelInputModule } from 'ngx-intl-tel-input';
|
|
18
|
+
import { PhoneNumberUtil, PhoneNumberFormat } from 'google-libphonenumber';
|
|
19
|
+
import { JwtHelperService } from '@auth0/angular-jwt';
|
|
20
|
+
|
|
21
|
+
var LoginState;
|
|
22
|
+
(function (LoginState) {
|
|
23
|
+
LoginState["LOGIN_SUCCESS"] = "AWL_MSG_LOGIN_SUCCESS";
|
|
24
|
+
LoginState["LOGIN_FAILED"] = "AWL_MSG_LOGIN_FAILED";
|
|
25
|
+
LoginState["LOGIN_EMPTYUSERNAME"] = "AWL_MSG_LOGIN_EMPTYUSERNAME";
|
|
26
|
+
LoginState["LOGIN_EMPTYPASSWORD"] = "AWL_MSG_LOGIN_EMPTYPASSWORD";
|
|
27
|
+
LoginState["LOGIN_GW_NOTCONFIGURED"] = "AWL_MSG_LOGIN_GW_NOTCONFIGURED";
|
|
28
|
+
LoginState["LOGGEDOUT"] = "AWL_MSG_LOGGEDOUT";
|
|
29
|
+
LoginState["WEBSOCKET_FAILURE"] = "AWL_MSG_LOGIN_WEBSOCKET_FAILURE";
|
|
30
|
+
LoginState["DEVICEACCESS_FAILURE"] = "AWL_MSG_DEVICEACCESS_FAILURE";
|
|
31
|
+
LoginState["RELOGGED_IN"] = "AWL_MSG_RELOGGED_IN";
|
|
32
|
+
LoginState["RECONNECTING"] = "AWL_MSG_RECONNECTING";
|
|
33
|
+
LoginState["FAILING_OVER"] = "AWL_MSG_FAILING_OVER";
|
|
34
|
+
LoginState["FAILOVER_SUCCESS"] = "AWL_MSG_FAIL_OVER_SUCCESS";
|
|
35
|
+
LoginState["FAILOVER_FAILED"] = "AWL_MSG_FAIL_OVER_FAILED";
|
|
36
|
+
LoginState["FAILING_BACK"] = "AWL_MSG_FAILING_BACK";
|
|
37
|
+
LoginState["FAILBACK_SUCCESS"] = "AWL_MSG_FAIL_BACK_SUCCESS";
|
|
38
|
+
LoginState["FAILBACK_FAILED"] = "AWL_MSG_FAIL_BACK_FAILED";
|
|
39
|
+
})(LoginState || (LoginState = {}));
|
|
40
|
+
var CallState;
|
|
41
|
+
(function (CallState) {
|
|
42
|
+
CallState["IDLE"] = "AWL_MSG_CALL_IDLE";
|
|
43
|
+
CallState["PROGRESSING"] = "AWL_MSG_CALL_PROGRESSING";
|
|
44
|
+
CallState["RINGING"] = "AWL_MSG_CALL_RINGING";
|
|
45
|
+
CallState["CONNECTED"] = "AWL_MSG_CALL_CONNECTED";
|
|
46
|
+
CallState["DISCONNECTED"] = "AWL_MSG_CALL_DISCONNECTED";
|
|
47
|
+
CallState["FAILED"] = "AWL_MSG_CALL_FAILED";
|
|
48
|
+
CallState["INCOMING"] = "AWL_MSG_CALL_INCOMING";
|
|
49
|
+
CallState["HELD"] = "AWL_MSG_CALL_HELD";
|
|
50
|
+
CallState["TRANSFERRED"] = "AWL_MSG_CALL_TRANSFERRED";
|
|
51
|
+
CallState["FAREND_UPDATE"] = "AWL_MSG_CALL_FAREND_UPDATE";
|
|
52
|
+
})(CallState || (CallState = {}));
|
|
53
|
+
class AvayaIPOCallListener {
|
|
54
|
+
service;
|
|
55
|
+
constructor(service) {
|
|
56
|
+
this.service = service;
|
|
57
|
+
}
|
|
58
|
+
onNewIncomingCall = (callId, callObj, autoAnswer) => {
|
|
59
|
+
const farEndNumber = callObj.getFarEndNumber();
|
|
60
|
+
this.service.zone.run(() => {
|
|
61
|
+
this.service.incomingCallSubject.next({
|
|
62
|
+
callId,
|
|
63
|
+
farEndNumber,
|
|
64
|
+
});
|
|
65
|
+
});
|
|
66
|
+
if (!this.service.activeCallsMap.has(callId)) {
|
|
67
|
+
this.service.assignToFirstAvailableSlot(callId, callObj);
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
onCallStateChange = (callId, callObj, event) => {
|
|
71
|
+
// Raw callback from AWL (WebSocket JSON-RPC). Useful to see the original event.
|
|
72
|
+
console.debug('[AWL] onCallStateChange raw →', { callId, event });
|
|
73
|
+
const state = callObj.getCallState();
|
|
74
|
+
this.service.zone.run(() => {
|
|
75
|
+
this.service.callStateSubject.next({ callId, state });
|
|
76
|
+
});
|
|
77
|
+
if (event === 'CallTransferSuccessful') {
|
|
78
|
+
console.debug('Call transfer successful');
|
|
79
|
+
}
|
|
80
|
+
this.service.handleCallState(callObj);
|
|
81
|
+
try {
|
|
82
|
+
const remote = callObj?.getRemoteStream?.();
|
|
83
|
+
// if (remote instanceof MediaStream) {
|
|
84
|
+
// console.log('Remote stream found', remote);
|
|
85
|
+
// this.service.remoteStream = remote;
|
|
86
|
+
// this.service.remoteStream$.next(remote);
|
|
87
|
+
// }
|
|
88
|
+
if (remote instanceof MediaStream) {
|
|
89
|
+
const payload = { callId, stream: remote };
|
|
90
|
+
this.service.remoteStream = payload;
|
|
91
|
+
this.service.remoteStream$.next(payload);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
catch (err) {
|
|
95
|
+
console.warn('Failed to extract media streams', err);
|
|
96
|
+
}
|
|
97
|
+
};
|
|
98
|
+
onCallTerminate = (callId, reason) => {
|
|
99
|
+
this.service.releaseCallById(callId);
|
|
100
|
+
console.debug('Call terminated: ', reason);
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
class AvayaIPOService {
|
|
104
|
+
zone;
|
|
105
|
+
http;
|
|
106
|
+
cli; // This should be the type of your Avaya IPO CLI instance
|
|
107
|
+
remoteStream = null;
|
|
108
|
+
remoteStream$ = new BehaviorSubject(null);
|
|
109
|
+
localStream = null;
|
|
110
|
+
incomingCallSubject = new Subject();
|
|
111
|
+
callStateSubject = new Subject();
|
|
112
|
+
callSlots = {
|
|
113
|
+
'1': null,
|
|
114
|
+
'2': null,
|
|
115
|
+
'3': null,
|
|
116
|
+
};
|
|
117
|
+
activeCallsMap = new Map();
|
|
118
|
+
activeCallId = null;
|
|
119
|
+
currentcallId = null;
|
|
120
|
+
agentId = '';
|
|
121
|
+
loginStatusSubject = new BehaviorSubject(false);
|
|
122
|
+
loginStatus$ = this.loginStatusSubject.asObservable();
|
|
123
|
+
DOM_TAGS = {
|
|
124
|
+
localVideo: 'lclVideo',
|
|
125
|
+
remoteVideo: 'rmtVideo',
|
|
126
|
+
remoteAudio: 'rmtAudio',
|
|
127
|
+
localAudio: 'lclAudio',
|
|
128
|
+
};
|
|
129
|
+
CONFIG_TEMPLATE = {
|
|
130
|
+
serviceType: 'phone', // value can be 'phone' or 'agent'
|
|
131
|
+
enableVideo: false,
|
|
132
|
+
Gateway: { ip: 'snugipose.avayalab.in', port: '9443' },
|
|
133
|
+
Stunserver: { ip: '', port: '3478' },
|
|
134
|
+
Turnserver: { ip: '', port: '3478', user: '', pwd: '' },
|
|
135
|
+
AppData: { applicationID: '', applicationUA: '', appInstanceID: '' },
|
|
136
|
+
disableResiliency: false,
|
|
137
|
+
};
|
|
138
|
+
agentStatusApiUrl = 'https://v2xla9n0t6.execute-api.us-west-2.amazonaws.com/dev';
|
|
139
|
+
CALL_HISTORY_API_URL = 'https://ktgxyy2x2h.execute-api.us-west-2.amazonaws.com/avayaCallHistory';
|
|
140
|
+
prewarmedStream;
|
|
141
|
+
isSafari() {
|
|
142
|
+
return /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
|
|
143
|
+
}
|
|
144
|
+
_rpcPatched = false;
|
|
145
|
+
constructor(zone, http) {
|
|
146
|
+
this.zone = zone;
|
|
147
|
+
this.http = http;
|
|
148
|
+
}
|
|
149
|
+
init() {
|
|
150
|
+
if (typeof AWL !== 'undefined') {
|
|
151
|
+
this.cli = new AWL.client();
|
|
152
|
+
const cliAny = this.cli;
|
|
153
|
+
Object.defineProperty(cliAny, 'jsonRpc', {
|
|
154
|
+
configurable: true,
|
|
155
|
+
set: (rpcClient) => {
|
|
156
|
+
console.log('[RPC-PATCH] jsonRpc assigned:', rpcClient);
|
|
157
|
+
// restore the real property
|
|
158
|
+
delete cliAny.jsonRpc;
|
|
159
|
+
cliAny.jsonRpc = rpcClient;
|
|
160
|
+
// now do the patch
|
|
161
|
+
this._doPatch();
|
|
162
|
+
},
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
else {
|
|
166
|
+
console.warn('AWL is not yet loaded');
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
logIn(username, password) {
|
|
170
|
+
this.cli.enableLogging();
|
|
171
|
+
// this.cli.disableLogging();
|
|
172
|
+
const appInstanceId = this.cli.generateAppInstanceID();
|
|
173
|
+
const config = {
|
|
174
|
+
...this.CONFIG_TEMPLATE,
|
|
175
|
+
AppData: {
|
|
176
|
+
...this.CONFIG_TEMPLATE.AppData,
|
|
177
|
+
appInstanceID: appInstanceId,
|
|
178
|
+
},
|
|
179
|
+
};
|
|
180
|
+
console.debug('Parsed config: ', config);
|
|
181
|
+
const res_setConfiguration = this.cli.setConfiguration(config, this.handleConfigChange, this.handleRegistrationChange, new AvayaIPOCallListener(this), this.handleTokenRenewal);
|
|
182
|
+
console.debug('res_setConfiguration: ', res_setConfiguration);
|
|
183
|
+
if (res_setConfiguration === 'AWL_MSG_SETCONFIG_SUCCESS') {
|
|
184
|
+
console.debug('SET CONFIG SUCCESS');
|
|
185
|
+
}
|
|
186
|
+
const res_setDomElements = this.cli.setDomElements(this.DOM_TAGS);
|
|
187
|
+
console.debug('res_setDomElements: ', res_setDomElements);
|
|
188
|
+
if (res_setDomElements === 'AWL_MSG_SETDOM_FAILED') {
|
|
189
|
+
console.debug('SET DOM FAILED');
|
|
190
|
+
}
|
|
191
|
+
else {
|
|
192
|
+
console.debug('SET DOM PASS');
|
|
193
|
+
}
|
|
194
|
+
const bumbConnection = 'true';
|
|
195
|
+
this.cli.logIn(username, password, bumbConnection);
|
|
196
|
+
this.setAgentId(username);
|
|
197
|
+
}
|
|
198
|
+
logOut() {
|
|
199
|
+
this.cli.logOut();
|
|
200
|
+
}
|
|
201
|
+
getAlternateServerConfig() {
|
|
202
|
+
return this.cli.getAlternateServerConfig();
|
|
203
|
+
}
|
|
204
|
+
getDeviceList() {
|
|
205
|
+
this.cli.getDeviceList((deviceList) => {
|
|
206
|
+
console.debug('Device List: ', deviceList);
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
setDeviceIds(deviceIds) {
|
|
210
|
+
console.debug('setDeviceIds: ', deviceIds);
|
|
211
|
+
this.cli.setDeviceIds(deviceIds);
|
|
212
|
+
}
|
|
213
|
+
async makeCall(destination) {
|
|
214
|
+
await this.prewarmMic();
|
|
215
|
+
console.debug('makeCall devicelist: ', this.getDeviceList());
|
|
216
|
+
const freeSlot = ['1', '2', '3'].find((s) => this.callSlots[s] === null);
|
|
217
|
+
if (!freeSlot) {
|
|
218
|
+
console.warn('All call slots are currently occupied.');
|
|
219
|
+
return null;
|
|
220
|
+
}
|
|
221
|
+
// const callObj = this.cli.makeCall(destination, 'audio');
|
|
222
|
+
const callObj = this.cli.makeCall(destination, 'video', {
|
|
223
|
+
headers: { 'X-Original-Caller': 'asees' },
|
|
224
|
+
});
|
|
225
|
+
if (!callObj) {
|
|
226
|
+
console.error('Error! Cannot make call.');
|
|
227
|
+
return null;
|
|
228
|
+
}
|
|
229
|
+
const callId = callObj.getCallId();
|
|
230
|
+
this.activeCallsMap.set(callId, callObj);
|
|
231
|
+
this.callSlots[freeSlot] = callObj;
|
|
232
|
+
this.activeCallId = callId;
|
|
233
|
+
return freeSlot;
|
|
234
|
+
}
|
|
235
|
+
dropCall() {
|
|
236
|
+
if (!this.activeCallId) {
|
|
237
|
+
console.warn('No active call to drop.');
|
|
238
|
+
return;
|
|
239
|
+
}
|
|
240
|
+
// Find the slot that contains the active call using activeCallId
|
|
241
|
+
let targetSlot = null;
|
|
242
|
+
for (const [slot, callObj] of Object.entries(this.callSlots)) {
|
|
243
|
+
if (callObj && callObj.getCallId() === this.activeCallId) {
|
|
244
|
+
targetSlot = slot;
|
|
245
|
+
break;
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
if (!targetSlot) {
|
|
249
|
+
console.error('Active call slot not found for activeCallId:', this.activeCallId);
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
// Drop the active call from the identified slot
|
|
253
|
+
const callId = this.callSlots[targetSlot].getCallId();
|
|
254
|
+
const res_dropCall = this.cli.dropCall(callId);
|
|
255
|
+
console.debug('res_dropCall: ', res_dropCall);
|
|
256
|
+
// Remove the call from the activeCalls map and the slot
|
|
257
|
+
this.activeCallsMap.delete(callId);
|
|
258
|
+
this.callSlots[targetSlot] = null;
|
|
259
|
+
// Reset activeCallId
|
|
260
|
+
this.activeCallId = null;
|
|
261
|
+
}
|
|
262
|
+
doMute() {
|
|
263
|
+
const res_doMute = this.cli.doMute(this.activeCallId);
|
|
264
|
+
console.debug('res_doMute: ', res_doMute);
|
|
265
|
+
}
|
|
266
|
+
doUnMute() {
|
|
267
|
+
const res_doUnMute = this.cli.doUnMute(this.activeCallId);
|
|
268
|
+
console.debug('res_doUnMute: ', res_doUnMute);
|
|
269
|
+
}
|
|
270
|
+
doHold() {
|
|
271
|
+
const res_doHold = this.cli.doHold(this.activeCallId);
|
|
272
|
+
console.debug('res_doHold: ', res_doHold);
|
|
273
|
+
}
|
|
274
|
+
doUnHold() {
|
|
275
|
+
const res_doUnHold = this.cli.doUnHold(this.activeCallId);
|
|
276
|
+
console.debug('res_doUnHold: ', res_doUnHold);
|
|
277
|
+
}
|
|
278
|
+
pauseVideo() {
|
|
279
|
+
const res_pauseVideo = this.cli.pauseVideo(this.activeCallId);
|
|
280
|
+
console.debug('res_pauseVideo: ', res_pauseVideo);
|
|
281
|
+
}
|
|
282
|
+
playVideo() {
|
|
283
|
+
const res_playVideo = this.cli.playVideo(this.activeCallId);
|
|
284
|
+
console.debug('res_playVideo: ', res_playVideo);
|
|
285
|
+
}
|
|
286
|
+
transferCall(target, type) {
|
|
287
|
+
const res_transferCall = this.cli.transferCall(target, this.activeCallId, type);
|
|
288
|
+
console.debug('res_transferCall: ', res_transferCall);
|
|
289
|
+
}
|
|
290
|
+
sendDTMF(digit) {
|
|
291
|
+
const res_sendDTMF = this.cli.sendDTMF(this.activeCallId, digit);
|
|
292
|
+
console.debug('res_sendDTMF: ', res_sendDTMF);
|
|
293
|
+
}
|
|
294
|
+
answerCall() {
|
|
295
|
+
if (this.callSlots['1']) {
|
|
296
|
+
this.activeCallId = this.callSlots['1'].getCallId();
|
|
297
|
+
const res_answerCall = this.cli.answerCall(this.activeCallId);
|
|
298
|
+
console.debug('res_answerCall: ', res_answerCall);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
addVideo() {
|
|
302
|
+
const res_addVideo = this.cli.addVideo(this.activeCallId);
|
|
303
|
+
console.debug('res_addVideo: ', res_addVideo);
|
|
304
|
+
}
|
|
305
|
+
removeVideo() {
|
|
306
|
+
const res_removeVideo = this.cli.removeVideo(this.activeCallId);
|
|
307
|
+
console.debug('res_removeVideo: ', res_removeVideo);
|
|
308
|
+
}
|
|
309
|
+
clearCallSlot() {
|
|
310
|
+
this.callSlots['1'] = null;
|
|
311
|
+
}
|
|
312
|
+
setAgentId(id) {
|
|
313
|
+
this.agentId = id;
|
|
314
|
+
}
|
|
315
|
+
agentIdle() {
|
|
316
|
+
const params = new HttpParams()
|
|
317
|
+
.set('action', 'release')
|
|
318
|
+
.set('agentId', this.agentId);
|
|
319
|
+
const headers = new HttpHeaders({ 'Content-Type': 'application/json' });
|
|
320
|
+
this.http.post(this.agentStatusApiUrl, {}, { params, headers }).subscribe((response) => console.debug('Agent marked as idle: ', response), (error) => console.error('Error updating agent status: ', error));
|
|
321
|
+
}
|
|
322
|
+
agentBusy() {
|
|
323
|
+
const params = new HttpParams()
|
|
324
|
+
.set('action', 'confirm')
|
|
325
|
+
.set('agentId', this.agentId);
|
|
326
|
+
const headers = new HttpHeaders({ 'Content-Type': 'application/json' });
|
|
327
|
+
this.http.post(this.agentStatusApiUrl, {}, { params, headers }).subscribe((response) => console.debug('Agent marked as busy: ', response), (error) => console.error('Error updating agent status: ', error));
|
|
328
|
+
}
|
|
329
|
+
agentAutoRelease() {
|
|
330
|
+
setInterval(() => {
|
|
331
|
+
const params = new HttpParams().set('action', 'auto-release');
|
|
332
|
+
const headers = new HttpHeaders({ 'Content-Type': 'application/json' });
|
|
333
|
+
this.http.get(this.agentStatusApiUrl, { params, headers }).subscribe((response) => console.debug('Auto-release triggered: ', response), (error) => console.error('Auto-release failed: ', error));
|
|
334
|
+
}, 2 * 60 * 1000);
|
|
335
|
+
}
|
|
336
|
+
handleConfigChange = (res_configChange) => {
|
|
337
|
+
console.debug('res_configChange: ', res_configChange);
|
|
338
|
+
};
|
|
339
|
+
handleRegistrationChange = (res_registrationChange) => {
|
|
340
|
+
// console.debug('res_registrationChange: ', res_registrationChange);
|
|
341
|
+
this.zone.run(() => {
|
|
342
|
+
this.loginStatusSubject.next(res_registrationChange);
|
|
343
|
+
});
|
|
344
|
+
console.debug('res_registrationChange: ', res_registrationChange);
|
|
345
|
+
if (res_registrationChange?.result === LoginState.LOGIN_SUCCESS) {
|
|
346
|
+
const rpc = this.cli.jsonRpc;
|
|
347
|
+
if (rpc && rpc.ws) {
|
|
348
|
+
rpc.ws.evtConnectionOpen.attach(() => {
|
|
349
|
+
console.log('[RPC-PATCH] WebSocket open — JSON-RPC ready');
|
|
350
|
+
this.patchRpcForTransfers(this.agentId);
|
|
351
|
+
});
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
};
|
|
355
|
+
handleTokenRenewal = (res_tokenRenewal) => {
|
|
356
|
+
console.debug('res_tokenRenewal: ', res_tokenRenewal);
|
|
357
|
+
if (res_tokenRenewal.result == 'AWL_MSG_TOKEN_RENEW_SUCCESS') {
|
|
358
|
+
console.debug('Token is successfully renewed');
|
|
359
|
+
}
|
|
360
|
+
else {
|
|
361
|
+
console.debug('Token renewal failed. reason: ', res_tokenRenewal.reason);
|
|
362
|
+
}
|
|
363
|
+
};
|
|
364
|
+
assignToFirstAvailableSlot(callId, callObj) {
|
|
365
|
+
const freeSlot = Object.entries(this.callSlots).find(([_, obj]) => obj === null);
|
|
366
|
+
if (freeSlot) {
|
|
367
|
+
this.callSlots[freeSlot[0]] = callObj;
|
|
368
|
+
this.activeCallsMap.set(callId, callObj);
|
|
369
|
+
// If we don't already have an active call tracked, set this incoming call as active.
|
|
370
|
+
if (!this.activeCallId) {
|
|
371
|
+
this.activeCallId = callId;
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
else {
|
|
375
|
+
console.warn('All call slots are occupied');
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
getSlotForCallId(callId) {
|
|
379
|
+
for (const [slot, callObj] of Object.entries(this.callSlots)) {
|
|
380
|
+
if (callObj && callObj.getCallId() === callId) {
|
|
381
|
+
return slot;
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
return null;
|
|
385
|
+
}
|
|
386
|
+
releaseCallById(callId) {
|
|
387
|
+
for (const [slot, obj] of Object.entries(this.callSlots)) {
|
|
388
|
+
if (obj?.getCallId() === callId) {
|
|
389
|
+
this.callSlots[slot] = null;
|
|
390
|
+
this.activeCallsMap.delete(callId);
|
|
391
|
+
break;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
handleCallState(callObj) {
|
|
396
|
+
console.debug(`Call state updated for Call ID: ${callObj?.getCallId()}`, callObj);
|
|
397
|
+
console.debug(`Call state updated for Call ID: ${callObj?.getSipUri()}`, callObj);
|
|
398
|
+
this.currentcallId = callObj.getCallId();
|
|
399
|
+
const state = callObj.getCallState();
|
|
400
|
+
if (!this.activeCallId) {
|
|
401
|
+
this.activeCallId = callObj.getCallId();
|
|
402
|
+
}
|
|
403
|
+
switch (state) {
|
|
404
|
+
case CallState.FAILED:
|
|
405
|
+
case CallState.DISCONNECTED:
|
|
406
|
+
console.debug('Call Failed or DISCONNECTED! Releasing call...');
|
|
407
|
+
this.releaseCallById(callObj.getCallId());
|
|
408
|
+
// Signal UI to reset to dialpad (ensure change detection runs)
|
|
409
|
+
this.zone.run(() => {
|
|
410
|
+
this.callStateSubject.next({
|
|
411
|
+
callId: callObj.getCallId(),
|
|
412
|
+
state: CallState.DISCONNECTED,
|
|
413
|
+
});
|
|
414
|
+
});
|
|
415
|
+
break;
|
|
416
|
+
case CallState.FAREND_UPDATE:
|
|
417
|
+
console.debug('FarEnd Updated:', callObj.getFarEndNumber(), callObj.getFarEndName(), callObj.getSipUri());
|
|
418
|
+
break;
|
|
419
|
+
default:
|
|
420
|
+
console.debug('Call state: ', state);
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
swapLines() {
|
|
424
|
+
let currentSlotKey = null;
|
|
425
|
+
// Find the slot key of the current active call
|
|
426
|
+
for (const [key, call] of Object.entries(this.callSlots)) {
|
|
427
|
+
if (call && call.getCallId() === this.activeCallId) {
|
|
428
|
+
currentSlotKey = key;
|
|
429
|
+
break;
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
if (!currentSlotKey) {
|
|
433
|
+
console.warn('Active call slot not found.');
|
|
434
|
+
return;
|
|
435
|
+
}
|
|
436
|
+
// Find another active call to swap to
|
|
437
|
+
for (const [key, call] of Object.entries(this.callSlots)) {
|
|
438
|
+
if (key === currentSlotKey)
|
|
439
|
+
continue;
|
|
440
|
+
if (call) {
|
|
441
|
+
const currentCall = this.callSlots[currentSlotKey];
|
|
442
|
+
console.log(`Holding call in slot ${currentSlotKey}`);
|
|
443
|
+
this.cli.doHold(currentCall?.getCallId());
|
|
444
|
+
console.log(`Unholding call in slot ${key}`);
|
|
445
|
+
this.cli.doUnHold(call.getCallId());
|
|
446
|
+
this.activeCallId = call.getCallId();
|
|
447
|
+
return;
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
console.warn('No other active calls to swap to.');
|
|
451
|
+
}
|
|
452
|
+
swapToSlot(targetSlot) {
|
|
453
|
+
// 1) Find the current slot
|
|
454
|
+
let currentSlotKey = null;
|
|
455
|
+
for (const [key, call] of Object.entries(this.callSlots)) {
|
|
456
|
+
if (call && call.getCallId() === this.activeCallId) {
|
|
457
|
+
currentSlotKey = key;
|
|
458
|
+
break;
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
if (!currentSlotKey) {
|
|
462
|
+
console.warn("Active call slot not found, can't swap.");
|
|
463
|
+
return;
|
|
464
|
+
}
|
|
465
|
+
// 2) If targetSlot is the same as currentSlotKey, do nothing
|
|
466
|
+
if (targetSlot === currentSlotKey) {
|
|
467
|
+
console.log(`Already on slot ${targetSlot}, no swap needed.`);
|
|
468
|
+
return;
|
|
469
|
+
}
|
|
470
|
+
// 3) If the target slot has a call
|
|
471
|
+
const targetCall = this.callSlots[targetSlot];
|
|
472
|
+
if (!targetCall) {
|
|
473
|
+
console.warn(`No call in slot ${targetSlot}, can't unhold that slot.`);
|
|
474
|
+
return;
|
|
475
|
+
}
|
|
476
|
+
// 4) Hold the old slot
|
|
477
|
+
const currentCall = this.callSlots[currentSlotKey];
|
|
478
|
+
console.log(`Holding call in slot ${currentSlotKey}`);
|
|
479
|
+
this.cli.doHold(currentCall?.getCallId());
|
|
480
|
+
// 5) Unhold the target slot
|
|
481
|
+
console.log(`Unholding call in slot ${targetSlot}`);
|
|
482
|
+
this.cli.doUnHold(targetCall.getCallId());
|
|
483
|
+
// 6) Update activeCallId
|
|
484
|
+
this.activeCallId = targetCall.getCallId();
|
|
485
|
+
console.log(`Swapped from slot ${currentSlotKey} to slot ${targetSlot}.`);
|
|
486
|
+
}
|
|
487
|
+
dropCallBySlot(targetSlot) {
|
|
488
|
+
// Check if there is a call in the requested slot.
|
|
489
|
+
const targetCall = this.callSlots[targetSlot];
|
|
490
|
+
if (!targetCall) {
|
|
491
|
+
console.warn(`No active call in slot ${targetSlot} to drop.`);
|
|
492
|
+
return;
|
|
493
|
+
}
|
|
494
|
+
// Retrieve the call ID from the target call.
|
|
495
|
+
const callId = targetCall.getCallId();
|
|
496
|
+
console.log(`Dropping call in slot ${targetSlot}, callId: ${callId}`);
|
|
497
|
+
// Issue the drop command through Avaya's CLI.
|
|
498
|
+
const res_dropCall = this.cli.dropCall(callId);
|
|
499
|
+
console.debug('dropCallBySlot response:', res_dropCall);
|
|
500
|
+
// Clean up: remove the call from the slot and from the activeCallsMap.
|
|
501
|
+
this.activeCallsMap.delete(callId);
|
|
502
|
+
this.callSlots[targetSlot] = null;
|
|
503
|
+
// If the activeCallId matches the dropped call, reset it.
|
|
504
|
+
if (this.activeCallId === callId) {
|
|
505
|
+
this.activeCallId = null;
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
mergeCall() {
|
|
509
|
+
const activeCallObjs = Object.entries(this.callSlots)
|
|
510
|
+
.filter(([_, callObj]) => callObj !== null)
|
|
511
|
+
.map(([_, callObj]) => callObj);
|
|
512
|
+
if (activeCallObjs.length < 2) {
|
|
513
|
+
console.warn('At least two active calls are required to perform merge.');
|
|
514
|
+
return;
|
|
515
|
+
}
|
|
516
|
+
const [sourceCall, targetCall] = activeCallObjs;
|
|
517
|
+
try {
|
|
518
|
+
const sourceCallId = sourceCall.getCallId();
|
|
519
|
+
const targetCallId = targetCall.getCallId();
|
|
520
|
+
console.log(`Merging calls: from ${sourceCallId} to ${targetCallId}`);
|
|
521
|
+
this.cli.transferCall(targetCallId, sourceCallId, 'Attended');
|
|
522
|
+
}
|
|
523
|
+
catch (error) {
|
|
524
|
+
console.error('Merge call failed:', error);
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
logCall(callData) {
|
|
528
|
+
// 1) Build the HttpParams with "action=log"
|
|
529
|
+
const params = new HttpParams().set('action', 'log');
|
|
530
|
+
// 2) Set headers to JSON
|
|
531
|
+
const headers = new HttpHeaders({ 'Content-Type': 'application/json' });
|
|
532
|
+
// 3) Make a POST request, passing callData in the body, and use params + headers
|
|
533
|
+
return this.http.post(this.CALL_HISTORY_API_URL, callData, {
|
|
534
|
+
params,
|
|
535
|
+
headers,
|
|
536
|
+
});
|
|
537
|
+
}
|
|
538
|
+
getActiveCallId() {
|
|
539
|
+
if (!this.activeCallId) {
|
|
540
|
+
return null;
|
|
541
|
+
}
|
|
542
|
+
const callObj = this.activeCallsMap.get(this.activeCallId);
|
|
543
|
+
if (!callObj) {
|
|
544
|
+
return null;
|
|
545
|
+
}
|
|
546
|
+
return callObj.getCallId();
|
|
547
|
+
}
|
|
548
|
+
async initLocalMic() {
|
|
549
|
+
try {
|
|
550
|
+
const micStream = await navigator.mediaDevices.getUserMedia({
|
|
551
|
+
audio: true,
|
|
552
|
+
});
|
|
553
|
+
console.log('Captured mic stream:', micStream);
|
|
554
|
+
this.localStream = micStream;
|
|
555
|
+
}
|
|
556
|
+
catch (err) {
|
|
557
|
+
console.error('Failed to capture microphone:', err);
|
|
558
|
+
}
|
|
559
|
+
}
|
|
560
|
+
/////////////////////////////////////////////////////////////
|
|
561
|
+
patchRpcForTransfers(agentId) {
|
|
562
|
+
if (this._rpcPatched) {
|
|
563
|
+
console.debug('[RPC-PATCH] already patched');
|
|
564
|
+
return;
|
|
565
|
+
}
|
|
566
|
+
const jsonRpc = this.cli.jsonRpc;
|
|
567
|
+
if (!jsonRpc || !jsonRpc.ws) {
|
|
568
|
+
console.warn('[RPC-PATCH] JSON-RPC not ready yet');
|
|
569
|
+
return;
|
|
570
|
+
}
|
|
571
|
+
console.debug('[RPC-PATCH] patching ws.send now');
|
|
572
|
+
this._rpcPatched = true;
|
|
573
|
+
const ws = jsonRpc.ws;
|
|
574
|
+
const origSend = ws.send.bind(ws);
|
|
575
|
+
ws.send = (raw) => {
|
|
576
|
+
// log every outbound frame so you can see your injection happen
|
|
577
|
+
try {
|
|
578
|
+
const msg = JSON.parse(raw);
|
|
579
|
+
if (msg.msgname === 'makeCall' || msg.msgname === 'transferCall') {
|
|
580
|
+
console.debug(`[RPC-PATCH] sending ${msg.msgname} →`, msg.params);
|
|
581
|
+
}
|
|
582
|
+
}
|
|
583
|
+
catch (_) {
|
|
584
|
+
/* not JSON, ignore */
|
|
585
|
+
}
|
|
586
|
+
return origSend(raw);
|
|
587
|
+
};
|
|
588
|
+
}
|
|
589
|
+
async makeCallWithCustomData(destination, dataToSend) {
|
|
590
|
+
// 0) Safari race fix
|
|
591
|
+
await this.prewarmMic();
|
|
592
|
+
this.patchRpcForTransfers(dataToSend);
|
|
593
|
+
// 2) find a free slot
|
|
594
|
+
const freeSlot = ['1', '2', '3'].find((s) => this.callSlots[s] === null);
|
|
595
|
+
if (!freeSlot) {
|
|
596
|
+
console.warn('All call slots are occupied.');
|
|
597
|
+
return null;
|
|
598
|
+
}
|
|
599
|
+
// 3) kick off the call via AWL
|
|
600
|
+
const callObj = this.cli.makeCall(destination, 'audio');
|
|
601
|
+
if (!callObj) {
|
|
602
|
+
console.error('Error! Cannot make call.');
|
|
603
|
+
return null;
|
|
604
|
+
}
|
|
605
|
+
// 4) book-keep exactly as you do today
|
|
606
|
+
const callId = callObj.getCallId();
|
|
607
|
+
this.activeCallsMap.set(callId, callObj);
|
|
608
|
+
this.callSlots[freeSlot] = callObj;
|
|
609
|
+
this.activeCallId = callId;
|
|
610
|
+
console.log(`📡 Started call in slot ${freeSlot} with customData="${dataToSend}"`);
|
|
611
|
+
return freeSlot;
|
|
612
|
+
}
|
|
613
|
+
transferCallWithTag(target, type) {
|
|
614
|
+
// 1) patch the JSON-RPC so the next transferCall has our tag
|
|
615
|
+
this.patchRpcForTransfers(this.agentId);
|
|
616
|
+
// 2) do the normal AWL transfer
|
|
617
|
+
const res = this.cli.transferCall(target, this.activeCallId, type);
|
|
618
|
+
console.debug('res_transferCall:', res);
|
|
619
|
+
}
|
|
620
|
+
_doPatch() {
|
|
621
|
+
// if (this._rpcPatched) return;
|
|
622
|
+
this._rpcPatched = true;
|
|
623
|
+
console.log('[RPC-PATCH] patching WebSocket now');
|
|
624
|
+
const jsonRpc = this.cli.jsonRpc;
|
|
625
|
+
const ws = jsonRpc.ws;
|
|
626
|
+
const origSend = ws.send.bind(ws);
|
|
627
|
+
ws.send = (data) => {
|
|
628
|
+
console.log('[RPC-PATCH] raw ws.send →', data);
|
|
629
|
+
try {
|
|
630
|
+
const msg = JSON.parse(data);
|
|
631
|
+
if (msg.msgname === 'makeCall' || msg.msgname === 'transferCall') {
|
|
632
|
+
// inject your field
|
|
633
|
+
msg.params.transferingAgent = this.agentId;
|
|
634
|
+
const patched = JSON.stringify(msg);
|
|
635
|
+
console.log(`[RPC-PATCH] sending ${msg.msgname} with params →`, msg.params);
|
|
636
|
+
return origSend(patched);
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
catch {
|
|
640
|
+
// non-JSON or other messages
|
|
641
|
+
}
|
|
642
|
+
return origSend(data);
|
|
643
|
+
};
|
|
644
|
+
}
|
|
645
|
+
/** Pre-warm microphone so AWL always has a local stream when it tries addStream() */
|
|
646
|
+
async prewarmMic() {
|
|
647
|
+
// Reuse if we already have a healthy stream
|
|
648
|
+
if (this.prewarmedStream && this.prewarmedStream.getTracks().length) {
|
|
649
|
+
return this.prewarmedStream;
|
|
650
|
+
}
|
|
651
|
+
// On Safari, first call should be simple (no exact device constraints)
|
|
652
|
+
const audio = this.isSafari() ? true : { echoCancellation: false, noiseSuppression: false, autoGainControl: false };
|
|
653
|
+
const stream = await navigator.mediaDevices.getUserMedia({ audio });
|
|
654
|
+
this.prewarmedStream = stream;
|
|
655
|
+
this.localStream = stream;
|
|
656
|
+
// Hand it to the polyfill fallback so addStream(undefined) still works
|
|
657
|
+
window.__LAST_LOCAL_STREAM = stream;
|
|
658
|
+
// (Optional) Nudge autoplay policy: attach to a muted hidden <audio> once
|
|
659
|
+
try {
|
|
660
|
+
const a = document.createElement('audio');
|
|
661
|
+
a.muted = true;
|
|
662
|
+
a.autoplay = true;
|
|
663
|
+
a.style.display = 'none';
|
|
664
|
+
a.srcObject = stream;
|
|
665
|
+
document.body.appendChild(a);
|
|
666
|
+
await a.play().catch(() => { });
|
|
667
|
+
// keep element for lifetime of the page to avoid GC issues
|
|
668
|
+
}
|
|
669
|
+
catch { }
|
|
670
|
+
return stream;
|
|
671
|
+
}
|
|
672
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOService, deps: [{ token: i0.NgZone }, { token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
673
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOService, providedIn: 'root' });
|
|
674
|
+
}
|
|
675
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOService, decorators: [{
|
|
676
|
+
type: Injectable,
|
|
677
|
+
args: [{ providedIn: 'root' }]
|
|
678
|
+
}], ctorParameters: () => [{ type: i0.NgZone }, { type: i1.HttpClient }] });
|
|
679
|
+
|
|
680
|
+
class CountryService {
|
|
681
|
+
httpClient;
|
|
682
|
+
countriesData;
|
|
683
|
+
// private static readonly COUNTRY_ENDPOINT = `https://assets.snugdesk.com/metadata/countries.json`;
|
|
684
|
+
static COUNTRY_ENDPOINT = `https://snugdesk-assets.s3.amazonaws.com/metadata/countries.json`;
|
|
685
|
+
constructor(httpClient) {
|
|
686
|
+
this.httpClient = httpClient;
|
|
687
|
+
}
|
|
688
|
+
async getAllCountries() {
|
|
689
|
+
if (!this.countriesData) {
|
|
690
|
+
this.countriesData = await lastValueFrom(this.httpClient.get(CountryService.COUNTRY_ENDPOINT));
|
|
691
|
+
}
|
|
692
|
+
return this.countriesData;
|
|
693
|
+
}
|
|
694
|
+
async getCountryFromCallingCode(countryCallingCode) {
|
|
695
|
+
await this.getAllCountries();
|
|
696
|
+
return this.countriesData?.find((c) => c.callingCode === countryCallingCode);
|
|
697
|
+
}
|
|
698
|
+
async getCountryFromCountryCode(countryCode) {
|
|
699
|
+
await this.getAllCountries();
|
|
700
|
+
return this.countriesData?.find((c) => c.code === countryCode);
|
|
701
|
+
}
|
|
702
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CountryService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
703
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CountryService, providedIn: 'root' });
|
|
704
|
+
}
|
|
705
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CountryService, decorators: [{
|
|
706
|
+
type: Injectable,
|
|
707
|
+
args: [{
|
|
708
|
+
providedIn: 'root',
|
|
709
|
+
}]
|
|
710
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }] });
|
|
711
|
+
|
|
712
|
+
class PhoneNumberLookupService {
|
|
713
|
+
httpClient;
|
|
714
|
+
constructor(httpClient) {
|
|
715
|
+
this.httpClient = httpClient;
|
|
716
|
+
}
|
|
717
|
+
async lookupPhoneNumber(phoneNumber, hlr) {
|
|
718
|
+
const apiEndpoint = hlr
|
|
719
|
+
? `https://api.snugdesk.com/phone-numbers/hlr/${encodeURIComponent(phoneNumber)}`
|
|
720
|
+
: `https://api.snugdesk.com/phone-numbers/${encodeURIComponent(phoneNumber)}`;
|
|
721
|
+
const res_lookupPhoneNumber = await lastValueFrom(this.httpClient.get(apiEndpoint));
|
|
722
|
+
// console.debug('res_lookupPhoneNumber: ', res_lookupPhoneNumber);
|
|
723
|
+
return res_lookupPhoneNumber;
|
|
724
|
+
}
|
|
725
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: PhoneNumberLookupService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
726
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: PhoneNumberLookupService, providedIn: 'root' });
|
|
727
|
+
}
|
|
728
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: PhoneNumberLookupService, decorators: [{
|
|
729
|
+
type: Injectable,
|
|
730
|
+
args: [{
|
|
731
|
+
providedIn: 'root',
|
|
732
|
+
}]
|
|
733
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }] });
|
|
734
|
+
|
|
735
|
+
// import { Injectable } from '@angular/core';
|
|
736
|
+
// import { HttpClient } from '@angular/common/http';
|
|
737
|
+
// import { lastValueFrom } from 'rxjs';
|
|
738
|
+
// import { CallMetadata } from '../models/active-call.model';
|
|
739
|
+
// export interface RecordingSession {
|
|
740
|
+
// audioContext: AudioContext;
|
|
741
|
+
// recordingStream: MediaStreamAudioDestinationNode;
|
|
742
|
+
// recorder: MediaRecorder;
|
|
743
|
+
// metadata: CallMetadata;
|
|
744
|
+
// }
|
|
745
|
+
// @Injectable({ providedIn: 'root' })
|
|
746
|
+
// export class RecordingManagerService {
|
|
747
|
+
// private sessions = new Map<string, RecordingSession>();
|
|
748
|
+
// private readonly MIME_TYPE_PREFERRED = 'audio/webm;codecs=opus';
|
|
749
|
+
// private readonly MIME_TYPE_FALLBACK = 'audio/webm';
|
|
750
|
+
// private readonly CHUNK_DURATION_MS = 5000;
|
|
751
|
+
// private readonly AUDIO_CONTEXT_CLOSE_DELAY_MS = 200;
|
|
752
|
+
// private readonly CALL_HISTORY_API_URL =
|
|
753
|
+
// 'https://ktgxyy2x2h.execute-api.us-west-2.amazonaws.com/avayaCallHistory';
|
|
754
|
+
// constructor(private http: HttpClient) {}
|
|
755
|
+
// // prettier-ignore
|
|
756
|
+
// startRecording(localMediaStream: MediaStream, remoteMediaStream: MediaStream, metadata: RecordingSession['metadata']): RecordingSession | undefined {
|
|
757
|
+
// // Stop any existing recording session for this callId
|
|
758
|
+
// this.stopRecording(metadata.callId);
|
|
759
|
+
// if (!localMediaStream || !remoteMediaStream) {
|
|
760
|
+
// console.warn(`[${metadata.callId}] Skipped start recording: missing remote or local stream`);
|
|
761
|
+
// return undefined;
|
|
762
|
+
// }
|
|
763
|
+
// const audioContext = new AudioContext();
|
|
764
|
+
// const recordingStream = audioContext.createMediaStreamDestination();
|
|
765
|
+
// // console.debug(`[${metadata.callId}] Connecting local and remote audio streams to recording context`);
|
|
766
|
+
// audioContext.createMediaStreamSource(localMediaStream).connect(recordingStream);
|
|
767
|
+
// audioContext.createMediaStreamSource(remoteMediaStream).connect(recordingStream);
|
|
768
|
+
// const mimeType = MediaRecorder.isTypeSupported(this.MIME_TYPE_PREFERRED)
|
|
769
|
+
// ? this.MIME_TYPE_PREFERRED
|
|
770
|
+
// : this.MIME_TYPE_FALLBACK;
|
|
771
|
+
// const recorder = new MediaRecorder(recordingStream.stream, { mimeType });
|
|
772
|
+
// recorder.ondataavailable = (evt: BlobEvent) => this.handleDataAvailable(evt, metadata);
|
|
773
|
+
// recorder.onerror = (err) => console.error(`[${metadata.callId}] MediaRecorder error:`, err);
|
|
774
|
+
// recorder.start(this.CHUNK_DURATION_MS);
|
|
775
|
+
// console.debug(`[${metadata.callId}] MediaRecorder started (chunk size: ${this.CHUNK_DURATION_MS / 1000}s)`);
|
|
776
|
+
// const session: RecordingSession = {
|
|
777
|
+
// audioContext,
|
|
778
|
+
// recordingStream,
|
|
779
|
+
// recorder,
|
|
780
|
+
// metadata,
|
|
781
|
+
// };
|
|
782
|
+
// this.sessions.set(metadata.callId, session);
|
|
783
|
+
// return session;
|
|
784
|
+
// }
|
|
785
|
+
// getSession(callId: string): RecordingSession | undefined {
|
|
786
|
+
// return this.sessions.get(callId);
|
|
787
|
+
// }
|
|
788
|
+
// stopRecording(callId: string): void {
|
|
789
|
+
// const session = this.sessions.get(callId);
|
|
790
|
+
// if (!session) return;
|
|
791
|
+
// const { recorder, audioContext } = session;
|
|
792
|
+
// if (recorder?.state === 'recording') {
|
|
793
|
+
// recorder.stop();
|
|
794
|
+
// console.debug(`[${callId}] MediaRecorder stopped`);
|
|
795
|
+
// }
|
|
796
|
+
// setTimeout(() => {
|
|
797
|
+
// if (audioContext.state !== 'closed') {
|
|
798
|
+
// audioContext.close();
|
|
799
|
+
// console.debug(`[${callId}] AudioContext closed`);
|
|
800
|
+
// }
|
|
801
|
+
// }, this.AUDIO_CONTEXT_CLOSE_DELAY_MS);
|
|
802
|
+
// this.sessions.delete(callId);
|
|
803
|
+
// }
|
|
804
|
+
// // prettier-ignore
|
|
805
|
+
// private async handleDataAvailable(evt: BlobEvent, metadata: RecordingSession['metadata']): Promise<void> {
|
|
806
|
+
// if (!evt.data || evt.data.size === 0) return;
|
|
807
|
+
// const filename = this.generateFilename(metadata.callId);
|
|
808
|
+
// // console.debug(`[${metadata.callId}] Recording chunk ready → preparing to upload as "${filename}"`);
|
|
809
|
+
// try {
|
|
810
|
+
// await this.uploadCallRecording(metadata, evt.data, filename);
|
|
811
|
+
// console.debug(`[${metadata.callId}] Upload successful → "${filename}"`);
|
|
812
|
+
// } catch (err) {
|
|
813
|
+
// console.error(`[${metadata.callId}] Upload failed → "${filename}"`, err);
|
|
814
|
+
// }
|
|
815
|
+
// }
|
|
816
|
+
// // prettier-ignore
|
|
817
|
+
// private async uploadCallRecording(metadata: RecordingSession['metadata'], file: Blob, filename?: string): Promise<{ success: boolean; message: string; s3Key?: string }> {
|
|
818
|
+
// try {
|
|
819
|
+
// if (!filename) {
|
|
820
|
+
// const timestamp = new Date()
|
|
821
|
+
// .toISOString()
|
|
822
|
+
// .replace(/[:.]/g, '')
|
|
823
|
+
// .replace('T', '-')
|
|
824
|
+
// .slice(0, 15);
|
|
825
|
+
// filename = `${timestamp}-${metadata.agentId}.webm`;
|
|
826
|
+
// console.debug(`[${metadata.callId}] filename generated: "${filename}"`);
|
|
827
|
+
// }
|
|
828
|
+
// // console.debug(`[${metadata.callId}] Converting Blob to Base64 for upload`);
|
|
829
|
+
// const base64 = await this.convertBlobToBase64(file);
|
|
830
|
+
// const body = base64.split(',')[1];
|
|
831
|
+
// const payload = {
|
|
832
|
+
// ...metadata,
|
|
833
|
+
// filename,
|
|
834
|
+
// isBase64: true,
|
|
835
|
+
// body,
|
|
836
|
+
// };
|
|
837
|
+
// const url = `${this.CALL_HISTORY_API_URL}?action=uploadRecording`;
|
|
838
|
+
// console.debug(`[${metadata.callId}] Sending recording chunk to ${url}`);
|
|
839
|
+
// console.debug(`[${metadata.callId}] Payload: `, payload);
|
|
840
|
+
// const res: any = await lastValueFrom(
|
|
841
|
+
// this.http.post(url, payload, {
|
|
842
|
+
// headers: { 'Content-Type': 'application/json' },
|
|
843
|
+
// responseType: 'json',
|
|
844
|
+
// })
|
|
845
|
+
// );
|
|
846
|
+
// console.debug(`[${metadata.callId}] Upload response: `, res);
|
|
847
|
+
// return {
|
|
848
|
+
// success: true,
|
|
849
|
+
// message: res?.message || 'Upload successful',
|
|
850
|
+
// s3Key: res?.s3Key,
|
|
851
|
+
// };
|
|
852
|
+
// } catch (err: any) {
|
|
853
|
+
// console.error(`[${metadata.callId}] Upload failed: `, err);
|
|
854
|
+
// return {
|
|
855
|
+
// success: false,
|
|
856
|
+
// message: err?.message || 'Upload failed',
|
|
857
|
+
// };
|
|
858
|
+
// }
|
|
859
|
+
// }
|
|
860
|
+
// private convertBlobToBase64(blob: Blob): Promise<string> {
|
|
861
|
+
// return new Promise((resolve, reject) => {
|
|
862
|
+
// const reader = new FileReader();
|
|
863
|
+
// reader.onerror = reject;
|
|
864
|
+
// reader.onload = () => resolve(reader.result as string);
|
|
865
|
+
// reader.readAsDataURL(blob);
|
|
866
|
+
// });
|
|
867
|
+
// }
|
|
868
|
+
// private generateFilename(callId: string): string {
|
|
869
|
+
// const now = new Date();
|
|
870
|
+
// const dateStr = now.toISOString().slice(0, 10).replace(/-/g, '');
|
|
871
|
+
// const timeStr = now.toTimeString().slice(0, 8).replace(/:/g, '');
|
|
872
|
+
// const msStr = now.getMilliseconds().toString().padStart(3, '0');
|
|
873
|
+
// const filename = `${callId}-${dateStr}-${timeStr}${msStr}.webm`;
|
|
874
|
+
// console.debug(`[${callId}] Generated filename: "${filename}"`);
|
|
875
|
+
// return filename;
|
|
876
|
+
// }
|
|
877
|
+
// }
|
|
878
|
+
class RecordingManagerService {
|
|
879
|
+
http;
|
|
880
|
+
sessions = new Map();
|
|
881
|
+
MIME_TYPE_PREFERRED = 'audio/webm;codecs=opus';
|
|
882
|
+
MIME_TYPE_FALLBACK = 'audio/webm';
|
|
883
|
+
CHUNK_DURATION_MS = 5000;
|
|
884
|
+
AUDIO_CONTEXT_CLOSE_DELAY_MS = 200;
|
|
885
|
+
CALL_HISTORY_API_URL = 'https://ktgxyy2x2h.execute-api.us-west-2.amazonaws.com/avayaCallHistory';
|
|
886
|
+
constructor(http) {
|
|
887
|
+
this.http = http;
|
|
888
|
+
}
|
|
889
|
+
// ---- Public API ----------------------------------------------------------
|
|
890
|
+
startRecording(localMediaStream, remoteMediaStream, metadata) {
|
|
891
|
+
// stop any previous session for this call
|
|
892
|
+
this.stopRecording(metadata.callId);
|
|
893
|
+
if (!localMediaStream || !remoteMediaStream) {
|
|
894
|
+
console.warn(`[${metadata.callId}] Skipped start recording: missing remote or local stream`);
|
|
895
|
+
return undefined;
|
|
896
|
+
}
|
|
897
|
+
const audioContext = new AudioContext();
|
|
898
|
+
const recordingStream = audioContext.createMediaStreamDestination();
|
|
899
|
+
audioContext.createMediaStreamSource(localMediaStream).connect(recordingStream);
|
|
900
|
+
audioContext.createMediaStreamSource(remoteMediaStream).connect(recordingStream);
|
|
901
|
+
const session = {
|
|
902
|
+
audioContext,
|
|
903
|
+
recordingStream,
|
|
904
|
+
metadata,
|
|
905
|
+
stopping: false,
|
|
906
|
+
chunkIndex: 0,
|
|
907
|
+
recorder: null,
|
|
908
|
+
chunkTimer: undefined
|
|
909
|
+
};
|
|
910
|
+
this.sessions.set(metadata.callId, session);
|
|
911
|
+
// kick off the chunk loop
|
|
912
|
+
this.startNextChunk(session);
|
|
913
|
+
return session;
|
|
914
|
+
}
|
|
915
|
+
getSession(callId) {
|
|
916
|
+
return this.sessions.get(callId);
|
|
917
|
+
}
|
|
918
|
+
stopRecording(callId) {
|
|
919
|
+
const session = this.sessions.get(callId);
|
|
920
|
+
if (!session)
|
|
921
|
+
return;
|
|
922
|
+
session.stopping = true;
|
|
923
|
+
if (session.chunkTimer) {
|
|
924
|
+
clearTimeout(session.chunkTimer);
|
|
925
|
+
session.chunkTimer = undefined;
|
|
926
|
+
}
|
|
927
|
+
const rec = session.recorder;
|
|
928
|
+
if (rec && rec.state === 'recording') {
|
|
929
|
+
try {
|
|
930
|
+
rec.stop(); // this will still produce the final (playable) chunk
|
|
931
|
+
}
|
|
932
|
+
catch { }
|
|
933
|
+
}
|
|
934
|
+
// close audio context a bit later
|
|
935
|
+
setTimeout(() => {
|
|
936
|
+
if (session.audioContext.state !== 'closed') {
|
|
937
|
+
session.audioContext.close();
|
|
938
|
+
console.debug(`[${session.metadata.callId}] AudioContext closed`);
|
|
939
|
+
}
|
|
940
|
+
}, this.AUDIO_CONTEXT_CLOSE_DELAY_MS);
|
|
941
|
+
this.sessions.delete(callId);
|
|
942
|
+
}
|
|
943
|
+
// ---- Internals -----------------------------------------------------------
|
|
944
|
+
/** Start a new MediaRecorder for the next chunk so the blob has its own headers */
|
|
945
|
+
startNextChunk(session) {
|
|
946
|
+
if (session.stopping)
|
|
947
|
+
return;
|
|
948
|
+
const mimeType = MediaRecorder.isTypeSupported(this.MIME_TYPE_PREFERRED)
|
|
949
|
+
? this.MIME_TYPE_PREFERRED
|
|
950
|
+
: this.MIME_TYPE_FALLBACK;
|
|
951
|
+
const recorder = new MediaRecorder(session.recordingStream.stream, { mimeType });
|
|
952
|
+
session.recorder = recorder;
|
|
953
|
+
const myChunkIdx = session.chunkIndex++;
|
|
954
|
+
recorder.ondataavailable = async (evt) => {
|
|
955
|
+
if (!evt.data || evt.data.size === 0)
|
|
956
|
+
return;
|
|
957
|
+
// give each chunk its own unique filename (includes index)
|
|
958
|
+
const filename = this.generateChunkFilename(session.metadata.callId, myChunkIdx);
|
|
959
|
+
try {
|
|
960
|
+
await this.uploadCallRecording(session.metadata, evt.data, filename);
|
|
961
|
+
console.debug(`[${session.metadata.callId}] ✅ uploaded chunk → ${filename}`);
|
|
962
|
+
}
|
|
963
|
+
catch (err) {
|
|
964
|
+
console.error(`[${session.metadata.callId}] ❌ upload failed → ${filename}`, err);
|
|
965
|
+
}
|
|
966
|
+
};
|
|
967
|
+
recorder.onerror = (err) => {
|
|
968
|
+
console.error(`[${session.metadata.callId}] MediaRecorder error:`, err);
|
|
969
|
+
};
|
|
970
|
+
// when this recorder is stopped, immediately start the next one (unless stopping)
|
|
971
|
+
recorder.onstop = () => {
|
|
972
|
+
if (session.stopping)
|
|
973
|
+
return;
|
|
974
|
+
// start the next chunk on the next tick to avoid overlap
|
|
975
|
+
setTimeout(() => this.startNextChunk(session), 0);
|
|
976
|
+
};
|
|
977
|
+
// Start recording this chunk (no timeslice!) and schedule a stop after N ms.
|
|
978
|
+
recorder.start();
|
|
979
|
+
console.debug(`[${session.metadata.callId}] MediaRecorder started for chunk #${myChunkIdx}`);
|
|
980
|
+
session.chunkTimer = window.setTimeout(() => {
|
|
981
|
+
if (!session.stopping && recorder.state === 'recording') {
|
|
982
|
+
// ensure we flush a complete, headered WebM
|
|
983
|
+
recorder.stop();
|
|
984
|
+
console.debug(`[${session.metadata.callId}] MediaRecorder stopped for chunk #${myChunkIdx}`);
|
|
985
|
+
}
|
|
986
|
+
}, this.CHUNK_DURATION_MS);
|
|
987
|
+
}
|
|
988
|
+
// Upload a single chunk (Blob) as base64 payload
|
|
989
|
+
async uploadCallRecording(metadata, file, filename) {
|
|
990
|
+
try {
|
|
991
|
+
if (!filename) {
|
|
992
|
+
filename = this.generateFilename(metadata.callId, metadata.agentId);
|
|
993
|
+
}
|
|
994
|
+
const base64 = await this.convertBlobToBase64(file);
|
|
995
|
+
const body = base64.split(',')[1];
|
|
996
|
+
const payload = {
|
|
997
|
+
...metadata,
|
|
998
|
+
filename,
|
|
999
|
+
isBase64: true,
|
|
1000
|
+
body,
|
|
1001
|
+
};
|
|
1002
|
+
const url = `${this.CALL_HISTORY_API_URL}?action=uploadRecording`;
|
|
1003
|
+
const res = await lastValueFrom(this.http.post(url, payload, {
|
|
1004
|
+
headers: { 'Content-Type': 'application/json' },
|
|
1005
|
+
responseType: 'json',
|
|
1006
|
+
}));
|
|
1007
|
+
return {
|
|
1008
|
+
success: true,
|
|
1009
|
+
message: res?.message || 'Upload successful',
|
|
1010
|
+
s3Key: res?.s3Key,
|
|
1011
|
+
};
|
|
1012
|
+
}
|
|
1013
|
+
catch (err) {
|
|
1014
|
+
return {
|
|
1015
|
+
success: false,
|
|
1016
|
+
message: err?.message || 'Upload failed',
|
|
1017
|
+
};
|
|
1018
|
+
}
|
|
1019
|
+
}
|
|
1020
|
+
convertBlobToBase64(blob) {
|
|
1021
|
+
return new Promise((resolve, reject) => {
|
|
1022
|
+
const reader = new FileReader();
|
|
1023
|
+
reader.onerror = reject;
|
|
1024
|
+
reader.onload = () => resolve(reader.result);
|
|
1025
|
+
reader.readAsDataURL(blob);
|
|
1026
|
+
});
|
|
1027
|
+
}
|
|
1028
|
+
generateFilename(callId, agentId) {
|
|
1029
|
+
const timestamp = new Date().toISOString().replace(/[:.]/g, '').replace('T', '-').slice(0, 15);
|
|
1030
|
+
return `${timestamp}-${agentId}-${callId}.webm`;
|
|
1031
|
+
}
|
|
1032
|
+
generateChunkFilename(callId, chunkIndex) {
|
|
1033
|
+
const now = new Date();
|
|
1034
|
+
const dateStr = now.toISOString().slice(0, 10).replace(/-/g, '');
|
|
1035
|
+
const timeStr = now.toTimeString().slice(0, 8).replace(/:/g, '');
|
|
1036
|
+
const msStr = now.getMilliseconds().toString().padStart(3, '0');
|
|
1037
|
+
const part = String(chunkIndex).padStart(4, '0');
|
|
1038
|
+
return `${callId}-${dateStr}-${timeStr}${msStr}-part${part}.webm`;
|
|
1039
|
+
}
|
|
1040
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: RecordingManagerService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1041
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: RecordingManagerService, providedIn: 'root' });
|
|
1042
|
+
}
|
|
1043
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: RecordingManagerService, decorators: [{
|
|
1044
|
+
type: Injectable,
|
|
1045
|
+
args: [{ providedIn: 'root' }]
|
|
1046
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }] });
|
|
1047
|
+
|
|
1048
|
+
// import { Injectable } from '@angular/core';
|
|
1049
|
+
// import { BehaviorSubject } from 'rxjs';
|
|
1050
|
+
// import { CallMetadata } from '../models/active-call.model';
|
|
1051
|
+
// @Injectable({ providedIn: 'root' })
|
|
1052
|
+
// export class CallSocketService {
|
|
1053
|
+
// private socket: WebSocket | null = null;
|
|
1054
|
+
// readonly message$ = new BehaviorSubject<any>(null);
|
|
1055
|
+
// open(metadata: CallMetadata): void {
|
|
1056
|
+
// if (this.socket?.readyState === WebSocket.OPEN) return;
|
|
1057
|
+
// if (this.socket?.readyState === WebSocket.CONNECTING) return;
|
|
1058
|
+
// this.socket = new WebSocket(
|
|
1059
|
+
// 'wss://hymtqh7k82.execute-api.us-west-2.amazonaws.com/callSession'
|
|
1060
|
+
// );
|
|
1061
|
+
// this.socket.onopen = () => {
|
|
1062
|
+
// console.warn('[WS] open----------------------------------------------');
|
|
1063
|
+
// this.socket!.send(JSON.stringify(metadata));
|
|
1064
|
+
// };
|
|
1065
|
+
// this.socket.onmessage = (ev) => {
|
|
1066
|
+
// try {
|
|
1067
|
+
// this.message$.next(JSON.parse(ev.data));
|
|
1068
|
+
// } catch {
|
|
1069
|
+
// this.message$.next(ev.data);
|
|
1070
|
+
// }
|
|
1071
|
+
// };
|
|
1072
|
+
// this.socket.onerror = (ev) => {
|
|
1073
|
+
// console.error('[WS] error', ev);
|
|
1074
|
+
// };
|
|
1075
|
+
// this.socket.onclose = () => {
|
|
1076
|
+
// console.warn('[WS] closed');
|
|
1077
|
+
// this.socket = null;
|
|
1078
|
+
// };
|
|
1079
|
+
// }
|
|
1080
|
+
// send(payload: object): void {
|
|
1081
|
+
// if (this.socket?.readyState === WebSocket.OPEN) {
|
|
1082
|
+
// this.socket.send(JSON.stringify(payload));
|
|
1083
|
+
// } else {
|
|
1084
|
+
// console.warn('[WS] not open; dropping', payload);
|
|
1085
|
+
// }
|
|
1086
|
+
// }
|
|
1087
|
+
// close(): void {
|
|
1088
|
+
// this.socket?.close();
|
|
1089
|
+
// this.socket = null;
|
|
1090
|
+
// }
|
|
1091
|
+
// }
|
|
1092
|
+
class CallSocketService {
|
|
1093
|
+
sockets = new Map();
|
|
1094
|
+
subjects = new Map();
|
|
1095
|
+
open(metadata) {
|
|
1096
|
+
const { callId } = metadata;
|
|
1097
|
+
const existing = this.sockets.get(callId);
|
|
1098
|
+
if (existing && (existing.readyState === WebSocket.OPEN || existing.readyState === WebSocket.CONNECTING)) {
|
|
1099
|
+
return;
|
|
1100
|
+
}
|
|
1101
|
+
const ws = new WebSocket('wss://hymtqh7k82.execute-api.us-west-2.amazonaws.com/callSession');
|
|
1102
|
+
this.sockets.set(callId, ws);
|
|
1103
|
+
if (!this.subjects.has(callId))
|
|
1104
|
+
this.subjects.set(callId, new BehaviorSubject(null));
|
|
1105
|
+
ws.onopen = () => {
|
|
1106
|
+
console.warn('[WS Open ]', { callId });
|
|
1107
|
+
ws.send(JSON.stringify(metadata));
|
|
1108
|
+
};
|
|
1109
|
+
ws.onmessage = (ev) => {
|
|
1110
|
+
const subj = this.subjects.get(callId);
|
|
1111
|
+
if (!subj)
|
|
1112
|
+
return;
|
|
1113
|
+
try {
|
|
1114
|
+
subj.next(JSON.parse(ev.data));
|
|
1115
|
+
}
|
|
1116
|
+
catch {
|
|
1117
|
+
subj.next(ev.data);
|
|
1118
|
+
}
|
|
1119
|
+
};
|
|
1120
|
+
ws.onerror = (ev) => {
|
|
1121
|
+
console.warn('[WS Error]', { callId, ev });
|
|
1122
|
+
};
|
|
1123
|
+
ws.onclose = () => {
|
|
1124
|
+
console.warn('[WS Closed]', { callId });
|
|
1125
|
+
this.sockets.delete(callId);
|
|
1126
|
+
// keep subject around if you want late subscribers to see last value.
|
|
1127
|
+
// If you want to free it too, uncomment the next line:
|
|
1128
|
+
// this.subjects.delete(callId);
|
|
1129
|
+
};
|
|
1130
|
+
}
|
|
1131
|
+
messages$(callId) {
|
|
1132
|
+
return this.subjects.get(callId);
|
|
1133
|
+
}
|
|
1134
|
+
send(callId, payload) {
|
|
1135
|
+
const ws = this.sockets.get(callId);
|
|
1136
|
+
if (ws?.readyState === WebSocket.OPEN) {
|
|
1137
|
+
ws.send(JSON.stringify(payload));
|
|
1138
|
+
}
|
|
1139
|
+
else {
|
|
1140
|
+
console.warn('[WS NotOpen ]', { callId, dropping: payload });
|
|
1141
|
+
}
|
|
1142
|
+
}
|
|
1143
|
+
close(callId) {
|
|
1144
|
+
const ws = this.sockets.get(callId);
|
|
1145
|
+
if (ws) {
|
|
1146
|
+
ws.close();
|
|
1147
|
+
this.sockets.delete(callId);
|
|
1148
|
+
// optional: also remove the subject
|
|
1149
|
+
// this.subjects.delete(callId);
|
|
1150
|
+
}
|
|
1151
|
+
}
|
|
1152
|
+
closeAll() {
|
|
1153
|
+
for (const [callId, ws] of this.sockets.entries()) {
|
|
1154
|
+
try {
|
|
1155
|
+
ws.close();
|
|
1156
|
+
}
|
|
1157
|
+
catch { }
|
|
1158
|
+
console.warn('[WS Closed]', { callId });
|
|
1159
|
+
}
|
|
1160
|
+
this.sockets.clear();
|
|
1161
|
+
// optional: also clear subjects
|
|
1162
|
+
// this.subjects.clear();
|
|
1163
|
+
}
|
|
1164
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CallSocketService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1165
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CallSocketService, providedIn: 'root' });
|
|
1166
|
+
}
|
|
1167
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CallSocketService, decorators: [{
|
|
1168
|
+
type: Injectable,
|
|
1169
|
+
args: [{ providedIn: 'root' }]
|
|
1170
|
+
}] });
|
|
1171
|
+
|
|
1172
|
+
class CallHistoryComponent {
|
|
1173
|
+
http;
|
|
1174
|
+
authenticationService;
|
|
1175
|
+
cdr;
|
|
1176
|
+
mode = 'dial';
|
|
1177
|
+
makeCallEv = new EventEmitter();
|
|
1178
|
+
transferEv = new EventEmitter();
|
|
1179
|
+
CALL_HISTORY_API_URL = 'https://ktgxyy2x2h.execute-api.us-west-2.amazonaws.com/avayaCallHistory';
|
|
1180
|
+
showHistoryLoader = false;
|
|
1181
|
+
callHistory = [];
|
|
1182
|
+
recentCallactiveTab = 'recentCall';
|
|
1183
|
+
constructor(http, authenticationService, cdr) {
|
|
1184
|
+
this.http = http;
|
|
1185
|
+
this.authenticationService = authenticationService;
|
|
1186
|
+
this.cdr = cdr;
|
|
1187
|
+
}
|
|
1188
|
+
async ngOnInit() {
|
|
1189
|
+
await this.loadCallHistory();
|
|
1190
|
+
}
|
|
1191
|
+
recentSelectTab(tab) {
|
|
1192
|
+
this.recentCallactiveTab = tab;
|
|
1193
|
+
}
|
|
1194
|
+
async loadCallHistory() {
|
|
1195
|
+
this.showHistoryLoader = true;
|
|
1196
|
+
try {
|
|
1197
|
+
// console.log('tennat id fetched ',this.authenticationService.tenant$.getValue());
|
|
1198
|
+
const tenantId = this.authenticationService.getTenantId();
|
|
1199
|
+
console.log('tennat id fetched ', tenantId);
|
|
1200
|
+
// const userId = this.authenticationService.getUserId();
|
|
1201
|
+
// console.log('tennat id fetched ',userId);
|
|
1202
|
+
const userId = '5001';
|
|
1203
|
+
const response = await this.getCallHistory(tenantId, userId);
|
|
1204
|
+
const items = response.items || [];
|
|
1205
|
+
this.callHistory = items.sort((a, b) => new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime());
|
|
1206
|
+
console.log('Fetched call history: ', this.callHistory);
|
|
1207
|
+
}
|
|
1208
|
+
catch (error) {
|
|
1209
|
+
console.error('Failed to load call history:', error);
|
|
1210
|
+
}
|
|
1211
|
+
finally {
|
|
1212
|
+
this.showHistoryLoader = false;
|
|
1213
|
+
this.cdr.detectChanges();
|
|
1214
|
+
}
|
|
1215
|
+
}
|
|
1216
|
+
getCallHistory(tenantId, agentId) {
|
|
1217
|
+
// GET /avayaCallHistory?action=history&tenantId=xxx&agentId=yyy
|
|
1218
|
+
return lastValueFrom(this.http.get(`${this.CALL_HISTORY_API_URL}?action=history&tenantId=${tenantId}&agentId=${agentId}`));
|
|
1219
|
+
}
|
|
1220
|
+
makeCall(phoneNumber) {
|
|
1221
|
+
this.makeCallEv.emit(phoneNumber);
|
|
1222
|
+
}
|
|
1223
|
+
get callerDropped() {
|
|
1224
|
+
return this.callHistory.filter((item) => {
|
|
1225
|
+
const duration = Number(item?.duration ?? 0);
|
|
1226
|
+
return item.disconnectSide === 'CALLER' && duration <= 0;
|
|
1227
|
+
});
|
|
1228
|
+
}
|
|
1229
|
+
selectFromHistory(item) {
|
|
1230
|
+
if (this.mode === 'transfer') {
|
|
1231
|
+
this.transferEv.emit(item);
|
|
1232
|
+
return;
|
|
1233
|
+
}
|
|
1234
|
+
const dialable = this.resolveDialable(item) || (item?.phoneNumber ?? '').toString().replace(/\s+/g, '');
|
|
1235
|
+
if (!dialable) {
|
|
1236
|
+
console.warn('No dialable number found in history item:', item);
|
|
1237
|
+
return;
|
|
1238
|
+
}
|
|
1239
|
+
this.makeCallEv.emit(dialable);
|
|
1240
|
+
}
|
|
1241
|
+
resolveDialable(item) {
|
|
1242
|
+
const clean = (v) => (v ?? '').toString().replace(/\s+/g, '');
|
|
1243
|
+
return (clean(item?.resolvedDialable) ||
|
|
1244
|
+
clean(item?.lookupPhoneNumber?.['local-number']) ||
|
|
1245
|
+
clean(item?.lookupPhoneNumber?.lookupPhoneNumber?.['local-number']) ||
|
|
1246
|
+
clean(item?.lookupPhoneNumber?.normalizedPhoneNumber) ||
|
|
1247
|
+
null);
|
|
1248
|
+
}
|
|
1249
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CallHistoryComponent, deps: [{ token: i1.HttpClient }, { token: i8.SnugdeskAuthenticationService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
1250
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.9", type: CallHistoryComponent, isStandalone: false, selector: "app-call-history", inputs: { mode: "mode" }, outputs: { makeCallEv: "makeCallEv", transferEv: "transferEv" }, ngImport: i0, template: "@if (showHistoryLoader) {\n<div class=\"history_loader\">\n <div class=\"spinner\"></div>\n <div class=\"loader_text\">Fetching call history...</div>\n</div>\n} @else {\n\n<div class=\"recent_container_outer\">\n <div class=\"recent_options_header\">\n <div class=\"recent_inner\">\n <button class=\"recent_btn\" (click)=\"recentSelectTab('recentCall')\"\n [class.active]=\"recentCallactiveTab === 'recentCall'\">\n Recent\n </button>\n <button class=\"recent_btn\" (click)=\"recentSelectTab('missedCall')\"\n [class.active]=\"recentCallactiveTab === 'missedCall'\">\n Missed Call\n </button>\n </div>\n </div>\n\n <h2 class=\"recent_heading\">Recent</h2>\n\n <div class=\"container_recent_search\">\n <input type=\"search\" placeholder=\"Type to search\" />\n </div>\n\n @if (recentCallactiveTab === 'recentCall') {\n <div class=\"recent_tab_container\">\n <div class=\"container_recent_list\">\n <div class=\"container_recent_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (call of callHistory; track $index) {\n <div class=\"outer_container_recent\">\n <div class=\"container_recent_item\">\n <!-- avatar -->\n <div class=\"container_recent_item_image\">\n <div class=\"container_recent_item_image_inner grd1\">\n {{ call.phoneNumber?.country?.code | uppercase }}\n </div>\n </div>\n\n <!-- middle -->\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">\n {{ call.phoneNumber?.normalizedPhoneNumber }}\n </div>\n\n <div class=\"recent_time_back\" [ngClass]=\"{\n missed_call_color: call.status === 'MISSED',\n caller_drop: call.disconnectSide === 'CALLER' && (call.duration ?? 0) <= 0\n }\">\n <!-- Direction icon (generic handset) -->\n <svg viewBox=\"0 0 512 512\" width=\"12px\" fill=\"currentColor\" aria-hidden=\"true\">\n <path\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n {{ call.direction }}\n @if (call.disconnectSide === 'CALLER' && (call.duration ?? 0) <= 0) {\n <span class=\"disconnect_badge\">Caller hung up</span>\n }\n </div>\n </div>\n\n <!-- right time -->\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">\n {{ call.timestamp | date : \"dd/MM/yyyy\" }}\n </span>\n </div>\n </div>\n\n <!-- call button -->\n <div class=\"calling_btn\">\n <div class=\"calling_btn_inner\">\n <!-- <button\n class=\"button_recent_call\"\n (click)=\"makeCall(call.phoneNumber?.normalizedPhoneNumber)\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button> -->\n\n @if (mode === 'transfer') {\n <button class=\"button_recent_call\" (click)=\"transferEv.emit(call)\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n } @else {\n <button class=\"button_recent_call\" (click)=\"\n makeCall(call.phoneNumber?.normalizedPhoneNumber)\n \">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n }\n </div>\n </div>\n </div>\n <!-- /middle -->\n </div>\n </div>\n } @if (callHistory.length === 0) {\n <div class=\"container_call_history_not_found\">\n <div class=\"icon_round_area_call_history\"></div>\n <h2>No Call History</h2>\n <p>Your call history is empty</p>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n } @if (recentCallactiveTab === 'missedCall') {\n <div class=\"recent_tab_container\">\n <div class=\"container_recent_list\">\n <div class=\"container_recent_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (call of callerDropped; track $index) {\n <div class=\"outer_container_recent\">\n <div class=\"container_recent_item\">\n <div class=\"container_recent_item_image\">\n <div class=\"container_recent_item_image_inner grd1\">\n {{ call.phoneNumber?.country?.code | uppercase }}\n </div>\n </div>\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">\n {{ call.phoneNumber?.normalizedPhoneNumber }}\n </div>\n <div class=\"recent_time_back caller_drop\">\n <svg fill=\"currentColor\" viewBox=\"0 0 640 512\" width=\"12px\">\n <path\n d=\"M232 0c9.7 0 18.5 5.8 22.2 14.8s1.7 19.3-5.2 26.2l-39 39 58.7 58.7C282.3 152.4 300.8 160 320 160s37.7-7.6 51.3-21.3L503 7c9.4-9.4 24.6-9.4 33.9 0s9.4 24.6 0 33.9L405.3 172.7C382.6 195.3 352 208 320 208s-62.6-12.7-85.3-35.3L176 113.9l-39 39c-6.9 6.9-17.2 8.9-26.2 5.2s-14.8-12.5-14.8-22.2L96 24c0-13.3 10.7-24 24-24L232 0zM51.4 489.9l-35.4-62c-9.7-16.9-8.3-38.1 5.5-51.7C72.6 325.9 178.1 256 320 256s247.4 69.9 298.5 120.2c13.9 13.6 15.2 34.8 5.5 51.7l-35.4 62c-7.4 12.9-22.7 19.1-37 14.8L438.8 470.8c-13.5-4.1-22.8-16.5-22.8-30.6l0-56.2c-62.3-20.8-129.7-20.8-192 0l0 56.2c0 14.1-9.3 26.6-22.8 30.6L88.4 504.7c-14.3 4.3-29.6-1.8-37-14.8z\" />\n </svg>\n Caller hung up\n </div>\n </div>\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">\n {{ call.timestamp | date : \"dd/MM/yyyy\" }}\n </span>\n </div>\n </div>\n <div class=\"calling_btn\">\n <div class=\"calling_btn_inner\">\n <button class=\"button_recent_call\" (click)=\"selectFromHistory(call)\">\n <svg viewBox=\"0 0 512 512\" height=\"15px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n } @if (callerDropped.length === 0) {\n <div class=\"container_call_history_not_found\">\n <div class=\"icon_round_area_call_history\"></div>\n <h2>No Caller-Hangup Calls</h2>\n <p>No inbound calls ended by caller.</p>\n </div>\n }\n </div>\n <!-- /.scrollbox-content -->\n </div>\n <!-- /.scrollbox -->\n </div>\n <!-- /.container_recent_list_inner -->\n </div>\n <!-- /.container_recent_list -->\n </div>\n <!-- /.recent_tab_container -->\n }\n</div>\n}\n", styles: [".recent_container_outer{margin:0;padding:0;width:100%;position:relative}.recent_options_header{margin:0;padding:15px 15px 10px;width:100%;text-align:center}.recent_inner{margin:0;padding:0 2px;width:100%;height:32px;background:#f0f0f4;border-radius:8px;display:flex;align-items:center;justify-content:center}.recent_btn{margin:0 1px;padding:0;width:50%;display:flex;align-items:center;justify-content:center;border:none;outline:none;background:#f0f0f4;height:26px;border-radius:8px;color:var(--sd-text-color);transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.recent_btn:hover{background:#e6e6eb;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.recent_btn.active{color:var(--sd-text-color-dark);background:var(--sd-background-color);font-weight:700}.recent_heading{font-family:var(--sd-text-font-family);font-size:20px;color:var(--sd-text-color-dark);font-weight:700;letter-spacing:-.65px;padding:0 0 15px 15px;line-height:normal;margin:0}.container_recent_search{margin:0 0 15px;padding:0 15px;width:100%}.container_recent_search input{padding:0 16px;width:100%;outline:none;display:block;border:1px solid var(--sd-border-color-dark);color:var(--sd-text-color-medium);font-size:12px;letter-spacing:-.25px;font-family:var(--sd-text-font-family);background:var(--sd-background-color);font-style:italic;font-weight:400;position:relative;border-radius:27px;height:39px}.recent_tab_container{margin:0;padding:0;width:100%;height:100%;position:relative}.container_recent_list{width:100%;margin:0;padding:0;position:relative}.container_recent_list_inner{width:100%;position:relative;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center;height:calc(100vh - 204px)}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.outer_container_recent{width:100%;margin:0;padding:0 15px}.container_recent_item{width:100%;position:relative;display:flex;flex-direction:row;height:72px;pointer-events:all;border-radius:8px;overflow:hidden;cursor:pointer}.calling_btn{display:none}.history_loader{width:100%;padding:28px 0;display:flex;flex-direction:column;align-items:center;gap:12px}.history_loader .spinner{width:36px;height:36px;border-radius:50%;border:4px solid #e5e7eb;border-top-color:#2563eb;animation:spin .8s linear infinite}.history_loader .loader_text{font-size:13px;color:var(--sd-text-color-medium)}@keyframes spin{to{transform:rotate(360deg)}}.container_recent_item:hover .calling_btn{display:block;position:absolute;text-align:right;right:0;width:78px;height:38px;background:var(--sd-background-color)}.container_recent_item_image{display:flex;align-items:center}.container_recent_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative;border-radius:50%;display:flex;align-items:center;justify-content:center;font-family:var(--sd-text-font-family);font-size:26px;color:var(--sd-text-color-inverse);font-weight:400;background:#2a7b9b}.grd1{background:#718ede}.container_recent_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;position:relative;border-top:1px solid #e9edef}.container_recent_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_recent_item_left .contact_field{font-size:13px;color:var(--sd-text-color-dark);margin-bottom:2px}.container_recent_item_left .missed_call_color{color:#e0261f}.container_recent_item_left .recent_time_back{font-size:12px;color:var(--sd-text-color-medium)}.container_recent_item_left .caller_drop{color:#e0261f}.disconnect_badge{display:inline-flex;align-items:center;margin-left:6px;padding:2px 6px;border-radius:8px;background:#e0261f1f;color:#e0261f;font-size:10px;font-weight:600;letter-spacing:.2px}.container_recent_item_right{display:flex;flex-direction:row;position:absolute;right:0}.container_recent_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_recent_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.calling_btn_inner{position:relative;width:100%;margin:0;padding:0;display:flex;justify-content:right}.button_recent_call{width:38px;height:38px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center;line-height:normal;border-radius:26px;outline:none;border:none;font-size:22px;font-weight:700;color:#6c0;background:#f1f1f1;border:1px solid #f1f1f1}.container_call_history_not_found{position:absolute;width:100%;height:100%;display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;align-items:center;background:#f8f8f8}.container_call_history_not_found h2{font-size:18px;font-weight:700;letter-spacing:-.65px;color:var(--sd-text-color-dark);text-align:center;padding-top:26px}.container_call_history_not_found p{font-size:12px;line-height:normal;color:var(--sd-text-color);font-weight:600;white-space:normal;text-align:center;padding-top:10px}.icon_round_area_call_history{width:165px;height:75px;background:var(--sd-background-color) url() center no-repeat;border-radius:38px;display:flex;grid-row:1}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "pipe", type: i3.UpperCasePipe, name: "uppercase" }, { kind: "pipe", type: i3.DatePipe, name: "date" }] });
|
|
1251
|
+
}
|
|
1252
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: CallHistoryComponent, decorators: [{
|
|
1253
|
+
type: Component,
|
|
1254
|
+
args: [{ selector: 'app-call-history', standalone: false, template: "@if (showHistoryLoader) {\n<div class=\"history_loader\">\n <div class=\"spinner\"></div>\n <div class=\"loader_text\">Fetching call history...</div>\n</div>\n} @else {\n\n<div class=\"recent_container_outer\">\n <div class=\"recent_options_header\">\n <div class=\"recent_inner\">\n <button class=\"recent_btn\" (click)=\"recentSelectTab('recentCall')\"\n [class.active]=\"recentCallactiveTab === 'recentCall'\">\n Recent\n </button>\n <button class=\"recent_btn\" (click)=\"recentSelectTab('missedCall')\"\n [class.active]=\"recentCallactiveTab === 'missedCall'\">\n Missed Call\n </button>\n </div>\n </div>\n\n <h2 class=\"recent_heading\">Recent</h2>\n\n <div class=\"container_recent_search\">\n <input type=\"search\" placeholder=\"Type to search\" />\n </div>\n\n @if (recentCallactiveTab === 'recentCall') {\n <div class=\"recent_tab_container\">\n <div class=\"container_recent_list\">\n <div class=\"container_recent_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (call of callHistory; track $index) {\n <div class=\"outer_container_recent\">\n <div class=\"container_recent_item\">\n <!-- avatar -->\n <div class=\"container_recent_item_image\">\n <div class=\"container_recent_item_image_inner grd1\">\n {{ call.phoneNumber?.country?.code | uppercase }}\n </div>\n </div>\n\n <!-- middle -->\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">\n {{ call.phoneNumber?.normalizedPhoneNumber }}\n </div>\n\n <div class=\"recent_time_back\" [ngClass]=\"{\n missed_call_color: call.status === 'MISSED',\n caller_drop: call.disconnectSide === 'CALLER' && (call.duration ?? 0) <= 0\n }\">\n <!-- Direction icon (generic handset) -->\n <svg viewBox=\"0 0 512 512\" width=\"12px\" fill=\"currentColor\" aria-hidden=\"true\">\n <path\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n {{ call.direction }}\n @if (call.disconnectSide === 'CALLER' && (call.duration ?? 0) <= 0) {\n <span class=\"disconnect_badge\">Caller hung up</span>\n }\n </div>\n </div>\n\n <!-- right time -->\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">\n {{ call.timestamp | date : \"dd/MM/yyyy\" }}\n </span>\n </div>\n </div>\n\n <!-- call button -->\n <div class=\"calling_btn\">\n <div class=\"calling_btn_inner\">\n <!-- <button\n class=\"button_recent_call\"\n (click)=\"makeCall(call.phoneNumber?.normalizedPhoneNumber)\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button> -->\n\n @if (mode === 'transfer') {\n <button class=\"button_recent_call\" (click)=\"transferEv.emit(call)\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n } @else {\n <button class=\"button_recent_call\" (click)=\"\n makeCall(call.phoneNumber?.normalizedPhoneNumber)\n \">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n }\n </div>\n </div>\n </div>\n <!-- /middle -->\n </div>\n </div>\n } @if (callHistory.length === 0) {\n <div class=\"container_call_history_not_found\">\n <div class=\"icon_round_area_call_history\"></div>\n <h2>No Call History</h2>\n <p>Your call history is empty</p>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n } @if (recentCallactiveTab === 'missedCall') {\n <div class=\"recent_tab_container\">\n <div class=\"container_recent_list\">\n <div class=\"container_recent_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (call of callerDropped; track $index) {\n <div class=\"outer_container_recent\">\n <div class=\"container_recent_item\">\n <div class=\"container_recent_item_image\">\n <div class=\"container_recent_item_image_inner grd1\">\n {{ call.phoneNumber?.country?.code | uppercase }}\n </div>\n </div>\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">\n {{ call.phoneNumber?.normalizedPhoneNumber }}\n </div>\n <div class=\"recent_time_back caller_drop\">\n <svg fill=\"currentColor\" viewBox=\"0 0 640 512\" width=\"12px\">\n <path\n d=\"M232 0c9.7 0 18.5 5.8 22.2 14.8s1.7 19.3-5.2 26.2l-39 39 58.7 58.7C282.3 152.4 300.8 160 320 160s37.7-7.6 51.3-21.3L503 7c9.4-9.4 24.6-9.4 33.9 0s9.4 24.6 0 33.9L405.3 172.7C382.6 195.3 352 208 320 208s-62.6-12.7-85.3-35.3L176 113.9l-39 39c-6.9 6.9-17.2 8.9-26.2 5.2s-14.8-12.5-14.8-22.2L96 24c0-13.3 10.7-24 24-24L232 0zM51.4 489.9l-35.4-62c-9.7-16.9-8.3-38.1 5.5-51.7C72.6 325.9 178.1 256 320 256s247.4 69.9 298.5 120.2c13.9 13.6 15.2 34.8 5.5 51.7l-35.4 62c-7.4 12.9-22.7 19.1-37 14.8L438.8 470.8c-13.5-4.1-22.8-16.5-22.8-30.6l0-56.2c-62.3-20.8-129.7-20.8-192 0l0 56.2c0 14.1-9.3 26.6-22.8 30.6L88.4 504.7c-14.3 4.3-29.6-1.8-37-14.8z\" />\n </svg>\n Caller hung up\n </div>\n </div>\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">\n {{ call.timestamp | date : \"dd/MM/yyyy\" }}\n </span>\n </div>\n </div>\n <div class=\"calling_btn\">\n <div class=\"calling_btn_inner\">\n <button class=\"button_recent_call\" (click)=\"selectFromHistory(call)\">\n <svg viewBox=\"0 0 512 512\" height=\"15px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n } @if (callerDropped.length === 0) {\n <div class=\"container_call_history_not_found\">\n <div class=\"icon_round_area_call_history\"></div>\n <h2>No Caller-Hangup Calls</h2>\n <p>No inbound calls ended by caller.</p>\n </div>\n }\n </div>\n <!-- /.scrollbox-content -->\n </div>\n <!-- /.scrollbox -->\n </div>\n <!-- /.container_recent_list_inner -->\n </div>\n <!-- /.container_recent_list -->\n </div>\n <!-- /.recent_tab_container -->\n }\n</div>\n}\n", styles: [".recent_container_outer{margin:0;padding:0;width:100%;position:relative}.recent_options_header{margin:0;padding:15px 15px 10px;width:100%;text-align:center}.recent_inner{margin:0;padding:0 2px;width:100%;height:32px;background:#f0f0f4;border-radius:8px;display:flex;align-items:center;justify-content:center}.recent_btn{margin:0 1px;padding:0;width:50%;display:flex;align-items:center;justify-content:center;border:none;outline:none;background:#f0f0f4;height:26px;border-radius:8px;color:var(--sd-text-color);transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.recent_btn:hover{background:#e6e6eb;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.recent_btn.active{color:var(--sd-text-color-dark);background:var(--sd-background-color);font-weight:700}.recent_heading{font-family:var(--sd-text-font-family);font-size:20px;color:var(--sd-text-color-dark);font-weight:700;letter-spacing:-.65px;padding:0 0 15px 15px;line-height:normal;margin:0}.container_recent_search{margin:0 0 15px;padding:0 15px;width:100%}.container_recent_search input{padding:0 16px;width:100%;outline:none;display:block;border:1px solid var(--sd-border-color-dark);color:var(--sd-text-color-medium);font-size:12px;letter-spacing:-.25px;font-family:var(--sd-text-font-family);background:var(--sd-background-color);font-style:italic;font-weight:400;position:relative;border-radius:27px;height:39px}.recent_tab_container{margin:0;padding:0;width:100%;height:100%;position:relative}.container_recent_list{width:100%;margin:0;padding:0;position:relative}.container_recent_list_inner{width:100%;position:relative;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center;height:calc(100vh - 204px)}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.outer_container_recent{width:100%;margin:0;padding:0 15px}.container_recent_item{width:100%;position:relative;display:flex;flex-direction:row;height:72px;pointer-events:all;border-radius:8px;overflow:hidden;cursor:pointer}.calling_btn{display:none}.history_loader{width:100%;padding:28px 0;display:flex;flex-direction:column;align-items:center;gap:12px}.history_loader .spinner{width:36px;height:36px;border-radius:50%;border:4px solid #e5e7eb;border-top-color:#2563eb;animation:spin .8s linear infinite}.history_loader .loader_text{font-size:13px;color:var(--sd-text-color-medium)}@keyframes spin{to{transform:rotate(360deg)}}.container_recent_item:hover .calling_btn{display:block;position:absolute;text-align:right;right:0;width:78px;height:38px;background:var(--sd-background-color)}.container_recent_item_image{display:flex;align-items:center}.container_recent_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative;border-radius:50%;display:flex;align-items:center;justify-content:center;font-family:var(--sd-text-font-family);font-size:26px;color:var(--sd-text-color-inverse);font-weight:400;background:#2a7b9b}.grd1{background:#718ede}.container_recent_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;position:relative;border-top:1px solid #e9edef}.container_recent_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_recent_item_left .contact_field{font-size:13px;color:var(--sd-text-color-dark);margin-bottom:2px}.container_recent_item_left .missed_call_color{color:#e0261f}.container_recent_item_left .recent_time_back{font-size:12px;color:var(--sd-text-color-medium)}.container_recent_item_left .caller_drop{color:#e0261f}.disconnect_badge{display:inline-flex;align-items:center;margin-left:6px;padding:2px 6px;border-radius:8px;background:#e0261f1f;color:#e0261f;font-size:10px;font-weight:600;letter-spacing:.2px}.container_recent_item_right{display:flex;flex-direction:row;position:absolute;right:0}.container_recent_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_recent_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.calling_btn_inner{position:relative;width:100%;margin:0;padding:0;display:flex;justify-content:right}.button_recent_call{width:38px;height:38px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center;line-height:normal;border-radius:26px;outline:none;border:none;font-size:22px;font-weight:700;color:#6c0;background:#f1f1f1;border:1px solid #f1f1f1}.container_call_history_not_found{position:absolute;width:100%;height:100%;display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;align-items:center;background:#f8f8f8}.container_call_history_not_found h2{font-size:18px;font-weight:700;letter-spacing:-.65px;color:var(--sd-text-color-dark);text-align:center;padding-top:26px}.container_call_history_not_found p{font-size:12px;line-height:normal;color:var(--sd-text-color);font-weight:600;white-space:normal;text-align:center;padding-top:10px}.icon_round_area_call_history{width:165px;height:75px;background:var(--sd-background-color) url() center no-repeat;border-radius:38px;display:flex;grid-row:1}\n"] }]
|
|
1255
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i8.SnugdeskAuthenticationService }, { type: i0.ChangeDetectorRef }], propDecorators: { mode: [{
|
|
1256
|
+
type: Input
|
|
1257
|
+
}], makeCallEv: [{
|
|
1258
|
+
type: Output
|
|
1259
|
+
}], transferEv: [{
|
|
1260
|
+
type: Output
|
|
1261
|
+
}] } });
|
|
1262
|
+
|
|
1263
|
+
class AuthenticationService {
|
|
1264
|
+
jwtHelper = new JwtHelperService();
|
|
1265
|
+
tenant$ = new BehaviorSubject(null);
|
|
1266
|
+
user$ = new BehaviorSubject(null);
|
|
1267
|
+
isAuthenticated$ = new BehaviorSubject(false);
|
|
1268
|
+
isAuthenticated() {
|
|
1269
|
+
try {
|
|
1270
|
+
const mySessionToken = this.getAuthSessionToken();
|
|
1271
|
+
// console.log(`Got sessionToken: ${sessionToken}`);
|
|
1272
|
+
if (!mySessionToken) {
|
|
1273
|
+
throw new Error('Session token not found');
|
|
1274
|
+
}
|
|
1275
|
+
else if (this.jwtHelper.isTokenExpired(mySessionToken)) {
|
|
1276
|
+
throw new Error('Session token expired');
|
|
1277
|
+
}
|
|
1278
|
+
this.isAuthenticated$.next(true);
|
|
1279
|
+
return true;
|
|
1280
|
+
}
|
|
1281
|
+
catch (err) {
|
|
1282
|
+
console.warn(err);
|
|
1283
|
+
this.isAuthenticated$.next(false);
|
|
1284
|
+
return false;
|
|
1285
|
+
}
|
|
1286
|
+
}
|
|
1287
|
+
getAuthSessionToken() {
|
|
1288
|
+
return sessionStorage.getItem('auth_sessionToken');
|
|
1289
|
+
}
|
|
1290
|
+
getAuthDomainToken() {
|
|
1291
|
+
return sessionStorage.getItem('auth_domainToken');
|
|
1292
|
+
}
|
|
1293
|
+
getDecodedAuthDomainToken() {
|
|
1294
|
+
const token = this.getAuthDomainToken();
|
|
1295
|
+
return token ? this.jwtHelper.decodeToken(token) : null;
|
|
1296
|
+
}
|
|
1297
|
+
getTenantId() {
|
|
1298
|
+
// return this.getDecodedAuthSessionToken().tenantId;
|
|
1299
|
+
return this.getDecodedAuthDomainToken()?.tenantId;
|
|
1300
|
+
}
|
|
1301
|
+
getDecodedAuthSessionToken() {
|
|
1302
|
+
if (!this.isAuthenticated()) {
|
|
1303
|
+
const err = new Error(`User not authenticated`);
|
|
1304
|
+
err.status = 403;
|
|
1305
|
+
throw err;
|
|
1306
|
+
}
|
|
1307
|
+
const token = this.getAuthSessionToken();
|
|
1308
|
+
if (!token) {
|
|
1309
|
+
const err = new Error(`Token missing`);
|
|
1310
|
+
err.status = 401;
|
|
1311
|
+
throw err;
|
|
1312
|
+
}
|
|
1313
|
+
return this.jwtHelper.decodeToken(token);
|
|
1314
|
+
}
|
|
1315
|
+
getUserId() {
|
|
1316
|
+
return this.getDecodedAuthSessionToken()?.userId;
|
|
1317
|
+
}
|
|
1318
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AuthenticationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1319
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AuthenticationService, providedIn: 'root' });
|
|
1320
|
+
}
|
|
1321
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AuthenticationService, decorators: [{
|
|
1322
|
+
type: Injectable,
|
|
1323
|
+
args: [{
|
|
1324
|
+
providedIn: 'root',
|
|
1325
|
+
}]
|
|
1326
|
+
}] });
|
|
1327
|
+
|
|
1328
|
+
class DialpadComponent {
|
|
1329
|
+
cdr;
|
|
1330
|
+
formBuilder;
|
|
1331
|
+
avayaIpoService;
|
|
1332
|
+
authenticationService;
|
|
1333
|
+
contacts = [];
|
|
1334
|
+
makeCallEv = new EventEmitter();
|
|
1335
|
+
audioElement;
|
|
1336
|
+
callStateSub;
|
|
1337
|
+
dialForm;
|
|
1338
|
+
showDirectoryPhonebook = false;
|
|
1339
|
+
incomingCallSubscription;
|
|
1340
|
+
incomingCallNumber;
|
|
1341
|
+
CountryISO = CountryISO;
|
|
1342
|
+
SearchCountryField = SearchCountryField;
|
|
1343
|
+
isRinging = false;
|
|
1344
|
+
audio;
|
|
1345
|
+
constructor(cdr, formBuilder, avayaIpoService, authenticationService) {
|
|
1346
|
+
this.cdr = cdr;
|
|
1347
|
+
this.formBuilder = formBuilder;
|
|
1348
|
+
this.avayaIpoService = avayaIpoService;
|
|
1349
|
+
this.authenticationService = authenticationService;
|
|
1350
|
+
this.dialForm = this.formBuilder.group({
|
|
1351
|
+
phoneNumber: ['', Validators.required],
|
|
1352
|
+
});
|
|
1353
|
+
}
|
|
1354
|
+
ngOnInit() {
|
|
1355
|
+
window.addEventListener('keydown', this.handleKeyboardInput);
|
|
1356
|
+
const phoneUtil = PhoneNumberUtil.getInstance();
|
|
1357
|
+
this.incomingCallSubscription =
|
|
1358
|
+
this.avayaIpoService.incomingCallSubject.subscribe((data) => {
|
|
1359
|
+
try {
|
|
1360
|
+
let normalizedNumber = data.farEndNumber.trim();
|
|
1361
|
+
// Convert numbers with "00XX" to "+XX"
|
|
1362
|
+
if (normalizedNumber.startsWith('00')) {
|
|
1363
|
+
normalizedNumber = '+' + normalizedNumber.substring(2);
|
|
1364
|
+
}
|
|
1365
|
+
let parsedNumber;
|
|
1366
|
+
if (normalizedNumber.startsWith('+') ||
|
|
1367
|
+
/^\d{1,4}/.test(normalizedNumber)) {
|
|
1368
|
+
parsedNumber = phoneUtil.parse(normalizedNumber);
|
|
1369
|
+
}
|
|
1370
|
+
else {
|
|
1371
|
+
parsedNumber = phoneUtil.parse(normalizedNumber, this.authenticationService.tenant$
|
|
1372
|
+
.getValue()
|
|
1373
|
+
.contactInformation.country.code.toUpperCase());
|
|
1374
|
+
}
|
|
1375
|
+
this.incomingCallNumber = phoneUtil.format(parsedNumber, PhoneNumberFormat.INTERNATIONAL);
|
|
1376
|
+
}
|
|
1377
|
+
catch (error) {
|
|
1378
|
+
console.warn('Invalid phone number:', data.farEndNumber);
|
|
1379
|
+
this.incomingCallNumber = data.farEndNumber; // Fallback
|
|
1380
|
+
}
|
|
1381
|
+
// this.incomingCallNumber = data.farEndNumber;
|
|
1382
|
+
this.isRinging = true;
|
|
1383
|
+
this.audio = this.audioElement.nativeElement;
|
|
1384
|
+
this.audio.src = '../../../assets/sounds/ringback_tone.mp3';
|
|
1385
|
+
this.audio.loop = true;
|
|
1386
|
+
this.audio.currentTime = 0;
|
|
1387
|
+
this.audio.play();
|
|
1388
|
+
// this.playRingtone();
|
|
1389
|
+
});
|
|
1390
|
+
this.callStateSub = this.avayaIpoService.callStateSubject.subscribe((evt) => {
|
|
1391
|
+
if (evt?.state === CallState.DISCONNECTED) {
|
|
1392
|
+
this.isRinging = false;
|
|
1393
|
+
this.audio?.pause?.();
|
|
1394
|
+
this.cdr.detectChanges();
|
|
1395
|
+
}
|
|
1396
|
+
});
|
|
1397
|
+
}
|
|
1398
|
+
ngOnDestroy() {
|
|
1399
|
+
window.removeEventListener('keydown', this.handleKeyboardInput);
|
|
1400
|
+
this.callStateSub?.unsubscribe?.();
|
|
1401
|
+
}
|
|
1402
|
+
makeCall() {
|
|
1403
|
+
if (this.dialForm.invalid) {
|
|
1404
|
+
console.warn('Dial form is invalid:', this.dialForm.value);
|
|
1405
|
+
return;
|
|
1406
|
+
}
|
|
1407
|
+
const phone = this.dialForm.value.phoneNumber;
|
|
1408
|
+
console.log('phone =======', phone);
|
|
1409
|
+
this.makeCallEv.emit(phone?.internationalNumber);
|
|
1410
|
+
this.dialForm.reset();
|
|
1411
|
+
this.showDirectoryPhonebook = false;
|
|
1412
|
+
}
|
|
1413
|
+
onClear() {
|
|
1414
|
+
const control = this.dialForm.get('phoneNumber');
|
|
1415
|
+
const current = control?.value?.number?.replace(/\s/g, '') || '';
|
|
1416
|
+
if (!current)
|
|
1417
|
+
return control?.reset();
|
|
1418
|
+
const updated = current.slice(0, -1);
|
|
1419
|
+
updated ? this.updatePhoneControl(updated) : control?.reset();
|
|
1420
|
+
}
|
|
1421
|
+
handleKeyboardInput = (event) => {
|
|
1422
|
+
const control = this.dialForm.get('phoneNumber');
|
|
1423
|
+
const current = control?.value?.number?.replace(/\s/g, '') || '';
|
|
1424
|
+
let updated = current;
|
|
1425
|
+
if (event.key === 'Enter') {
|
|
1426
|
+
this.makeCall();
|
|
1427
|
+
return;
|
|
1428
|
+
}
|
|
1429
|
+
else if (event.key === 'Backspace') {
|
|
1430
|
+
if (!current)
|
|
1431
|
+
return control?.reset();
|
|
1432
|
+
updated = current.slice(0, -1);
|
|
1433
|
+
}
|
|
1434
|
+
else if (/^[0-9]$/.test(event.key)) {
|
|
1435
|
+
updated += event.key;
|
|
1436
|
+
}
|
|
1437
|
+
else {
|
|
1438
|
+
return;
|
|
1439
|
+
}
|
|
1440
|
+
updated ? this.updatePhoneControl(updated) : control?.reset();
|
|
1441
|
+
};
|
|
1442
|
+
handleKeypadInput(digit) {
|
|
1443
|
+
const control = this.dialForm.get('phoneNumber');
|
|
1444
|
+
const current = control?.value?.number?.replace(/\s/g, '') || '';
|
|
1445
|
+
this.updatePhoneControl(current + digit);
|
|
1446
|
+
}
|
|
1447
|
+
updatePhoneControl(value) {
|
|
1448
|
+
setTimeout(() => {
|
|
1449
|
+
this.dialForm.get('phoneNumber')?.setValue(value);
|
|
1450
|
+
this.cdr.detectChanges();
|
|
1451
|
+
});
|
|
1452
|
+
}
|
|
1453
|
+
declineAvayaCall() {
|
|
1454
|
+
this.avayaIpoService.dropCall();
|
|
1455
|
+
this.isRinging = false;
|
|
1456
|
+
this.audio?.pause?.();
|
|
1457
|
+
this.avayaIpoService.agentIdle();
|
|
1458
|
+
this.cdr.detectChanges();
|
|
1459
|
+
}
|
|
1460
|
+
acceptAvayaCAll() {
|
|
1461
|
+
this.avayaIpoService.answerCall();
|
|
1462
|
+
// this.headerComponent.showAvayaIpo = true;
|
|
1463
|
+
this.isRinging = false;
|
|
1464
|
+
this.audio.pause();
|
|
1465
|
+
}
|
|
1466
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: DialpadComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1$1.FormBuilder }, { token: AvayaIPOService }, { token: AuthenticationService }], target: i0.ɵɵFactoryTarget.Component });
|
|
1467
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.9", type: DialpadComponent, isStandalone: false, selector: "app-dialpad", inputs: { contacts: "contacts" }, outputs: { makeCallEv: "makeCallEv" }, viewQueries: [{ propertyName: "audioElement", first: true, predicate: ["audioElement"], descendants: true }], ngImport: i0, template: "<div class=\"container_pd_sd\">\n <audio id=\"sd_call_ring\" #audioElement></audio>\n\n @if (isRinging) {\n <div class=\"incoming_wrapper\">\n <div class=\"incoming_strip slide-right\">\n <div class=\"incoming_avatar\">\n <img\n src=\"../../../assets/images/incomingCall_avatar.gif\"\n alt=\"Incoming call avatar\"\n />\n </div>\n <div class=\"incoming_details\">\n <div class=\"incoming_label\">Incoming call</div>\n <div class=\"incoming_number\">\n {{ incomingCallNumber || 'Unknown number' }}\n </div>\n <div class=\"incoming_subtext\">Web call is ringing</div>\n </div>\n <div class=\"call_action_buttons incoming_actions\">\n <button\n type=\"button\"\n class=\"call_action_button decline\"\n (click)=\"declineAvayaCall()\"\n >\n <span class=\"icon_pill\">\n <svg height=\"18\" viewBox=\"0 0 640 512\">\n <path\n class=\"icon_color\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\"\n />\n </svg>\n </span>\n <span class=\"label\">Decline</span>\n </button>\n <button\n type=\"button\"\n class=\"call_action_button accept\"\n (click)=\"acceptAvayaCAll()\"\n >\n <span class=\"icon_pill\">\n <svg height=\"18\" viewBox=\"0 0 512 512\">\n <path\n class=\"icon_color\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\"\n />\n </svg>\n </span>\n <span class=\"label\">Answer</span>\n </button>\n </div>\n </div>\n </div>\n } @else {\n <div\n class=\"container_phone_number_input\"\n [formGroup]=\"dialForm\"\n (ngSubmit)=\"makeCall()\"\n >\n @if (contacts?.length > 0) {\n <button\n class=\"button_phonebook\"\n (click)=\"showDirectoryPhonebook = !showDirectoryPhonebook\"\n >\n <svg\n fill=\"currentColor\"\n height=\"22px\"\n style=\"enable-background: new 0 0 611.985 611.986\"\n version=\"1.1\"\n viewBox=\"0 0 611.985 611.986\"\n width=\"40.985px\"\n routerLinkActive=\"active\"\n xml:space=\"preserve\"\n >\n <g>\n <path\n d=\"M143.021,158.99c42.525,0,79.456-36.931,79.456-79.456S185.546,0.078,143.021,0.078\n S63.565,37.009,63.565,79.534S100.496,158.99,143.021,158.99z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M278.097,381.467c0-127.13-65.98-190.695-139.048-190.695S0,254.338,0,381.467\n C0,468.583,278.097,468.583,278.097,381.467z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M373.444,220.807c42.525,0,79.456-36.931,79.456-79.456s-36.931-79.456-79.456-79.456\n s-79.456,36.931-79.456,79.456S330.919,220.807,373.444,220.807z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M369.471,252.59c-21.962,0-43.033,6.452-62.071,17.957c11.537,31.051,18.37,67.792,18.37,110.921\n c0,42.843-28.541,76.151-77.517,95.22c60.768,51.424,260.267,40.459,260.267-33.403\n C508.52,316.155,442.539,252.59,369.471,252.59z\"\n fill=\"currentColor\"\n />\n </g>\n </svg>\n </button>\n }\n\n\n <ngx-intl-tel-input\n formControlName=\"phoneNumber\"\n name=\"phoneNumber\"\n [cssClass]=\"'phone_number_sd'\"\n [preferredCountries]=\"['in']\"\n [customPlaceholder]=\"'Phone Number'\"\n [enableAutoCountrySelect]=\"false\"\n [enablePlaceholder]=\"false\"\n [searchCountryFlag]=\"true\"\n [searchCountryField]=\"[SearchCountryField.Iso2, SearchCountryField.Name]\"\n [selectFirstCountry]=\"false\"\n [selectedCountryISO]=\"CountryISO.India\"\n [maxLength]=\"15\"\n [phoneValidation]=\"true\"\n [separateDialCode]=\"true\"\n (keydown)=\"handleKeyboardInput($event)\"\n >\n </ngx-intl-tel-input>\n <!-- <input type=\"text\" class=\"phone_number_sd\"> -->\n <button class=\"button_backspace\" (click)=\"onClear()\"></button>\n </div>\n }\n</div>\n\n@if (!isRinging && showDirectoryPhonebook) {\n <div class=\"container_phonebook\">\n <div class=\"container_phonebook_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (contact of contacts; track contact.id ?? contact.phone ?? $index) {\n <div class=\"container_contact_item\">\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg\n viewBox=\"0 0 212 212\"\n height=\"49px\"\n width=\"49px\"\n version=\"1.1\"\n x=\"0px\"\n y=\"0px\"\n enable-background=\"new 0 0 212 212\"\n >\n <path\n fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"\n ></path>\n <g>\n <path\n fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\"\n ></path>\n <path\n fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\"\n ></path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ contact.name }}</div>\n <div class=\"contact_field\">{{ contact.phone }}</div>\n </div>\n </div>\n\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">\n <div class=\"container_rating\">\n @for (star of [1,2,3,4,5]; track star) {\n <span [class.filled]=\"star <= (contact.rating || 0)\">\u2605</span>\n }\n </div>\n </span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n}\n\n@if (!isRinging) {\n <div class=\"container_dialpad\">\n <div class=\"container_dialpad_content\">\n <div class=\"container_dialpad_digits\">\n <!-- <div class=\"active-calls-container\"></div> -->\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('1')\">\n 1\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('2')\">\n 2\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('3')\">\n 3\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('4')\">\n 4\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('5')\">\n 5\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('6')\">\n 6\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('7')\">\n 7\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('8')\">\n 8\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('9')\">\n 9\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path\n fill=\"currentColor\"\n d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86 102.4l122 70.4L208 32z\"\n />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('0')\">\n 0\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path\n fill=\"currentColor\"\n d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3 36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4 26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2 384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64 192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\"\n />\n </svg>\n </button>\n </div>\n </div>\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n <button\n class=\"button_dialpad_digit button_call_green\"\n (click)=\"makeCall()\"\n [disabled]=\"!this.dialForm.valid\"\n [ngClass]=\"{ button_call_disabled: !this.dialForm.valid }\"\n >\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path\n fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\"\n />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n}\n", styles: [".container_pd_sd{width:100%;padding:0 30px;margin:0}.container_dialpad{width:100%;margin:0 auto;padding:32px 0;position:relative;background:var(--sd-background-color);-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center}.container_dialpad_header{width:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99;background:var(--sd-background-color)}.container_dialpad_header_inner1{margin:0;width:100%;padding:0 15px;position:relative}.container_dialpad_header_inner2{width:100%;margin:0;padding:15px;background:#f7f7f7;background:linear-gradient(180deg,#f7f7f7 72%,#fff);border-radius:12px 12px 0 0;position:relative}.container_phone_number_input{width:100%;margin:0;padding:0;position:relative}.container_phone_number_input .phone_number_sd{width:100%;display:flex;height:48px;outline:none;text-align:center;align-items:center;border:1px solid var(--sd-border-color-light);border-radius:27px;font-size:20px;font-weight:400;color:#000;padding:0 50px}.container_phone_number_input .button_phonebook{position:absolute;width:38px;height:38px;background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;border-radius:50%;border:none;outline:none;display:flex;align-items:center;justify-content:center;top:5px;left:8px;color:#a6a6a6}.container_phone_number_input .button_phonebook:hover,.container_phone_number_input .button_phonebook .sec_button{background:#f0f0f0;color:#29abe0}.container_phone_number_input .button_backspace{width:48px;height:48px;display:block;background:url() center no-repeat;position:absolute;right:0;top:0;outline:none;border:none}.container_phonebook{position:absolute;width:100%;left:0;right:0;top:149px;background:var(--sd-background-color);height:386px;overflow:hidden}.container_phonebook_inner{width:100%;position:relative;height:100%;padding:0}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.container_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:8px;overflow:hidden;margin-bottom:10px;cursor:pointer}.container_contact_item_image{display:flex}.container_contact_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative}.container_contact_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_contact_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_contact_item_left .contact_field{font-size:13px;margin-bottom:2px}.contact_dark{color:var(--sd-text-color-dark)}.contact_gray{color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_grey{font-size:12px;color:var(--sd-text-color);padding-top:2px;display:flex}.container_contact_item_left .contact_field_green{font-size:13px;margin-bottom:2px;color:#6c0}.container_contact_item_left .contact_field_highlighted{display:flex;font-size:11px;color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_highlighted span{background:#6c0;border-radius:3px;text-transform:uppercase;color:var(--sd-text-color-inverse);font-size:9px;padding:2px 4px}.container_contact_item_right{display:flex;flex-direction:row}.container_contact_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_contact_item_right .right_field_highlighted{font-size:16px;color:#29abe0}.container_contact_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .right_recent_content{display:inline-block;padding:0;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .container_rating{font-size:11px;cursor:pointer}.container_contact_item_right .container_rating span{color:gray;transition:color .3s}.container_contact_item_right .container_rating span.filled{color:gold}.container_dialpad_content{width:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);padding:21px 0 15px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center}.container_dialpad_digits{width:100%;display:flex;justify-content:center;margin-bottom:10px;flex-direction:column}.container_dialpad_row{display:flex;column-gap:10px;flex-direction:row;flex-shrink:0;justify-content:center;margin-bottom:10px}.button_dialpad_digit{display:block;width:50px;height:50px;display:flex;column-gap:20px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;font-size:22px;font-weight:700;color:var(--sd-text-color-dark);background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.button_dialpad_digit:hover{background:#000;color:var(--sd-text-color-inverse)}.button_call_green{background:#6c0;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.button_call_green:hover{background:#5ab400;color:var(--sd-text-color-inverse);width:60px;height:60px}.button_call_red{background:#e0261f;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s;z-index:9999}.button_call_red:hover{background:#cf211a}.button_call_disabled{background:#eee;color:var(--sd-text-color-light);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.container_dialpad_buttons{width:100%;display:flex;justify-content:center;padding-top:0;margin-bottom:6px}.call_action_buttons{display:flex;gap:14px;align-items:center;justify-content:center;margin-left:auto}.call_action_button{display:inline-flex;align-items:center;gap:10px;padding:12px 18px;border-radius:999px;border:none;color:#fff;font-weight:600;letter-spacing:.2px;cursor:pointer;box-shadow:0 8px 20px #00000029;transition:transform .2s ease,box-shadow .2s ease,filter .2s ease}.call_action_button .icon_pill{width:42px;height:42px;display:grid;place-items:center;border-radius:50%;background:#ffffff29}.call_action_button.accept{background:linear-gradient(135deg,#2bb673,#0fa958)}.call_action_button.decline{background:linear-gradient(135deg,#f45c43,#c81d1d)}.call_action_button:hover{transform:translateY(-1px);box-shadow:0 10px 24px #0003;filter:saturate(1.05)}.call_action_button:active{transform:translateY(0);box-shadow:0 6px 16px #00000029}.incoming_wrapper{width:100%;display:flex;justify-content:center;padding:8px 0 6px}.incoming_strip{display:flex;flex-direction:column;align-items:center;gap:14px;width:100%}.incoming_avatar{position:relative;width:72px;height:72px;border-radius:50%;overflow:hidden}.incoming_avatar img{width:100%;height:100%;object-fit:cover;display:block;border-radius:50%}.incoming_details{display:flex;flex-direction:column;align-items:center;gap:4px}.incoming_label{font-size:12px;text-transform:uppercase;letter-spacing:.4px;color:#6b7280}.incoming_number{font-size:20px;font-weight:700;letter-spacing:.1px;color:var(--sd-text-color-dark)}.incoming_subtext{color:#6b7280;font-size:13px}.incoming_actions{margin-left:0;margin-top:6px}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i5.NgxIntlTelInputComponent, selector: "ngx-intl-tel-input", inputs: ["value", "preferredCountries", "enablePlaceholder", "customPlaceholder", "numberFormat", "cssClass", "onlyCountries", "enableAutoCountrySelect", "searchCountryFlag", "searchCountryField", "searchCountryPlaceholder", "maxLength", "selectFirstCountry", "selectedCountryISO", "phoneValidation", "inputId", "separateDialCode"], outputs: ["countryChange"] }, { kind: "directive", type: i5.NativeElementInjectorDirective, selector: "[ngModel], [formControl], [formControlName]" }] });
|
|
1468
|
+
}
|
|
1469
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: DialpadComponent, decorators: [{
|
|
1470
|
+
type: Component,
|
|
1471
|
+
args: [{ selector: 'app-dialpad', standalone: false, template: "<div class=\"container_pd_sd\">\n <audio id=\"sd_call_ring\" #audioElement></audio>\n\n @if (isRinging) {\n <div class=\"incoming_wrapper\">\n <div class=\"incoming_strip slide-right\">\n <div class=\"incoming_avatar\">\n <img\n src=\"../../../assets/images/incomingCall_avatar.gif\"\n alt=\"Incoming call avatar\"\n />\n </div>\n <div class=\"incoming_details\">\n <div class=\"incoming_label\">Incoming call</div>\n <div class=\"incoming_number\">\n {{ incomingCallNumber || 'Unknown number' }}\n </div>\n <div class=\"incoming_subtext\">Web call is ringing</div>\n </div>\n <div class=\"call_action_buttons incoming_actions\">\n <button\n type=\"button\"\n class=\"call_action_button decline\"\n (click)=\"declineAvayaCall()\"\n >\n <span class=\"icon_pill\">\n <svg height=\"18\" viewBox=\"0 0 640 512\">\n <path\n class=\"icon_color\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\"\n />\n </svg>\n </span>\n <span class=\"label\">Decline</span>\n </button>\n <button\n type=\"button\"\n class=\"call_action_button accept\"\n (click)=\"acceptAvayaCAll()\"\n >\n <span class=\"icon_pill\">\n <svg height=\"18\" viewBox=\"0 0 512 512\">\n <path\n class=\"icon_color\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\"\n />\n </svg>\n </span>\n <span class=\"label\">Answer</span>\n </button>\n </div>\n </div>\n </div>\n } @else {\n <div\n class=\"container_phone_number_input\"\n [formGroup]=\"dialForm\"\n (ngSubmit)=\"makeCall()\"\n >\n @if (contacts?.length > 0) {\n <button\n class=\"button_phonebook\"\n (click)=\"showDirectoryPhonebook = !showDirectoryPhonebook\"\n >\n <svg\n fill=\"currentColor\"\n height=\"22px\"\n style=\"enable-background: new 0 0 611.985 611.986\"\n version=\"1.1\"\n viewBox=\"0 0 611.985 611.986\"\n width=\"40.985px\"\n routerLinkActive=\"active\"\n xml:space=\"preserve\"\n >\n <g>\n <path\n d=\"M143.021,158.99c42.525,0,79.456-36.931,79.456-79.456S185.546,0.078,143.021,0.078\n S63.565,37.009,63.565,79.534S100.496,158.99,143.021,158.99z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M278.097,381.467c0-127.13-65.98-190.695-139.048-190.695S0,254.338,0,381.467\n C0,468.583,278.097,468.583,278.097,381.467z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M373.444,220.807c42.525,0,79.456-36.931,79.456-79.456s-36.931-79.456-79.456-79.456\n s-79.456,36.931-79.456,79.456S330.919,220.807,373.444,220.807z\"\n fill=\"currentColor\"\n />\n <path\n d=\"M369.471,252.59c-21.962,0-43.033,6.452-62.071,17.957c11.537,31.051,18.37,67.792,18.37,110.921\n c0,42.843-28.541,76.151-77.517,95.22c60.768,51.424,260.267,40.459,260.267-33.403\n C508.52,316.155,442.539,252.59,369.471,252.59z\"\n fill=\"currentColor\"\n />\n </g>\n </svg>\n </button>\n }\n\n\n <ngx-intl-tel-input\n formControlName=\"phoneNumber\"\n name=\"phoneNumber\"\n [cssClass]=\"'phone_number_sd'\"\n [preferredCountries]=\"['in']\"\n [customPlaceholder]=\"'Phone Number'\"\n [enableAutoCountrySelect]=\"false\"\n [enablePlaceholder]=\"false\"\n [searchCountryFlag]=\"true\"\n [searchCountryField]=\"[SearchCountryField.Iso2, SearchCountryField.Name]\"\n [selectFirstCountry]=\"false\"\n [selectedCountryISO]=\"CountryISO.India\"\n [maxLength]=\"15\"\n [phoneValidation]=\"true\"\n [separateDialCode]=\"true\"\n (keydown)=\"handleKeyboardInput($event)\"\n >\n </ngx-intl-tel-input>\n <!-- <input type=\"text\" class=\"phone_number_sd\"> -->\n <button class=\"button_backspace\" (click)=\"onClear()\"></button>\n </div>\n }\n</div>\n\n@if (!isRinging && showDirectoryPhonebook) {\n <div class=\"container_phonebook\">\n <div class=\"container_phonebook_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (contact of contacts; track contact.id ?? contact.phone ?? $index) {\n <div class=\"container_contact_item\">\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg\n viewBox=\"0 0 212 212\"\n height=\"49px\"\n width=\"49px\"\n version=\"1.1\"\n x=\"0px\"\n y=\"0px\"\n enable-background=\"new 0 0 212 212\"\n >\n <path\n fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"\n ></path>\n <g>\n <path\n fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\"\n ></path>\n <path\n fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\"\n ></path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ contact.name }}</div>\n <div class=\"contact_field\">{{ contact.phone }}</div>\n </div>\n </div>\n\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">\n <div class=\"container_rating\">\n @for (star of [1,2,3,4,5]; track star) {\n <span [class.filled]=\"star <= (contact.rating || 0)\">\u2605</span>\n }\n </div>\n </span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n}\n\n@if (!isRinging) {\n <div class=\"container_dialpad\">\n <div class=\"container_dialpad_content\">\n <div class=\"container_dialpad_digits\">\n <!-- <div class=\"active-calls-container\"></div> -->\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('1')\">\n 1\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('2')\">\n 2\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('3')\">\n 3\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('4')\">\n 4\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('5')\">\n 5\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('6')\">\n 6\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('7')\">\n 7\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('8')\">\n 8\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('9')\">\n 9\n </button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path\n fill=\"currentColor\"\n d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86 102.4l122 70.4L208 32z\"\n />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('0')\">\n 0\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"handleKeypadInput('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path\n fill=\"currentColor\"\n d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3 36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4 26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2 384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64 192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\"\n />\n </svg>\n </button>\n </div>\n </div>\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n <button\n class=\"button_dialpad_digit button_call_green\"\n (click)=\"makeCall()\"\n [disabled]=\"!this.dialForm.valid\"\n [ngClass]=\"{ button_call_disabled: !this.dialForm.valid }\"\n >\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path\n fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\"\n />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n}\n", styles: [".container_pd_sd{width:100%;padding:0 30px;margin:0}.container_dialpad{width:100%;margin:0 auto;padding:32px 0;position:relative;background:var(--sd-background-color);-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center}.container_dialpad_header{width:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99;background:var(--sd-background-color)}.container_dialpad_header_inner1{margin:0;width:100%;padding:0 15px;position:relative}.container_dialpad_header_inner2{width:100%;margin:0;padding:15px;background:#f7f7f7;background:linear-gradient(180deg,#f7f7f7 72%,#fff);border-radius:12px 12px 0 0;position:relative}.container_phone_number_input{width:100%;margin:0;padding:0;position:relative}.container_phone_number_input .phone_number_sd{width:100%;display:flex;height:48px;outline:none;text-align:center;align-items:center;border:1px solid var(--sd-border-color-light);border-radius:27px;font-size:20px;font-weight:400;color:#000;padding:0 50px}.container_phone_number_input .button_phonebook{position:absolute;width:38px;height:38px;background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;border-radius:50%;border:none;outline:none;display:flex;align-items:center;justify-content:center;top:5px;left:8px;color:#a6a6a6}.container_phone_number_input .button_phonebook:hover,.container_phone_number_input .button_phonebook .sec_button{background:#f0f0f0;color:#29abe0}.container_phone_number_input .button_backspace{width:48px;height:48px;display:block;background:url() center no-repeat;position:absolute;right:0;top:0;outline:none;border:none}.container_phonebook{position:absolute;width:100%;left:0;right:0;top:149px;background:var(--sd-background-color);height:386px;overflow:hidden}.container_phonebook_inner{width:100%;position:relative;height:100%;padding:0}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.container_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:8px;overflow:hidden;margin-bottom:10px;cursor:pointer}.container_contact_item_image{display:flex}.container_contact_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative}.container_contact_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_contact_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_contact_item_left .contact_field{font-size:13px;margin-bottom:2px}.contact_dark{color:var(--sd-text-color-dark)}.contact_gray{color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_grey{font-size:12px;color:var(--sd-text-color);padding-top:2px;display:flex}.container_contact_item_left .contact_field_green{font-size:13px;margin-bottom:2px;color:#6c0}.container_contact_item_left .contact_field_highlighted{display:flex;font-size:11px;color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_highlighted span{background:#6c0;border-radius:3px;text-transform:uppercase;color:var(--sd-text-color-inverse);font-size:9px;padding:2px 4px}.container_contact_item_right{display:flex;flex-direction:row}.container_contact_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_contact_item_right .right_field_highlighted{font-size:16px;color:#29abe0}.container_contact_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .right_recent_content{display:inline-block;padding:0;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .container_rating{font-size:11px;cursor:pointer}.container_contact_item_right .container_rating span{color:gray;transition:color .3s}.container_contact_item_right .container_rating span.filled{color:gold}.container_dialpad_content{width:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);padding:21px 0 15px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center}.container_dialpad_digits{width:100%;display:flex;justify-content:center;margin-bottom:10px;flex-direction:column}.container_dialpad_row{display:flex;column-gap:10px;flex-direction:row;flex-shrink:0;justify-content:center;margin-bottom:10px}.button_dialpad_digit{display:block;width:50px;height:50px;display:flex;column-gap:20px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;font-size:22px;font-weight:700;color:var(--sd-text-color-dark);background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.button_dialpad_digit:hover{background:#000;color:var(--sd-text-color-inverse)}.button_call_green{background:#6c0;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.button_call_green:hover{background:#5ab400;color:var(--sd-text-color-inverse);width:60px;height:60px}.button_call_red{background:#e0261f;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s;z-index:9999}.button_call_red:hover{background:#cf211a}.button_call_disabled{background:#eee;color:var(--sd-text-color-light);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.container_dialpad_buttons{width:100%;display:flex;justify-content:center;padding-top:0;margin-bottom:6px}.call_action_buttons{display:flex;gap:14px;align-items:center;justify-content:center;margin-left:auto}.call_action_button{display:inline-flex;align-items:center;gap:10px;padding:12px 18px;border-radius:999px;border:none;color:#fff;font-weight:600;letter-spacing:.2px;cursor:pointer;box-shadow:0 8px 20px #00000029;transition:transform .2s ease,box-shadow .2s ease,filter .2s ease}.call_action_button .icon_pill{width:42px;height:42px;display:grid;place-items:center;border-radius:50%;background:#ffffff29}.call_action_button.accept{background:linear-gradient(135deg,#2bb673,#0fa958)}.call_action_button.decline{background:linear-gradient(135deg,#f45c43,#c81d1d)}.call_action_button:hover{transform:translateY(-1px);box-shadow:0 10px 24px #0003;filter:saturate(1.05)}.call_action_button:active{transform:translateY(0);box-shadow:0 6px 16px #00000029}.incoming_wrapper{width:100%;display:flex;justify-content:center;padding:8px 0 6px}.incoming_strip{display:flex;flex-direction:column;align-items:center;gap:14px;width:100%}.incoming_avatar{position:relative;width:72px;height:72px;border-radius:50%;overflow:hidden}.incoming_avatar img{width:100%;height:100%;object-fit:cover;display:block;border-radius:50%}.incoming_details{display:flex;flex-direction:column;align-items:center;gap:4px}.incoming_label{font-size:12px;text-transform:uppercase;letter-spacing:.4px;color:#6b7280}.incoming_number{font-size:20px;font-weight:700;letter-spacing:.1px;color:var(--sd-text-color-dark)}.incoming_subtext{color:#6b7280;font-size:13px}.incoming_actions{margin-left:0;margin-top:6px}\n"] }]
|
|
1472
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1$1.FormBuilder }, { type: AvayaIPOService }, { type: AuthenticationService }], propDecorators: { contacts: [{
|
|
1473
|
+
type: Input
|
|
1474
|
+
}], makeCallEv: [{
|
|
1475
|
+
type: Output
|
|
1476
|
+
}], audioElement: [{
|
|
1477
|
+
type: ViewChild,
|
|
1478
|
+
args: ['audioElement']
|
|
1479
|
+
}] } });
|
|
1480
|
+
|
|
1481
|
+
class DeviceSelectorComponent {
|
|
1482
|
+
selectedAudioInputId;
|
|
1483
|
+
selectedAudioOutputId;
|
|
1484
|
+
change = new EventEmitter();
|
|
1485
|
+
audioInputs = [];
|
|
1486
|
+
audioOutputs = [];
|
|
1487
|
+
ngOnInit() {
|
|
1488
|
+
if (navigator?.mediaDevices) {
|
|
1489
|
+
navigator.mediaDevices.addEventListener('devicechange', this.handleDeviceChange);
|
|
1490
|
+
this.refreshDeviceList();
|
|
1491
|
+
}
|
|
1492
|
+
else {
|
|
1493
|
+
console.warn('MediaDevices API not supported');
|
|
1494
|
+
}
|
|
1495
|
+
}
|
|
1496
|
+
ngOnDestroy() {
|
|
1497
|
+
navigator.mediaDevices.removeEventListener('devicechange', this.handleDeviceChange);
|
|
1498
|
+
}
|
|
1499
|
+
handleDeviceChange = async () => {
|
|
1500
|
+
await this.refreshDeviceList();
|
|
1501
|
+
};
|
|
1502
|
+
async refreshDeviceList() {
|
|
1503
|
+
const devices = await navigator.mediaDevices.enumerateDevices();
|
|
1504
|
+
this.audioInputs = devices.filter((d) => d.kind === 'audioinput');
|
|
1505
|
+
this.audioOutputs = devices.filter((d) => d.kind === 'audiooutput');
|
|
1506
|
+
// Fallback to first device if previous selection is no longer available
|
|
1507
|
+
if (!this.audioInputs.some((d) => d.deviceId === this.selectedAudioInputId)) {
|
|
1508
|
+
this.selectedAudioInputId = this.audioInputs[0]?.deviceId ?? '';
|
|
1509
|
+
}
|
|
1510
|
+
if (!this.audioOutputs.some((d) => d.deviceId === this.selectedAudioOutputId)) {
|
|
1511
|
+
this.selectedAudioOutputId = this.audioOutputs[0]?.deviceId ?? '';
|
|
1512
|
+
}
|
|
1513
|
+
this.emitDeviceChange();
|
|
1514
|
+
}
|
|
1515
|
+
onMicChange(deviceId) {
|
|
1516
|
+
this.selectedAudioInputId = deviceId;
|
|
1517
|
+
this.emitDeviceChange();
|
|
1518
|
+
}
|
|
1519
|
+
onSpeakerChange(deviceId) {
|
|
1520
|
+
this.selectedAudioOutputId = deviceId;
|
|
1521
|
+
this.emitDeviceChange();
|
|
1522
|
+
}
|
|
1523
|
+
emitDeviceChange() {
|
|
1524
|
+
this.change.emit({
|
|
1525
|
+
audioInputId: this.selectedAudioInputId ?? '',
|
|
1526
|
+
audioOutputId: this.selectedAudioOutputId ?? '',
|
|
1527
|
+
videoInputId: undefined,
|
|
1528
|
+
});
|
|
1529
|
+
}
|
|
1530
|
+
trackByDeviceId(index, device) {
|
|
1531
|
+
return device.deviceId;
|
|
1532
|
+
}
|
|
1533
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: DeviceSelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
1534
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.9", type: DeviceSelectorComponent, isStandalone: false, selector: "app-device-selector", inputs: { selectedAudioInputId: "selectedAudioInputId", selectedAudioOutputId: "selectedAudioOutputId" }, outputs: { change: "change" }, ngImport: i0, template: "<div class=\"sd_container_device_selector\">\n <h2 class=\"sd_tab_heading\">Preferences</h2>\n\n <div class=\"sd_tab_content\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"sd_tab_pd_sd\">\n <h3>Audio settings</h3>\n\n <div class=\"sd_container_selector_item\">\n <label for=\"audioInputSelect\">\n Microphone<br />\n <p>The device used to capture your voice</p>\n </label>\n\n <div class=\"sd_container_item_input\">\n <div class=\"sd_container_item_icon\">\n <svg width=\"14px\" viewBox=\"0 0 384 512\">\n <path\n d=\"M192 0C139 0 96 43 96 96l0 160c0 53 43 96 96 96s96-43 96-96l0-160c0-53-43-96-96-96zM64 216c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l144 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6c85.8-11.7 152-85.3 152-174.4l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 70.7-57.3 128-128 128S64 326.7 64 256l0-40z\"\n />\n </svg>\n </div>\n\n <select\n id=\"audioInputSelect\"\n [(ngModel)]=\"selectedAudioInputId\"\n (change)=\"onMicChange(selectedAudioInputId)\"\n >\n @for (mic of audioInputs; track mic.deviceId; let i = $index) {\n <option [value]=\"mic.deviceId\">\n {{ mic.label || \"Microphone \" + (i + 1) }}\n </option>\n } @empty {\n <option disabled>No microphones found</option>\n }\n </select>\n </div>\n </div>\n\n <div class=\"sd_container_selector_item\">\n <label for=\"audioOutputSelect\">\n Speakers<br />\n <p>The device that rings when you get a call.</p>\n </label>\n\n <div class=\"sd_container_item_input\">\n <div class=\"sd_container_item_icon\">\n <svg width=\"21px\" viewBox=\"0 0 576 512\">\n <path\n d=\"M333.1 34.8C344.6 40 352 51.4 352 64l0 384c0 12.6-7.4 24-18.9 29.2s-25 3.1-34.4-5.3L163.8 352 96 352c-35.3 0-64-28.7-64-64l0-64c0-35.3 28.7-64 64-64l67.8 0L298.7 40.1c9.4-8.4 22.9-10.4 34.4-5.3zm172 72.2c43.2 35.2 70.9 88.9 70.9 149s-27.7 113.8-70.9 149c-10.3 8.4-25.4 6.8-33.8-3.5s-6.8-25.4 3.5-33.8C507.3 341.3 528 301.1 528 256s-20.7-85.3-53.2-111.8c-10.3-8.4-11.8-23.5-3.5-33.8s23.5-11.8 33.8-3.5zm-60.5 74.5C466.1 199.1 480 225.9 480 256s-13.9 56.9-35.4 74.5c-10.3 8.4-25.4 6.8-33.8-3.5s-6.8-25.4 3.5-33.8C425.1 284.4 432 271 432 256s-6.9-28.4-17.7-37.3c-10.3-8.4-11.8-23.5-3.5-33.8s23.5-11.8 33.8-3.5z\"\n />\n </svg>\n </div>\n\n <select\n id=\"audioOutputSelect\"\n [(ngModel)]=\"selectedAudioOutputId\"\n (change)=\"onSpeakerChange(selectedAudioOutputId)\"\n >\n @for (spk of audioOutputs; track spk.deviceId; let i = $index) {\n <option [value]=\"spk.deviceId\">\n {{ spk.label || \"Speaker \" + (i + 1) }}\n </option>\n } @empty {\n <option disabled>No speakers found</option>\n }\n </select>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>\n", styles: [".sd_container_device_selector{margin:0;padding:15px 0 0;width:100%;position:relative}.sd_tab_heading{font-family:var(--sd-text-font-family);font-size:20px;color:var(--sd-text-color-dark);font-weight:700;letter-spacing:-.65px;padding:0 0 15px 15px;line-height:normal;margin:0}.sd_tab_content{margin:0;padding:0 0 15px;width:100%;position:relative;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center;height:calc(100vh - 108px)}.sd_tab_pd_sd{margin:0;padding:0 15px;width:100%}.sd_tab_content h3{font-family:var(--sd-text-font-family);font-size:16px;color:var(--sd-text-color-dark);letter-spacing:-.26px;font-weight:700;padding-bottom:10px}.sd_container_selector_item{width:100%;background:var(--sd-background-color);border-radius:8px;padding:15px;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;margin-bottom:15px;position:relative}.sd_container_selector_item label{font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;display:block;color:var(--sd-text-color-dark);margin-bottom:8px}.sd_container_selector_item label p{font-size:12px;color:var(--sd-text-color-medium);margin:4px 0 0;line-height:1.4}.sd_container_selector_item select{font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;display:block;color:var(--sd-text-color);width:100%;border:1px solid var(--sd-border-color-dark);padding:12px 8px 12px 63px;border-radius:4px;outline:none;position:relative;box-sizing:border-box}.sd_container_item_input{margin:0;padding:0;width:100%;position:relative}.sd_container_item_icon{position:absolute;width:54px;height:44px;display:flex;align-items:center;justify-content:center;background:#f1f1f1;top:1px;left:1px;border-bottom-left-radius:3px;border-top-left-radius:3px;z-index:9}\n"], dependencies: [{ kind: "directive", type: i1$1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1$1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1$1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i5.NativeElementInjectorDirective, selector: "[ngModel], [formControl], [formControlName]" }] });
|
|
1535
|
+
}
|
|
1536
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: DeviceSelectorComponent, decorators: [{
|
|
1537
|
+
type: Component,
|
|
1538
|
+
args: [{ selector: 'app-device-selector', standalone: false, template: "<div class=\"sd_container_device_selector\">\n <h2 class=\"sd_tab_heading\">Preferences</h2>\n\n <div class=\"sd_tab_content\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"sd_tab_pd_sd\">\n <h3>Audio settings</h3>\n\n <div class=\"sd_container_selector_item\">\n <label for=\"audioInputSelect\">\n Microphone<br />\n <p>The device used to capture your voice</p>\n </label>\n\n <div class=\"sd_container_item_input\">\n <div class=\"sd_container_item_icon\">\n <svg width=\"14px\" viewBox=\"0 0 384 512\">\n <path\n d=\"M192 0C139 0 96 43 96 96l0 160c0 53 43 96 96 96s96-43 96-96l0-160c0-53-43-96-96-96zM64 216c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l144 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6c85.8-11.7 152-85.3 152-174.4l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 70.7-57.3 128-128 128S64 326.7 64 256l0-40z\"\n />\n </svg>\n </div>\n\n <select\n id=\"audioInputSelect\"\n [(ngModel)]=\"selectedAudioInputId\"\n (change)=\"onMicChange(selectedAudioInputId)\"\n >\n @for (mic of audioInputs; track mic.deviceId; let i = $index) {\n <option [value]=\"mic.deviceId\">\n {{ mic.label || \"Microphone \" + (i + 1) }}\n </option>\n } @empty {\n <option disabled>No microphones found</option>\n }\n </select>\n </div>\n </div>\n\n <div class=\"sd_container_selector_item\">\n <label for=\"audioOutputSelect\">\n Speakers<br />\n <p>The device that rings when you get a call.</p>\n </label>\n\n <div class=\"sd_container_item_input\">\n <div class=\"sd_container_item_icon\">\n <svg width=\"21px\" viewBox=\"0 0 576 512\">\n <path\n d=\"M333.1 34.8C344.6 40 352 51.4 352 64l0 384c0 12.6-7.4 24-18.9 29.2s-25 3.1-34.4-5.3L163.8 352 96 352c-35.3 0-64-28.7-64-64l0-64c0-35.3 28.7-64 64-64l67.8 0L298.7 40.1c9.4-8.4 22.9-10.4 34.4-5.3zm172 72.2c43.2 35.2 70.9 88.9 70.9 149s-27.7 113.8-70.9 149c-10.3 8.4-25.4 6.8-33.8-3.5s-6.8-25.4 3.5-33.8C507.3 341.3 528 301.1 528 256s-20.7-85.3-53.2-111.8c-10.3-8.4-11.8-23.5-3.5-33.8s23.5-11.8 33.8-3.5zm-60.5 74.5C466.1 199.1 480 225.9 480 256s-13.9 56.9-35.4 74.5c-10.3 8.4-25.4 6.8-33.8-3.5s-6.8-25.4 3.5-33.8C425.1 284.4 432 271 432 256s-6.9-28.4-17.7-37.3c-10.3-8.4-11.8-23.5-3.5-33.8s23.5-11.8 33.8-3.5z\"\n />\n </svg>\n </div>\n\n <select\n id=\"audioOutputSelect\"\n [(ngModel)]=\"selectedAudioOutputId\"\n (change)=\"onSpeakerChange(selectedAudioOutputId)\"\n >\n @for (spk of audioOutputs; track spk.deviceId; let i = $index) {\n <option [value]=\"spk.deviceId\">\n {{ spk.label || \"Speaker \" + (i + 1) }}\n </option>\n } @empty {\n <option disabled>No speakers found</option>\n }\n </select>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>\n", styles: [".sd_container_device_selector{margin:0;padding:15px 0 0;width:100%;position:relative}.sd_tab_heading{font-family:var(--sd-text-font-family);font-size:20px;color:var(--sd-text-color-dark);font-weight:700;letter-spacing:-.65px;padding:0 0 15px 15px;line-height:normal;margin:0}.sd_tab_content{margin:0;padding:0 0 15px;width:100%;position:relative;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex;justify-content:center;align-items:center;height:calc(100vh - 108px)}.sd_tab_pd_sd{margin:0;padding:0 15px;width:100%}.sd_tab_content h3{font-family:var(--sd-text-font-family);font-size:16px;color:var(--sd-text-color-dark);letter-spacing:-.26px;font-weight:700;padding-bottom:10px}.sd_container_selector_item{width:100%;background:var(--sd-background-color);border-radius:8px;padding:15px;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;margin-bottom:15px;position:relative}.sd_container_selector_item label{font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;display:block;color:var(--sd-text-color-dark);margin-bottom:8px}.sd_container_selector_item label p{font-size:12px;color:var(--sd-text-color-medium);margin:4px 0 0;line-height:1.4}.sd_container_selector_item select{font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;display:block;color:var(--sd-text-color);width:100%;border:1px solid var(--sd-border-color-dark);padding:12px 8px 12px 63px;border-radius:4px;outline:none;position:relative;box-sizing:border-box}.sd_container_item_input{margin:0;padding:0;width:100%;position:relative}.sd_container_item_icon{position:absolute;width:54px;height:44px;display:flex;align-items:center;justify-content:center;background:#f1f1f1;top:1px;left:1px;border-bottom-left-radius:3px;border-top-left-radius:3px;z-index:9}\n"] }]
|
|
1539
|
+
}], propDecorators: { selectedAudioInputId: [{
|
|
1540
|
+
type: Input
|
|
1541
|
+
}], selectedAudioOutputId: [{
|
|
1542
|
+
type: Input
|
|
1543
|
+
}], change: [{
|
|
1544
|
+
type: Output
|
|
1545
|
+
}] } });
|
|
1546
|
+
|
|
1547
|
+
class AvayaIPOWidgetComponent {
|
|
1548
|
+
http;
|
|
1549
|
+
cdr;
|
|
1550
|
+
zone;
|
|
1551
|
+
formBuilder;
|
|
1552
|
+
countryService;
|
|
1553
|
+
phoneNumberLookupService;
|
|
1554
|
+
avayaIPOService;
|
|
1555
|
+
recordingManagerService;
|
|
1556
|
+
callSocket;
|
|
1557
|
+
authenticationService;
|
|
1558
|
+
tenantService;
|
|
1559
|
+
userService;
|
|
1560
|
+
tenantId;
|
|
1561
|
+
userId;
|
|
1562
|
+
isVisible;
|
|
1563
|
+
containerHeightObservable;
|
|
1564
|
+
containerWidthObservable;
|
|
1565
|
+
notificationEvent = new EventEmitter();
|
|
1566
|
+
isRecording = false;
|
|
1567
|
+
loginForm;
|
|
1568
|
+
transferForm;
|
|
1569
|
+
dtmfForm;
|
|
1570
|
+
isRinging = false;
|
|
1571
|
+
showActiveCall = false;
|
|
1572
|
+
loginSubscription;
|
|
1573
|
+
incomingCallSubscription;
|
|
1574
|
+
callDuration = 0;
|
|
1575
|
+
timer;
|
|
1576
|
+
isCalling = false;
|
|
1577
|
+
lookupPhoneNumber = null;
|
|
1578
|
+
ringtone;
|
|
1579
|
+
isReconnectingInProgress = false;
|
|
1580
|
+
isFailoverInProgress = false;
|
|
1581
|
+
isFailbackInProgress = false;
|
|
1582
|
+
showLoginForm = false;
|
|
1583
|
+
showLoginPassword = false;
|
|
1584
|
+
showLoginLoader = false;
|
|
1585
|
+
hasAvayaSocketError = false;
|
|
1586
|
+
hasAvayaLoginError = false;
|
|
1587
|
+
hasAvayaLoginConflictError = false;
|
|
1588
|
+
hasAvayaDeviceError = false;
|
|
1589
|
+
callStateSubscription;
|
|
1590
|
+
isMuted = false;
|
|
1591
|
+
isOnHold = false;
|
|
1592
|
+
isactiveCall = false;
|
|
1593
|
+
phoneNumber = '';
|
|
1594
|
+
isDailpadOpen = false;
|
|
1595
|
+
isCallTransferOpen = false;
|
|
1596
|
+
activeTab = 'tab1';
|
|
1597
|
+
contacts = [];
|
|
1598
|
+
searchText = '';
|
|
1599
|
+
extensionNumber = '';
|
|
1600
|
+
isDirectoryOpen = false;
|
|
1601
|
+
istransfer = false;
|
|
1602
|
+
rating = 4; // TODO: Remove thiss
|
|
1603
|
+
isConfereceList = false;
|
|
1604
|
+
conferenceParticipants;
|
|
1605
|
+
/* conferenceParticipants = [
|
|
1606
|
+
{ name: 'Pankaj Kumar Singh', duration: '00:10' },
|
|
1607
|
+
{ name: 'Saurabh Mehta', duration: '00:10' },
|
|
1608
|
+
{ name: 'Rajat Verma', duration: '00:10' }
|
|
1609
|
+
]; */
|
|
1610
|
+
currentView = 'transferListpage';
|
|
1611
|
+
redialTimeout;
|
|
1612
|
+
showCallDisconnected = false;
|
|
1613
|
+
lastDisconnectedCall = null;
|
|
1614
|
+
activeCalls = [];
|
|
1615
|
+
currentActiveCallIndex = 0;
|
|
1616
|
+
isAddingNewCall = false;
|
|
1617
|
+
isredialling = false;
|
|
1618
|
+
wasInbound = false;
|
|
1619
|
+
showHistoryPopup = false;
|
|
1620
|
+
noOfCalls = 0;
|
|
1621
|
+
isCallingFromContact = false;
|
|
1622
|
+
bottomActiveTab = 'keypad_sd';
|
|
1623
|
+
disconnecttriggered = false;
|
|
1624
|
+
lastRemoteStream = null;
|
|
1625
|
+
lastLocalStream = null;
|
|
1626
|
+
showInitLoader = false;
|
|
1627
|
+
tenant;
|
|
1628
|
+
user;
|
|
1629
|
+
internalContacts = [
|
|
1630
|
+
{ id: 1, name: 'Lovish Snug', phone: '6283675354', role: 'Agent', ext: '8002' },
|
|
1631
|
+
{ id: 2, name: 'Rakesh Kumar', phone: '(888) 9876549', role: 'Administrator', ext: '3005' },
|
|
1632
|
+
{ id: 3, name: 'Sohan Singh', phone: '(876) 8800965', role: 'Manager', ext: '3005' },
|
|
1633
|
+
{ id: 4, name: 'Pardeep Mishra', phone: '(876) 8800965', role: 'Agent', ext: '3005' },
|
|
1634
|
+
{ id: 5, name: 'Varun Ji', phone: '(876) 8800965', role: 'Manager', ext: '3005' }
|
|
1635
|
+
];
|
|
1636
|
+
directoryContacts = [
|
|
1637
|
+
{ id: 101, name: 'Rajeev Kumar Singh', phone: '(654) 9876549', rating: 4 },
|
|
1638
|
+
{ id: 102, name: 'Amit Sharma', phone: '(987) 6543210', rating: 5 },
|
|
1639
|
+
{ id: 103, name: 'Sunita Verma', phone: '(765) 4321987', rating: 3 }
|
|
1640
|
+
];
|
|
1641
|
+
recognition;
|
|
1642
|
+
transcriptText = '';
|
|
1643
|
+
showTranscriptionPopup = false;
|
|
1644
|
+
recognitionKeepAlive;
|
|
1645
|
+
transcriptHistory = [];
|
|
1646
|
+
recognitionActive = false;
|
|
1647
|
+
transcriptSegments = []; // final lines w/ timestamps
|
|
1648
|
+
transcriptStartedAt = 0;
|
|
1649
|
+
transcriptEndedAt = 0;
|
|
1650
|
+
autoDownloadSummaryOnEnd = true; // toggle if you want auto-download
|
|
1651
|
+
summaryText = '';
|
|
1652
|
+
displayCallingNumber;
|
|
1653
|
+
redialnumber;
|
|
1654
|
+
isTransferring = false;
|
|
1655
|
+
constructor(http, cdr, zone, formBuilder, countryService, phoneNumberLookupService, avayaIPOService, recordingManagerService, callSocket, authenticationService, tenantService, userService) {
|
|
1656
|
+
this.http = http;
|
|
1657
|
+
this.cdr = cdr;
|
|
1658
|
+
this.zone = zone;
|
|
1659
|
+
this.formBuilder = formBuilder;
|
|
1660
|
+
this.countryService = countryService;
|
|
1661
|
+
this.phoneNumberLookupService = phoneNumberLookupService;
|
|
1662
|
+
this.avayaIPOService = avayaIPOService;
|
|
1663
|
+
this.recordingManagerService = recordingManagerService;
|
|
1664
|
+
this.callSocket = callSocket;
|
|
1665
|
+
this.authenticationService = authenticationService;
|
|
1666
|
+
this.tenantService = tenantService;
|
|
1667
|
+
this.userService = userService;
|
|
1668
|
+
this.loginForm = this.formBuilder.group({
|
|
1669
|
+
ipoExtensionNo: ['', Validators.required],
|
|
1670
|
+
ipoPassword: ['', Validators.required],
|
|
1671
|
+
});
|
|
1672
|
+
this.transferForm = this.formBuilder.group({
|
|
1673
|
+
target: ['', Validators.required],
|
|
1674
|
+
});
|
|
1675
|
+
this.dtmfForm = this.formBuilder.group({
|
|
1676
|
+
dtmf: ['', Validators.required],
|
|
1677
|
+
});
|
|
1678
|
+
}
|
|
1679
|
+
async ngOnInit() {
|
|
1680
|
+
if (!('serviceWorker' in navigator)) {
|
|
1681
|
+
// Service Worker isn't supported on this browser, disable or hide UI.
|
|
1682
|
+
return;
|
|
1683
|
+
}
|
|
1684
|
+
if (!('PushManager' in window)) {
|
|
1685
|
+
// Push isn't supported on this browser, disable or hide UI.
|
|
1686
|
+
return;
|
|
1687
|
+
}
|
|
1688
|
+
this.loadScript('https://snugdesk-assets.s3.us-east-1.amazonaws.com/js/awl.min.js').then(() => {
|
|
1689
|
+
this.avayaIPOService.init();
|
|
1690
|
+
});
|
|
1691
|
+
this.showInitLoader = true;
|
|
1692
|
+
try {
|
|
1693
|
+
if (!this.authenticationService.isAuthenticated()) {
|
|
1694
|
+
const res_login = await this.authenticationService.login(this.tenantId, this.userId);
|
|
1695
|
+
console.debug('Login response:', res_login);
|
|
1696
|
+
}
|
|
1697
|
+
this.tenant = sessionStorage.getItem('sd-tenant');
|
|
1698
|
+
if (!this.tenant) {
|
|
1699
|
+
this.tenant = await this.tenantService.getTenantById(this.tenantId);
|
|
1700
|
+
console.debug('Fetched tenant:', this.tenant);
|
|
1701
|
+
sessionStorage.setItem('sd-tenant', JSON.stringify(this.tenant));
|
|
1702
|
+
}
|
|
1703
|
+
this.user = sessionStorage.getItem('sd-user');
|
|
1704
|
+
if (!this.user) {
|
|
1705
|
+
this.user = await this.userService.getUserById(this.userId);
|
|
1706
|
+
console.debug('Fetched user:', this.user);
|
|
1707
|
+
sessionStorage.setItem('sd-user', JSON.stringify(this.user));
|
|
1708
|
+
}
|
|
1709
|
+
}
|
|
1710
|
+
catch (error) {
|
|
1711
|
+
console.error(error);
|
|
1712
|
+
// this.hasInitConfigurationErrors = true;
|
|
1713
|
+
// this.snackBar.open(
|
|
1714
|
+
// `Oops! WhatsApp settings unavailable or invalid configuration detected`,
|
|
1715
|
+
// 'Close',
|
|
1716
|
+
// {
|
|
1717
|
+
// duration: 5000,
|
|
1718
|
+
// panelClass: ['snackbar-error'],
|
|
1719
|
+
// horizontalPosition: 'center',
|
|
1720
|
+
// verticalPosition: 'top',
|
|
1721
|
+
// }
|
|
1722
|
+
// );
|
|
1723
|
+
}
|
|
1724
|
+
finally {
|
|
1725
|
+
// Select the first WhatsApp phone number by default
|
|
1726
|
+
this.showInitLoader = false;
|
|
1727
|
+
this.cdr.detectChanges();
|
|
1728
|
+
}
|
|
1729
|
+
return;
|
|
1730
|
+
}
|
|
1731
|
+
loadScript(src) {
|
|
1732
|
+
return new Promise((resolve, reject) => {
|
|
1733
|
+
const script = document.createElement('script');
|
|
1734
|
+
script.src = src;
|
|
1735
|
+
script.async = true;
|
|
1736
|
+
script.onload = () => resolve();
|
|
1737
|
+
script.onerror = () => reject();
|
|
1738
|
+
document.body.appendChild(script);
|
|
1739
|
+
});
|
|
1740
|
+
}
|
|
1741
|
+
ngAfterViewInit() {
|
|
1742
|
+
this.loginSubscription = this.avayaIPOService.loginStatus$.subscribe(async (response) => {
|
|
1743
|
+
this.zone.run(async () => {
|
|
1744
|
+
console.debug('Got Avaya IPO Login response: ', response);
|
|
1745
|
+
if (!response) {
|
|
1746
|
+
this.showLoginForm = true;
|
|
1747
|
+
this.cdr.detectChanges();
|
|
1748
|
+
// this.hasAvayaDeviceError = true;
|
|
1749
|
+
return;
|
|
1750
|
+
}
|
|
1751
|
+
this.showLoginLoader = false;
|
|
1752
|
+
switch (response.result) {
|
|
1753
|
+
case LoginState.LOGIN_SUCCESS:
|
|
1754
|
+
console.debug('Login successful:', response);
|
|
1755
|
+
this.showLoginForm = false;
|
|
1756
|
+
this.hasAvayaLoginError = false;
|
|
1757
|
+
this.hasAvayaLoginConflictError = false;
|
|
1758
|
+
this.avayaIPOService.agentIdle();
|
|
1759
|
+
await this.avayaIPOService.initLocalMic();
|
|
1760
|
+
await this.fetchContacts();
|
|
1761
|
+
// await this.loadCallHistory();
|
|
1762
|
+
break;
|
|
1763
|
+
case LoginState.LOGGEDOUT:
|
|
1764
|
+
this.loginForm.reset();
|
|
1765
|
+
this.showLoginForm = true;
|
|
1766
|
+
this.hasAvayaLoginError = false;
|
|
1767
|
+
this.hasAvayaLoginConflictError = false;
|
|
1768
|
+
this.hasAvayaDeviceError = false;
|
|
1769
|
+
this.hasAvayaSocketError = false;
|
|
1770
|
+
this.avayaIPOService.agentBusy();
|
|
1771
|
+
break;
|
|
1772
|
+
case LoginState.LOGIN_EMPTYUSERNAME:
|
|
1773
|
+
case LoginState.LOGIN_EMPTYPASSWORD:
|
|
1774
|
+
this.showLoginForm = true;
|
|
1775
|
+
this.hasAvayaLoginError = true;
|
|
1776
|
+
this.avayaIPOService.agentBusy();
|
|
1777
|
+
break;
|
|
1778
|
+
case LoginState.LOGIN_FAILED:
|
|
1779
|
+
this.showLoginForm = true;
|
|
1780
|
+
if (response.reason === 'Invalid Credentials') {
|
|
1781
|
+
this.hasAvayaLoginError = true;
|
|
1782
|
+
}
|
|
1783
|
+
else
|
|
1784
|
+
this.hasAvayaLoginConflictError = true;
|
|
1785
|
+
this.avayaIPOService.agentBusy();
|
|
1786
|
+
break;
|
|
1787
|
+
case LoginState.LOGIN_GW_NOTCONFIGURED:
|
|
1788
|
+
case LoginState.WEBSOCKET_FAILURE:
|
|
1789
|
+
this.showLoginForm = true;
|
|
1790
|
+
this.hasAvayaSocketError = true;
|
|
1791
|
+
// this.loginForm.disable();
|
|
1792
|
+
this.avayaIPOService.agentBusy();
|
|
1793
|
+
break;
|
|
1794
|
+
case LoginState.DEVICEACCESS_FAILURE:
|
|
1795
|
+
this.showLoginForm = true;
|
|
1796
|
+
this.hasAvayaDeviceError = true;
|
|
1797
|
+
this.avayaIPOService.agentBusy();
|
|
1798
|
+
break;
|
|
1799
|
+
case LoginState.RECONNECTING:
|
|
1800
|
+
this.isReconnectingInProgress = true;
|
|
1801
|
+
this.avayaIPOService.agentBusy();
|
|
1802
|
+
break;
|
|
1803
|
+
case LoginState.RELOGGED_IN:
|
|
1804
|
+
this.isReconnectingInProgress = false;
|
|
1805
|
+
this.avayaIPOService.agentIdle();
|
|
1806
|
+
break;
|
|
1807
|
+
case LoginState.FAILING_OVER:
|
|
1808
|
+
this.isFailoverInProgress = true;
|
|
1809
|
+
this.avayaIPOService.agentBusy();
|
|
1810
|
+
break;
|
|
1811
|
+
case LoginState.FAILOVER_SUCCESS:
|
|
1812
|
+
this.isFailoverInProgress = false;
|
|
1813
|
+
this.avayaIPOService.agentIdle();
|
|
1814
|
+
break;
|
|
1815
|
+
case LoginState.FAILOVER_FAILED:
|
|
1816
|
+
this.isFailoverInProgress = false;
|
|
1817
|
+
this.showLoginForm = true;
|
|
1818
|
+
this.hasAvayaSocketError = true;
|
|
1819
|
+
this.avayaIPOService.agentBusy();
|
|
1820
|
+
break;
|
|
1821
|
+
case LoginState.FAILING_BACK:
|
|
1822
|
+
this.isFailbackInProgress = true;
|
|
1823
|
+
this.avayaIPOService.agentBusy();
|
|
1824
|
+
break;
|
|
1825
|
+
case LoginState.FAILBACK_SUCCESS:
|
|
1826
|
+
this.isFailbackInProgress = false;
|
|
1827
|
+
this.avayaIPOService.agentIdle();
|
|
1828
|
+
break;
|
|
1829
|
+
case LoginState.FAILBACK_FAILED:
|
|
1830
|
+
this.isFailbackInProgress = false;
|
|
1831
|
+
this.showLoginForm = true;
|
|
1832
|
+
this.hasAvayaSocketError = true;
|
|
1833
|
+
this.avayaIPOService.agentBusy();
|
|
1834
|
+
break;
|
|
1835
|
+
default:
|
|
1836
|
+
console.warn('Unhandled Avaya IPO Login state:', response.result);
|
|
1837
|
+
}
|
|
1838
|
+
this.cdr.detectChanges();
|
|
1839
|
+
});
|
|
1840
|
+
});
|
|
1841
|
+
this.avayaIPOService.remoteStream$
|
|
1842
|
+
.pipe(filter((r) => !!r && r.stream.getAudioTracks().length > 0))
|
|
1843
|
+
.subscribe((r) => {
|
|
1844
|
+
this.lastRemoteStream = r.stream;
|
|
1845
|
+
this.lastLocalStream = this.avayaIPOService.localStream;
|
|
1846
|
+
});
|
|
1847
|
+
this.callStateSubscription =
|
|
1848
|
+
this.avayaIPOService.callStateSubject.subscribe((evt) => {
|
|
1849
|
+
this.zone.run(() => {
|
|
1850
|
+
// 1) early‐exit on null/undefined
|
|
1851
|
+
if (!evt || !evt.callId) {
|
|
1852
|
+
console.debug('ignoring empty callState event', evt);
|
|
1853
|
+
return;
|
|
1854
|
+
}
|
|
1855
|
+
// 2) now it’s safe to destructure
|
|
1856
|
+
const { state, callId } = evt;
|
|
1857
|
+
const idx = this.activeCalls.findIndex((c) => c.callId === callId);
|
|
1858
|
+
if (idx === -1) {
|
|
1859
|
+
console.warn('Event for unknown call', { state, callId });
|
|
1860
|
+
return;
|
|
1861
|
+
}
|
|
1862
|
+
console.log('Call stateeeeeeeeeee', state);
|
|
1863
|
+
switch (state) {
|
|
1864
|
+
case CallState.CONNECTED:
|
|
1865
|
+
this.currentActiveCallIndex = idx;
|
|
1866
|
+
this.avayaIPOService.agentBusy();
|
|
1867
|
+
this.handleCallConnect();
|
|
1868
|
+
this.cdr.detectChanges();
|
|
1869
|
+
break;
|
|
1870
|
+
case CallState.DISCONNECTED:
|
|
1871
|
+
console.log(this.disconnecttriggered);
|
|
1872
|
+
this.avayaIPOService.agentIdle();
|
|
1873
|
+
if (!this.disconnecttriggered) {
|
|
1874
|
+
this.lastRemoteStream = null;
|
|
1875
|
+
this.lastLocalStream = null;
|
|
1876
|
+
this.handleCallDisconnect(idx, 'CALLER');
|
|
1877
|
+
}
|
|
1878
|
+
this.showActiveCall = false;
|
|
1879
|
+
this.bottomActiveTab = 'keypad_sd';
|
|
1880
|
+
this.avayaIPOService.activeCallId = null;
|
|
1881
|
+
this.isactiveCall = false;
|
|
1882
|
+
break;
|
|
1883
|
+
case CallState.INCOMING:
|
|
1884
|
+
console.log(this.disconnecttriggered);
|
|
1885
|
+
this.avayaIPOService.agentIdle();
|
|
1886
|
+
if (!this.disconnecttriggered) {
|
|
1887
|
+
this.lastRemoteStream = null;
|
|
1888
|
+
this.lastLocalStream = null;
|
|
1889
|
+
this.handleCallDisconnect(idx, 'AGENT');
|
|
1890
|
+
}
|
|
1891
|
+
this.showActiveCall = false;
|
|
1892
|
+
this.bottomActiveTab = 'keypad_sd';
|
|
1893
|
+
this.avayaIPOService.activeCallId = null;
|
|
1894
|
+
this.isactiveCall = false;
|
|
1895
|
+
break;
|
|
1896
|
+
case CallState.HELD:
|
|
1897
|
+
this.recordingManagerService.stopRecording(callId);
|
|
1898
|
+
this.lastRemoteStream = null;
|
|
1899
|
+
this.lastLocalStream = null;
|
|
1900
|
+
console.log('**************** HOLD HOLD HOLD *****************');
|
|
1901
|
+
break;
|
|
1902
|
+
case CallState.PROGRESSING:
|
|
1903
|
+
this.currentActiveCallIndex = idx;
|
|
1904
|
+
this.showActiveCall = true;
|
|
1905
|
+
break;
|
|
1906
|
+
default:
|
|
1907
|
+
console.warn('Unhandled call state:', state, 'idx=', idx);
|
|
1908
|
+
}
|
|
1909
|
+
this.cdr.detectChanges();
|
|
1910
|
+
});
|
|
1911
|
+
});
|
|
1912
|
+
this.incomingCallSubscription = this.avayaIPOService.incomingCallSubject
|
|
1913
|
+
.pipe(filter((evt) => !!evt && typeof evt.callId === 'string'))
|
|
1914
|
+
.subscribe(async (evt) => {
|
|
1915
|
+
this.zone.run(async () => {
|
|
1916
|
+
const { callId, farEndNumber } = evt;
|
|
1917
|
+
console.log('📞 Incoming call!', callId, farEndNumber);
|
|
1918
|
+
// TODO: rename displayInfo to this.lookupPhoneNumber
|
|
1919
|
+
const displayInfo = await this.getLookupPhoneNumber(farEndNumber);
|
|
1920
|
+
console.log(displayInfo);
|
|
1921
|
+
const slot = this.avayaIPOService.getSlotForCallId(callId);
|
|
1922
|
+
if (!slot) {
|
|
1923
|
+
console.warn('Couldn’t find slot for incoming callId', callId);
|
|
1924
|
+
return;
|
|
1925
|
+
}
|
|
1926
|
+
this.activeCalls.push({
|
|
1927
|
+
callId,
|
|
1928
|
+
phoneNumber: displayInfo.lookupPhoneNumber?.['local-number'].replace(/\s+/g, ''),
|
|
1929
|
+
lookupPhoneNumber: displayInfo,
|
|
1930
|
+
duration: 0,
|
|
1931
|
+
isMuted: false,
|
|
1932
|
+
isOnHold: false,
|
|
1933
|
+
startTime: Date.now(),
|
|
1934
|
+
slot,
|
|
1935
|
+
});
|
|
1936
|
+
this.phoneNumber = displayInfo.lookupPhoneNumber?.['local-number'].replace(/\s+/g, '');
|
|
1937
|
+
this.currentActiveCallIndex = this.activeCalls.length - 1;
|
|
1938
|
+
this.isactiveCall = true;
|
|
1939
|
+
this.wasInbound = true;
|
|
1940
|
+
// Ensure UI is on dialpad tab and active-call view when ringing
|
|
1941
|
+
this.bottomActiveTab = 'keypad_sd';
|
|
1942
|
+
this.showActiveCall = true;
|
|
1943
|
+
this.cdr.detectChanges();
|
|
1944
|
+
});
|
|
1945
|
+
});
|
|
1946
|
+
}
|
|
1947
|
+
ngOnDestroy() {
|
|
1948
|
+
this.loginSubscription?.unsubscribe();
|
|
1949
|
+
this.incomingCallSubscription?.unsubscribe();
|
|
1950
|
+
this.callStateSubscription?.unsubscribe();
|
|
1951
|
+
// this.clearTimer(); todo
|
|
1952
|
+
}
|
|
1953
|
+
// prettier-ignore
|
|
1954
|
+
async getLookupPhoneNumber(phoneNumber) {
|
|
1955
|
+
if (!phoneNumber || typeof phoneNumber !== 'string') {
|
|
1956
|
+
console.warn('lookupPhoneNumber called with an invalid phoneNumber: ', phoneNumber);
|
|
1957
|
+
return {
|
|
1958
|
+
phoneNumber: '',
|
|
1959
|
+
};
|
|
1960
|
+
}
|
|
1961
|
+
let normalizedPhoneNumber = phoneNumber.trim();
|
|
1962
|
+
if (normalizedPhoneNumber.startsWith('00')) {
|
|
1963
|
+
normalizedPhoneNumber = '+' + normalizedPhoneNumber.substring(2);
|
|
1964
|
+
}
|
|
1965
|
+
try {
|
|
1966
|
+
const res_lookupPhoneNumber = await this.phoneNumberLookupService.lookupPhoneNumber(normalizedPhoneNumber);
|
|
1967
|
+
if (!res_lookupPhoneNumber?.data?.valid) {
|
|
1968
|
+
throw new Error('Invalid res_lookupPhoneNumber: ', res_lookupPhoneNumber);
|
|
1969
|
+
}
|
|
1970
|
+
const countryCode = res_lookupPhoneNumber?.data
|
|
1971
|
+
? res_lookupPhoneNumber.data['country-code']
|
|
1972
|
+
: null;
|
|
1973
|
+
const res_getCountryFromCountryCode = countryCode
|
|
1974
|
+
? await this.countryService.getCountryFromCountryCode(countryCode)
|
|
1975
|
+
: null;
|
|
1976
|
+
// console.debug('res_getCountryFromCountryCode:', res_getCountryFromCountryCode);
|
|
1977
|
+
return {
|
|
1978
|
+
normalizedPhoneNumber: normalizedPhoneNumber,
|
|
1979
|
+
lookupPhoneNumber: res_lookupPhoneNumber?.data,
|
|
1980
|
+
country: res_getCountryFromCountryCode,
|
|
1981
|
+
};
|
|
1982
|
+
}
|
|
1983
|
+
catch (err) {
|
|
1984
|
+
console.warn(`Error looking up phone number ${phoneNumber}: `, err);
|
|
1985
|
+
return {
|
|
1986
|
+
normalizedPhoneNumber: normalizedPhoneNumber,
|
|
1987
|
+
};
|
|
1988
|
+
}
|
|
1989
|
+
}
|
|
1990
|
+
onLogin() {
|
|
1991
|
+
this.loginForm.markAllAsTouched();
|
|
1992
|
+
this.hasAvayaLoginError = false;
|
|
1993
|
+
this.hasAvayaDeviceError = false;
|
|
1994
|
+
this.hasAvayaSocketError = false;
|
|
1995
|
+
if (this.loginForm.invalid) {
|
|
1996
|
+
console.warn(`Invalid form data`);
|
|
1997
|
+
return;
|
|
1998
|
+
}
|
|
1999
|
+
this.showLoginLoader = true;
|
|
2000
|
+
const { ipoExtensionNo, ipoPassword } = this.loginForm.value;
|
|
2001
|
+
this.extensionNumber = ipoExtensionNo;
|
|
2002
|
+
try {
|
|
2003
|
+
this.avayaIPOService.logIn(ipoExtensionNo, ipoPassword);
|
|
2004
|
+
}
|
|
2005
|
+
catch (err) {
|
|
2006
|
+
console.error(err);
|
|
2007
|
+
this.showLoginLoader = false;
|
|
2008
|
+
}
|
|
2009
|
+
}
|
|
2010
|
+
onLogout() {
|
|
2011
|
+
this.avayaIPOService.logOut();
|
|
2012
|
+
}
|
|
2013
|
+
async makeCall(phoneNumber) {
|
|
2014
|
+
this.redialnumber = phoneNumber;
|
|
2015
|
+
console.log('phone number passed', phoneNumber);
|
|
2016
|
+
this.isCalling = true;
|
|
2017
|
+
this.displayCallingNumber = phoneNumber;
|
|
2018
|
+
this.showActiveCall = true;
|
|
2019
|
+
this.cdr.detectChanges();
|
|
2020
|
+
if (!phoneNumber) {
|
|
2021
|
+
alert('Please enter a number to call.');
|
|
2022
|
+
this.isCalling = false;
|
|
2023
|
+
return;
|
|
2024
|
+
}
|
|
2025
|
+
const initPhoneNumber = phoneNumber.replace(/\s+/g, '');
|
|
2026
|
+
this.lookupPhoneNumber = await this.getLookupPhoneNumber(phoneNumber);
|
|
2027
|
+
console.log('lookupPhoneNumber: ', this.lookupPhoneNumber);
|
|
2028
|
+
this.phoneNumber = this.lookupPhoneNumber?.lookupPhoneNumber?.['local-number']?.replace(/\s+/g, '');
|
|
2029
|
+
console.log('this.phoneNumber: ', this.phoneNumber);
|
|
2030
|
+
// hold the existing call if any
|
|
2031
|
+
if (this.activeCalls.length > 0) {
|
|
2032
|
+
const currentCall = this.activeCalls[this.currentActiveCallIndex];
|
|
2033
|
+
currentCall.isOnHold = true;
|
|
2034
|
+
this.avayaIPOService.doHold();
|
|
2035
|
+
}
|
|
2036
|
+
// don't duplicate the same number
|
|
2037
|
+
const alreadyExists = this.activeCalls.some((c) => c.phoneNumber === phoneNumber);
|
|
2038
|
+
if (alreadyExists) {
|
|
2039
|
+
console.warn('Call already in progress for this number.');
|
|
2040
|
+
this.isCalling = false;
|
|
2041
|
+
return;
|
|
2042
|
+
}
|
|
2043
|
+
const dataToSend = '*************custom data in sdp*************';
|
|
2044
|
+
try {
|
|
2045
|
+
// 🔴 FIX: await the Promise returned by the service
|
|
2046
|
+
const slot = await this.avayaIPOService.makeCallWithCustomData(this.phoneNumber, dataToSend);
|
|
2047
|
+
if (!slot) {
|
|
2048
|
+
console.warn('Could not place call: no free slot or call failed to start.');
|
|
2049
|
+
this.isCalling = false;
|
|
2050
|
+
return;
|
|
2051
|
+
}
|
|
2052
|
+
this.activeCalls.push({
|
|
2053
|
+
callId: this.avayaIPOService.activeCallId,
|
|
2054
|
+
phoneNumber,
|
|
2055
|
+
lookupPhoneNumber: this.lookupPhoneNumber,
|
|
2056
|
+
duration: 0,
|
|
2057
|
+
isMuted: false,
|
|
2058
|
+
isOnHold: false,
|
|
2059
|
+
startTime: Date.now(),
|
|
2060
|
+
slot, // ✅ now a "1" | "2" | "3" value, not a Promise
|
|
2061
|
+
});
|
|
2062
|
+
this.currentActiveCallIndex = this.activeCalls.length - 1;
|
|
2063
|
+
this.noOfCalls = this.activeCalls.length;
|
|
2064
|
+
// Log call initiation so the same IDs are present before connect/disconnect logs
|
|
2065
|
+
const callId = this.avayaIPOService.activeCallId;
|
|
2066
|
+
const callLogPayload = {
|
|
2067
|
+
tenantId: this.tenantId,
|
|
2068
|
+
agentId: this.extensionNumber,
|
|
2069
|
+
callId,
|
|
2070
|
+
phoneNumber: this.lookupPhoneNumber ?? phoneNumber,
|
|
2071
|
+
direction: 'outbound',
|
|
2072
|
+
duration: 0,
|
|
2073
|
+
status: 'initiated',
|
|
2074
|
+
timestamp: new Date().toISOString(),
|
|
2075
|
+
callinginfo: 'calling',
|
|
2076
|
+
};
|
|
2077
|
+
this.avayaIPOService.logCall(callLogPayload).subscribe({
|
|
2078
|
+
error: (err) => console.error('logCall (initiated) failed', err),
|
|
2079
|
+
});
|
|
2080
|
+
this.avayaIPOService.agentBusy();
|
|
2081
|
+
this.isCallingFromContact = false;
|
|
2082
|
+
this.isAddingNewCall = false;
|
|
2083
|
+
}
|
|
2084
|
+
catch (err) {
|
|
2085
|
+
console.error('Failed to start call:', err);
|
|
2086
|
+
this.isCalling = false;
|
|
2087
|
+
}
|
|
2088
|
+
}
|
|
2089
|
+
async makeCallFromContact(phoneNumber) {
|
|
2090
|
+
this.isCallingFromContact = true;
|
|
2091
|
+
this.makeCall(phoneNumber);
|
|
2092
|
+
}
|
|
2093
|
+
async cancelCall() {
|
|
2094
|
+
// Mirror manual disconnect button behavior
|
|
2095
|
+
this.disconnecttriggered = true;
|
|
2096
|
+
this.avayaIPOService.agentIdle();
|
|
2097
|
+
await this.dropCurrentCall();
|
|
2098
|
+
}
|
|
2099
|
+
async dropCurrentCall() {
|
|
2100
|
+
console.log('***************dropping call button triggered******************');
|
|
2101
|
+
this.disconnecttriggered = true;
|
|
2102
|
+
console.log('Cancelling active call. Active calls before:', this.activeCalls);
|
|
2103
|
+
console.log('Current active call index:', this.currentActiveCallIndex);
|
|
2104
|
+
const index = this.currentActiveCallIndex;
|
|
2105
|
+
if (index < 0 || index >= this.activeCalls.length) {
|
|
2106
|
+
console.warn('Active call index out of range:', index);
|
|
2107
|
+
return;
|
|
2108
|
+
}
|
|
2109
|
+
const activeCall = this.activeCalls[index];
|
|
2110
|
+
const targetSlot = activeCall.slot;
|
|
2111
|
+
if (!targetSlot) {
|
|
2112
|
+
console.warn('Active call at index', index, 'does not have a valid slot.');
|
|
2113
|
+
return;
|
|
2114
|
+
}
|
|
2115
|
+
console.log(`Dropping active call from slot ${targetSlot} (index ${index}).`);
|
|
2116
|
+
await this.avayaIPOService.dropCallBySlot(targetSlot);
|
|
2117
|
+
// Log the manual drop so it shares the same IDs as other call logs
|
|
2118
|
+
const dropLog = {
|
|
2119
|
+
tenantId: this.tenantId,
|
|
2120
|
+
agentId: this.extensionNumber,
|
|
2121
|
+
callId: activeCall.callId,
|
|
2122
|
+
phoneNumber: activeCall.lookupPhoneNumber ?? activeCall.phoneNumber,
|
|
2123
|
+
duration: activeCall.duration,
|
|
2124
|
+
status: 'disconnected',
|
|
2125
|
+
timestamp: new Date().toISOString(),
|
|
2126
|
+
callinginfo: 'calling',
|
|
2127
|
+
disconnectSide: 'AGENT',
|
|
2128
|
+
};
|
|
2129
|
+
this.avayaIPOService.logCall(dropLog).subscribe({
|
|
2130
|
+
error: (err) => console.error('logCall (dropCurrentCall) failed', err),
|
|
2131
|
+
});
|
|
2132
|
+
this.handleCallDisconnect(index, 'CALLER');
|
|
2133
|
+
// Force UI back to dialpad after manual drop
|
|
2134
|
+
this.zone.run(() => {
|
|
2135
|
+
this.showActiveCall = this.activeCalls.length > 0;
|
|
2136
|
+
if (!this.showActiveCall) {
|
|
2137
|
+
this.bottomActiveTab = 'keypad_sd';
|
|
2138
|
+
this.avayaIPOService.activeCallId = null;
|
|
2139
|
+
}
|
|
2140
|
+
this.cdr.detectChanges();
|
|
2141
|
+
});
|
|
2142
|
+
}
|
|
2143
|
+
onMute() {
|
|
2144
|
+
try {
|
|
2145
|
+
this.avayaIPOService.doMute();
|
|
2146
|
+
this.isMuted = !this.isMuted;
|
|
2147
|
+
}
|
|
2148
|
+
catch {
|
|
2149
|
+
console.log("couldn't mute");
|
|
2150
|
+
}
|
|
2151
|
+
}
|
|
2152
|
+
onUnmute() {
|
|
2153
|
+
try {
|
|
2154
|
+
this.avayaIPOService.doUnMute();
|
|
2155
|
+
this.isMuted = !this.isMuted;
|
|
2156
|
+
}
|
|
2157
|
+
catch {
|
|
2158
|
+
console.log("couldn't unmute");
|
|
2159
|
+
}
|
|
2160
|
+
}
|
|
2161
|
+
toggleHold() {
|
|
2162
|
+
if (!this.activeCalls[this.currentActiveCallIndex])
|
|
2163
|
+
return;
|
|
2164
|
+
if (this.activeCalls[this.currentActiveCallIndex].isOnHold) {
|
|
2165
|
+
this.avayaIPOService.doUnHold();
|
|
2166
|
+
this.activeCalls[this.currentActiveCallIndex].isOnHold = false;
|
|
2167
|
+
}
|
|
2168
|
+
else {
|
|
2169
|
+
this.avayaIPOService.doHold();
|
|
2170
|
+
this.activeCalls[this.currentActiveCallIndex].isOnHold = true;
|
|
2171
|
+
}
|
|
2172
|
+
}
|
|
2173
|
+
toggleMute() {
|
|
2174
|
+
if (!this.activeCalls[this.currentActiveCallIndex])
|
|
2175
|
+
return;
|
|
2176
|
+
if (this.activeCalls[this.currentActiveCallIndex].isMuted) {
|
|
2177
|
+
this.avayaIPOService.doUnMute();
|
|
2178
|
+
this.activeCalls[this.currentActiveCallIndex].isMuted = false;
|
|
2179
|
+
}
|
|
2180
|
+
else {
|
|
2181
|
+
this.avayaIPOService.doMute();
|
|
2182
|
+
this.activeCalls[this.currentActiveCallIndex].isMuted = true;
|
|
2183
|
+
}
|
|
2184
|
+
}
|
|
2185
|
+
onPauseVideo() {
|
|
2186
|
+
this.avayaIPOService.pauseVideo();
|
|
2187
|
+
}
|
|
2188
|
+
onResumeVideo() {
|
|
2189
|
+
this.avayaIPOService.playVideo();
|
|
2190
|
+
}
|
|
2191
|
+
onDigitClick(digit) {
|
|
2192
|
+
const currentValue = this.transferForm.get('target')?.value || '';
|
|
2193
|
+
this.transferForm.patchValue({ target: currentValue + digit });
|
|
2194
|
+
}
|
|
2195
|
+
onBackspace() {
|
|
2196
|
+
const currentValue = this.transferForm.get('target')?.value || '';
|
|
2197
|
+
if (currentValue.length > 0) {
|
|
2198
|
+
this.transferForm.patchValue({ target: currentValue.slice(0, -1) });
|
|
2199
|
+
}
|
|
2200
|
+
}
|
|
2201
|
+
onTransferCall() {
|
|
2202
|
+
this.istransfer = true;
|
|
2203
|
+
if (this.transferForm.valid) {
|
|
2204
|
+
const { target } = this.transferForm.value;
|
|
2205
|
+
console.log('Transfer Target:', target);
|
|
2206
|
+
const type = 'unAttended';
|
|
2207
|
+
this.avayaIPOService.transferCall(target, type);
|
|
2208
|
+
}
|
|
2209
|
+
else {
|
|
2210
|
+
console.error('Transfer form is invalid.');
|
|
2211
|
+
}
|
|
2212
|
+
}
|
|
2213
|
+
onSendDTMF(digit) {
|
|
2214
|
+
this.avayaIPOService.sendDTMF(digit);
|
|
2215
|
+
const currentValue = this.dtmfForm.get('dtmf')?.value || '';
|
|
2216
|
+
this.dtmfForm.patchValue({ dtmf: currentValue + digit });
|
|
2217
|
+
}
|
|
2218
|
+
onAnswerCall() {
|
|
2219
|
+
this.avayaIPOService.answerCall();
|
|
2220
|
+
this.isRinging = false;
|
|
2221
|
+
// this.stopRingtone();
|
|
2222
|
+
this.showActiveCall = true;
|
|
2223
|
+
}
|
|
2224
|
+
onAddVideo() {
|
|
2225
|
+
this.avayaIPOService.addVideo();
|
|
2226
|
+
}
|
|
2227
|
+
onRemoveVideo() {
|
|
2228
|
+
this.avayaIPOService.removeVideo();
|
|
2229
|
+
}
|
|
2230
|
+
allowOnlyNumbers(event) {
|
|
2231
|
+
const input = event.target;
|
|
2232
|
+
input.value = input.value.replace(/[^0-9]/g, '');
|
|
2233
|
+
}
|
|
2234
|
+
startTimer(index) {
|
|
2235
|
+
const call = this.activeCalls[index];
|
|
2236
|
+
if (!call || call.timer)
|
|
2237
|
+
return;
|
|
2238
|
+
call.timer = setInterval(() => {
|
|
2239
|
+
call.duration++;
|
|
2240
|
+
// this.cdr.detectChanges();
|
|
2241
|
+
this.cdr.markForCheck();
|
|
2242
|
+
}, 1000);
|
|
2243
|
+
}
|
|
2244
|
+
clearTimer(index) {
|
|
2245
|
+
const call = this.activeCalls[index];
|
|
2246
|
+
if (call?.timer) {
|
|
2247
|
+
clearInterval(call.timer);
|
|
2248
|
+
call.timer = null;
|
|
2249
|
+
}
|
|
2250
|
+
}
|
|
2251
|
+
formatTime(totalSeconds) {
|
|
2252
|
+
const minutes = Math.floor(totalSeconds / 60);
|
|
2253
|
+
const seconds = totalSeconds % 60;
|
|
2254
|
+
return (String(minutes).padStart(2, '0') + ':' + String(seconds).padStart(2, '0'));
|
|
2255
|
+
}
|
|
2256
|
+
menuDailpad() {
|
|
2257
|
+
this.isDailpadOpen = !this.isDailpadOpen;
|
|
2258
|
+
this.cdr.detectChanges();
|
|
2259
|
+
}
|
|
2260
|
+
async selectTab(tab) {
|
|
2261
|
+
this.activeTab = tab;
|
|
2262
|
+
/* if (tab === 'tab1') {
|
|
2263
|
+
await this.loadCallHistory();
|
|
2264
|
+
} */
|
|
2265
|
+
}
|
|
2266
|
+
async fetchContacts() {
|
|
2267
|
+
const size = 50;
|
|
2268
|
+
let start = 0;
|
|
2269
|
+
let allContacts = [];
|
|
2270
|
+
while (true) {
|
|
2271
|
+
const queryString = `q=(and (term field%3Dtenant_id '${this.tenantId}'))&size=${size}&start=${start}&q.parser=structured`;
|
|
2272
|
+
try {
|
|
2273
|
+
const res_searchEntities = await this.searchEntities(queryString);
|
|
2274
|
+
const hits = res_searchEntities?.hits?.hit;
|
|
2275
|
+
if (!Array.isArray(hits) || hits.length === 0) {
|
|
2276
|
+
break;
|
|
2277
|
+
}
|
|
2278
|
+
allContacts = [...allContacts, ...hits];
|
|
2279
|
+
start += size;
|
|
2280
|
+
}
|
|
2281
|
+
catch (error) {
|
|
2282
|
+
console.error('Error fetching contacts:', error);
|
|
2283
|
+
break;
|
|
2284
|
+
}
|
|
2285
|
+
}
|
|
2286
|
+
this.contacts = allContacts;
|
|
2287
|
+
}
|
|
2288
|
+
searchEntities(queryString) {
|
|
2289
|
+
return lastValueFrom(this.http.get('https://search.snugdesk.com/entities?' + queryString));
|
|
2290
|
+
}
|
|
2291
|
+
handleCallConnect() {
|
|
2292
|
+
console.log("handle call connect ----------------------------");
|
|
2293
|
+
const active = this.activeCalls[this.currentActiveCallIndex];
|
|
2294
|
+
if (!active)
|
|
2295
|
+
return;
|
|
2296
|
+
const { callId, phoneNumber } = active;
|
|
2297
|
+
const callMetadata = {
|
|
2298
|
+
tenantId: this.tenantId,
|
|
2299
|
+
agentId: this.extensionNumber,
|
|
2300
|
+
callId,
|
|
2301
|
+
phoneNumber,
|
|
2302
|
+
};
|
|
2303
|
+
this.callSocket.open(callMetadata);
|
|
2304
|
+
// Update UI and call state
|
|
2305
|
+
this.startTimer(this.currentActiveCallIndex);
|
|
2306
|
+
this.isRinging = false;
|
|
2307
|
+
this.isactiveCall = true;
|
|
2308
|
+
console.log("this.isactiveCall = ", this.isactiveCall);
|
|
2309
|
+
this.isCalling = false;
|
|
2310
|
+
this.showActiveCall = true;
|
|
2311
|
+
// Log call as answered/connected so the same ids can be updated later on disconnect
|
|
2312
|
+
const callLogPayload = {
|
|
2313
|
+
tenantId: this.tenantId,
|
|
2314
|
+
agentId: this.extensionNumber,
|
|
2315
|
+
callId,
|
|
2316
|
+
phoneNumber: active.lookupPhoneNumber ?? active.phoneNumber,
|
|
2317
|
+
direction: this.wasInbound ? 'inbound' : 'outbound',
|
|
2318
|
+
duration: active.duration,
|
|
2319
|
+
status: 'connected',
|
|
2320
|
+
timestamp: new Date().toISOString(),
|
|
2321
|
+
callinginfo: 'calling',
|
|
2322
|
+
};
|
|
2323
|
+
this.avayaIPOService.logCall(callLogPayload).subscribe({
|
|
2324
|
+
error: (err) => console.error('logCall (connected) failed', err),
|
|
2325
|
+
});
|
|
2326
|
+
this.cdr.detectChanges();
|
|
2327
|
+
const local = this.lastLocalStream || undefined;
|
|
2328
|
+
const remote = this.lastRemoteStream || undefined;
|
|
2329
|
+
const hasAudio = (stream) => stream instanceof MediaStream && stream.getAudioTracks().length > 0;
|
|
2330
|
+
if (hasAudio(remote) && hasAudio(local)) {
|
|
2331
|
+
console.debug(`[${callId}] Starting recording with available local & remote streams`);
|
|
2332
|
+
this.recordingManagerService.startRecording(local, remote, callMetadata);
|
|
2333
|
+
this.initTranscription();
|
|
2334
|
+
this.recognition.start();
|
|
2335
|
+
this.showTranscriptionPopup = true;
|
|
2336
|
+
this.transcriptSegments = [];
|
|
2337
|
+
this.transcriptHistory = [];
|
|
2338
|
+
this.transcriptText = '';
|
|
2339
|
+
this.transcriptStartedAt = Date.now();
|
|
2340
|
+
return;
|
|
2341
|
+
}
|
|
2342
|
+
console.debug(`[${callId}] Waiting for remote stream...`);
|
|
2343
|
+
const remoteStreamSubscription = this.avayaIPOService.remoteStream$
|
|
2344
|
+
.pipe(filter((r) => !!r && r.callId === callId), filter((r) => r.stream.getAudioTracks().length > 0))
|
|
2345
|
+
.subscribe({
|
|
2346
|
+
next: ({ stream: freshRemote }) => {
|
|
2347
|
+
const freshLocal = this.avayaIPOService.localStream;
|
|
2348
|
+
console.debug(`[${callId}] Remote stream received! Starting recording.`);
|
|
2349
|
+
this.recordingManagerService.startRecording(freshLocal, freshRemote, callMetadata);
|
|
2350
|
+
this.initTranscription();
|
|
2351
|
+
this.recognition.start();
|
|
2352
|
+
this.showTranscriptionPopup = true;
|
|
2353
|
+
this.transcriptSegments = [];
|
|
2354
|
+
this.transcriptHistory = [];
|
|
2355
|
+
this.transcriptText = '';
|
|
2356
|
+
this.transcriptStartedAt = Date.now();
|
|
2357
|
+
remoteStreamSubscription.unsubscribe(); // stop listening after the first hit
|
|
2358
|
+
},
|
|
2359
|
+
error: (err) => {
|
|
2360
|
+
console.error(`[${callId}] Error receiving remote stream:`, err);
|
|
2361
|
+
remoteStreamSubscription.unsubscribe();
|
|
2362
|
+
},
|
|
2363
|
+
});
|
|
2364
|
+
}
|
|
2365
|
+
handleCallDisconnect(index, disconnectSide) {
|
|
2366
|
+
console.log('+++++++++++++++++handle disconnect trigered++++++++++++++');
|
|
2367
|
+
this.disconnecttriggered = false;
|
|
2368
|
+
this.isAddingNewCall = false;
|
|
2369
|
+
this.isCalling = false;
|
|
2370
|
+
this.isRecording = false;
|
|
2371
|
+
console.log(`Disconnecting call at index ${index}. Active calls before:`, this.activeCalls);
|
|
2372
|
+
if (index < 0 || index >= this.activeCalls.length) {
|
|
2373
|
+
console.warn(`Invalid index ${index}, activeCalls.length = ${this.activeCalls.length}`);
|
|
2374
|
+
return;
|
|
2375
|
+
}
|
|
2376
|
+
const disconnectingCall = this.activeCalls[index];
|
|
2377
|
+
this.clearTimer(index);
|
|
2378
|
+
const callId = this.activeCalls[this.currentActiveCallIndex].callId;
|
|
2379
|
+
this.recordingManagerService.stopRecording(disconnectingCall.callId);
|
|
2380
|
+
console.log('***********stopping recording in disconnect***********');
|
|
2381
|
+
this.callSocket.close(callId);
|
|
2382
|
+
// const callData = {
|
|
2383
|
+
// tenantId: this.tenantId,
|
|
2384
|
+
// agentId: this.extensionNumber,
|
|
2385
|
+
// callId: this.avayaIPOService.currentcallId,
|
|
2386
|
+
// // phoneNumber: disconnectingCall.phoneNumber,
|
|
2387
|
+
// phoneNumber:
|
|
2388
|
+
// this.activeCalls[this.currentActiveCallIndex].lookupPhoneNumber,
|
|
2389
|
+
// direction: this.wasInbound ? 'inbound' : 'outbound',
|
|
2390
|
+
// duration: disconnectingCall.duration,
|
|
2391
|
+
// status: 'disconnected',
|
|
2392
|
+
// timestamp: new Date().toISOString(),
|
|
2393
|
+
// callinginfo: 'calling',
|
|
2394
|
+
// };
|
|
2395
|
+
const callData = {
|
|
2396
|
+
tenantId: this.tenantId,
|
|
2397
|
+
agentId: this.extensionNumber,
|
|
2398
|
+
callId: disconnectingCall.callId,
|
|
2399
|
+
phoneNumber: disconnectingCall.lookupPhoneNumber,
|
|
2400
|
+
direction: this.wasInbound ? 'inbound' : 'outbound',
|
|
2401
|
+
duration: disconnectingCall.duration,
|
|
2402
|
+
status: 'disconnected',
|
|
2403
|
+
timestamp: new Date().toISOString(),
|
|
2404
|
+
callinginfo: 'calling',
|
|
2405
|
+
disconnectSide: disconnectSide
|
|
2406
|
+
};
|
|
2407
|
+
this.avayaIPOService.logCall(callData).subscribe(() => { }, (err) => console.error(err));
|
|
2408
|
+
this.lastDisconnectedCall = {
|
|
2409
|
+
...disconnectingCall,
|
|
2410
|
+
};
|
|
2411
|
+
this.activeCalls.splice(index, 1);
|
|
2412
|
+
console.log('Active calls after removal:', this.activeCalls);
|
|
2413
|
+
if (index === this.currentActiveCallIndex) {
|
|
2414
|
+
if (this.activeCalls.length > 0) {
|
|
2415
|
+
this.currentActiveCallIndex = 0;
|
|
2416
|
+
const newActive = this.activeCalls[0];
|
|
2417
|
+
this.lookupPhoneNumber = newActive.phoneNumber;
|
|
2418
|
+
// this.lookupPhoneNumber = newActive.lookupPhoneNumber; TODO to check whats happening here
|
|
2419
|
+
this.startTimer(0);
|
|
2420
|
+
this.isMuted = newActive.isMuted;
|
|
2421
|
+
this.isOnHold = false;
|
|
2422
|
+
this.showActiveCall = true;
|
|
2423
|
+
this.isactiveCall = true;
|
|
2424
|
+
this.avayaIPOService.activeCallId = newActive.callId;
|
|
2425
|
+
this.avayaIPOService.doUnHold();
|
|
2426
|
+
newActive.isOnHold = false;
|
|
2427
|
+
for (let i = 1; i < this.activeCalls.length; i++) {
|
|
2428
|
+
const other = this.activeCalls[i];
|
|
2429
|
+
this.avayaIPOService.activeCallId = other.callId;
|
|
2430
|
+
this.avayaIPOService.doHold();
|
|
2431
|
+
other.isOnHold = true;
|
|
2432
|
+
}
|
|
2433
|
+
console.log(`Switched active call to slot ${newActive.slot} (index=0)`);
|
|
2434
|
+
}
|
|
2435
|
+
else {
|
|
2436
|
+
this.showCallDisconnected = true;
|
|
2437
|
+
this.isactiveCall = false;
|
|
2438
|
+
this.isMuted = false;
|
|
2439
|
+
this.isOnHold = false;
|
|
2440
|
+
console.log('flag value for isCalling', this.isCalling);
|
|
2441
|
+
this.showActiveCall = false; // return to dialpad view when no calls remain
|
|
2442
|
+
this.bottomActiveTab = 'keypad_sd';
|
|
2443
|
+
this.avayaIPOService.activeCallId = null;
|
|
2444
|
+
if (!this.isCalling) {
|
|
2445
|
+
this.phoneNumber = '';
|
|
2446
|
+
this.lookupPhoneNumber = null;
|
|
2447
|
+
}
|
|
2448
|
+
this.istransfer = false;
|
|
2449
|
+
this.showCallDisconnected = false;
|
|
2450
|
+
this.cdr.detectChanges();
|
|
2451
|
+
}
|
|
2452
|
+
}
|
|
2453
|
+
else if (index < this.currentActiveCallIndex) {
|
|
2454
|
+
this.currentActiveCallIndex--;
|
|
2455
|
+
}
|
|
2456
|
+
this.transferForm.reset();
|
|
2457
|
+
this.dtmfForm.reset();
|
|
2458
|
+
this.isDailpadOpen = false;
|
|
2459
|
+
this.isCallTransferOpen = false;
|
|
2460
|
+
this.isRinging = false;
|
|
2461
|
+
setTimeout(() => {
|
|
2462
|
+
this.showActiveCall = this.activeCalls.length > 0;
|
|
2463
|
+
this.cdr.detectChanges();
|
|
2464
|
+
}, 3000);
|
|
2465
|
+
console.log('handleCallDisconnect() done. new active index =', this.currentActiveCallIndex);
|
|
2466
|
+
console.log('active call div decider ', this.showActiveCall);
|
|
2467
|
+
this.stopTranscription(disconnectingCall.callId);
|
|
2468
|
+
}
|
|
2469
|
+
goToDiv(div) {
|
|
2470
|
+
this.currentView = div;
|
|
2471
|
+
}
|
|
2472
|
+
restrictNumeric(e) {
|
|
2473
|
+
const allowedKeys = [
|
|
2474
|
+
'Enter',
|
|
2475
|
+
'Backspace',
|
|
2476
|
+
'ArrowLeft',
|
|
2477
|
+
'ArrowRight',
|
|
2478
|
+
'Tab',
|
|
2479
|
+
'Delete',
|
|
2480
|
+
];
|
|
2481
|
+
return /^[0-9]$/.test(e.key) || allowedKeys.includes(e.key);
|
|
2482
|
+
}
|
|
2483
|
+
onRedial() {
|
|
2484
|
+
this.isredialling = true;
|
|
2485
|
+
if (!this.phoneNumber) {
|
|
2486
|
+
console.log('returninggggggg');
|
|
2487
|
+
return;
|
|
2488
|
+
}
|
|
2489
|
+
if (this.redialTimeout) {
|
|
2490
|
+
console.log('redialingggggggggggggggg');
|
|
2491
|
+
this.istransfer = false;
|
|
2492
|
+
this.showCallDisconnected = false;
|
|
2493
|
+
clearTimeout(this.redialTimeout);
|
|
2494
|
+
this.redialTimeout = null;
|
|
2495
|
+
}
|
|
2496
|
+
this.isCalling = true;
|
|
2497
|
+
this.makeCall(this.redialnumber);
|
|
2498
|
+
}
|
|
2499
|
+
swapLinesToIndex(index) {
|
|
2500
|
+
const slotMap = ['1', '2', '3'];
|
|
2501
|
+
if (index < 0 || index >= slotMap.length) {
|
|
2502
|
+
console.warn('Index out of range for slotMap');
|
|
2503
|
+
return;
|
|
2504
|
+
}
|
|
2505
|
+
const targetSlot = slotMap[index];
|
|
2506
|
+
this.avayaIPOService.swapToSlot(targetSlot);
|
|
2507
|
+
this.activeCalls.forEach((call, i) => {
|
|
2508
|
+
if (call.slot === targetSlot) {
|
|
2509
|
+
call.isOnHold = false;
|
|
2510
|
+
this.currentActiveCallIndex = i;
|
|
2511
|
+
}
|
|
2512
|
+
else {
|
|
2513
|
+
call.isOnHold = true;
|
|
2514
|
+
}
|
|
2515
|
+
});
|
|
2516
|
+
// this.currentActiveCallIndex = index;x
|
|
2517
|
+
}
|
|
2518
|
+
mergeCall() {
|
|
2519
|
+
console.log('******************* merging ******************');
|
|
2520
|
+
this.avayaIPOService.mergeCall();
|
|
2521
|
+
}
|
|
2522
|
+
onAddCall() {
|
|
2523
|
+
this.isCallTransferOpen = false;
|
|
2524
|
+
this.isAddingNewCall = true;
|
|
2525
|
+
// this.dialForm.reset();
|
|
2526
|
+
}
|
|
2527
|
+
getDisplayStatus(call) {
|
|
2528
|
+
if (call.direction === 'inbound' &&
|
|
2529
|
+
call.status === 'disconnected' &&
|
|
2530
|
+
call.duration === 0) {
|
|
2531
|
+
return 'Missed';
|
|
2532
|
+
}
|
|
2533
|
+
// If status is "connected"
|
|
2534
|
+
if (call.status === 'connected') {
|
|
2535
|
+
return 'Connected';
|
|
2536
|
+
}
|
|
2537
|
+
// If status is "disconnected" but not missed
|
|
2538
|
+
if (call.status === 'disconnected') {
|
|
2539
|
+
return 'Disconnected';
|
|
2540
|
+
}
|
|
2541
|
+
// Otherwise show the raw status
|
|
2542
|
+
return call.status;
|
|
2543
|
+
}
|
|
2544
|
+
bottomSelectTab(tab) {
|
|
2545
|
+
this.bottomActiveTab = tab;
|
|
2546
|
+
}
|
|
2547
|
+
/** Central place to notify AvayaIPOService about current picks */
|
|
2548
|
+
handleDeviceSelection(event) {
|
|
2549
|
+
this.avayaIPOService.setDeviceIds({
|
|
2550
|
+
defaultId: true,
|
|
2551
|
+
audioInputID: event.audioInputId || undefined,
|
|
2552
|
+
audioOutputID: event.audioOutputId || undefined,
|
|
2553
|
+
videoInputID: event.videoInputId || undefined,
|
|
2554
|
+
});
|
|
2555
|
+
}
|
|
2556
|
+
createMixedStream(local, remote) {
|
|
2557
|
+
const audioContext = new AudioContext();
|
|
2558
|
+
const destination = audioContext.createMediaStreamDestination();
|
|
2559
|
+
// Local stream
|
|
2560
|
+
if (local) {
|
|
2561
|
+
const localSource = audioContext.createMediaStreamSource(local);
|
|
2562
|
+
localSource.connect(destination);
|
|
2563
|
+
}
|
|
2564
|
+
// Remote stream
|
|
2565
|
+
if (remote) {
|
|
2566
|
+
const remoteSource = audioContext.createMediaStreamSource(remote);
|
|
2567
|
+
remoteSource.connect(destination);
|
|
2568
|
+
}
|
|
2569
|
+
return destination.stream;
|
|
2570
|
+
}
|
|
2571
|
+
initTranscription() {
|
|
2572
|
+
const SpeechRecognition = window.SpeechRecognition ||
|
|
2573
|
+
window.webkitSpeechRecognition;
|
|
2574
|
+
if (!SpeechRecognition) {
|
|
2575
|
+
console.error('Web Speech API is not supported in this browser.');
|
|
2576
|
+
return;
|
|
2577
|
+
}
|
|
2578
|
+
this.recognition = new SpeechRecognition();
|
|
2579
|
+
this.recognition.lang = 'en-US'; // change as needed
|
|
2580
|
+
this.recognition.continuous = true;
|
|
2581
|
+
this.recognition.interimResults = true;
|
|
2582
|
+
this.recognition.onresult = (event) => {
|
|
2583
|
+
let interimTranscript = '';
|
|
2584
|
+
for (let i = event.resultIndex; i < event.results.length; ++i) {
|
|
2585
|
+
if (event.results[i].isFinal) {
|
|
2586
|
+
const text = event.results[i][0].transcript.trim();
|
|
2587
|
+
this.transcriptHistory.push(text);
|
|
2588
|
+
// ⬇️ ADD: store finals w/ timestamp for summary
|
|
2589
|
+
this.transcriptSegments.push({ ts: Date.now(), text });
|
|
2590
|
+
}
|
|
2591
|
+
else {
|
|
2592
|
+
interimTranscript += event.results[i][0].transcript;
|
|
2593
|
+
}
|
|
2594
|
+
}
|
|
2595
|
+
this.transcriptText = [...this.transcriptHistory, interimTranscript].join(' ');
|
|
2596
|
+
console.log('Live Transcription:', this.transcriptText, this.transcriptHistory);
|
|
2597
|
+
this.cdr.detectChanges();
|
|
2598
|
+
};
|
|
2599
|
+
this.recognition.onerror = (event) => {
|
|
2600
|
+
console.error('Transcription error:', event.error);
|
|
2601
|
+
};
|
|
2602
|
+
this.recognition.onend = () => {
|
|
2603
|
+
if (this.showTranscriptionPopup) {
|
|
2604
|
+
console.log('Recognition ended, restarting…');
|
|
2605
|
+
try {
|
|
2606
|
+
this.recognition.start();
|
|
2607
|
+
}
|
|
2608
|
+
catch (err) {
|
|
2609
|
+
console.warn('Error restarting recognition:', err);
|
|
2610
|
+
}
|
|
2611
|
+
}
|
|
2612
|
+
};
|
|
2613
|
+
// 👇 Manual keep-alive every 25s
|
|
2614
|
+
if (this.recognitionKeepAlive) {
|
|
2615
|
+
clearInterval(this.recognitionKeepAlive);
|
|
2616
|
+
}
|
|
2617
|
+
this.recognitionKeepAlive = setInterval(() => {
|
|
2618
|
+
if (this.showTranscriptionPopup && this.recognition) {
|
|
2619
|
+
console.log('Force restarting recognition (keep-alive)…');
|
|
2620
|
+
try {
|
|
2621
|
+
this.recognition.stop(); // will trigger onend → restart
|
|
2622
|
+
}
|
|
2623
|
+
catch (err) {
|
|
2624
|
+
console.warn('Error stopping recognition:', err);
|
|
2625
|
+
}
|
|
2626
|
+
}
|
|
2627
|
+
}, 10000);
|
|
2628
|
+
}
|
|
2629
|
+
stopTranscription(callId) {
|
|
2630
|
+
// mark end BEFORE clearing anything
|
|
2631
|
+
this.transcriptEndedAt = Date.now();
|
|
2632
|
+
if (this.recognitionKeepAlive) {
|
|
2633
|
+
clearInterval(this.recognitionKeepAlive);
|
|
2634
|
+
this.recognitionKeepAlive = null;
|
|
2635
|
+
}
|
|
2636
|
+
if (this.recognition) {
|
|
2637
|
+
try {
|
|
2638
|
+
this.recognition.onend = null; // prevent auto-restart during teardown
|
|
2639
|
+
this.recognition.stop();
|
|
2640
|
+
}
|
|
2641
|
+
catch { }
|
|
2642
|
+
}
|
|
2643
|
+
// ⬇️ NEW: build + download markdown summary (if there is content)
|
|
2644
|
+
if (this.autoDownloadSummaryOnEnd && this.transcriptSegments.length > 0) {
|
|
2645
|
+
this.generateAndDownloadSummary(callId);
|
|
2646
|
+
}
|
|
2647
|
+
// cleanup live state (leave segments cleared after download)
|
|
2648
|
+
this.transcriptText = '';
|
|
2649
|
+
this.transcriptHistory = [];
|
|
2650
|
+
this.transcriptSegments = [];
|
|
2651
|
+
this.transcriptStartedAt = 0;
|
|
2652
|
+
this.transcriptEndedAt = 0;
|
|
2653
|
+
this.showTranscriptionPopup = false;
|
|
2654
|
+
}
|
|
2655
|
+
// Build + download a Markdown summary using lightweight extractive heuristics
|
|
2656
|
+
generateAndDownloadSummary(callId) {
|
|
2657
|
+
const fullText = this.transcriptSegments.map(s => s.text).join(' ');
|
|
2658
|
+
if (!fullText.trim())
|
|
2659
|
+
return;
|
|
2660
|
+
const bullets = this.buildExtractiveSummary(fullText, 5); // 5 highlight bullets
|
|
2661
|
+
const actions = this.extractActionItems(fullText);
|
|
2662
|
+
const keyInfo = this.extractKeyInfo(fullText);
|
|
2663
|
+
const mdLines = [];
|
|
2664
|
+
mdLines.push('# Call Summary');
|
|
2665
|
+
mdLines.push(`**Call ID:** ${callId ?? '-'}`);
|
|
2666
|
+
mdLines.push(`**Agent:** ${this.extensionNumber} `);
|
|
2667
|
+
mdLines.push(`**Tenant:** ${this.tenantId}`);
|
|
2668
|
+
mdLines.push(`**Start:** ${this.transcriptStartedAt ? new Date(this.transcriptStartedAt).toISOString() : '-'}`);
|
|
2669
|
+
mdLines.push(`**End:** ${this.transcriptEndedAt ? new Date(this.transcriptEndedAt).toISOString() : '-'}`);
|
|
2670
|
+
const durMs = (this.transcriptStartedAt && this.transcriptEndedAt)
|
|
2671
|
+
? (this.transcriptEndedAt - this.transcriptStartedAt) : 0;
|
|
2672
|
+
mdLines.push(`**Duration:** ${this.fmtHMS(durMs)}`);
|
|
2673
|
+
mdLines.push('');
|
|
2674
|
+
mdLines.push('## Highlights');
|
|
2675
|
+
if (bullets.length)
|
|
2676
|
+
bullets.forEach(b => mdLines.push(`- ${b}`));
|
|
2677
|
+
else
|
|
2678
|
+
mdLines.push('- (no highlights)');
|
|
2679
|
+
if (actions.length) {
|
|
2680
|
+
mdLines.push('', '## Action Items');
|
|
2681
|
+
actions.forEach(a => mdLines.push(`- [ ] ${a}`));
|
|
2682
|
+
}
|
|
2683
|
+
if (keyInfo.length) {
|
|
2684
|
+
mdLines.push('', '## Key Info');
|
|
2685
|
+
keyInfo.forEach(k => mdLines.push(`- ${k}`));
|
|
2686
|
+
}
|
|
2687
|
+
// Optional appendix: full transcript with timestamps
|
|
2688
|
+
mdLines.push('', '## Transcript (final lines)');
|
|
2689
|
+
for (const seg of this.transcriptSegments) {
|
|
2690
|
+
const offset = seg.ts - (this.transcriptStartedAt || seg.ts);
|
|
2691
|
+
mdLines.push(`[${this.fmtHMS(offset)}] ${seg.text}`);
|
|
2692
|
+
}
|
|
2693
|
+
const mdBlob = new Blob([mdLines.join('\n')], { type: 'text/markdown;charset=utf-8' });
|
|
2694
|
+
const name = `call-${callId || 'unknown'}-summary.md`;
|
|
2695
|
+
this.downloadBlobFile(mdBlob, name);
|
|
2696
|
+
// store in memory if you also want to show it somewhere
|
|
2697
|
+
this.summaryText = mdLines.join('\n');
|
|
2698
|
+
}
|
|
2699
|
+
// === extractive “enough” summarizer ===
|
|
2700
|
+
buildExtractiveSummary(text, maxBullets = 5) {
|
|
2701
|
+
const sents = this.splitSentences(text);
|
|
2702
|
+
if (sents.length <= maxBullets)
|
|
2703
|
+
return sents;
|
|
2704
|
+
const stop = new Set(('i,me,my,myself,we,our,ours,ourselves,you,your,yours,' +
|
|
2705
|
+
'he,him,his,she,her,hers,it,its,they,them,their,theirs,what,which,who,whom,' +
|
|
2706
|
+
'this,that,these,those,am,is,are,was,were,be,been,being,have,has,had,do,does,' +
|
|
2707
|
+
'did,doing,a,an,the,and,but,if,or,because,as,until,while,of,at,by,for,with,' +
|
|
2708
|
+
'about,against,between,into,through,during,before,after,above,below,to,from,' +
|
|
2709
|
+
'up,down,in,out,on,off,over,under,again,further,then,once,here,there,when,' +
|
|
2710
|
+
'where,why,how,all,any,both,each,few,more,most,other,some,such,no,nor,not,' +
|
|
2711
|
+
'only,own,same,so,than,too,very,can,will,just,don,should,now').split(','));
|
|
2712
|
+
const tokenize = (s) => s.toLowerCase().replace(/[^a-z0-9\s]/g, ' ')
|
|
2713
|
+
.split(/\s+/).filter(w => w && !stop.has(w));
|
|
2714
|
+
const freq = new Map();
|
|
2715
|
+
for (const s of sents)
|
|
2716
|
+
for (const w of tokenize(s))
|
|
2717
|
+
freq.set(w, (freq.get(w) || 0) + 1);
|
|
2718
|
+
const scored = sents.map((s, i) => {
|
|
2719
|
+
const words = tokenize(s);
|
|
2720
|
+
const score = words.reduce((acc, w) => acc + (freq.get(w) || 0), 0) / Math.max(1, words.length);
|
|
2721
|
+
return { i, s, score };
|
|
2722
|
+
});
|
|
2723
|
+
return scored
|
|
2724
|
+
.sort((a, b) => b.score - a.score)
|
|
2725
|
+
.slice(0, maxBullets)
|
|
2726
|
+
.sort((a, b) => a.i - b.i)
|
|
2727
|
+
.map(x => x.s.trim());
|
|
2728
|
+
}
|
|
2729
|
+
splitSentences(text) {
|
|
2730
|
+
return text
|
|
2731
|
+
.replace(/\s+/g, ' ')
|
|
2732
|
+
.split(/(?<=[.!?])\s+(?=[A-Z0-9])/g)
|
|
2733
|
+
.map(s => s.trim())
|
|
2734
|
+
.filter(Boolean);
|
|
2735
|
+
}
|
|
2736
|
+
extractActionItems(text) {
|
|
2737
|
+
const sents = this.splitSentences(text);
|
|
2738
|
+
const pat = /(need to|please|send|share|call|schedule|arrange|follow up|email|whatsapp|provide|confirm|deliver|prepare|create|set up|book|assign|remind)/i;
|
|
2739
|
+
return sents.filter(s => pat.test(s)).slice(0, 12);
|
|
2740
|
+
}
|
|
2741
|
+
extractKeyInfo(text) {
|
|
2742
|
+
const out = [];
|
|
2743
|
+
const phones = text.match(/(?:\+?\d[\s-]?){8,15}\d/g) || [];
|
|
2744
|
+
const emails = text.match(/[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,}/gi) || [];
|
|
2745
|
+
const amounts = text.match(/(?:₹|\$|INR\s?)\s?\d[\d,]*(?:\.\d+)?/g) || [];
|
|
2746
|
+
const dates = text.match(/\b(?:\d{1,2}[-/]\d{1,2}[-/]\d{2,4}|(?:Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)[a-z]*\s+\d{1,2}(?:,\s*\d{4})?)\b/gi) || [];
|
|
2747
|
+
if (phones.length)
|
|
2748
|
+
out.push(`Phones: ${Array.from(new Set(phones)).join(', ')}`);
|
|
2749
|
+
if (emails.length)
|
|
2750
|
+
out.push(`Emails: ${Array.from(new Set(emails)).join(', ')}`);
|
|
2751
|
+
if (amounts.length)
|
|
2752
|
+
out.push(`Amounts: ${Array.from(new Set(amounts)).join(', ')}`);
|
|
2753
|
+
if (dates.length)
|
|
2754
|
+
out.push(`Dates: ${Array.from(new Set(dates)).join(', ')}`);
|
|
2755
|
+
return out;
|
|
2756
|
+
}
|
|
2757
|
+
downloadBlobFile(blob, filename) {
|
|
2758
|
+
const url = URL.createObjectURL(blob);
|
|
2759
|
+
const a = document.createElement('a');
|
|
2760
|
+
a.href = url;
|
|
2761
|
+
a.download = filename;
|
|
2762
|
+
document.body.appendChild(a);
|
|
2763
|
+
a.click();
|
|
2764
|
+
a.remove();
|
|
2765
|
+
URL.revokeObjectURL(url);
|
|
2766
|
+
}
|
|
2767
|
+
fmtHMS(ms) {
|
|
2768
|
+
const totalSec = Math.floor(ms / 1000);
|
|
2769
|
+
const h = Math.floor(totalSec / 3600);
|
|
2770
|
+
const m = Math.floor((totalSec % 3600) / 60);
|
|
2771
|
+
const s = totalSec % 60;
|
|
2772
|
+
return (h > 0 ? String(h).padStart(2, '0') + ':' : '') +
|
|
2773
|
+
String(m).padStart(2, '0') + ':' + String(s).padStart(2, '0');
|
|
2774
|
+
}
|
|
2775
|
+
// add this method in the component class
|
|
2776
|
+
transferToFromHistory(item) {
|
|
2777
|
+
// prefer a previously-resolved dialable if you store one
|
|
2778
|
+
const prefer = (v) => (v ?? '').toString().replace(/\s+/g, '');
|
|
2779
|
+
let target = prefer(item.resolvedDialable) ||
|
|
2780
|
+
prefer(item.lookupPhoneNumber?.['local-number']) ||
|
|
2781
|
+
prefer(item.lookupPhoneNumber?.lookupPhoneNumber?.['local-number']) ||
|
|
2782
|
+
prefer(item.lookupPhoneNumber?.normalizedPhoneNumber) ||
|
|
2783
|
+
prefer(item.phoneNumber);
|
|
2784
|
+
if (!target) {
|
|
2785
|
+
console.warn('Transfer aborted: no dialable number found in recent item:', item);
|
|
2786
|
+
return;
|
|
2787
|
+
}
|
|
2788
|
+
// IMPORTANT: transfer, not makeCall
|
|
2789
|
+
this.avayaIPOService.transferCall(target, 'unAttended'); // or 'Blind'
|
|
2790
|
+
this.isCallTransferOpen = false; // close transfer UI if you want
|
|
2791
|
+
}
|
|
2792
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetComponent, deps: [{ token: i1.HttpClient }, { token: i0.ChangeDetectorRef }, { token: i0.NgZone }, { token: i1$1.FormBuilder }, { token: CountryService }, { token: PhoneNumberLookupService }, { token: AvayaIPOService }, { token: RecordingManagerService }, { token: CallSocketService }, { token: i8.SnugdeskAuthenticationService }, { token: i8.TenantService }, { token: i8.UserService }], target: i0.ɵɵFactoryTarget.Component });
|
|
2793
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.9", type: AvayaIPOWidgetComponent, isStandalone: false, selector: "snugdesk-avaya-ipo-widget", inputs: { tenantId: "tenantId", userId: "userId", isVisible: "isVisible", containerHeightObservable: "containerHeightObservable", containerWidthObservable: "containerWidthObservable" }, outputs: { notificationEvent: "notificationEvent" }, ngImport: i0, template: "@if (isReconnectingInProgress || isFailoverInProgress || isFailbackInProgress) {\n <div class=\"container_connection_status_overlay\">\n <div class=\"container_connection_status\">\n <i class=\"fa-solid fa-circle-dot\"></i>\n\n @if (isReconnectingInProgress) {\n <span>Reconnecting to the server...</span>\n }\n\n @if (isFailoverInProgress) {\n <span>Failover in progress...</span>\n }\n\n @if (isFailbackInProgress) {\n <span>Failback in progress...</span>\n }\n\n <div class=\"container_connection_status_loader\">\n <div class=\"container_three_dot_loader\">\n <span class=\"dot\"></span>\n <span class=\"dot\"></span>\n <span class=\"dot\"></span>\n </div>\n </div>\n </div>\n </div>\n}\n\n\n\n@if (showLoginForm) {\n <!-- Login Form -->\n <div class=\"container_avaya_login\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"container_avaya_login_inner\">\n <div class=\"container_avaya_login_content\">\n\n <div class=\"container_avaya_logo\">\n <div class=\"container_avaya_logo_image\"></div>\n <h3>IP Office<span>TM</span></h3>\n </div>\n\n @if (hasAvayaSocketError || hasAvayaLoginError || hasAvayaLoginConflictError || hasAvayaDeviceError) {\n <div class=\"container_avaya_login_error\">\n <p class=\"message_label error_message\">\n @if (hasAvayaSocketError) {\n We're having trouble connecting to the server\n }\n @if (hasAvayaLoginError) {\n Invalid extension or password!\n }\n @if (hasAvayaLoginConflictError) {\n Another user is logged in with this extension\n }\n @if (hasAvayaDeviceError) {\n We're having trouble accessing your device\n }\n </p>\n </div>\n }\n\n <div class=\"container_avaya_login_form\">\n <form [formGroup]=\"loginForm\" (ngSubmit)=\"onLogin()\">\n\n <!-- Extension -->\n <div class=\"form_field\">\n <mat-form-field class=\"properties_form_field mat-form-field-outline\">\n <mat-label class=\"mdc-floating-disabled\">Extension</mat-label>\n <input matInput formControlName=\"ipoExtensionNo\" autocomplete=\"ipoExtensionNo\"\n (keypress)=\"restrictNumeric($event)\" />\n\n @if (loginForm.controls['ipoExtensionNo'].invalid && loginForm.controls['ipoExtensionNo'].touched) {\n <mat-error>\n @if (loginForm.controls['ipoExtensionNo'].errors?.['required']) {\n <small>Extension is required.</small>\n }\n </mat-error>\n }\n </mat-form-field>\n </div>\n\n <!-- Password -->\n <div class=\"form_field\">\n <mat-form-field class=\"properties_form_field mat-form-field-outline\">\n <mat-label class=\"mdc-floating-disabled\">Password</mat-label>\n <input matInput formControlName=\"ipoPassword\" autocomplete=\"avaya-ipo-password\"\n (keypress)=\"restrictNumeric($event)\" [type]=\"showLoginPassword ? 'text' : 'password'\" />\n\n <mat-icon matSuffix (click)=\"showLoginPassword = !showLoginPassword && !hasAvayaSocketError\">\n @if (showLoginPassword) {\n <span class=\"container_icon_eye\">\n <i class=\"fa-solid fa-eye\"></i>\n </span>\n } @else {\n <span class=\"container_icon_eye\">\n <i class=\"fa-solid fa-eye-slash\"></i>\n </span>\n }\n </mat-icon>\n\n @if (loginForm.controls['ipoPassword'].invalid && loginForm.controls['ipoPassword'].touched) {\n <mat-error>\n @if (loginForm.controls['ipoPassword'].errors?.['required']) {\n <small>Password is required.</small>\n }\n </mat-error>\n }\n </mat-form-field>\n </div>\n\n <!-- Login button -->\n <div class=\"avaya_login_button_area\">\n <button type=\"submit\" class=\"button_submit_login\"\n [ngClass]=\"{ 'button_submit_login_loader': showLoginLoader }\"\n [disabled]=\"showLoginLoader\">\n @if (showLoginLoader) {\n <span class=\"button_submit_loader\"></span>\n } @else {\n <span>Login</span>\n }\n </button>\n </div>\n\n </form>\n </div>\n\n </div>\n </div>\n </div>\n </div>\n </div>\n} @else {\n <!-- Softphone UI -->\n <ng-container *ngTemplateOutlet=\"showSoftphoneDiv\"></ng-container>\n}\n\n<!-- Footer (always visible) -->\n<div class=\"container_avaya_footer\">© 2025 SNUG Technologies Pvt. Ltd.\n\n\n\n</div>\n\n\n\n\n\n<ng-template #showSoftphoneDiv>\n\n@if (!showActiveCall) {\n <div class=\"container_avaya_softphone_inner\">\n <!-- ------------------- Keypad work start ---------- -->\n @if (bottomActiveTab === 'keypad_sd') {\n <div class=\"tab-content_sd\">\n <div class=\"container_dialpad_header\">\n <div class=\"container_dialpad_header_inner2\">\n <div class=\"container_avaya_extension\">\n <div class=\"container_avaya_extension_left\">\n <div class=\"extension_label\">Extension</div>\n <div class=\"extension_number\">{{ extensionNumber }}</div>\n </div>\n <div class=\"container_avaya_extension_right\">\n <button title=\"Sign out\" (click)=\"onLogout()\">\n <svg width=\"16px\" fill=\"currentColor\" viewBox=\"0 0 512 512\">\n <path\n d=\"M377.9 105.9L500.7 228.7c7.2 7.2 11.3 17.1 11.3 27.3s-4.1 20.1-11.3 27.3L377.9 406.1c-6.4 6.4-15 9.9-24 9.9c-18.7 0-33.9-15.2-33.9-33.9l0-62.1-128 0c-17.7 0-32-14.3-32-32l0-64c0-17.7 14.3-32 32-32l128 0 0-62.1c0-18.7 15.2-33.9 33.9-33.9c9 0 17.6 3.6 24 9.9zM160 96L96 96c-17.7 0-32 14.3-32 32l0 256c0 17.7 14.3 32 32 32l64 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-64 0c-53 0-96-43-96-96L0 128C0 75 43 32 96 32l64 0c17.7 0 32 14.3 32 32s-14.3 32-32 32z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n <div class=\"dialpad_container_sd\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <app-dialpad [contacts]=\"contacts\" (makeCallEv)=\"makeCall($event)\"></app-dialpad>\n <!-- to do list -->\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- ------------------- Keypad work end ---------- -->\n\n @if (bottomActiveTab === 'recent_sd') {\n <div class=\"tab-content_sd\">\n <app-call-history (makeCallEv)=\"makeCallFromContact($event)\"></app-call-history>\n </div>\n }\n\n @if (bottomActiveTab === 'settings_sd') {\n <div class=\"tab-content_sd\">\n <app-device-selector (change)=\"handleDeviceSelection($event)\"></app-device-selector>\n </div>\n }\n </div>\n\n <div class=\"tabs_sd\">\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('recent_sd')\"\n [class.active]=\"bottomActiveTab === 'recent_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"19px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M464 256A208 208 0 1 1 48 256a208 208 0 1 1 416 0zM0 256a256 256 0 1 0 512 0A256 256 0 1 0 0 256zM232 120l0 136c0 8 4 15.5 10.7 20l96 64c11 7.4 25.9 4.4 33.3-6.7s4.4-25.9-6.7-33.3L280 243.2 280 120c0-13.3-10.7-24-24-24s-24 10.7-24 24z\" />\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Recent</span>\n </button>\n\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('keypad_sd')\"\n [class.active]=\"bottomActiveTab === 'keypad_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"20px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Keypad</span>\n </button>\n\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('settings_sd')\"\n [class.active]=\"bottomActiveTab === 'settings_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"19px\" viewBox=\"-0.03 0 16.079 16.079\">\n <g>\n <path fill=\"currentColor\"\n d=\"M8.182 1.083a6.99 6.99 0 0 0-6.248 3.493c-1.929 3.342-.776 7.629 2.57 9.562 3.346 1.934 7.634.792 9.562-2.55 1.929-3.343.776-7.634-2.57-9.567a6.98 6.98 0 0 0-3.314-.938zM8 2.08a6 6 0 0 1 6 6 6 6 0 0 1-6 6 6 6 0 0 1-6-6 6 6 0 0 1 6-6z\" />\n <path fill=\"currentColor\"\n d=\"M9.322 0L6.69.393v1.354a6.49 6.477 43.146 0 1 2.632.005V0zM3.937 1.19L1.93 2.897l.988 1.177a6.49 6.477 43.146 0 1 2.017-1.69zm8.128.013l-.993 1.184a6.49 6.477 43.146 0 1 .17.09 6.49 6.477 43.146 0 1 1.845 1.603l1.006-1.197zM.455 5.42l-.44 2.596 1.515.267a6.49 6.477 43.146 0 1 .455-2.593zm15.086.003l-1.523.269a6.49 6.477 43.146 0 1 .464 2.59l1.533-.27zM1.858 10.118l-1.351.78 1.33 2.271 1.339-.772a6.49 6.477 43.146 0 1-1.317-2.28zm12.301.003a6.49 6.477 43.146 0 1-.534 1.215 6.49 6.477 43.146 0 1-.774 1.069l1.338.773 1.302-2.288zm-9.557 3.471l-.534 1.47 2.48.884.525-1.446a6.49 6.477 43.146 0 1-2.303-.8 6.49 6.477 43.146 0 1-.168-.108zm6.814.016a6.49 6.477 43.146 0 1-2.475.897l.53 1.457 2.468-.917z\" />\n <path fill=\"currentColor\"\n d=\"M7.648 3.093a4.989 4.989 0 0 0-3.982 2.483 5.013 5.013 0 0 0 1.836 6.834 5.002 5.002 0 0 0 6.83-1.827 5.01 5.01 0 0 0-1.836-6.832 4.976 4.976 0 0 0-2.848-.658zM8 4.08a4 4 0 0 1 4 4 4 4 0 0 1-4 4 4 4 0 0 1-4-4 4 4 0 0 1 4-4z\" />\n </g>\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Preferences</span>\n </button>\n </div>\n} @else {\n <ng-container *ngTemplateOutlet=\"showActiveCallDiv\"></ng-container>\n}\n\n\n<!-- ////////////// -->\n\n <!-- Active Call Div -->\n \n <ng-template #showActiveCallDiv>\n\n\n @if (showTranscriptionPopup) {\n <div class=\"transcription-popup\">\n <div class=\"transcription-header\">\n <span>Live Transcription</span>\n <button class=\"close-btn\" (click)=\"showTranscriptionPopup = false\">\u00D7</button>\n </div>\n <div class=\"transcription-body\">\n <p>{{ transcriptText }}</p>\n </div>\n </div>\n}\n\n\n\n\n\n\n\n<div class=\"container_call\">\n <div class=\"container_call_header\">\n <div class=\"container_call_header_inner\">\n\n @if (!conferenceParticipants || conferenceParticipants.length <= 1) {\n <!-- ------------ add call list swap and merge area start ---------------- -->\n @if (activeCalls.length > 1) {\n <div class=\"container_swap_merge_list\">\n <div class=\"container_conference_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"outer_container_swap_merge\">\n @for (call of activeCalls; track call.phoneNumber; let i = $index) {\n @if (i !== currentActiveCallIndex) {\n <div class=\"container_swap_merge_item\"\n (click)=\"swapLinesToIndex(i)\"\n [ngClass]=\"{ 'active-call': i === currentActiveCallIndex, 'other-call': i !== currentActiveCallIndex }\">\n\n <!-- Avatar -->\n <div class=\"container_recent_item_image\">\n <div class=\"container_swap_merge_item_image_inner\">\n <!-- svg avatar here -->\n </div>\n </div>\n\n <!-- Call Info -->\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">{{ call.phoneNumber }}</div>\n @if (call.isOnHold) {\n <div class=\"onhold_st\">On hold</div>\n }\n </div>\n\n <!-- Action Buttons -->\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <button class=\"button_swap_merge\">\n <div class=\"button_icon\">\n <!-- merge svg -->\n </div>\n <div class=\"button_text\">Merge</div>\n </button>\n </div>\n <div class=\"right_field\">\n <button class=\"button_swap_merge\">\n <div class=\"button_icon\">\n <!-- swap svg -->\n </div>\n <div class=\"button_text\">Swap</div>\n </button>\n </div>\n </div>\n </div>\n </div>\n }\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n <!-- ------------ add call list swap and merge area end ---------------- -->\n\n @if (isCalling) {\n <div class=\"container_call_info\">Calling...</div>\n } @else {\n @if (isactiveCall) {\n <div class=\"container_call_info\">\n {{ formatTime(activeCalls[currentActiveCallIndex].duration) }}\n </div>\n }\n @if (showCallDisconnected) {\n <div class=\"container_call_info_blinker\">\n <!-- disconnected svg -->\n {{ this.istransfer ? 'Call Transferred' : 'Call ended' }}\n </div>\n }\n }\n\n <div class=\"container_call_ani\">\n @if (isCalling && displayCallingNumber && activeCalls.length === 0) {\n <!-- show what the user dialed while call is spinning up -->\n <div class=\"container_call_ani_number\">\n {{ displayCallingNumber }}\n </div>} \n @else if (activeCalls.length > 0) {\n <div class=\"container_call_ani_number\">\n {{ activeCalls[currentActiveCallIndex].lookupPhoneNumber.country?.flag }}\n {{ activeCalls[currentActiveCallIndex].phoneNumber }}\n \n </div>\n <div class=\"container_call_ani_name\">\n {{ activeCalls[currentActiveCallIndex].lookupPhoneNumber.phoneNumber?.location }}\n </div>\n } @else {\n <div class=\"container_call_ani_number\">\n {{ lastDisconnectedCall?.lookupPhoneNumber?.country?.flag }}\n {{ lastDisconnectedCall?.phoneNumber }}\n </div>\n <div class=\"container_call_ani_name\">\n {{ lastDisconnectedCall?.lookupPhoneNumber?.phoneNumber?.location }}\n </div>\n }\n\n <div class=\"container_call_ani_image\">\n <div class=\"container_call_ani_image_inner\" [ngClass]=\"{ 'call_animation': isactiveCall }\">\n <!-- big avatar svg -->\n </div>\n </div>\n </div>\n\n } @else {\n <!-- Conference Call Placeholder -->\n <!-- @if block can render conference call template here -->\n }\n\n </div>\n </div>\n\n <div class=\"container_call_content\">\n <div class=\"container_call_dnis\">\n <span>Ext. {{ extensionNumber }}</span>\n </div>\n\n <div class=\"container_call_options\">\n <div class=\"container_dialpad_row margin_bottom_16\">\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall && activeCalls.length < 4 }\"\n (click)=\"onAddCall()\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"> \n <svg fill=\"#000000\" width=\"20\" viewBox=\"0 0 52 52\" enable-background=\"new 0 0 52 52\"\n xml:space=\"preserve\">\n <path fill=\"currentColor\" d=\"M30,29h16.5c0.8,0,1.5-0.7,1.5-1.5v-3c0-0.8-0.7-1.5-1.5-1.5H30c-0.6,0-1-0.4-1-1V5.5C29,4.7,28.3,4,27.5,4\n h-3C23.7,4,23,4.7,23,5.5V22c0,0.6-0.4,1-1,1H5.5C4.7,23,4,23.7,4,24.5v3C4,28.3,4.7,29,5.5,29H22c0.6,0,1,0.4,1,1v16.5\n c0,0.8,0.7,1.5,1.5,1.5h3c0.8,0,1.5-0.7,1.5-1.5V30C29,29.4,29.4,29,30,29z\" />\n </svg>\n </div>\n <div class=\"button_text\">Add</div>\n </button>\n\n @if (activeCalls.length > 0) {\n <button class=\"button_call_option\" (click)=\"onAddCall()\" type=\"button\">\n <div class=\"button_icon\"> <svg width=\"20\" viewBox=\"0 0 52 52\" aria-hidden=\"true\">\n <path fill=\"currentColor\"\n d=\"M30,29h16.5c0.8,0,1.5-0.7,1.5-1.5v-3c0-0.8-0.7-1.5-1.5-1.5H30c-0.6,0-1-0.4-1-1V5.5C29,4.7,28.3,4,27.5,4h-3\n C23.7,4,23,4.7,23,5.5V22c0,0.6-0.4,1-1,1H5.5C4.7,23,4,23.7,4,24.5v3C4,28.3,4.7,29,5.5,29H22c0.6,0,1,0.4,1,1v16.5\n c0,0.8,0.7,1.5,1.5,1.5h3c0.8,0,1.5-0.7,1.5-1.5V30C29,29.4,29.4,29,30,29z\"/>\n </svg></div>\n <div class=\"button_text\">Conference</div>\n </button>\n }\n\n @if (!isRecording) {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"> <svg width=\"22px\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path fill=\"currentColor\" d=\"M3 10L3 14M7.5 6L7.5 18M12 3V21M16.5 6V18M21 10V14\" stroke=\"#000000\"\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg></div>\n <div class=\"button_text\">Record</div>\n </button>\n } @else {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"><svg width=\"22px\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path fill=\"currentColor\" d=\"M3 10L3 14M7.5 6L7.5 18M12 3V21M16.5 6V18M21 10V14\" stroke=\"#000000\"\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg></div>\n <div class=\"button_text\">Stop</div>\n </button>\n }\n\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\"\n (click)=\"isCallTransferOpen = !isCallTransferOpen\">\n <div class=\"button_icon\"> <svg width=\"25px\" height=\"25px\" viewBox=\"0 0 48 48\" version=\"1\" enable-background=\"new 0 0 48 48\">\n <path fill=\"currentColor\"\n d=\"M39.2,8.4l-1.8,1.8c-6.3,6.5-5.4,22,0,27.6l1.8,1.8c0.5,0.5,1.3,0.5,1.8,0l3.6-3.7c0.5-0.5,0.5-1.3,0-1.8 l-3.4-3.4h-4.8c-1.3-1.3-1.3-12.1,0-13.4h4.8l3.3-3.4c0.5-0.5,0.5-1.3,0-1.8L41,8.4C40.5,7.9,39.7,7.9,39.2,8.4z\" />\n <path fill=\"currentColor\"\n d=\"M11.2,8.4l-1.8,1.8c-6.3,6.5-5.4,22,0,27.6l1.8,1.8c0.5,0.5,1.3,0.5,1.8,0l3.6-3.7c0.5-0.5,0.5-1.3,0-1.8 l-3.4-3.4H8.5c-1.3-1.3-1.3-12.1,0-13.4h4.8l3.3-3.4c0.5-0.5,0.5-1.3,0-1.8L13,8.4C12.5,7.9,11.7,7.9,11.2,8.4z\" />\n <g fill=\"currentColor\">\n <polygon points=\"25.3,18.6 30.7,24 25.3,29.4\" />\n <rect x=\"16\" y=\"22\" width=\"11\" height=\"4\" />\n </g>\n </svg></div>\n <div class=\"button_text\">Transfer</div>\n </button>\n\n @if (isCallTransferOpen) {\n <ng-container *ngTemplateOutlet=\"callTransferTemplate\"></ng-container>\n }\n </div>\n\n <!-- Mute / Unmute -->\n <div class=\"container_dialpad_row\">\n @if (activeCalls.length > 0 && !this.activeCalls[this.currentActiveCallIndex].isMuted) {\n <button class=\"button_call_option button_elevated\"\n (click)=\"toggleMute()\"\n [ngClass]=\"{ 'button_disabled': showCallDisconnected }\">\n <div class=\"button_icon\"> <svg viewBox=\"0 0 384 512\" width=\"16\">\n <path fill=\"currentColor\"\n d=\"M192 0C139 0 96 43 96 96l0 160c0 53 43 96 96 96s96-43 96-96l0-160c0-53-43-96-96-96zM64 216c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l72 0 72 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6c85.8-11.7 152-85.3 152-174.4l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 70.7-57.3 128-128 128s-128-57.3-128-128l0-40z\" />\n </svg></div>\n <div class=\"button_text\">Mute</div>\n </button>\n } @else if (activeCalls.length > 0 && this.activeCalls[this.currentActiveCallIndex].isMuted) {\n <button class=\"button_call_option button_active button_elevated\" (click)=\"toggleMute()\">\n <div class=\"button_icon\"> <svg width=\"26\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M38.8 5.1C28.4-3.1 13.3-1.2 5.1 9.2S-1.2 34.7 9.2 42.9l592 464c10.4 8.2 25.5 6.3 33.7-4.1s6.3-25.5-4.1-33.7L472.1 344.7c15.2-26 23.9-56.3 23.9-88.7l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 21.2-5.1 41.1-14.2 58.7L416 300.8 416 96c0-53-43-96-96-96s-96 43-96 96l0 54.3L38.8 5.1zM344 430.4c20.4-2.8 39.7-9.1 57.3-18.2l-43.1-33.9C346.1 382 333.3 384 320 384c-70.7 0-128-57.3-128-128l0-8.7L144.7 210c-.5 1.9-.7 3.9-.7 6l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l72 0 72 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6z\" />\n </svg></div>\n <div class=\"button_text\">Unmute</div>\n </button>\n }\n\n <!-- Hold / Unhold -->\n @if (activeCalls.length > 0 && !this.activeCalls[this.currentActiveCallIndex].isOnHold) {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\"\n (click)=\"toggleHold()\">\n <div class=\"button_icon\"> <svg viewBox=\"0 0 320 512\" width=\"14\">\n <path fill=\"currentColor\" d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\" />\n </svg></div>\n <div class=\"button_text\">Hold</div>\n </button>\n } @else if (activeCalls.length > 0 && this.activeCalls[this.currentActiveCallIndex].isOnHold) {\n <button class=\"button_call_option button_active button_elevated\" (click)=\"toggleHold()\">\n <div class=\"button_icon\">\n <svg viewBox=\"0 0 320 512\" width=\"14\">\n <path fill=\"currentColor\" d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\" />\n </svg></div>\n <div class=\"button_text\">Unhold</div>\n </button>\n \n }\n\n <!-- Keypad -->\n <button class=\"button_call_option button_elevated\"\n [class.button_active]=\"isDailpadOpen\"\n (click)=\"menuDailpad()\"\n [ngClass]=\"{ 'button_disabled': showCallDisconnected }\">\n <div class=\"button_icon\"> <svg width=\"22px\" height=\"22px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n </svg></div>\n <div class=\"button_text\">{{ isDailpadOpen ? 'Hide' : 'Keypad' }}</div>\n </button>\n\n @if (isDailpadOpen) {\n <div class=\"container_popup_dialpad\">\n <div class=\"container_popup_dialpad_inner\">\n <div [formGroup]=\"dtmfForm\">\n <div class=\"container_dtmf_input\">\n <input type=\"tel\" formControlName=\"dtmf\">\n </div>\n <div class=\"container_dialpad_digits\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('1')\">1</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('2')\">2</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('3')\">3</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('4')\">4</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('5')\">5</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('6')\">6</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('7')\">7</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('8')\">8</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('9')\">9</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path fill=\"currentColor\"\n d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86 102.4l122 70.4L208 32z\" />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('0')\">0</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path fill=\"currentColor\"\n d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3 36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4 26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2 384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64 192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n\n <!-- Call Control Buttons -->\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n @if (!showCallDisconnected || isCalling) {\n <button class=\"button_dialpad_digit button_call_red\" (click)=\"dropCurrentCall()\">\n <svg width=\"25\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\" />\n </svg>\n </button>\n } @else if (showCallDisconnected && !isCalling) {\n <button class=\"button_dialpad_digit button_call_green\"\n (click)=\"onRedial()\"\n [ngClass]=\"{ 'button_call_disabled': !phoneNumber }\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n }\n </div>\n </div>\n </div>\n</div>\n</ng-template>\n\n\n <!-- ////////////////// -->\n <!-- Conference Call Template -->\n <ng-template #conferenceCallTemplate>\n<div class=\"container_active_call_header\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">Conference call</div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_green\">\n <svg viewBox=\"0 0 512 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </div>\n <div class=\"container_active_call_status_green\">\n with {{ conferenceParticipants?.length }} people\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"container_active_call_options\">\n <div class=\"button_active_call_options button_active_call_options_highlighted\">\n <span (click)=\"isConfereceList = !isConfereceList\" class=\"container_active_call_options_button\">\n <svg viewBox=\"0 0 448 512\" width=\"16px\">\n <path fill=\"currentColor\"\n d=\"M201.4 374.6c12.5 12.5 32.8 12.5 45.3 0l160-160c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L224 306.7 86.6 169.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l160 160z\" />\n </svg>\n </span>\n </div>\n </div>\n</div>\n\n<!-- Conference List -->\n@if (isConfereceList) {\n <div class=\"container_conference_contact_list\">\n <div class=\"container_conference_contact_list_inner\">\n <div class=\"scrollbox container_conference_contact_list_scrollbox\">\n <div class=\"scrollbox-content\">\n\n @for (person of conferenceParticipants; track person.id) {\n <div class=\"container_conference_contact_item\">\n <!-- Avatar -->\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"49px\" width=\"49px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n\n <!-- Name and Time -->\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ person.name }}</div>\n <div class=\"contact_field_green\">{{ person.duration }}</div>\n </div>\n </div>\n\n <!-- End Button -->\n <div class=\"container_contact_item_right\">\n <div class=\"right_field\">\n <button class=\"button_disconnect_call\">\n <svg width=\"22\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\">\n </path>\n </svg>\n </button>\n </div>\n </div>\n </div>\n\n @if (!$last) {\n <div class=\"container_separator\"></div>\n }\n }\n\n </div>\n </div>\n </div>\n </div>\n}\n\n</ng-template>\n\n\n <ng-template #callTransferTemplate>\n <div class=\"container_transfer_call\">\n <div class=\"container_transfer_call_inner\">\n <div class=\"container_button_close\">\n <span (click)=\"isCallTransferOpen = false\" title=\"Close\">\n <i class=\"fa-sharp fa-solid fa-xmark\"></i>\n </span>\n </div>\n\n <!-- First Div -->\n <div class=\"container_transfer_call_content\" @slideInOut>\n @if (currentView === 'transferListpage') {\n <div class=\"container_active_call_duration\">{{ formatTime(callDuration)}}</div>\n\n <div class=\"container_active_call_header margin_bottom_16 shadow_st\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">{{ lookupPhoneNumber?.country?.flag }}\n {{ lookupPhoneNumber?.phoneNumber }}</div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_green\">\n <svg viewBox=\"0 0 512 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </div>\n <div class=\"container_active_call_status_green\">Active call</div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">Ext. {{extensionNumber}}</span>\n </div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div class=\"container_transfer_tab_buttons\">\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab1')\" [class.active]=\"activeTab === 'tab1'\">\n <svg width=\"18px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M464 256A208 208 0 1 1 48 256a208 208 0 1 1 416 0zM0 256a256 256 0 1 0 512 0A256 256 0 1 0 0 256zM232 120l0 136c0 8 4 15.5 10.7 20l96 64c11 7.4 25.9 4.4 33.3-6.7s4.4-25.9-6.7-33.3L280 243.2 280 120c0-13.3-10.7-24-24-24s-24 10.7-24 24z\" />\n </svg> Recent</button>\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab2')\" [class.active]=\"activeTab === 'tab2'\">\n <svg width=\"20px\" viewBox=\"0 0 24 24\">\n <path fill=\"currentColor\"\n d=\"M15.71,12.71a6,6,0,1,0-7.42,0,10,10,0,0,0-6.22,8.18,1,1,0,0,0,2,.22,8,8,0,0,1,15.9,0,1,1,0,0,0,1,.89h.11a1,1,0,0,0,.88-1.1A10,10,0,0,0,15.71,12.71ZM12,12a4,4,0,1,1,4-4A4,4,0,0,1,12,12Z\" />\n </svg> Internal</button>\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab3')\" [class.active]=\"activeTab === 'tab3'\">\n <svg width=\"18px\" height=\"18px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" /></svg>\n Dail</button>\n </div>\n\n <!-- Tab 1 -->\n @if (activeTab === 'tab1') {\n <!-- <div class=\"container_transfer_tab_content\">\n <app-call-history (makeCallEv)=\"makeCallFromContact($event)\"></app-call-history>\n </div> -->\n <div class=\"container_transfer_tab_content\">\n <app-call-history [mode]=\"'transfer'\" (transferEv)=\"transferToFromHistory($event)\"> </app-call-history>\n</div>\n }\n\n <!-- Tab 2 -->\n @if (activeTab === 'tab2') {\n <div class=\"container_transfer_tab_content\">\n <div class=\"container_transfer_internal_search\">\n <input type=\"search\" placeholder=\"Type to search\">\n </div>\n <div class=\"container_transfer_contact_list\">\n <div class=\"container_transfer_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <!-- Instead of repeating static contacts, render via array -->\n @for (contact of internalContacts; track contact.ext) {\n <div class=\"container_contact_item\" (click)=\"goToDiv('transferDetails'); makeCallFromContact(contact.phone)\">\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"49px\" width=\"49px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ contact.name }}</div>\n <div class=\"contact_field\">{{ contact.phone }}</div>\n @if (contact.role) {\n <div class=\"contact_field_grey\">\n <div class=\"contact_field_highlighted\"><span>{{ contact.role }}</span></div>\n </div>\n }\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">Ext. {{ contact.ext }}</span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- Tab 3 -->\n @if (activeTab === 'tab3') {\n <div class=\"container_transfer_tab_content\">\n <form [formGroup]=\"transferForm\">\n <div class=\"container_transfer_contact_list\">\n <div class=\"container_phone_number_input margin_bottom_16\">\n <button class=\"button_phonebook\" (click)=\"isDirectoryOpen = !isDirectoryOpen\">\uD83D\uDCD6</button>\n <input placeholder=\"Phone Number\" type=\"tel\" formControlName=\"target\">\n <button class=\"button_backspace\" (click)=\"onBackspace()\"></button>\n </div>\n\n @if (isDirectoryOpen) {\n <div class=\"container_transfer_internal_contact_list\">\n <div class=\"container_transfer_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (directoryContact of directoryContacts; track directoryContact.id) {\n <div class=\"container_contact_item\">\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ directoryContact.name }}</div>\n <div class=\"contact_field\">{{ directoryContact.phone }}</div>\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">\n <div class=\"container_rating\">\n @for (star of [1,2,3,4,5]; track $index) {\n <span [class.filled]=\"star <= directoryContact.rating\">\u2605</span>\n }\n </div>\n </span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- Dialpad -->\n <div class=\"container_dialpad_digits\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('1')\">1</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('2')\">2</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('3')\">3</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('4')\">4</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('5')\">5</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('6')\">6</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('7')\">7</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('8')\">8</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('9')\">9</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path fill=\"currentColor\" d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9\n 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7\n 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8\n 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32\n 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9\n 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38\n 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86\n 102.4l122 70.4L208 32z\" />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('0')\">0</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path fill=\"currentColor\" d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3\n36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4\n26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9\n0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5\n69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1\n0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2\n384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64\n192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4\n19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\" />\n </svg></button>\n </div>\n </div>\n\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n @if (isAddingNewCall) {\n <button class=\"button_dialpad_digit button_call_green\"\n (click)=\"makeCall(transferForm.value.target)\"\n [disabled]=\"!phoneNumber\"\n [ngClass]=\"{ 'button_call_disabled': !phoneNumber }\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n } @else {\n <button type=\"submit\" class=\"button_dialpad_digit button_call_green\" (click)=\"onTransferCall()\"> <svg height=\"20px\" viewBox=\"0 0 52 52\" enable-background=\"new 0 0 52 52\" xml:space=\"preserve\">\n <g>\n <path fill=\"currentColor\" d=\"M48.5,37.9L42.4,33c-1.4-1.1-3.4-1.2-4.8-0.1l-5.2,3.8c-0.6,0.5-1.5,0.4-2.1-0.2l-7.8-7l-7-7.8\n c-0.6-0.6-0.6-1.4-0.2-2.1l3.8-5.2c1.1-1.4,1-3.4-0.1-4.8l-4.9-6.1c-1.5-1.8-4.2-2-5.9-0.3L3,8.4c-0.8,0.8-1.2,1.9-1.2,3\n c0.5,10.2,5.1,19.9,11.9,26.7S30.2,49.5,40.4,50c1.1,0.1,2.2-0.4,3-1.2l5.2-5.2C50.5,42.1,50.4,39.3,48.5,37.9z\" />\n <path fill=\"currentColor\" d=\"M48.4,2H33c-1,0-1.3,1.1-0.5,1.9l4.9,5l-9,9.1c-0.5,0.5-0.5,1.4,0,1.9l3.7,3.7c0.5,0.5,1.3,0.5,1.9,0\n l9.1-9.1l5.1,4.9C48.9,20.3,50,20,50,19V3.7C50,3,49.1,2,48.4,2z\" />\n </g>\n </svg></button>\n }\n </div>\n </div>\n </div>\n </form>\n </div>\n }\n }\n</div>\n\n <!-- Second Div -->\n @if (currentView === 'transferDetails') {\n <div class=\"container_transfer_call_content\" @slideInOut>\n <div class=\"container_active_call_duration\">00:10</div>\n <div class=\"container_active_call_header shadow_st\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"></path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 ...\"></path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 ...\"></path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">\n {{ lookupPhoneNumber?.country?.flag }} {{ lookupPhoneNumber?.phoneNumber }}\n </div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_grey\">\n <svg viewBox=\"0 0 320 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\"></path>\n </svg>\n </div>\n <div class=\"container_active_call_status_grey\">On hold</div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">Ext. {{ extensionNumber }}</span>\n </div>\n </div>\n </div>\n\n <div class=\"container_transfer_call_details\">\n <div class=\"container_call_ani_number padding_top_nt\">+91 (654) 9876549</div>\n <div class=\"container_call_ani_name\">Rajeev kumar singh</div>\n <div class=\"container_call_ani_image\">\n <div class=\"container_call_ani_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"85\" width=\"85\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"></path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 ...\"></path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 ...\"></path>\n </g>\n </svg>\n </div>\n </div>\n </div>\n\n <div class=\"container_call_dnis\">\n <span>Ext. {{ extensionNumber }}</span>\n </div>\n\n <div class=\"container_call_options margin_bottom_nt\">\n <div class=\"container_dialpad_row margin_bottom_16\">\n <button class=\"button_call_option\" (click)=\"mergeCall()\">\n <div class=\"button_icon\"><svg width=\"22\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M32 64C14.3 64 0 78.3 0 96s14.3 32 32 32l97.2 0c9.7 0 18.9 4.4 25 12L247 256 154.2 372c-6.1 7.6-15.3 12-25 12L32 384c-17.7 0-32 14.3-32 32s14.3 32 32 32l97.2 0c29.2 0 56.7-13.3 75-36l99.2-124 80.6 0 0 32c0 12.9 7.8 24.6 19.8 29.6s25.7 2.2 34.9-6.9l64-64c6-6 9.4-14.1 9.4-22.6s-3.4-16.6-9.4-22.6l-64-64c-9.2-9.2-22.9-11.9-34.9-6.9s-19.8 16.6-19.8 29.6l0 32-80.6 0L204.2 100c-18.2-22.8-45.8-36-75-36L32 64z\"></path>\n </svg></div>\n <div class=\"button_text\">Merge</div>\n </button>\n\n <button class=\"button_call_option\">\n <div class=\"button_icon\"> <svg width=\"26px\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M150.6 73.4c-12.5-12.5-32.8-12.5-45.3 0l-96 96c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L96 173.3 96 320c0 53 43 96 96 96l112 0c17.7 0 32-14.3 32-32s-14.3-32-32-32l-112 0c-17.7 0-32-14.3-32-32l0-146.7 41.4 41.4c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3l-96-96zM336 96c-17.7 0-32 14.3-32 32s14.3 32 32 32l112 0c17.7 0 32-14.3 32-32l0 146.7-41.4-41.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l96 96c12.5 12.5 32.8 12.5 45.3 0l96-96c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L544 338.7 544 192c0-53-43-96-96-96L336 96z\"></path>\n </svg></div>\n <div class=\"button_text\">Swap</div>\n </button>\n\n <button class=\"button_call_option\">\n <div class=\"button_icon\"><svg width=\"19px\" viewBox=\"0 0 52 52\">\n <path fill=\"currentColor\"\n d=\"M48.5,37.9L42.4,33c-1.4-1.1-3.4-1.2-4.8-0.1l-5.2,3.8c-0.6,0.5-1.5,0.4-2.1-0.2l-7.8-7l-7-7.8 c-0.6-0.6-0.6-1.4-0.2-2.1l3.8-5.2c1.1-1.4,1-3.4-0.1-4.8l-4.9-6.1c-1.5-1.8-4.2-2-5.9-0.3L3,8.4c-0.8,0.8-1.2,1.9-1.2,3 c0.5,10.2,5.1,19.9,11.9,26.7S30.2,49.5,40.4,50c1.1,0.1,2.2-0.4,3-1.2l5.2-5.2C50.5,42.1,50.4,39.3,48.5,37.9z\"></path>\n <path fill=\"currentColor\"\n d=\"M48.4,2H33c-1,0-1.3,1.1-0.5,1.9l4.9,5l-9,9.1c-0.5,0.5-0.5,1.4,0,1.9l3.7,3.7c0.5,0.5,1.3,0.5,1.9,0 l9.1-9.1l5.1,4.9C48.9,20.3,50,20,50,19V3.7C50,3,49.1,2,48.4,2z\"></path>\n </svg>\n</div>\n <div class=\"button_text\">Transfer</div>\n </button>\n </div>\n </div>\n\n <div class=\"container_transfer_disconnect\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit button_call_red\" (click)=\"goToDiv('transferListpage')\">\n <svg width=\"36px\" viewBox=\"0 0 76 76\">\n <path fill=\"currentColor\"\n d=\"M 16.2013,39.582L 15.4175,39.4096L 14.8094,38.9315L 14.4029,38.2193L 14.25,37.364L 14.25,34.9055C 14.25,33.184 14.6023,31.6057 15.3069,30.1704C 16.0116,28.7352 16.9698,27.4788 18.1818,26.4013C 19.3937,25.3237 20.8095,24.4424 22.429,23.7573C 24.0486,23.0722 25.7732,22.6277 27.6031,22.4239L 31.304,22.0792L 34.9918,21.8256L 39.109,21.715L 42.966,21.8451L 46.101,22.0987L 48.9628,22.489C 50.7276,22.7318 52.3883,23.199 53.945,23.8906C 55.5017,24.5822 56.8567,25.4625 58.0101,26.5313C 59.1635,27.6002 60.0752,28.8468 60.7451,30.2712C 61.4151,31.6956 61.75,33.2599 61.75,34.964L 61.75,37.4486L 61.5874,38.2681L 61.1386,38.9511L 60.4785,39.4096L 59.6752,39.582L 51.8897,39.582L 51.0994,39.4096L 50.4588,38.9413L 50.0327,38.2584L 49.8799,37.4226L 49.8799,32.1217L 49.7075,31.1656L 49.2295,30.3818L 48.5075,29.8452C 48.23,29.7108 47.9286,29.6436 47.6034,29.6436L 28.7413,29.6436C 28.4161,29.6436 28.0974,29.7108 27.7852,29.8452L 26.9722,30.3818L 26.4128,31.1851C 26.2697,31.4973 26.1982,31.8225 26.1982,32.1607L 26.1982,37.364L 26.0356,38.2193L 25.5803,38.9315L 24.9104,39.4096L 24.1038,39.582L 16.2013,39.582 Z M 34.8333,41.1667L 41.1666,41.1667L 41.1667,52.6458L 45.9167,47.8958L 45.9167,54.2292L 38,62.5417L 30.0833,54.2292L 30.0833,47.8958L 34.8333,52.6458L 34.8333,41.1667 Z \" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n}\n\n\n\n\n\n\n\n\n\n\n\n </div>\n </div>\n </ng-template>\n\n \n\n\n</ng-template>\n\n<div id=\"videoPanel\" style=\"display: none\">\n <audio #rmtAudio id=\"rmtAudio\" autoplay></audio>\n <audio #lclAudio id=\"lclAudio\" autoplay muted></audio>\n <video #rmtVideo id=\"rmtVideo\" autoplay></video>\n <video #lclVideo id=\"lclVideo\" autoplay muted controls title=\"Local Video\"></video>\n</div>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Open+Sans:300,400,500,600,700,800\";@import\"https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css\";:host{--sd-app-color: #ff6633;--sd-app-color-dark: #ff5805;--sd-app-color-light: #fe8c66;--sd-text-font-family: \"Open Sans\", \"sans-serif\", \"Helvetica Neue\", \"Helvetica\", \"Arial\";--sd-text-color: #666;--sd-text-color-inverse: #fff;--sd-text-color-light: #ccc;--sd-text-color-medium: #999;--sd-text-color-dark: #333;--sd-background-color: #fff;--sd-border-color-light: #e3e3e3;--sd-border-color-dark: #c9c9c9}.message_label{width:100%;font-size:13px;font-family:var(--sd-text-font-family);letter-spacing:-.25px;border-radius:4px;font-weight:600;border:1px solid #f5c6cb;padding:8px 11px;position:relative}.error_message{color:#e1143c;background-color:#f8d7da;border-color:#dfadb2}.success_message{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert_message{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.properties_form_field{width:100%;font-size:13px!important;font-family:var(--sd-text-font-family)!important;font-weight:600!important;letter-spacing:-.24px!important;color:#555!important}.container_avaya_login{width:100%;height:calc(100vh - 51px);padding:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.container_avaya_login_inner{margin:0 auto;width:100%;max-width:370px;padding:65px 20px 20px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex}.container_connection_status_overlay{width:100%;height:100%;position:absolute;left:0;right:0;bottom:0;background-color:#6669;display:flex;justify-content:center;z-index:9999999;padding:20px}.container_connection_status{width:100%;background:var(--sd-background-color);padding:0 10px;border-radius:8px;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);display:flex;height:42px;align-items:center;position:relative;box-shadow:#32325d40 0 6px 12px -2px,#0000004d 0 3px 7px -3px}.container_connection_status span{padding:0 0 0 8px}.container_connection_status .fa-circle-dot{color:#f66060}.container_connection_status_loader{position:absolute;right:15px}.container_avaya_login_content{position:relative;width:100%}.container_avaya_logo{margin:0;padding:0 26px 24px;width:100%;text-align:center}.container_avaya_logo_image{margin:0 auto;width:120px;height:35px;background:var(--sd-background-color) url() center no-repeat;background-size:120px}.container_avaya_logo h3{margin:0;padding:6px 0 0;font-family:var(--sd-text-font-family);font-size:18px;font-weight:700;letter-spacing:-.45px;color:var(--sd-text-color-dark);position:relative}.container_avaya_logo h3 span{font-size:8px;position:absolute;top:8px}.container_avaya_login_error{width:100%;white-space:normal}.container_avaya_login_form{width:100%;margin:0;padding:26px 0 0;position:relative}.container_avaya_login_form form{width:100%;margin:0;padding:0}.container_avaya_login_form .form_field{margin:0;padding:0;width:100%}.avaya_login_button_area{padding:10px 0 0}.container_icon_eye{position:absolute;right:15px}.container_avaya_login_form .button_submit_login{font:14px / 48px var(--sd-text-font-family);display:inline-block;height:48px;background:#cd1f1e;color:var(--sd-text-color-inverse);border:1px solid #cd1f1e;font-weight:600;padding:0 12px;border-radius:4px;outline:none;letter-spacing:-.24px;cursor:pointer;width:100%;position:relative}.container_avaya_login_form .button_submit_login_loader{background:#cd1f1e!important;color:var(--sd-text-color-inverse)!important;border:1px solid #cd1f1e!important;cursor:auto}.container_avaya_login_form .button_submit_login:disabled{background:#f1f1f1;color:var(--sd-text-color-medium);border:1px solid #f1f1f1}.button_submit_loader:after{content:\"\";position:absolute;width:16px;height:16px;inset:0;margin:auto;border:3px solid transparent;border-top-color:#fff;border-radius:50%;animation:button_submit_loader_spinner 1s ease infinite}@keyframes button_submit_loader_spinner{0%{transform:rotate(0)}to{transform:rotate(1turn)}}.container_avaya_softphone{width:100%;height:100%;position:relative}.container_avaya_softphone_inner{width:100%;position:absolute;top:0;left:0;right:0;padding:0;height:calc(100% - 51px);font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);line-height:normal;background:var(--sd-background-color);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background:#000}.container_call{margin:0 auto;padding:0;position:relative;width:100%;height:100%;background:var(--sd-background-color)}.container_dialpad_header{width:100%;position:relative;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99999;background:var(--sd-background-color);padding:15px 15px 0}.container_dialpad_header_inner2{width:100%;margin:0;padding:15px 15px 0;background:#f7f7f7;background:linear-gradient(180deg,#f7f7f7 72%,#fff);border-radius:12px 12px 0 0;position:relative}.container_avaya_extension{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:8px 10px;border:1px solid var(--sd-border-color-light);border-radius:12px;margin-bottom:15px;background:#f0f0f0}.container_avaya_extension_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center}.container_avaya_extension_left .extension_label{display:flex;align-items:center;line-height:normal;text-align:left}.container_avaya_extension_left .extension_number{display:flex;align-items:center;line-height:20px}.container_avaya_extension_right{flex-basis:auto;flex-direction:column;flex-grow:1;text-align:right;justify-items:flex-end}.container_avaya_extension_right button{background:none;border:none;outline:none;color:#a6a6a6;font-weight:600;border-radius:50%;align-items:center;justify-content:center;-webkit-transition-duration:.3s;transition-duration:.3s;background:#f0f0f0;height:38px;width:38px}.container_avaya_extension_right button:hover{color:var(--sd-text-color-dark);background:var(--sd-border-color-light)}.dialpad_container_sd{position:absolute;width:100%;right:0;left:0;top:101px;height:calc(100vh - 101px);-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.container_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:8px;overflow:hidden;margin-bottom:10px;cursor:pointer}.container_contact_item_image{display:flex}.container_contact_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative}.container_contact_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_contact_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_contact_item_left .contact_field{font-size:13px;margin-bottom:2px}.contact_dark{color:var(--sd-text-color-dark)}.contact_gray{color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_grey{font-size:12px;color:var(--sd-text-color);padding-top:2px;display:flex}.container_contact_item_left .contact_field_green{font-size:13px;margin-bottom:2px;color:#6c0}.container_contact_item_left .contact_field_highlighted{display:flex;font-size:11px;color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_highlighted span{background:#6c0;border-radius:3px;text-transform:uppercase;color:var(--sd-text-color-inverse);font-size:9px;padding:2px 4px}.container_contact_item_right{display:flex;flex-direction:row}.container_contact_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_contact_item_right .right_field_highlighted{font-size:16px;color:#29abe0}.container_contact_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .right_recent_content{display:inline-block;padding:0;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .container_rating{font-size:11px;cursor:pointer}.container_contact_item_right .container_rating span{color:gray;transition:color .3s}.container_contact_item_right .container_rating span.filled{color:gold}.container_dialpad_digits{width:100%;display:flex;justify-content:center;margin-bottom:10px;flex-direction:column}.container_dialpad_row{display:flex;column-gap:10px;flex-direction:row;flex-shrink:0;justify-content:center;margin-bottom:10px}.button_dialpad_digit{display:block;width:50px;height:50px;display:flex;column-gap:20px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;font-size:22px;font-weight:700;color:var(--sd-text-color-dark);background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.button_dialpad_digit:hover{background:#000;color:var(--sd-text-color-inverse)}.button_call_green{background:#6c0;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.button_call_green:hover{background:#5ab400;color:var(--sd-text-color-inverse);width:60px;height:60px}.button_call_red{background:#e0261f;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s;z-index:9999}.button_call_red:hover{background:#cf211a}.button_call_disabled{background:#eee;color:var(--sd-text-color-light);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.container_dialpad_buttons{width:100%;display:flex;justify-content:center;padding-top:0;margin-bottom:6px}.container_call_header{width:100%;position:absolute;top:0;left:0;right:0;height:221px;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99;background:var(--sd-background-color)}.container_call_header_inner{width:100%;margin:0;padding:10px 15px 15px;position:relative}.container_call_info{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:var(--sd-text-color)}.container_call_info_blinker{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:red;animation:blinker 1s linear infinite}@keyframes blinker{26%{opacity:0}}.container_call_ani{width:100%;margin:0;padding:0}.container_call_ani_number{text-align:center;font-size:20px;font-weight:600;padding-top:26px}.container_call_ani_name{text-align:center;font-size:13px;font-weight:600;padding-top:2px}.container_call_ani_image{margin:0;padding:24px 0 0;width:100%}.container_call_ani_image_inner{display:flex;align-items:center;flex:0 0 auto;padding:0;width:85px;height:85px;position:relative;margin:0 auto;border-radius:100%}.container_call_content{width:100%;position:absolute;top:221px;left:0;right:0;height:calc(100% - 221px);font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);padding-top:20px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center}.container_call_dnis{width:100%;text-align:center;padding-bottom:16px}.container_call_dnis span{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 16px}.container_call_options{width:100%;display:flex;justify-content:center;margin-bottom:16px;flex-direction:column}.button_call_option{display:block;width:80px;height:60px;display:flex;flex-direction:column;flex-shrink:0;column-gap:0px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;outline:none;border:none;font-size:12px;font-weight:600;color:var(--sd-text-color-dark);background:none;-webkit-transition-duration:.3s;transition-duration:.3s;position:relative}.button_call_option .button_icon{display:flex;justify-content:center;align-items:center;height:40px}.button_call_option .button_text{display:flex;justify-content:center;color:var(--sd-text-color)}.button_active{color:#67cc02}.button_elevated{position:relative;z-index:9999}.button_disabled{color:gray;cursor:not-allowed;pointer-events:none;opacity:.6}.button_call_option[disabled],.button_disabled{opacity:.5;pointer-events:none;display:inline-flex}.container_popup_dialpad{position:absolute;z-index:999;left:0;right:0;bottom:0}.container_popup_dialpad_inner{background:#f1f1f1;background:linear-gradient(180deg,#f1f1f1 44%,#fff);margin:0 auto;padding:10px 15px 20px;border-radius:8px;width:88%;position:relative;height:475px}.container_popup_dialpad .button_close{position:absolute;right:-15px;top:-17px;z-index:9}.container_popup_dialpad .button_close span{width:32px;height:32px;background:var(--sd-background-color);display:flex;align-items:center;justify-content:center;border-radius:50%;cursor:pointer;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.container_popup_dialpad .button_close span:hover{background:#f1f1f1;color:var(--sd-text-color-dark)}.container_popup_dialpad .button_close span .fa-xmark{font-size:1.4em;font-weight:400}.container_dtmf_input{width:100%;margin:0;padding:0 0 10px;position:relative}.container_dtmf_input input{width:100%;display:flex;height:38px;outline:none;text-align:center;align-items:center;border:1px solid #f1f1f1;font-size:26px;font-weight:400;color:transparent;text-shadow:0 0 0 #000;padding:0;background:#f1f1f1}.container_active_call_header{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:12px;overflow:hidden}.container_active_call_avatar{display:flex}@keyframes play1{0%{transform:scale(1)}15%{box-shadow:0 0 0 2px #57b84666}25%{box-shadow:0 0 0 3px #98d48266,0 0 0 20px #fff3}25%{box-shadow:0 0 0 5px #bde4a966,0 0 0 30px #fff3}}.container_active_call_avatar_inner{margin:0 auto;padding:0;width:50px;height:50px;background:#dfe6e8;color:var(--sd-text-color-dark);position:relative;animation:play1 2s ease infinite;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;backface-visibility:hidden;border-radius:100%;overflow:hidden;display:flex;justify-content:center;align-items:center}.container_active_call_info{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_active_call_info_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_active_call_duration{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:var(--sd-text-color);margin-bottom:10px}.container_active_call_number{font-size:13px;margin-bottom:2px}.container_active_call_details{font-size:13px;color:#6c0;padding-top:2px}.container_active_call_details_inner{display:flex;margin:0;padding:0;justify-content:left}.container_active_call_icon_grey{display:flex;width:18px;height:16px;justify-content:center;align-items:center;color:var(--sd-text-color-medium)}.container_active_call_icon_green{display:flex;width:18px;height:16px;justify-content:center;align-items:center;color:#6c0}.container_active_call_status_green{display:flex;justify-content:center;align-items:center;color:#6c0;padding-left:6px;font-size:12px}.container_active_call_status_grey{display:flex;justify-content:center;align-items:center;color:var(--sd-text-color-medium);padding-left:6px;font-size:12px}.container_active_call_options{display:flex;flex-direction:row}.container_active_call_options .button_active_call_options{display:flex;flex-basis:auto;justify-content:right}.container_active_call_options .button_active_call_options_highlighted{font-size:16px;color:#29abe0}.container_active_call_options_button{display:flex;border-radius:26px;border:none;padding:0;font-size:11px;color:var(--sd-text-color);width:32px;height:32px;justify-content:center;align-items:center}.container_active_call_options_button:hover{background:#f1f1f1}.container_conference_contact_list{position:relative;width:100%;background:#f7f7f7;border-top:1px solid #eee;border-bottom:1px solid #eee;height:149px;padding:0;margin-top:16px}.container_swap_merge_contact_list{position:relative;width:100%;background:var(--sd-background-color);height:149px;padding:0;margin-top:32px}.container_conference_contact_list_inner{margin:0;padding:0;position:relative;width:100%;height:100%}.container_conference_contact_list_scrollbox{padding:12px 15px}.container_conference_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:0 10px;overflow:hidden;cursor:pointer}.button_disconnect_call{display:block;width:42px;height:42px;display:flex;margin:0;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;color:#e0261f;background:#f7f7f7;-webkit-transition-duration:.3s;transition-duration:.3s}.button_disconnect_call:hover{color:var(--sd-text-color-inverse);background:#e0261f;-webkit-transition-duration:.3s;transition-duration:.3s}.container_separator{height:1px;width:100%;background:#ededed;margin:11px 0}.container_transfer_call{position:absolute;z-index:99999;left:0;right:0;bottom:-51px;width:100%;background:var(--sd-background-color)}.container_transfer_call_inner{background:var(--sd-background-color);box-shadow:#11111a1a 0 4px 16px,#11111a0d 0 8px 32px;padding:20px 15px 10px;border-radius:0;width:100%;position:relative;height:596px;overflow:hidden}.container_transfer_call_inner h2{font-family:var(--sd-text-font-family);font-size:20px;font-weight:700;color:var(--sd-text-color-dark);letter-spacing:-1px;padding:0;line-height:normal;text-transform:capitalize;margin-bottom:15px}.container_button_close{position:absolute;right:4px;top:4px}.container_button_close span{width:36px;height:36px;background:var(--sd-background-color);display:flex;align-items:center;justify-content:center;border-radius:50%;cursor:pointer}.container_button_close span:hover{background:var(--sd-background-color)}.container_button_close .fa-xmark{font-size:1.6em;font-weight:400;color:var(--sd-text-color-light)}.container_button_close span:hover .fa-xmark:hover{color:var(--sd-text-color-dark)}.container_transfer_call_content{margin:0;padding:5px 0 0;width:100%;background:var(--sd-background-color);min-height:485px}.container_transfer_call_content h3{font-family:var(--sd-text-font-family);font-size:14px;font-weight:700;color:var(--sd-text-color);letter-spacing:-.26px;padding:8px 0 0;line-height:normal;text-transform:capitalize;margin-bottom:15px}.container_transfer_call_content h3 span{color:var(--sd-text-color-dark)}.container_transfer_call_details{width:100%;margin:0;padding:10px 15px 69px;position:relative}.container_transfer_tab_buttons{display:flex;gap:10px;align-items:center;margin-bottom:15px}.button_transfer_tab{padding:10px;border:none;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);background-color:var(--sd-border-color-light);cursor:pointer;width:105px;border-radius:28px;outline:none}.button_transfer_tab.active{background-color:var(--sd-app-color);color:#fff}.container_transfer_tab_content{margin:0;padding:0;width:100%}.container_transfer_internal_search{margin:0 0 15px;padding:0;width:100%}.container_transfer_internal_search input{padding:0 16px;width:100%;outline:none;display:block;border:1px solid var(--sd-border-color-dark);color:var(--sd-text-color-medium);font-size:12px;letter-spacing:-.25px;font-family:var(--sd-text-font-family);background:var(--sd-background-color);font-style:italic;font-weight:400;position:relative;border-radius:27px;height:48px}.container_transfer_contact_list{width:100%;margin:0;padding:0;position:relative;height:336px}.container_transfer_contact_list_inner{width:100%;position:relative;height:100%}.container_transfer_internal_contact_list{position:absolute;width:100%;left:0;right:0;background:var(--sd-background-color);height:332px}.container_transfer_disconnect{width:100%;display:flex;justify-content:center;padding-top:3px;margin-bottom:6px}.container_avaya_footer{width:100%;padding:0;background:#f7f7f7;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color-medium);position:relative;display:flex;height:51px;justify-content:center;align-items:center;border-top:1px solid var(--sd-border-color-light);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;z-index:999;position:absolute;bottom:0;left:0}.margin_bottom_16{margin-bottom:16px}.margin_bottom_nt{margin-bottom:10px}.padding_top_nt{padding-top:7px}.shadow_st{box-shadow:#00000026 0 2px 8px}.call_animation{animation:play 2s ease infinite;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;backface-visibility:hidden}.mini-call-switcher{display:flex;justify-content:center;margin-top:10px}.mini-call{padding:6px 12px;margin:0 6px;background:#eaeaea;border-radius:8px;cursor:pointer;font-size:14px}.mini-call.active{background-color:#c6f6d5;font-weight:700}.history-backdrop{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;z-index:999}.history-popup{position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);background:var(--sd-background-color);padding:20px;z-index:1000;box-shadow:0 2px 10px #0000004d;border-radius:5px;width:600px;max-height:80vh;overflow-y:auto}.history-content{position:relative}.history-content h3{margin-top:0}.close-btn{background:transparent;border:none;font-size:1.5rem;position:absolute;top:0;right:0;cursor:pointer}.history-content table{width:100%;border-collapse:collapse;margin-top:1rem}.history-content th,.history-content td{padding:.5rem;border:1px solid var(--sd-text-color-light)}@keyframes play{0%{transform:scale(1)}15%{box-shadow:0 0 0 4px #57b84680}25%{box-shadow:0 0 0 5px #98d48266,0 0 0 20px #fff3}25%{box-shadow:0 0 0 10px #c2e5b180,0 0 0 30px #fff3}}::-webkit-input-placeholder{color:#ddd}::-moz-placeholder{color:var(--sd-text-color-dark)}:-ms-input-placeholder{color:var(--sd-text-color-dark)}:-moz-placeholder{color:var(--sd-text-color-dark)}.slide_st{left:-100px;-webkit-animation:slide .5s forwards;-webkit-animation-delay:2s;animation:slide .5s forwards;animation-delay:0s}@-webkit-keyframes slide{to{left:0}}@keyframes slide{to{left:0}}.tabs_sd{width:100%;position:absolute;bottom:0;left:0;right:0;display:flex;height:51px;border-top:1px solid var(--sd-border-color-light);background:#f9f9f9;z-index:999999}.tab_heading_btn{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;align-items:center;padding:0;border:none;cursor:pointer;width:25%;font-family:var(--sd-text-font-family);font-size:11px;font-weight:600;line-height:normal;color:var(--sd-text-color-medium);background:none;transition:background-color .3s linear;-webkit-transition:background-color .3s linear;-o-transition:background-color .3s linear;-ms-transition:background-color .3s linear}.nav_icon{display:flex;align-items:center;justify-content:center;height:22px;margin-bottom:1px}.nav_heading_sd{display:flex;align-items:center}.tab_heading_btn.active{color:var(--sd-app-color);background:url() top center no-repeat}.tab-content_sd{width:100%;height:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;line-height:normal;background:var(--sd-background-color);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;position:relative}.container_transfer_fav_list{width:100%;margin:0;padding:0;position:relative;height:376px}.outer_container_swap_merge{width:100%;margin:0;padding:0}.container_swap_merge_item{width:100%;position:relative;display:flex;flex-direction:row;height:72px;border-radius:8px;overflow:hidden}.container_recent_item_image{display:flex;align-items:center}.container_swap_merge_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative;border-radius:50%;display:flex;align-items:center;justify-content:center;font-family:var(--sd-text-font-family);font-size:26px;color:var(--sd-text-color-inverse);font-weight:400}.container_recent_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;position:relative;border-top:1px solid #e9edef}.container_recent_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_recent_item_left .contact_field{font-size:13px;color:var(--sd-text-color-dark);margin-bottom:2px}.onhold_st{font-family:var(--sd-text-font-family);font-size:12px;color:var(--sd-text-color-medium);font-weight:600}.container_recent_item_right{display:flex;flex-direction:row;position:absolute;right:0}.container_recent_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.button_swap_merge{display:block;width:32px;height:45px;display:flex;flex-direction:column;flex-shrink:0;column-gap:0px;margin:0 0 0 28px;align-items:center;justify-content:center;line-height:normal;outline:none;border:none;font-size:12px;font-weight:600;color:var(--sd-text-color-dark);background:none;-webkit-transition-duration:.3s;transition-duration:.3s;position:relative;pointer-events:all;cursor:pointer}.button_swap_merge .button_icon{display:flex;justify-content:center;align-items:center;height:40px}.button_swap_merge .button_text{display:flex;justify-content:center;color:var(--sd-text-color-dark)}@media only screen and (max-width: 767px){.previewcontent{width:95%!important}.bgSecWrap .numberArrow{left:0!important;top:-20px!important}.transcription-popup{position:fixed;bottom:20px;right:20px;width:320px;max-height:250px;background:#fff;border-radius:10px;box-shadow:0 4px 12px #0003;padding:12px;z-index:1000;display:flex;flex-direction:column}.transcription-header{display:flex;justify-content:space-between;align-items:center;font-weight:700;font-size:14px;margin-bottom:8px}.transcription-body{overflow-y:auto;font-size:13px;line-height:1.4;color:#333}.close-btn{border:none;background:transparent;font-size:18px;cursor:pointer}}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1$1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i10.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i10.MatLabel, selector: "mat-label" }, { kind: "directive", type: i10.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i10.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: i11.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly", "disabledInteractive"], exportAs: ["matInput"] }, { kind: "directive", type: i5.NativeElementInjectorDirective, selector: "[ngModel], [formControl], [formControlName]" }, { kind: "component", type: CallHistoryComponent, selector: "app-call-history", inputs: ["mode"], outputs: ["makeCallEv", "transferEv"] }, { kind: "component", type: DialpadComponent, selector: "app-dialpad", inputs: ["contacts"], outputs: ["makeCallEv"] }, { kind: "component", type: DeviceSelectorComponent, selector: "app-device-selector", inputs: ["selectedAudioInputId", "selectedAudioOutputId"], outputs: ["change"] }], animations: [
|
|
2794
|
+
trigger('slideInOut', [
|
|
2795
|
+
transition(':enter', [
|
|
2796
|
+
style({ transform: 'translateX(100%)', opacity: 0 }),
|
|
2797
|
+
animate('300ms ease-in', style({ transform: 'translateX(0)', opacity: 1 })),
|
|
2798
|
+
]),
|
|
2799
|
+
transition(':leave', [
|
|
2800
|
+
animate('300ms ease-out', style({ transform: 'translateX(-100%)', opacity: 0 })),
|
|
2801
|
+
]),
|
|
2802
|
+
]),
|
|
2803
|
+
] });
|
|
2804
|
+
}
|
|
2805
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetComponent, decorators: [{
|
|
2806
|
+
type: Component,
|
|
2807
|
+
args: [{ selector: 'snugdesk-avaya-ipo-widget', animations: [
|
|
2808
|
+
trigger('slideInOut', [
|
|
2809
|
+
transition(':enter', [
|
|
2810
|
+
style({ transform: 'translateX(100%)', opacity: 0 }),
|
|
2811
|
+
animate('300ms ease-in', style({ transform: 'translateX(0)', opacity: 1 })),
|
|
2812
|
+
]),
|
|
2813
|
+
transition(':leave', [
|
|
2814
|
+
animate('300ms ease-out', style({ transform: 'translateX(-100%)', opacity: 0 })),
|
|
2815
|
+
]),
|
|
2816
|
+
]),
|
|
2817
|
+
], standalone: false, template: "@if (isReconnectingInProgress || isFailoverInProgress || isFailbackInProgress) {\n <div class=\"container_connection_status_overlay\">\n <div class=\"container_connection_status\">\n <i class=\"fa-solid fa-circle-dot\"></i>\n\n @if (isReconnectingInProgress) {\n <span>Reconnecting to the server...</span>\n }\n\n @if (isFailoverInProgress) {\n <span>Failover in progress...</span>\n }\n\n @if (isFailbackInProgress) {\n <span>Failback in progress...</span>\n }\n\n <div class=\"container_connection_status_loader\">\n <div class=\"container_three_dot_loader\">\n <span class=\"dot\"></span>\n <span class=\"dot\"></span>\n <span class=\"dot\"></span>\n </div>\n </div>\n </div>\n </div>\n}\n\n\n\n@if (showLoginForm) {\n <!-- Login Form -->\n <div class=\"container_avaya_login\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"container_avaya_login_inner\">\n <div class=\"container_avaya_login_content\">\n\n <div class=\"container_avaya_logo\">\n <div class=\"container_avaya_logo_image\"></div>\n <h3>IP Office<span>TM</span></h3>\n </div>\n\n @if (hasAvayaSocketError || hasAvayaLoginError || hasAvayaLoginConflictError || hasAvayaDeviceError) {\n <div class=\"container_avaya_login_error\">\n <p class=\"message_label error_message\">\n @if (hasAvayaSocketError) {\n We're having trouble connecting to the server\n }\n @if (hasAvayaLoginError) {\n Invalid extension or password!\n }\n @if (hasAvayaLoginConflictError) {\n Another user is logged in with this extension\n }\n @if (hasAvayaDeviceError) {\n We're having trouble accessing your device\n }\n </p>\n </div>\n }\n\n <div class=\"container_avaya_login_form\">\n <form [formGroup]=\"loginForm\" (ngSubmit)=\"onLogin()\">\n\n <!-- Extension -->\n <div class=\"form_field\">\n <mat-form-field class=\"properties_form_field mat-form-field-outline\">\n <mat-label class=\"mdc-floating-disabled\">Extension</mat-label>\n <input matInput formControlName=\"ipoExtensionNo\" autocomplete=\"ipoExtensionNo\"\n (keypress)=\"restrictNumeric($event)\" />\n\n @if (loginForm.controls['ipoExtensionNo'].invalid && loginForm.controls['ipoExtensionNo'].touched) {\n <mat-error>\n @if (loginForm.controls['ipoExtensionNo'].errors?.['required']) {\n <small>Extension is required.</small>\n }\n </mat-error>\n }\n </mat-form-field>\n </div>\n\n <!-- Password -->\n <div class=\"form_field\">\n <mat-form-field class=\"properties_form_field mat-form-field-outline\">\n <mat-label class=\"mdc-floating-disabled\">Password</mat-label>\n <input matInput formControlName=\"ipoPassword\" autocomplete=\"avaya-ipo-password\"\n (keypress)=\"restrictNumeric($event)\" [type]=\"showLoginPassword ? 'text' : 'password'\" />\n\n <mat-icon matSuffix (click)=\"showLoginPassword = !showLoginPassword && !hasAvayaSocketError\">\n @if (showLoginPassword) {\n <span class=\"container_icon_eye\">\n <i class=\"fa-solid fa-eye\"></i>\n </span>\n } @else {\n <span class=\"container_icon_eye\">\n <i class=\"fa-solid fa-eye-slash\"></i>\n </span>\n }\n </mat-icon>\n\n @if (loginForm.controls['ipoPassword'].invalid && loginForm.controls['ipoPassword'].touched) {\n <mat-error>\n @if (loginForm.controls['ipoPassword'].errors?.['required']) {\n <small>Password is required.</small>\n }\n </mat-error>\n }\n </mat-form-field>\n </div>\n\n <!-- Login button -->\n <div class=\"avaya_login_button_area\">\n <button type=\"submit\" class=\"button_submit_login\"\n [ngClass]=\"{ 'button_submit_login_loader': showLoginLoader }\"\n [disabled]=\"showLoginLoader\">\n @if (showLoginLoader) {\n <span class=\"button_submit_loader\"></span>\n } @else {\n <span>Login</span>\n }\n </button>\n </div>\n\n </form>\n </div>\n\n </div>\n </div>\n </div>\n </div>\n </div>\n} @else {\n <!-- Softphone UI -->\n <ng-container *ngTemplateOutlet=\"showSoftphoneDiv\"></ng-container>\n}\n\n<!-- Footer (always visible) -->\n<div class=\"container_avaya_footer\">© 2025 SNUG Technologies Pvt. Ltd.\n\n\n\n</div>\n\n\n\n\n\n<ng-template #showSoftphoneDiv>\n\n@if (!showActiveCall) {\n <div class=\"container_avaya_softphone_inner\">\n <!-- ------------------- Keypad work start ---------- -->\n @if (bottomActiveTab === 'keypad_sd') {\n <div class=\"tab-content_sd\">\n <div class=\"container_dialpad_header\">\n <div class=\"container_dialpad_header_inner2\">\n <div class=\"container_avaya_extension\">\n <div class=\"container_avaya_extension_left\">\n <div class=\"extension_label\">Extension</div>\n <div class=\"extension_number\">{{ extensionNumber }}</div>\n </div>\n <div class=\"container_avaya_extension_right\">\n <button title=\"Sign out\" (click)=\"onLogout()\">\n <svg width=\"16px\" fill=\"currentColor\" viewBox=\"0 0 512 512\">\n <path\n d=\"M377.9 105.9L500.7 228.7c7.2 7.2 11.3 17.1 11.3 27.3s-4.1 20.1-11.3 27.3L377.9 406.1c-6.4 6.4-15 9.9-24 9.9c-18.7 0-33.9-15.2-33.9-33.9l0-62.1-128 0c-17.7 0-32-14.3-32-32l0-64c0-17.7 14.3-32 32-32l128 0 0-62.1c0-18.7 15.2-33.9 33.9-33.9c9 0 17.6 3.6 24 9.9zM160 96L96 96c-17.7 0-32 14.3-32 32l0 256c0 17.7 14.3 32 32 32l64 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-64 0c-53 0-96-43-96-96L0 128C0 75 43 32 96 32l64 0c17.7 0 32 14.3 32 32s-14.3 32-32 32z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n <div class=\"dialpad_container_sd\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <app-dialpad [contacts]=\"contacts\" (makeCallEv)=\"makeCall($event)\"></app-dialpad>\n <!-- to do list -->\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- ------------------- Keypad work end ---------- -->\n\n @if (bottomActiveTab === 'recent_sd') {\n <div class=\"tab-content_sd\">\n <app-call-history (makeCallEv)=\"makeCallFromContact($event)\"></app-call-history>\n </div>\n }\n\n @if (bottomActiveTab === 'settings_sd') {\n <div class=\"tab-content_sd\">\n <app-device-selector (change)=\"handleDeviceSelection($event)\"></app-device-selector>\n </div>\n }\n </div>\n\n <div class=\"tabs_sd\">\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('recent_sd')\"\n [class.active]=\"bottomActiveTab === 'recent_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"19px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M464 256A208 208 0 1 1 48 256a208 208 0 1 1 416 0zM0 256a256 256 0 1 0 512 0A256 256 0 1 0 0 256zM232 120l0 136c0 8 4 15.5 10.7 20l96 64c11 7.4 25.9 4.4 33.3-6.7s4.4-25.9-6.7-33.3L280 243.2 280 120c0-13.3-10.7-24-24-24s-24 10.7-24 24z\" />\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Recent</span>\n </button>\n\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('keypad_sd')\"\n [class.active]=\"bottomActiveTab === 'keypad_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"20px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\"></path>\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Keypad</span>\n </button>\n\n <button class=\"tab_heading_btn\" (click)=\"bottomSelectTab('settings_sd')\"\n [class.active]=\"bottomActiveTab === 'settings_sd'\">\n <span class=\"nav_icon\">\n <svg width=\"19px\" viewBox=\"-0.03 0 16.079 16.079\">\n <g>\n <path fill=\"currentColor\"\n d=\"M8.182 1.083a6.99 6.99 0 0 0-6.248 3.493c-1.929 3.342-.776 7.629 2.57 9.562 3.346 1.934 7.634.792 9.562-2.55 1.929-3.343.776-7.634-2.57-9.567a6.98 6.98 0 0 0-3.314-.938zM8 2.08a6 6 0 0 1 6 6 6 6 0 0 1-6 6 6 6 0 0 1-6-6 6 6 0 0 1 6-6z\" />\n <path fill=\"currentColor\"\n d=\"M9.322 0L6.69.393v1.354a6.49 6.477 43.146 0 1 2.632.005V0zM3.937 1.19L1.93 2.897l.988 1.177a6.49 6.477 43.146 0 1 2.017-1.69zm8.128.013l-.993 1.184a6.49 6.477 43.146 0 1 .17.09 6.49 6.477 43.146 0 1 1.845 1.603l1.006-1.197zM.455 5.42l-.44 2.596 1.515.267a6.49 6.477 43.146 0 1 .455-2.593zm15.086.003l-1.523.269a6.49 6.477 43.146 0 1 .464 2.59l1.533-.27zM1.858 10.118l-1.351.78 1.33 2.271 1.339-.772a6.49 6.477 43.146 0 1-1.317-2.28zm12.301.003a6.49 6.477 43.146 0 1-.534 1.215 6.49 6.477 43.146 0 1-.774 1.069l1.338.773 1.302-2.288zm-9.557 3.471l-.534 1.47 2.48.884.525-1.446a6.49 6.477 43.146 0 1-2.303-.8 6.49 6.477 43.146 0 1-.168-.108zm6.814.016a6.49 6.477 43.146 0 1-2.475.897l.53 1.457 2.468-.917z\" />\n <path fill=\"currentColor\"\n d=\"M7.648 3.093a4.989 4.989 0 0 0-3.982 2.483 5.013 5.013 0 0 0 1.836 6.834 5.002 5.002 0 0 0 6.83-1.827 5.01 5.01 0 0 0-1.836-6.832 4.976 4.976 0 0 0-2.848-.658zM8 4.08a4 4 0 0 1 4 4 4 4 0 0 1-4 4 4 4 0 0 1-4-4 4 4 0 0 1 4-4z\" />\n </g>\n </svg>\n </span>\n <span class=\"nav_heading_sd\">Preferences</span>\n </button>\n </div>\n} @else {\n <ng-container *ngTemplateOutlet=\"showActiveCallDiv\"></ng-container>\n}\n\n\n<!-- ////////////// -->\n\n <!-- Active Call Div -->\n \n <ng-template #showActiveCallDiv>\n\n\n @if (showTranscriptionPopup) {\n <div class=\"transcription-popup\">\n <div class=\"transcription-header\">\n <span>Live Transcription</span>\n <button class=\"close-btn\" (click)=\"showTranscriptionPopup = false\">\u00D7</button>\n </div>\n <div class=\"transcription-body\">\n <p>{{ transcriptText }}</p>\n </div>\n </div>\n}\n\n\n\n\n\n\n\n<div class=\"container_call\">\n <div class=\"container_call_header\">\n <div class=\"container_call_header_inner\">\n\n @if (!conferenceParticipants || conferenceParticipants.length <= 1) {\n <!-- ------------ add call list swap and merge area start ---------------- -->\n @if (activeCalls.length > 1) {\n <div class=\"container_swap_merge_list\">\n <div class=\"container_conference_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <div class=\"outer_container_swap_merge\">\n @for (call of activeCalls; track call.phoneNumber; let i = $index) {\n @if (i !== currentActiveCallIndex) {\n <div class=\"container_swap_merge_item\"\n (click)=\"swapLinesToIndex(i)\"\n [ngClass]=\"{ 'active-call': i === currentActiveCallIndex, 'other-call': i !== currentActiveCallIndex }\">\n\n <!-- Avatar -->\n <div class=\"container_recent_item_image\">\n <div class=\"container_swap_merge_item_image_inner\">\n <!-- svg avatar here -->\n </div>\n </div>\n\n <!-- Call Info -->\n <div class=\"container_recent_item_left\">\n <div class=\"container_recent_item_left_inner\">\n <div class=\"contact_field\">{{ call.phoneNumber }}</div>\n @if (call.isOnHold) {\n <div class=\"onhold_st\">On hold</div>\n }\n </div>\n\n <!-- Action Buttons -->\n <div class=\"container_recent_item_right\">\n <div class=\"right_field\">\n <button class=\"button_swap_merge\">\n <div class=\"button_icon\">\n <!-- merge svg -->\n </div>\n <div class=\"button_text\">Merge</div>\n </button>\n </div>\n <div class=\"right_field\">\n <button class=\"button_swap_merge\">\n <div class=\"button_icon\">\n <!-- swap svg -->\n </div>\n <div class=\"button_text\">Swap</div>\n </button>\n </div>\n </div>\n </div>\n </div>\n }\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n <!-- ------------ add call list swap and merge area end ---------------- -->\n\n @if (isCalling) {\n <div class=\"container_call_info\">Calling...</div>\n } @else {\n @if (isactiveCall) {\n <div class=\"container_call_info\">\n {{ formatTime(activeCalls[currentActiveCallIndex].duration) }}\n </div>\n }\n @if (showCallDisconnected) {\n <div class=\"container_call_info_blinker\">\n <!-- disconnected svg -->\n {{ this.istransfer ? 'Call Transferred' : 'Call ended' }}\n </div>\n }\n }\n\n <div class=\"container_call_ani\">\n @if (isCalling && displayCallingNumber && activeCalls.length === 0) {\n <!-- show what the user dialed while call is spinning up -->\n <div class=\"container_call_ani_number\">\n {{ displayCallingNumber }}\n </div>} \n @else if (activeCalls.length > 0) {\n <div class=\"container_call_ani_number\">\n {{ activeCalls[currentActiveCallIndex].lookupPhoneNumber.country?.flag }}\n {{ activeCalls[currentActiveCallIndex].phoneNumber }}\n \n </div>\n <div class=\"container_call_ani_name\">\n {{ activeCalls[currentActiveCallIndex].lookupPhoneNumber.phoneNumber?.location }}\n </div>\n } @else {\n <div class=\"container_call_ani_number\">\n {{ lastDisconnectedCall?.lookupPhoneNumber?.country?.flag }}\n {{ lastDisconnectedCall?.phoneNumber }}\n </div>\n <div class=\"container_call_ani_name\">\n {{ lastDisconnectedCall?.lookupPhoneNumber?.phoneNumber?.location }}\n </div>\n }\n\n <div class=\"container_call_ani_image\">\n <div class=\"container_call_ani_image_inner\" [ngClass]=\"{ 'call_animation': isactiveCall }\">\n <!-- big avatar svg -->\n </div>\n </div>\n </div>\n\n } @else {\n <!-- Conference Call Placeholder -->\n <!-- @if block can render conference call template here -->\n }\n\n </div>\n </div>\n\n <div class=\"container_call_content\">\n <div class=\"container_call_dnis\">\n <span>Ext. {{ extensionNumber }}</span>\n </div>\n\n <div class=\"container_call_options\">\n <div class=\"container_dialpad_row margin_bottom_16\">\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall && activeCalls.length < 4 }\"\n (click)=\"onAddCall()\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"> \n <svg fill=\"#000000\" width=\"20\" viewBox=\"0 0 52 52\" enable-background=\"new 0 0 52 52\"\n xml:space=\"preserve\">\n <path fill=\"currentColor\" d=\"M30,29h16.5c0.8,0,1.5-0.7,1.5-1.5v-3c0-0.8-0.7-1.5-1.5-1.5H30c-0.6,0-1-0.4-1-1V5.5C29,4.7,28.3,4,27.5,4\n h-3C23.7,4,23,4.7,23,5.5V22c0,0.6-0.4,1-1,1H5.5C4.7,23,4,23.7,4,24.5v3C4,28.3,4.7,29,5.5,29H22c0.6,0,1,0.4,1,1v16.5\n c0,0.8,0.7,1.5,1.5,1.5h3c0.8,0,1.5-0.7,1.5-1.5V30C29,29.4,29.4,29,30,29z\" />\n </svg>\n </div>\n <div class=\"button_text\">Add</div>\n </button>\n\n @if (activeCalls.length > 0) {\n <button class=\"button_call_option\" (click)=\"onAddCall()\" type=\"button\">\n <div class=\"button_icon\"> <svg width=\"20\" viewBox=\"0 0 52 52\" aria-hidden=\"true\">\n <path fill=\"currentColor\"\n d=\"M30,29h16.5c0.8,0,1.5-0.7,1.5-1.5v-3c0-0.8-0.7-1.5-1.5-1.5H30c-0.6,0-1-0.4-1-1V5.5C29,4.7,28.3,4,27.5,4h-3\n C23.7,4,23,4.7,23,5.5V22c0,0.6-0.4,1-1,1H5.5C4.7,23,4,23.7,4,24.5v3C4,28.3,4.7,29,5.5,29H22c0.6,0,1,0.4,1,1v16.5\n c0,0.8,0.7,1.5,1.5,1.5h3c0.8,0,1.5-0.7,1.5-1.5V30C29,29.4,29.4,29,30,29z\"/>\n </svg></div>\n <div class=\"button_text\">Conference</div>\n </button>\n }\n\n @if (!isRecording) {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"> <svg width=\"22px\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path fill=\"currentColor\" d=\"M3 10L3 14M7.5 6L7.5 18M12 3V21M16.5 6V18M21 10V14\" stroke=\"#000000\"\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg></div>\n <div class=\"button_text\">Record</div>\n </button>\n } @else {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\">\n <div class=\"button_icon\"><svg width=\"22px\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path fill=\"currentColor\" d=\"M3 10L3 14M7.5 6L7.5 18M12 3V21M16.5 6V18M21 10V14\" stroke=\"#000000\"\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg></div>\n <div class=\"button_text\">Stop</div>\n </button>\n }\n\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\"\n (click)=\"isCallTransferOpen = !isCallTransferOpen\">\n <div class=\"button_icon\"> <svg width=\"25px\" height=\"25px\" viewBox=\"0 0 48 48\" version=\"1\" enable-background=\"new 0 0 48 48\">\n <path fill=\"currentColor\"\n d=\"M39.2,8.4l-1.8,1.8c-6.3,6.5-5.4,22,0,27.6l1.8,1.8c0.5,0.5,1.3,0.5,1.8,0l3.6-3.7c0.5-0.5,0.5-1.3,0-1.8 l-3.4-3.4h-4.8c-1.3-1.3-1.3-12.1,0-13.4h4.8l3.3-3.4c0.5-0.5,0.5-1.3,0-1.8L41,8.4C40.5,7.9,39.7,7.9,39.2,8.4z\" />\n <path fill=\"currentColor\"\n d=\"M11.2,8.4l-1.8,1.8c-6.3,6.5-5.4,22,0,27.6l1.8,1.8c0.5,0.5,1.3,0.5,1.8,0l3.6-3.7c0.5-0.5,0.5-1.3,0-1.8 l-3.4-3.4H8.5c-1.3-1.3-1.3-12.1,0-13.4h4.8l3.3-3.4c0.5-0.5,0.5-1.3,0-1.8L13,8.4C12.5,7.9,11.7,7.9,11.2,8.4z\" />\n <g fill=\"currentColor\">\n <polygon points=\"25.3,18.6 30.7,24 25.3,29.4\" />\n <rect x=\"16\" y=\"22\" width=\"11\" height=\"4\" />\n </g>\n </svg></div>\n <div class=\"button_text\">Transfer</div>\n </button>\n\n @if (isCallTransferOpen) {\n <ng-container *ngTemplateOutlet=\"callTransferTemplate\"></ng-container>\n }\n </div>\n\n <!-- Mute / Unmute -->\n <div class=\"container_dialpad_row\">\n @if (activeCalls.length > 0 && !this.activeCalls[this.currentActiveCallIndex].isMuted) {\n <button class=\"button_call_option button_elevated\"\n (click)=\"toggleMute()\"\n [ngClass]=\"{ 'button_disabled': showCallDisconnected }\">\n <div class=\"button_icon\"> <svg viewBox=\"0 0 384 512\" width=\"16\">\n <path fill=\"currentColor\"\n d=\"M192 0C139 0 96 43 96 96l0 160c0 53 43 96 96 96s96-43 96-96l0-160c0-53-43-96-96-96zM64 216c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l72 0 72 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6c85.8-11.7 152-85.3 152-174.4l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 70.7-57.3 128-128 128s-128-57.3-128-128l0-40z\" />\n </svg></div>\n <div class=\"button_text\">Mute</div>\n </button>\n } @else if (activeCalls.length > 0 && this.activeCalls[this.currentActiveCallIndex].isMuted) {\n <button class=\"button_call_option button_active button_elevated\" (click)=\"toggleMute()\">\n <div class=\"button_icon\"> <svg width=\"26\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M38.8 5.1C28.4-3.1 13.3-1.2 5.1 9.2S-1.2 34.7 9.2 42.9l592 464c10.4 8.2 25.5 6.3 33.7-4.1s6.3-25.5-4.1-33.7L472.1 344.7c15.2-26 23.9-56.3 23.9-88.7l0-40c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 40c0 21.2-5.1 41.1-14.2 58.7L416 300.8 416 96c0-53-43-96-96-96s-96 43-96 96l0 54.3L38.8 5.1zM344 430.4c20.4-2.8 39.7-9.1 57.3-18.2l-43.1-33.9C346.1 382 333.3 384 320 384c-70.7 0-128-57.3-128-128l0-8.7L144.7 210c-.5 1.9-.7 3.9-.7 6l0 40c0 89.1 66.2 162.7 152 174.4l0 33.6-48 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l72 0 72 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0 0-33.6z\" />\n </svg></div>\n <div class=\"button_text\">Unmute</div>\n </button>\n }\n\n <!-- Hold / Unhold -->\n @if (activeCalls.length > 0 && !this.activeCalls[this.currentActiveCallIndex].isOnHold) {\n <button class=\"button_call_option button_elevated\"\n [ngClass]=\"{ 'button_disabled': !isactiveCall }\"\n [disabled]=\"!isactiveCall\"\n (click)=\"toggleHold()\">\n <div class=\"button_icon\"> <svg viewBox=\"0 0 320 512\" width=\"14\">\n <path fill=\"currentColor\" d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\" />\n </svg></div>\n <div class=\"button_text\">Hold</div>\n </button>\n } @else if (activeCalls.length > 0 && this.activeCalls[this.currentActiveCallIndex].isOnHold) {\n <button class=\"button_call_option button_active button_elevated\" (click)=\"toggleHold()\">\n <div class=\"button_icon\">\n <svg viewBox=\"0 0 320 512\" width=\"14\">\n <path fill=\"currentColor\" d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\" />\n </svg></div>\n <div class=\"button_text\">Unhold</div>\n </button>\n \n }\n\n <!-- Keypad -->\n <button class=\"button_call_option button_elevated\"\n [class.button_active]=\"isDailpadOpen\"\n (click)=\"menuDailpad()\"\n [ngClass]=\"{ 'button_disabled': showCallDisconnected }\">\n <div class=\"button_icon\"> <svg width=\"22px\" height=\"22px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n </svg></div>\n <div class=\"button_text\">{{ isDailpadOpen ? 'Hide' : 'Keypad' }}</div>\n </button>\n\n @if (isDailpadOpen) {\n <div class=\"container_popup_dialpad\">\n <div class=\"container_popup_dialpad_inner\">\n <div [formGroup]=\"dtmfForm\">\n <div class=\"container_dtmf_input\">\n <input type=\"tel\" formControlName=\"dtmf\">\n </div>\n <div class=\"container_dialpad_digits\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('1')\">1</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('2')\">2</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('3')\">3</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('4')\">4</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('5')\">5</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('6')\">6</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('7')\">7</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('8')\">8</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('9')\">9</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path fill=\"currentColor\"\n d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86 102.4l122 70.4L208 32z\" />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('0')\">0</button>\n <button class=\"button_dialpad_digit\" (click)=\"onSendDTMF('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path fill=\"currentColor\"\n d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3 36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4 26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1 0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2 384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64 192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n\n <!-- Call Control Buttons -->\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n @if (!showCallDisconnected || isCalling) {\n <button class=\"button_dialpad_digit button_call_red\" (click)=\"dropCurrentCall()\">\n <svg width=\"25\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\" />\n </svg>\n </button>\n } @else if (showCallDisconnected && !isCalling) {\n <button class=\"button_dialpad_digit button_call_green\"\n (click)=\"onRedial()\"\n [ngClass]=\"{ 'button_call_disabled': !phoneNumber }\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n }\n </div>\n </div>\n </div>\n</div>\n</ng-template>\n\n\n <!-- ////////////////// -->\n <!-- Conference Call Template -->\n <ng-template #conferenceCallTemplate>\n<div class=\"container_active_call_header\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">Conference call</div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_green\">\n <svg viewBox=\"0 0 512 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </div>\n <div class=\"container_active_call_status_green\">\n with {{ conferenceParticipants?.length }} people\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"container_active_call_options\">\n <div class=\"button_active_call_options button_active_call_options_highlighted\">\n <span (click)=\"isConfereceList = !isConfereceList\" class=\"container_active_call_options_button\">\n <svg viewBox=\"0 0 448 512\" width=\"16px\">\n <path fill=\"currentColor\"\n d=\"M201.4 374.6c12.5 12.5 32.8 12.5 45.3 0l160-160c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L224 306.7 86.6 169.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l160 160z\" />\n </svg>\n </span>\n </div>\n </div>\n</div>\n\n<!-- Conference List -->\n@if (isConfereceList) {\n <div class=\"container_conference_contact_list\">\n <div class=\"container_conference_contact_list_inner\">\n <div class=\"scrollbox container_conference_contact_list_scrollbox\">\n <div class=\"scrollbox-content\">\n\n @for (person of conferenceParticipants; track person.id) {\n <div class=\"container_conference_contact_item\">\n <!-- Avatar -->\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"49px\" width=\"49px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n\n <!-- Name and Time -->\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ person.name }}</div>\n <div class=\"contact_field_green\">{{ person.duration }}</div>\n </div>\n </div>\n\n <!-- End Button -->\n <div class=\"container_contact_item_right\">\n <div class=\"right_field\">\n <button class=\"button_disconnect_call\">\n <svg width=\"22\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M11.7 266.3l41.9 94.3c6.1 13.7 20.8 21.3 35.5 18.4l109.2-21.8c15-3 25.7-16.1 25.7-31.4V240c62.3-20.8 129.7-20.8 192 0v85.8c0 15.3 10.8 28.4 25.7 31.4L550.9 379c14.7 2.9 29.4-4.7 35.5-18.4l41.9-94.3c7.2-16.2 5.1-35.1-7.4-47.7C570.8 168.1 464.2 96 320 96S69.2 168.1 19.1 218.6c-12.5 12.6-14.6 31.5-7.4 47.7z\">\n </path>\n </svg>\n </button>\n </div>\n </div>\n </div>\n\n @if (!$last) {\n <div class=\"container_separator\"></div>\n }\n }\n\n </div>\n </div>\n </div>\n </div>\n}\n\n</ng-template>\n\n\n <ng-template #callTransferTemplate>\n <div class=\"container_transfer_call\">\n <div class=\"container_transfer_call_inner\">\n <div class=\"container_button_close\">\n <span (click)=\"isCallTransferOpen = false\" title=\"Close\">\n <i class=\"fa-sharp fa-solid fa-xmark\"></i>\n </span>\n </div>\n\n <!-- First Div -->\n <div class=\"container_transfer_call_content\" @slideInOut>\n @if (currentView === 'transferListpage') {\n <div class=\"container_active_call_duration\">{{ formatTime(callDuration)}}</div>\n\n <div class=\"container_active_call_header margin_bottom_16 shadow_st\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">{{ lookupPhoneNumber?.country?.flag }}\n {{ lookupPhoneNumber?.phoneNumber }}</div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_green\">\n <svg viewBox=\"0 0 512 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </div>\n <div class=\"container_active_call_status_green\">Active call</div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">Ext. {{extensionNumber}}</span>\n </div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div class=\"container_transfer_tab_buttons\">\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab1')\" [class.active]=\"activeTab === 'tab1'\">\n <svg width=\"18px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M464 256A208 208 0 1 1 48 256a208 208 0 1 1 416 0zM0 256a256 256 0 1 0 512 0A256 256 0 1 0 0 256zM232 120l0 136c0 8 4 15.5 10.7 20l96 64c11 7.4 25.9 4.4 33.3-6.7s4.4-25.9-6.7-33.3L280 243.2 280 120c0-13.3-10.7-24-24-24s-24 10.7-24 24z\" />\n </svg> Recent</button>\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab2')\" [class.active]=\"activeTab === 'tab2'\">\n <svg width=\"20px\" viewBox=\"0 0 24 24\">\n <path fill=\"currentColor\"\n d=\"M15.71,12.71a6,6,0,1,0-7.42,0,10,10,0,0,0-6.22,8.18,1,1,0,0,0,2,.22,8,8,0,0,1,15.9,0,1,1,0,0,0,1,.89h.11a1,1,0,0,0,.88-1.1A10,10,0,0,0,15.71,12.71ZM12,12a4,4,0,1,1,4-4A4,4,0,0,1,12,12Z\" />\n </svg> Internal</button>\n <button class=\"button_transfer_tab\" (click)=\"selectTab('tab3')\" [class.active]=\"activeTab === 'tab3'\">\n <svg width=\"18px\" height=\"18px\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\" d=\"M256,400a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M256,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M384,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,272a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,144a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" />\n <path fill=\"currentColor\" d=\"M128,16a48,48,0,1,0,48,48,48,48,0,0,0-48-48Z\" /></svg>\n Dail</button>\n </div>\n\n <!-- Tab 1 -->\n @if (activeTab === 'tab1') {\n <!-- <div class=\"container_transfer_tab_content\">\n <app-call-history (makeCallEv)=\"makeCallFromContact($event)\"></app-call-history>\n </div> -->\n <div class=\"container_transfer_tab_content\">\n <app-call-history [mode]=\"'transfer'\" (transferEv)=\"transferToFromHistory($event)\"> </app-call-history>\n</div>\n }\n\n <!-- Tab 2 -->\n @if (activeTab === 'tab2') {\n <div class=\"container_transfer_tab_content\">\n <div class=\"container_transfer_internal_search\">\n <input type=\"search\" placeholder=\"Type to search\">\n </div>\n <div class=\"container_transfer_contact_list\">\n <div class=\"container_transfer_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n <!-- Instead of repeating static contacts, render via array -->\n @for (contact of internalContacts; track contact.ext) {\n <div class=\"container_contact_item\" (click)=\"goToDiv('transferDetails'); makeCallFromContact(contact.phone)\">\n <div class=\"container_contact_item_image\">\n <div class=\"container_contact_item_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"49px\" width=\"49px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\">\n </path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 c-0.964-1.144-2.024-2.326-3.184-3.527c-1.741-1.802-3.71-3.646-5.924-5.47c-2.952-2.431-6.339-4.824-10.204-7.026 c-1.877-1.07-3.873-2.092-5.98-3.055c-0.062-0.028-0.118-0.059-0.18-0.087c-9.792-4.44-22.106-7.529-37.416-7.529 s-27.624,3.089-37.416,7.529c-0.338,0.153-0.653,0.318-0.985,0.474c-1.431,0.674-2.806,1.376-4.128,2.101 c-0.716,0.393-1.417,0.792-2.101,1.197c-3.421,2.027-6.475,4.191-9.15,6.395c-2.213,1.823-4.182,3.668-5.924,5.47 c-1.161,1.201-2.22,2.384-3.184,3.527c-0.964,1.144-1.832,2.25-2.609,3.299c-0.778,1.049-1.464,2.04-2.065,2.955 c-0.557,0.848-1.033,1.622-1.447,2.324c-0.033,0.056-0.073,0.119-0.104,0.174c-0.435,0.744-0.79,1.392-1.07,1.926 c-0.559,1.068-0.818,1.678-0.818,1.678v0.398c18.285,17.927,43.322,28.985,70.945,28.985c27.678,0,52.761-11.103,71.055-29.095 v-0.289c0,0-0.619-1.45-1.992-3.778C174.594,173.238,174.117,172.463,173.561,171.615z\">\n </path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 c1.772-0.532,3.482-1.188,5.12-1.958c2.184-1.027,4.242-2.258,6.15-3.67c2.863-2.119,5.39-4.646,7.509-7.509 c0.706-0.954,1.367-1.945,1.98-2.971c0.919-1.539,1.729-3.155,2.422-4.84c0.462-1.123,0.872-2.277,1.226-3.458 c0.177-0.591,0.341-1.188,0.49-1.792c0.299-1.208,0.542-2.443,0.725-3.701c0.275-1.887,0.417-3.827,0.417-5.811 c0-1.984-0.142-3.925-0.417-5.811c-0.184-1.258-0.426-2.493-0.725-3.701c-0.15-0.604-0.313-1.202-0.49-1.793 c-0.354-1.181-0.764-2.335-1.226-3.458c-0.693-1.685-1.504-3.301-2.422-4.84c-0.613-1.026-1.274-2.017-1.98-2.971 c-2.119-2.863-4.646-5.39-7.509-7.509c-1.909-1.412-3.966-2.643-6.15-3.67c-1.638-0.77-3.348-1.426-5.12-1.958 c-1.181-0.355-2.39-0.655-3.624-0.896c-2.468-0.484-5.035-0.737-7.68-0.737c-21.162,0-37.345,16.183-37.345,37.345 C68.657,109.317,84.84,125.5,106.002,125.5z\">\n </path>\n </g>\n </svg>\n </div>\n </div>\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ contact.name }}</div>\n <div class=\"contact_field\">{{ contact.phone }}</div>\n @if (contact.role) {\n <div class=\"contact_field_grey\">\n <div class=\"contact_field_highlighted\"><span>{{ contact.role }}</span></div>\n </div>\n }\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field\">\n <span class=\"right_field_content\">Ext. {{ contact.ext }}</span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- Tab 3 -->\n @if (activeTab === 'tab3') {\n <div class=\"container_transfer_tab_content\">\n <form [formGroup]=\"transferForm\">\n <div class=\"container_transfer_contact_list\">\n <div class=\"container_phone_number_input margin_bottom_16\">\n <button class=\"button_phonebook\" (click)=\"isDirectoryOpen = !isDirectoryOpen\">\uD83D\uDCD6</button>\n <input placeholder=\"Phone Number\" type=\"tel\" formControlName=\"target\">\n <button class=\"button_backspace\" (click)=\"onBackspace()\"></button>\n </div>\n\n @if (isDirectoryOpen) {\n <div class=\"container_transfer_internal_contact_list\">\n <div class=\"container_transfer_contact_list_inner\">\n <div class=\"scrollbox\">\n <div class=\"scrollbox-content\">\n @for (directoryContact of directoryContacts; track directoryContact.id) {\n <div class=\"container_contact_item\">\n <div class=\"container_contact_item_left\">\n <div class=\"container_contact_item_left_inner\">\n <div class=\"contact_field\">{{ directoryContact.name }}</div>\n <div class=\"contact_field\">{{ directoryContact.phone }}</div>\n </div>\n </div>\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">\n <div class=\"container_rating\">\n @for (star of [1,2,3,4,5]; track $index) {\n <span [class.filled]=\"star <= directoryContact.rating\">\u2605</span>\n }\n </div>\n </span>\n </div>\n </div>\n </div>\n }\n </div>\n </div>\n </div>\n </div>\n }\n\n <!-- Dialpad -->\n <div class=\"container_dialpad_digits\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('1')\">1</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('2')\">2</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('3')\">3</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('4')\">4</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('5')\">5</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('6')\">6</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('7')\">7</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('8')\">8</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('9')\">9</button>\n </div>\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('*')\">\n <svg viewBox=\"0 0 512 512\" height=\"17px\">\n <path fill=\"currentColor\" d=\"M208 32c0-17.7 14.3-32 32-32l32 0c17.7 0 32 14.3 32 32l0 140.9\n 122-70.4c15.3-8.8 34.9-3.6 43.7 11.7l16 27.7c8.8 15.3 3.6 34.9-11.7\n 43.7L352 256l122 70.4c15.3 8.8 20.6 28.4 11.7 43.7l-16 27.7c-8.8\n 15.3-28.4 20.6-43.7 11.7L304 339.1 304 480c0 17.7-14.3 32-32 32l-32\n 0c-17.7 0-32-14.3-32-32l0-140.9L86 409.6c-15.3 8.8-34.9\n 3.6-43.7-11.7l-16-27.7c-8.8-15.3-3.6-34.9 11.7-43.7L160 256 38\n 185.6c-15.3-8.8-20.5-28.4-11.7-43.7l16-27.7C51.1 98.8 70.7 93.6 86\n 102.4l122 70.4L208 32z\" />\n </svg>\n </button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('0')\">0</button>\n <button class=\"button_dialpad_digit\" (click)=\"onDigitClick('#')\">\n <svg height=\"19px\" viewBox=\"0 0 448 512\">\n <path fill=\"currentColor\" d=\"M181.3 32.4c17.4 2.9 29.2 19.4 26.3\n36.8L197.8 128l95.1 0 11.5-69.3c2.9-17.4 19.4-29.2 36.8-26.3s29.2 19.4\n26.3 36.8L357.8 128l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9\n0L325.8 320l58.2 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-68.9 0-11.5\n69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8l9.8-58.7-95.1\n0-11.5 69.3c-2.9 17.4-19.4 29.2-36.8 26.3s-29.2-19.4-26.3-36.8L90.2\n384 32 384c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 21.3-128L64\n192c-17.7 0-32-14.3-32-32s14.3-32 32-32l68.9 0 11.5-69.3c2.9-17.4\n19.4-29.2 36.8-26.3zM187.1 192L165.8 320l95.1 0 21.3-128-95.1 0z\" />\n </svg></button>\n </div>\n </div>\n\n <div class=\"container_dialpad_buttons\">\n <div class=\"container_dialpad_row\">\n @if (isAddingNewCall) {\n <button class=\"button_dialpad_digit button_call_green\"\n (click)=\"makeCall(transferForm.value.target)\"\n [disabled]=\"!phoneNumber\"\n [ngClass]=\"{ 'button_call_disabled': !phoneNumber }\">\n <svg viewBox=\"0 0 512 512\" height=\"18px\">\n <path fill=\"currentColor\"\n d=\"M164.9 24.6c-7.7-18.6-28-28.5-47.4-23.2l-88 24C12.1 30.2 0 46 0 64C0 311.4 200.6 512 448 512c18 0 33.8-12.1 38.6-29.5l24-88c5.3-19.4-4.6-39.7-23.2-47.4l-96-40c-16.3-6.8-35.2-2.1-46.3 11.6L304.7 368C234.3 334.7 177.3 277.7 144 207.3L193.3 167c13.7-11.2 18.4-30 11.6-46.3l-40-96z\" />\n </svg>\n </button>\n } @else {\n <button type=\"submit\" class=\"button_dialpad_digit button_call_green\" (click)=\"onTransferCall()\"> <svg height=\"20px\" viewBox=\"0 0 52 52\" enable-background=\"new 0 0 52 52\" xml:space=\"preserve\">\n <g>\n <path fill=\"currentColor\" d=\"M48.5,37.9L42.4,33c-1.4-1.1-3.4-1.2-4.8-0.1l-5.2,3.8c-0.6,0.5-1.5,0.4-2.1-0.2l-7.8-7l-7-7.8\n c-0.6-0.6-0.6-1.4-0.2-2.1l3.8-5.2c1.1-1.4,1-3.4-0.1-4.8l-4.9-6.1c-1.5-1.8-4.2-2-5.9-0.3L3,8.4c-0.8,0.8-1.2,1.9-1.2,3\n c0.5,10.2,5.1,19.9,11.9,26.7S30.2,49.5,40.4,50c1.1,0.1,2.2-0.4,3-1.2l5.2-5.2C50.5,42.1,50.4,39.3,48.5,37.9z\" />\n <path fill=\"currentColor\" d=\"M48.4,2H33c-1,0-1.3,1.1-0.5,1.9l4.9,5l-9,9.1c-0.5,0.5-0.5,1.4,0,1.9l3.7,3.7c0.5,0.5,1.3,0.5,1.9,0\n l9.1-9.1l5.1,4.9C48.9,20.3,50,20,50,19V3.7C50,3,49.1,2,48.4,2z\" />\n </g>\n </svg></button>\n }\n </div>\n </div>\n </div>\n </form>\n </div>\n }\n }\n</div>\n\n <!-- Second Div -->\n @if (currentView === 'transferDetails') {\n <div class=\"container_transfer_call_content\" @slideInOut>\n <div class=\"container_active_call_duration\">00:10</div>\n <div class=\"container_active_call_header shadow_st\">\n <div class=\"container_active_call_avatar\">\n <div class=\"container_active_call_avatar_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"45px\" width=\"45px\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"></path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 ...\"></path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 ...\"></path>\n </g>\n </svg>\n </div>\n </div>\n\n <div class=\"container_active_call_info\">\n <div class=\"container_active_call_info_inner\">\n <div class=\"container_active_call_number\">\n {{ lookupPhoneNumber?.country?.flag }} {{ lookupPhoneNumber?.phoneNumber }}\n </div>\n <div class=\"container_active_call_details\">\n <div class=\"container_active_call_details_inner\">\n <div class=\"container_active_call_icon_grey\">\n <svg viewBox=\"0 0 320 512\" width=\"12\">\n <path fill=\"currentColor\"\n d=\"M128 64L0 64 0 448l128 0 0-384zm192 0L192 64l0 384 128 0 0-384z\"></path>\n </svg>\n </div>\n <div class=\"container_active_call_status_grey\">On hold</div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"container_contact_item_right\">\n <div class=\"right_field right_field_highlighted\">\n <span class=\"right_field_content\">Ext. {{ extensionNumber }}</span>\n </div>\n </div>\n </div>\n\n <div class=\"container_transfer_call_details\">\n <div class=\"container_call_ani_number padding_top_nt\">+91 (654) 9876549</div>\n <div class=\"container_call_ani_name\">Rajeev kumar singh</div>\n <div class=\"container_call_ani_image\">\n <div class=\"container_call_ani_image_inner\">\n <svg viewBox=\"0 0 212 212\" height=\"85\" width=\"85\" version=\"1.1\" x=\"0px\" y=\"0px\"\n enable-background=\"new 0 0 212 212\">\n <path fill=\"#DFE5E7\"\n d=\"M106.251,0.5C164.653,0.5,212,47.846,212,106.25S164.653,212,106.25,212C47.846,212,0.5,164.654,0.5,106.25 S47.846,0.5,106.251,0.5z\"></path>\n <g>\n <path fill=\"#FFFFFF\"\n d=\"M173.561,171.615c-0.601-0.915-1.287-1.907-2.065-2.955c-0.777-1.049-1.645-2.155-2.608-3.299 ...\"></path>\n <path fill=\"#FFFFFF\"\n d=\"M106.002,125.5c2.645,0,5.212-0.253,7.68-0.737c1.234-0.242,2.443-0.542,3.624-0.896 ...\"></path>\n </g>\n </svg>\n </div>\n </div>\n </div>\n\n <div class=\"container_call_dnis\">\n <span>Ext. {{ extensionNumber }}</span>\n </div>\n\n <div class=\"container_call_options margin_bottom_nt\">\n <div class=\"container_dialpad_row margin_bottom_16\">\n <button class=\"button_call_option\" (click)=\"mergeCall()\">\n <div class=\"button_icon\"><svg width=\"22\" viewBox=\"0 0 512 512\">\n <path fill=\"currentColor\"\n d=\"M32 64C14.3 64 0 78.3 0 96s14.3 32 32 32l97.2 0c9.7 0 18.9 4.4 25 12L247 256 154.2 372c-6.1 7.6-15.3 12-25 12L32 384c-17.7 0-32 14.3-32 32s14.3 32 32 32l97.2 0c29.2 0 56.7-13.3 75-36l99.2-124 80.6 0 0 32c0 12.9 7.8 24.6 19.8 29.6s25.7 2.2 34.9-6.9l64-64c6-6 9.4-14.1 9.4-22.6s-3.4-16.6-9.4-22.6l-64-64c-9.2-9.2-22.9-11.9-34.9-6.9s-19.8 16.6-19.8 29.6l0 32-80.6 0L204.2 100c-18.2-22.8-45.8-36-75-36L32 64z\"></path>\n </svg></div>\n <div class=\"button_text\">Merge</div>\n </button>\n\n <button class=\"button_call_option\">\n <div class=\"button_icon\"> <svg width=\"26px\" viewBox=\"0 0 640 512\">\n <path fill=\"currentColor\"\n d=\"M150.6 73.4c-12.5-12.5-32.8-12.5-45.3 0l-96 96c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L96 173.3 96 320c0 53 43 96 96 96l112 0c17.7 0 32-14.3 32-32s-14.3-32-32-32l-112 0c-17.7 0-32-14.3-32-32l0-146.7 41.4 41.4c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3l-96-96zM336 96c-17.7 0-32 14.3-32 32s14.3 32 32 32l112 0c17.7 0 32-14.3 32-32l0 146.7-41.4-41.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l96 96c12.5 12.5 32.8 12.5 45.3 0l96-96c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L544 338.7 544 192c0-53-43-96-96-96L336 96z\"></path>\n </svg></div>\n <div class=\"button_text\">Swap</div>\n </button>\n\n <button class=\"button_call_option\">\n <div class=\"button_icon\"><svg width=\"19px\" viewBox=\"0 0 52 52\">\n <path fill=\"currentColor\"\n d=\"M48.5,37.9L42.4,33c-1.4-1.1-3.4-1.2-4.8-0.1l-5.2,3.8c-0.6,0.5-1.5,0.4-2.1-0.2l-7.8-7l-7-7.8 c-0.6-0.6-0.6-1.4-0.2-2.1l3.8-5.2c1.1-1.4,1-3.4-0.1-4.8l-4.9-6.1c-1.5-1.8-4.2-2-5.9-0.3L3,8.4c-0.8,0.8-1.2,1.9-1.2,3 c0.5,10.2,5.1,19.9,11.9,26.7S30.2,49.5,40.4,50c1.1,0.1,2.2-0.4,3-1.2l5.2-5.2C50.5,42.1,50.4,39.3,48.5,37.9z\"></path>\n <path fill=\"currentColor\"\n d=\"M48.4,2H33c-1,0-1.3,1.1-0.5,1.9l4.9,5l-9,9.1c-0.5,0.5-0.5,1.4,0,1.9l3.7,3.7c0.5,0.5,1.3,0.5,1.9,0 l9.1-9.1l5.1,4.9C48.9,20.3,50,20,50,19V3.7C50,3,49.1,2,48.4,2z\"></path>\n </svg>\n</div>\n <div class=\"button_text\">Transfer</div>\n </button>\n </div>\n </div>\n\n <div class=\"container_transfer_disconnect\">\n <div class=\"container_dialpad_row\">\n <button class=\"button_dialpad_digit button_call_red\" (click)=\"goToDiv('transferListpage')\">\n <svg width=\"36px\" viewBox=\"0 0 76 76\">\n <path fill=\"currentColor\"\n d=\"M 16.2013,39.582L 15.4175,39.4096L 14.8094,38.9315L 14.4029,38.2193L 14.25,37.364L 14.25,34.9055C 14.25,33.184 14.6023,31.6057 15.3069,30.1704C 16.0116,28.7352 16.9698,27.4788 18.1818,26.4013C 19.3937,25.3237 20.8095,24.4424 22.429,23.7573C 24.0486,23.0722 25.7732,22.6277 27.6031,22.4239L 31.304,22.0792L 34.9918,21.8256L 39.109,21.715L 42.966,21.8451L 46.101,22.0987L 48.9628,22.489C 50.7276,22.7318 52.3883,23.199 53.945,23.8906C 55.5017,24.5822 56.8567,25.4625 58.0101,26.5313C 59.1635,27.6002 60.0752,28.8468 60.7451,30.2712C 61.4151,31.6956 61.75,33.2599 61.75,34.964L 61.75,37.4486L 61.5874,38.2681L 61.1386,38.9511L 60.4785,39.4096L 59.6752,39.582L 51.8897,39.582L 51.0994,39.4096L 50.4588,38.9413L 50.0327,38.2584L 49.8799,37.4226L 49.8799,32.1217L 49.7075,31.1656L 49.2295,30.3818L 48.5075,29.8452C 48.23,29.7108 47.9286,29.6436 47.6034,29.6436L 28.7413,29.6436C 28.4161,29.6436 28.0974,29.7108 27.7852,29.8452L 26.9722,30.3818L 26.4128,31.1851C 26.2697,31.4973 26.1982,31.8225 26.1982,32.1607L 26.1982,37.364L 26.0356,38.2193L 25.5803,38.9315L 24.9104,39.4096L 24.1038,39.582L 16.2013,39.582 Z M 34.8333,41.1667L 41.1666,41.1667L 41.1667,52.6458L 45.9167,47.8958L 45.9167,54.2292L 38,62.5417L 30.0833,54.2292L 30.0833,47.8958L 34.8333,52.6458L 34.8333,41.1667 Z \" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n}\n\n\n\n\n\n\n\n\n\n\n\n </div>\n </div>\n </ng-template>\n\n \n\n\n</ng-template>\n\n<div id=\"videoPanel\" style=\"display: none\">\n <audio #rmtAudio id=\"rmtAudio\" autoplay></audio>\n <audio #lclAudio id=\"lclAudio\" autoplay muted></audio>\n <video #rmtVideo id=\"rmtVideo\" autoplay></video>\n <video #lclVideo id=\"lclVideo\" autoplay muted controls title=\"Local Video\"></video>\n</div>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Open+Sans:300,400,500,600,700,800\";@import\"https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css\";:host{--sd-app-color: #ff6633;--sd-app-color-dark: #ff5805;--sd-app-color-light: #fe8c66;--sd-text-font-family: \"Open Sans\", \"sans-serif\", \"Helvetica Neue\", \"Helvetica\", \"Arial\";--sd-text-color: #666;--sd-text-color-inverse: #fff;--sd-text-color-light: #ccc;--sd-text-color-medium: #999;--sd-text-color-dark: #333;--sd-background-color: #fff;--sd-border-color-light: #e3e3e3;--sd-border-color-dark: #c9c9c9}.message_label{width:100%;font-size:13px;font-family:var(--sd-text-font-family);letter-spacing:-.25px;border-radius:4px;font-weight:600;border:1px solid #f5c6cb;padding:8px 11px;position:relative}.error_message{color:#e1143c;background-color:#f8d7da;border-color:#dfadb2}.success_message{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert_message{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.properties_form_field{width:100%;font-size:13px!important;font-family:var(--sd-text-font-family)!important;font-weight:600!important;letter-spacing:-.24px!important;color:#555!important}.container_avaya_login{width:100%;height:calc(100vh - 51px);padding:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.container_avaya_login_inner{margin:0 auto;width:100%;max-width:370px;padding:65px 20px 20px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;display:flex}.container_connection_status_overlay{width:100%;height:100%;position:absolute;left:0;right:0;bottom:0;background-color:#6669;display:flex;justify-content:center;z-index:9999999;padding:20px}.container_connection_status{width:100%;background:var(--sd-background-color);padding:0 10px;border-radius:8px;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);display:flex;height:42px;align-items:center;position:relative;box-shadow:#32325d40 0 6px 12px -2px,#0000004d 0 3px 7px -3px}.container_connection_status span{padding:0 0 0 8px}.container_connection_status .fa-circle-dot{color:#f66060}.container_connection_status_loader{position:absolute;right:15px}.container_avaya_login_content{position:relative;width:100%}.container_avaya_logo{margin:0;padding:0 26px 24px;width:100%;text-align:center}.container_avaya_logo_image{margin:0 auto;width:120px;height:35px;background:var(--sd-background-color) url() center no-repeat;background-size:120px}.container_avaya_logo h3{margin:0;padding:6px 0 0;font-family:var(--sd-text-font-family);font-size:18px;font-weight:700;letter-spacing:-.45px;color:var(--sd-text-color-dark);position:relative}.container_avaya_logo h3 span{font-size:8px;position:absolute;top:8px}.container_avaya_login_error{width:100%;white-space:normal}.container_avaya_login_form{width:100%;margin:0;padding:26px 0 0;position:relative}.container_avaya_login_form form{width:100%;margin:0;padding:0}.container_avaya_login_form .form_field{margin:0;padding:0;width:100%}.avaya_login_button_area{padding:10px 0 0}.container_icon_eye{position:absolute;right:15px}.container_avaya_login_form .button_submit_login{font:14px / 48px var(--sd-text-font-family);display:inline-block;height:48px;background:#cd1f1e;color:var(--sd-text-color-inverse);border:1px solid #cd1f1e;font-weight:600;padding:0 12px;border-radius:4px;outline:none;letter-spacing:-.24px;cursor:pointer;width:100%;position:relative}.container_avaya_login_form .button_submit_login_loader{background:#cd1f1e!important;color:var(--sd-text-color-inverse)!important;border:1px solid #cd1f1e!important;cursor:auto}.container_avaya_login_form .button_submit_login:disabled{background:#f1f1f1;color:var(--sd-text-color-medium);border:1px solid #f1f1f1}.button_submit_loader:after{content:\"\";position:absolute;width:16px;height:16px;inset:0;margin:auto;border:3px solid transparent;border-top-color:#fff;border-radius:50%;animation:button_submit_loader_spinner 1s ease infinite}@keyframes button_submit_loader_spinner{0%{transform:rotate(0)}to{transform:rotate(1turn)}}.container_avaya_softphone{width:100%;height:100%;position:relative}.container_avaya_softphone_inner{width:100%;position:absolute;top:0;left:0;right:0;padding:0;height:calc(100% - 51px);font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);line-height:normal;background:var(--sd-background-color);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background:#000}.container_call{margin:0 auto;padding:0;position:relative;width:100%;height:100%;background:var(--sd-background-color)}.container_dialpad_header{width:100%;position:relative;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99999;background:var(--sd-background-color);padding:15px 15px 0}.container_dialpad_header_inner2{width:100%;margin:0;padding:15px 15px 0;background:#f7f7f7;background:linear-gradient(180deg,#f7f7f7 72%,#fff);border-radius:12px 12px 0 0;position:relative}.container_avaya_extension{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:8px 10px;border:1px solid var(--sd-border-color-light);border-radius:12px;margin-bottom:15px;background:#f0f0f0}.container_avaya_extension_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center}.container_avaya_extension_left .extension_label{display:flex;align-items:center;line-height:normal;text-align:left}.container_avaya_extension_left .extension_number{display:flex;align-items:center;line-height:20px}.container_avaya_extension_right{flex-basis:auto;flex-direction:column;flex-grow:1;text-align:right;justify-items:flex-end}.container_avaya_extension_right button{background:none;border:none;outline:none;color:#a6a6a6;font-weight:600;border-radius:50%;align-items:center;justify-content:center;-webkit-transition-duration:.3s;transition-duration:.3s;background:#f0f0f0;height:38px;width:38px}.container_avaya_extension_right button:hover{color:var(--sd-text-color-dark);background:var(--sd-border-color-light)}.dialpad_container_sd{position:absolute;width:100%;right:0;left:0;top:101px;height:calc(100vh - 101px);-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0}.scrollbox{height:100%;overflow-y:scroll;overflow-x:hidden;visibility:hidden;padding:0 0 7px;position:relative;scroll-behavior:smooth;scrollbar-width:thin;width:100%}.scrollbox-content,.scrollbox:hover,.scrollbox:focus{visibility:visible}.container_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:8px;overflow:hidden;margin-bottom:10px;cursor:pointer}.container_contact_item_image{display:flex}.container_contact_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative}.container_contact_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_contact_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_contact_item_left .contact_field{font-size:13px;margin-bottom:2px}.contact_dark{color:var(--sd-text-color-dark)}.contact_gray{color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_grey{font-size:12px;color:var(--sd-text-color);padding-top:2px;display:flex}.container_contact_item_left .contact_field_green{font-size:13px;margin-bottom:2px;color:#6c0}.container_contact_item_left .contact_field_highlighted{display:flex;font-size:11px;color:var(--sd-text-color-medium)}.container_contact_item_left .contact_field_highlighted span{background:#6c0;border-radius:3px;text-transform:uppercase;color:var(--sd-text-color-inverse);font-size:9px;padding:2px 4px}.container_contact_item_right{display:flex;flex-direction:row}.container_contact_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.container_contact_item_right .right_field_highlighted{font-size:16px;color:#29abe0}.container_contact_item_right .right_field_content{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 8px;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .right_recent_content{display:inline-block;padding:0;font-size:11px;color:var(--sd-text-color)}.container_contact_item_right .container_rating{font-size:11px;cursor:pointer}.container_contact_item_right .container_rating span{color:gray;transition:color .3s}.container_contact_item_right .container_rating span.filled{color:gold}.container_dialpad_digits{width:100%;display:flex;justify-content:center;margin-bottom:10px;flex-direction:column}.container_dialpad_row{display:flex;column-gap:10px;flex-direction:row;flex-shrink:0;justify-content:center;margin-bottom:10px}.button_dialpad_digit{display:block;width:50px;height:50px;display:flex;column-gap:20px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;font-size:22px;font-weight:700;color:var(--sd-text-color-dark);background:var(--sd-background-color);-webkit-transition-duration:.3s;transition-duration:.3s;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.button_dialpad_digit:hover{background:#000;color:var(--sd-text-color-inverse)}.button_call_green{background:#6c0;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.button_call_green:hover{background:#5ab400;color:var(--sd-text-color-inverse);width:60px;height:60px}.button_call_red{background:#e0261f;color:var(--sd-text-color-inverse);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s;z-index:9999}.button_call_red:hover{background:#cf211a}.button_call_disabled{background:#eee;color:var(--sd-text-color-light);width:60px;height:60px;-webkit-transition-duration:.3s;transition-duration:.3s}.container_dialpad_buttons{width:100%;display:flex;justify-content:center;padding-top:0;margin-bottom:6px}.container_call_header{width:100%;position:absolute;top:0;left:0;right:0;height:221px;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);z-index:99;background:var(--sd-background-color)}.container_call_header_inner{width:100%;margin:0;padding:10px 15px 15px;position:relative}.container_call_info{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:var(--sd-text-color)}.container_call_info_blinker{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:red;animation:blinker 1s linear infinite}@keyframes blinker{26%{opacity:0}}.container_call_ani{width:100%;margin:0;padding:0}.container_call_ani_number{text-align:center;font-size:20px;font-weight:600;padding-top:26px}.container_call_ani_name{text-align:center;font-size:13px;font-weight:600;padding-top:2px}.container_call_ani_image{margin:0;padding:24px 0 0;width:100%}.container_call_ani_image_inner{display:flex;align-items:center;flex:0 0 auto;padding:0;width:85px;height:85px;position:relative;margin:0 auto;border-radius:100%}.container_call_content{width:100%;position:absolute;top:221px;left:0;right:0;height:calc(100% - 221px);font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);padding-top:20px;display:flex;flex-basis:auto;flex-direction:column;align-items:center;justify-content:center}.container_call_dnis{width:100%;text-align:center;padding-bottom:16px}.container_call_dnis span{display:inline-block;border-radius:26px;border:1px solid #e5e5e5;padding:5px 16px}.container_call_options{width:100%;display:flex;justify-content:center;margin-bottom:16px;flex-direction:column}.button_call_option{display:block;width:80px;height:60px;display:flex;flex-direction:column;flex-shrink:0;column-gap:0px;margin:0 10px;align-items:center;justify-content:center;line-height:normal;outline:none;border:none;font-size:12px;font-weight:600;color:var(--sd-text-color-dark);background:none;-webkit-transition-duration:.3s;transition-duration:.3s;position:relative}.button_call_option .button_icon{display:flex;justify-content:center;align-items:center;height:40px}.button_call_option .button_text{display:flex;justify-content:center;color:var(--sd-text-color)}.button_active{color:#67cc02}.button_elevated{position:relative;z-index:9999}.button_disabled{color:gray;cursor:not-allowed;pointer-events:none;opacity:.6}.button_call_option[disabled],.button_disabled{opacity:.5;pointer-events:none;display:inline-flex}.container_popup_dialpad{position:absolute;z-index:999;left:0;right:0;bottom:0}.container_popup_dialpad_inner{background:#f1f1f1;background:linear-gradient(180deg,#f1f1f1 44%,#fff);margin:0 auto;padding:10px 15px 20px;border-radius:8px;width:88%;position:relative;height:475px}.container_popup_dialpad .button_close{position:absolute;right:-15px;top:-17px;z-index:9}.container_popup_dialpad .button_close span{width:32px;height:32px;background:var(--sd-background-color);display:flex;align-items:center;justify-content:center;border-radius:50%;cursor:pointer;box-shadow:#0000001a 0 0 5px,#0000001a 0 0 1px;border:1px solid #ebebeb}.container_popup_dialpad .button_close span:hover{background:#f1f1f1;color:var(--sd-text-color-dark)}.container_popup_dialpad .button_close span .fa-xmark{font-size:1.4em;font-weight:400}.container_dtmf_input{width:100%;margin:0;padding:0 0 10px;position:relative}.container_dtmf_input input{width:100%;display:flex;height:38px;outline:none;text-align:center;align-items:center;border:1px solid #f1f1f1;font-size:26px;font-weight:400;color:transparent;text-shadow:0 0 0 #000;padding:0;background:#f1f1f1}.container_active_call_header{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:10px;border:1px solid var(--sd-border-color-light);border-radius:12px;overflow:hidden}.container_active_call_avatar{display:flex}@keyframes play1{0%{transform:scale(1)}15%{box-shadow:0 0 0 2px #57b84666}25%{box-shadow:0 0 0 3px #98d48266,0 0 0 20px #fff3}25%{box-shadow:0 0 0 5px #bde4a966,0 0 0 30px #fff3}}.container_active_call_avatar_inner{margin:0 auto;padding:0;width:50px;height:50px;background:#dfe6e8;color:var(--sd-text-color-dark);position:relative;animation:play1 2s ease infinite;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;backface-visibility:hidden;border-radius:100%;overflow:hidden;display:flex;justify-content:center;align-items:center}.container_active_call_info{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1}.container_active_call_info_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_active_call_duration{display:flex;justify-content:center;font-size:14px;font-weight:600;width:100%;color:var(--sd-text-color);margin-bottom:10px}.container_active_call_number{font-size:13px;margin-bottom:2px}.container_active_call_details{font-size:13px;color:#6c0;padding-top:2px}.container_active_call_details_inner{display:flex;margin:0;padding:0;justify-content:left}.container_active_call_icon_grey{display:flex;width:18px;height:16px;justify-content:center;align-items:center;color:var(--sd-text-color-medium)}.container_active_call_icon_green{display:flex;width:18px;height:16px;justify-content:center;align-items:center;color:#6c0}.container_active_call_status_green{display:flex;justify-content:center;align-items:center;color:#6c0;padding-left:6px;font-size:12px}.container_active_call_status_grey{display:flex;justify-content:center;align-items:center;color:var(--sd-text-color-medium);padding-left:6px;font-size:12px}.container_active_call_options{display:flex;flex-direction:row}.container_active_call_options .button_active_call_options{display:flex;flex-basis:auto;justify-content:right}.container_active_call_options .button_active_call_options_highlighted{font-size:16px;color:#29abe0}.container_active_call_options_button{display:flex;border-radius:26px;border:none;padding:0;font-size:11px;color:var(--sd-text-color);width:32px;height:32px;justify-content:center;align-items:center}.container_active_call_options_button:hover{background:#f1f1f1}.container_conference_contact_list{position:relative;width:100%;background:#f7f7f7;border-top:1px solid #eee;border-bottom:1px solid #eee;height:149px;padding:0;margin-top:16px}.container_swap_merge_contact_list{position:relative;width:100%;background:var(--sd-background-color);height:149px;padding:0;margin-top:32px}.container_conference_contact_list_inner{margin:0;padding:0;position:relative;width:100%;height:100%}.container_conference_contact_list_scrollbox{padding:12px 15px}.container_conference_contact_item{width:100%;position:relative;display:flex;flex-direction:row;align-items:center;padding:0 10px;overflow:hidden;cursor:pointer}.button_disconnect_call{display:block;width:42px;height:42px;display:flex;margin:0;align-items:center;justify-content:center;line-height:normal;border-radius:50%;outline:none;border:none;color:#e0261f;background:#f7f7f7;-webkit-transition-duration:.3s;transition-duration:.3s}.button_disconnect_call:hover{color:var(--sd-text-color-inverse);background:#e0261f;-webkit-transition-duration:.3s;transition-duration:.3s}.container_separator{height:1px;width:100%;background:#ededed;margin:11px 0}.container_transfer_call{position:absolute;z-index:99999;left:0;right:0;bottom:-51px;width:100%;background:var(--sd-background-color)}.container_transfer_call_inner{background:var(--sd-background-color);box-shadow:#11111a1a 0 4px 16px,#11111a0d 0 8px 32px;padding:20px 15px 10px;border-radius:0;width:100%;position:relative;height:596px;overflow:hidden}.container_transfer_call_inner h2{font-family:var(--sd-text-font-family);font-size:20px;font-weight:700;color:var(--sd-text-color-dark);letter-spacing:-1px;padding:0;line-height:normal;text-transform:capitalize;margin-bottom:15px}.container_button_close{position:absolute;right:4px;top:4px}.container_button_close span{width:36px;height:36px;background:var(--sd-background-color);display:flex;align-items:center;justify-content:center;border-radius:50%;cursor:pointer}.container_button_close span:hover{background:var(--sd-background-color)}.container_button_close .fa-xmark{font-size:1.6em;font-weight:400;color:var(--sd-text-color-light)}.container_button_close span:hover .fa-xmark:hover{color:var(--sd-text-color-dark)}.container_transfer_call_content{margin:0;padding:5px 0 0;width:100%;background:var(--sd-background-color);min-height:485px}.container_transfer_call_content h3{font-family:var(--sd-text-font-family);font-size:14px;font-weight:700;color:var(--sd-text-color);letter-spacing:-.26px;padding:8px 0 0;line-height:normal;text-transform:capitalize;margin-bottom:15px}.container_transfer_call_content h3 span{color:var(--sd-text-color-dark)}.container_transfer_call_details{width:100%;margin:0;padding:10px 15px 69px;position:relative}.container_transfer_tab_buttons{display:flex;gap:10px;align-items:center;margin-bottom:15px}.button_transfer_tab{padding:10px;border:none;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color);background-color:var(--sd-border-color-light);cursor:pointer;width:105px;border-radius:28px;outline:none}.button_transfer_tab.active{background-color:var(--sd-app-color);color:#fff}.container_transfer_tab_content{margin:0;padding:0;width:100%}.container_transfer_internal_search{margin:0 0 15px;padding:0;width:100%}.container_transfer_internal_search input{padding:0 16px;width:100%;outline:none;display:block;border:1px solid var(--sd-border-color-dark);color:var(--sd-text-color-medium);font-size:12px;letter-spacing:-.25px;font-family:var(--sd-text-font-family);background:var(--sd-background-color);font-style:italic;font-weight:400;position:relative;border-radius:27px;height:48px}.container_transfer_contact_list{width:100%;margin:0;padding:0;position:relative;height:336px}.container_transfer_contact_list_inner{width:100%;position:relative;height:100%}.container_transfer_internal_contact_list{position:absolute;width:100%;left:0;right:0;background:var(--sd-background-color);height:332px}.container_transfer_disconnect{width:100%;display:flex;justify-content:center;padding-top:3px;margin-bottom:6px}.container_avaya_footer{width:100%;padding:0;background:#f7f7f7;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;color:var(--sd-text-color-medium);position:relative;display:flex;height:51px;justify-content:center;align-items:center;border-top:1px solid var(--sd-border-color-light);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;z-index:999;position:absolute;bottom:0;left:0}.margin_bottom_16{margin-bottom:16px}.margin_bottom_nt{margin-bottom:10px}.padding_top_nt{padding-top:7px}.shadow_st{box-shadow:#00000026 0 2px 8px}.call_animation{animation:play 2s ease infinite;-webkit-backface-visibility:hidden;-moz-backface-visibility:hidden;-ms-backface-visibility:hidden;backface-visibility:hidden}.mini-call-switcher{display:flex;justify-content:center;margin-top:10px}.mini-call{padding:6px 12px;margin:0 6px;background:#eaeaea;border-radius:8px;cursor:pointer;font-size:14px}.mini-call.active{background-color:#c6f6d5;font-weight:700}.history-backdrop{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;z-index:999}.history-popup{position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);background:var(--sd-background-color);padding:20px;z-index:1000;box-shadow:0 2px 10px #0000004d;border-radius:5px;width:600px;max-height:80vh;overflow-y:auto}.history-content{position:relative}.history-content h3{margin-top:0}.close-btn{background:transparent;border:none;font-size:1.5rem;position:absolute;top:0;right:0;cursor:pointer}.history-content table{width:100%;border-collapse:collapse;margin-top:1rem}.history-content th,.history-content td{padding:.5rem;border:1px solid var(--sd-text-color-light)}@keyframes play{0%{transform:scale(1)}15%{box-shadow:0 0 0 4px #57b84680}25%{box-shadow:0 0 0 5px #98d48266,0 0 0 20px #fff3}25%{box-shadow:0 0 0 10px #c2e5b180,0 0 0 30px #fff3}}::-webkit-input-placeholder{color:#ddd}::-moz-placeholder{color:var(--sd-text-color-dark)}:-ms-input-placeholder{color:var(--sd-text-color-dark)}:-moz-placeholder{color:var(--sd-text-color-dark)}.slide_st{left:-100px;-webkit-animation:slide .5s forwards;-webkit-animation-delay:2s;animation:slide .5s forwards;animation-delay:0s}@-webkit-keyframes slide{to{left:0}}@keyframes slide{to{left:0}}.tabs_sd{width:100%;position:absolute;bottom:0;left:0;right:0;display:flex;height:51px;border-top:1px solid var(--sd-border-color-light);background:#f9f9f9;z-index:999999}.tab_heading_btn{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;align-items:center;padding:0;border:none;cursor:pointer;width:25%;font-family:var(--sd-text-font-family);font-size:11px;font-weight:600;line-height:normal;color:var(--sd-text-color-medium);background:none;transition:background-color .3s linear;-webkit-transition:background-color .3s linear;-o-transition:background-color .3s linear;-ms-transition:background-color .3s linear}.nav_icon{display:flex;align-items:center;justify-content:center;height:22px;margin-bottom:1px}.nav_heading_sd{display:flex;align-items:center}.tab_heading_btn.active{color:var(--sd-app-color);background:url() top center no-repeat}.tab-content_sd{width:100%;height:100%;font-family:var(--sd-text-font-family);font-size:13px;font-weight:600;line-height:normal;background:var(--sd-background-color);-webkit-touch-callout:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;position:relative}.container_transfer_fav_list{width:100%;margin:0;padding:0;position:relative;height:376px}.outer_container_swap_merge{width:100%;margin:0;padding:0}.container_swap_merge_item{width:100%;position:relative;display:flex;flex-direction:row;height:72px;border-radius:8px;overflow:hidden}.container_recent_item_image{display:flex;align-items:center}.container_swap_merge_item_image_inner{margin:0 auto;padding:0;width:50px;height:50px;position:relative;border-radius:50%;display:flex;align-items:center;justify-content:center;font-family:var(--sd-text-font-family);font-size:26px;color:var(--sd-text-color-inverse);font-weight:400}.container_recent_item_left{display:flex;flex-basis:auto;flex-direction:column;flex-grow:1;justify-content:center;min-width:0;position:relative;border-top:1px solid #e9edef}.container_recent_item_left_inner{display:flex;flex-basis:1;flex-direction:column;padding:0 0 0 10px}.container_recent_item_left .contact_field{font-size:13px;color:var(--sd-text-color-dark);margin-bottom:2px}.onhold_st{font-family:var(--sd-text-font-family);font-size:12px;color:var(--sd-text-color-medium);font-weight:600}.container_recent_item_right{display:flex;flex-direction:row;position:absolute;right:0}.container_recent_item_right .right_field{display:flex;flex-basis:auto;justify-content:right}.button_swap_merge{display:block;width:32px;height:45px;display:flex;flex-direction:column;flex-shrink:0;column-gap:0px;margin:0 0 0 28px;align-items:center;justify-content:center;line-height:normal;outline:none;border:none;font-size:12px;font-weight:600;color:var(--sd-text-color-dark);background:none;-webkit-transition-duration:.3s;transition-duration:.3s;position:relative;pointer-events:all;cursor:pointer}.button_swap_merge .button_icon{display:flex;justify-content:center;align-items:center;height:40px}.button_swap_merge .button_text{display:flex;justify-content:center;color:var(--sd-text-color-dark)}@media only screen and (max-width: 767px){.previewcontent{width:95%!important}.bgSecWrap .numberArrow{left:0!important;top:-20px!important}.transcription-popup{position:fixed;bottom:20px;right:20px;width:320px;max-height:250px;background:#fff;border-radius:10px;box-shadow:0 4px 12px #0003;padding:12px;z-index:1000;display:flex;flex-direction:column}.transcription-header{display:flex;justify-content:space-between;align-items:center;font-weight:700;font-size:14px;margin-bottom:8px}.transcription-body{overflow-y:auto;font-size:13px;line-height:1.4;color:#333}.close-btn{border:none;background:transparent;font-size:18px;cursor:pointer}}\n"] }]
|
|
2818
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i0.ChangeDetectorRef }, { type: i0.NgZone }, { type: i1$1.FormBuilder }, { type: CountryService }, { type: PhoneNumberLookupService }, { type: AvayaIPOService }, { type: RecordingManagerService }, { type: CallSocketService }, { type: i8.SnugdeskAuthenticationService }, { type: i8.TenantService }, { type: i8.UserService }], propDecorators: { tenantId: [{
|
|
2819
|
+
type: Input,
|
|
2820
|
+
args: ['tenantId']
|
|
2821
|
+
}], userId: [{
|
|
2822
|
+
type: Input,
|
|
2823
|
+
args: ['userId']
|
|
2824
|
+
}], isVisible: [{
|
|
2825
|
+
type: Input,
|
|
2826
|
+
args: ['isVisible']
|
|
2827
|
+
}], containerHeightObservable: [{
|
|
2828
|
+
type: Input,
|
|
2829
|
+
args: ['containerHeightObservable']
|
|
2830
|
+
}], containerWidthObservable: [{
|
|
2831
|
+
type: Input,
|
|
2832
|
+
args: ['containerWidthObservable']
|
|
2833
|
+
}], notificationEvent: [{
|
|
2834
|
+
type: Output
|
|
2835
|
+
}] } });
|
|
2836
|
+
|
|
2837
|
+
class AvayaIPOWidgetModule {
|
|
2838
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
|
2839
|
+
static ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetModule, declarations: [AvayaIPOWidgetComponent,
|
|
2840
|
+
CallHistoryComponent,
|
|
2841
|
+
DialpadComponent,
|
|
2842
|
+
DeviceSelectorComponent], imports: [CommonModule,
|
|
2843
|
+
FormsModule,
|
|
2844
|
+
ReactiveFormsModule,
|
|
2845
|
+
MatFormFieldModule,
|
|
2846
|
+
MatInputModule,
|
|
2847
|
+
MatError,
|
|
2848
|
+
MatFormField,
|
|
2849
|
+
MatLabel,
|
|
2850
|
+
NgxIntlTelInputModule], exports: [AvayaIPOWidgetComponent] });
|
|
2851
|
+
static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetModule, imports: [CommonModule,
|
|
2852
|
+
FormsModule,
|
|
2853
|
+
ReactiveFormsModule,
|
|
2854
|
+
MatFormFieldModule,
|
|
2855
|
+
MatInputModule,
|
|
2856
|
+
MatFormField,
|
|
2857
|
+
NgxIntlTelInputModule] });
|
|
2858
|
+
}
|
|
2859
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.9", ngImport: i0, type: AvayaIPOWidgetModule, decorators: [{
|
|
2860
|
+
type: NgModule,
|
|
2861
|
+
args: [{
|
|
2862
|
+
declarations: [
|
|
2863
|
+
AvayaIPOWidgetComponent,
|
|
2864
|
+
CallHistoryComponent,
|
|
2865
|
+
DialpadComponent,
|
|
2866
|
+
DeviceSelectorComponent,
|
|
2867
|
+
],
|
|
2868
|
+
imports: [
|
|
2869
|
+
CommonModule,
|
|
2870
|
+
FormsModule,
|
|
2871
|
+
ReactiveFormsModule,
|
|
2872
|
+
MatFormFieldModule,
|
|
2873
|
+
MatInputModule,
|
|
2874
|
+
MatError,
|
|
2875
|
+
MatFormField,
|
|
2876
|
+
MatLabel,
|
|
2877
|
+
NgxIntlTelInputModule,
|
|
2878
|
+
],
|
|
2879
|
+
exports: [AvayaIPOWidgetComponent],
|
|
2880
|
+
}]
|
|
2881
|
+
}] });
|
|
2882
|
+
|
|
2883
|
+
/*
|
|
2884
|
+
* Public API Surface of avaya-ipo-widget
|
|
2885
|
+
*/
|
|
2886
|
+
|
|
2887
|
+
/**
|
|
2888
|
+
* Generated bundle index. Do not edit.
|
|
2889
|
+
*/
|
|
2890
|
+
|
|
2891
|
+
export { AvayaIPOWidgetComponent, AvayaIPOWidgetModule };
|
|
2892
|
+
//# sourceMappingURL=snugdesk-avaya-ipo-widget.mjs.map
|