@vgroup/dialbox 0.0.83 → 0.0.85
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/components/call-progress/incoming-call/incoming-call.component.mjs +52 -31
- package/esm2020/lib/dialbox.component.mjs +3 -33
- package/esm2020/lib/service/twilio.service.mjs +53 -83
- package/fesm2015/vgroup-dialbox.mjs +104 -142
- package/fesm2015/vgroup-dialbox.mjs.map +1 -1
- package/fesm2020/vgroup-dialbox.mjs +103 -143
- package/fesm2020/vgroup-dialbox.mjs.map +1 -1
- package/lib/components/call-progress/incoming-call/incoming-call.component.d.ts +6 -2
- package/lib/dialbox.component.d.ts +0 -2
- package/lib/service/twilio.service.d.ts +8 -7
- package/package.json +1 -1
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
import { HttpParams } from '@angular/common/http';
|
|
2
|
-
import { HttpHeaders } from "@angular/common/http";
|
|
1
|
+
import { HttpHeaders, HttpParams } from '@angular/common/http';
|
|
3
2
|
import { Injectable } from '@angular/core';
|
|
3
|
+
import { BehaviorSubject, of, throwError } from 'rxjs';
|
|
4
|
+
import { catchError, shareReplay, tap } from 'rxjs/operators';
|
|
4
5
|
import { Device } from '@twilio/voice-sdk';
|
|
5
|
-
import {
|
|
6
|
-
import { environment } from "../environments/environments";
|
|
6
|
+
import { environment } from '../environments/environments';
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
8
8
|
import * as i1 from "@angular/common/http";
|
|
9
9
|
import * as i2 from "./extension.service";
|
|
@@ -30,85 +30,60 @@ export class TwilioService {
|
|
|
30
30
|
this.isAvailableNumber = new BehaviorSubject(false);
|
|
31
31
|
this.callerIdList = new BehaviorSubject([]);
|
|
32
32
|
this.triggerSMSReload = new BehaviorSubject(false);
|
|
33
|
-
this.
|
|
34
|
-
this.
|
|
35
|
-
|
|
36
|
-
this.
|
|
37
|
-
|
|
38
|
-
|
|
33
|
+
this.tokenInitialized = false;
|
|
34
|
+
this.tokenInitialization$ = null;
|
|
35
|
+
this.lastCallSid = null;
|
|
36
|
+
this.lastCallTime = 0;
|
|
37
|
+
this.CALL_DEBOUNCE_MS = 1000; // 1 second debounce
|
|
38
|
+
this.initializeToken();
|
|
39
|
+
}
|
|
40
|
+
initializeToken() {
|
|
41
|
+
if (this.tokenInitialized) {
|
|
42
|
+
return this.tokenInitialization$ || of(null);
|
|
39
43
|
}
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
}, 2000);
|
|
44
|
+
this.tokenInitialization$ = this.extensionService.getIncomingCallToken().pipe(tap((data) => {
|
|
45
|
+
this.incomingCallToken = data.token;
|
|
46
|
+
localStorage.setItem('in-token', data.token);
|
|
47
|
+
this.tokenInitialized = true;
|
|
48
|
+
this.initializeDevice();
|
|
49
|
+
}), catchError(error => {
|
|
50
|
+
console.error('Error initializing Twilio token:', error);
|
|
51
|
+
return throwError(() => error);
|
|
52
|
+
}), shareReplay(1) // Cache the result for subsequent subscribers
|
|
53
|
+
);
|
|
54
|
+
return this.tokenInitialization$;
|
|
52
55
|
}
|
|
53
|
-
|
|
54
|
-
if (this.
|
|
55
|
-
|
|
56
|
+
initializeTwilioDevice() {
|
|
57
|
+
if (this.tokenInitialized && this.incomingCallToken) {
|
|
58
|
+
this.initializeDevice();
|
|
56
59
|
}
|
|
60
|
+
else {
|
|
61
|
+
this.initializeToken().subscribe();
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
initializeDevice() {
|
|
57
65
|
if (this.device) {
|
|
58
66
|
this.device.destroy();
|
|
59
67
|
}
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
// Prevent multiple initializations
|
|
63
|
-
if (this.deviceInitialized || this.initializationInProgress || !this.token) {
|
|
68
|
+
if (!this.incomingCallToken) {
|
|
69
|
+
console.error('No Twilio token available');
|
|
64
70
|
return;
|
|
65
71
|
}
|
|
66
|
-
this.
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
try {
|
|
70
|
-
this.incomingCallToken = data.token;
|
|
71
|
-
localStorage.setItem('in-token', data.token);
|
|
72
|
-
// Destroy existing device if it exists
|
|
73
|
-
if (this.device) {
|
|
74
|
-
this.device.destroy();
|
|
75
|
-
}
|
|
76
|
-
// Initialize Twilio Device with required options
|
|
77
|
-
const deviceOptions = {
|
|
78
|
-
allowIncomingWhileBusy: true,
|
|
79
|
-
closeProtection: true
|
|
80
|
-
};
|
|
81
|
-
// Only add debug in development
|
|
82
|
-
if (!environment.production) {
|
|
83
|
-
deviceOptions.debug = false;
|
|
84
|
-
}
|
|
85
|
-
this.device = new Device(this.incomingCallToken, deviceOptions);
|
|
86
|
-
this.device.register();
|
|
87
|
-
this.deviceInitialized = true;
|
|
88
|
-
this.initializationInProgress = false;
|
|
89
|
-
console.log('Twilio Device initialized successfully');
|
|
90
|
-
this.setupDeviceEventHandlers();
|
|
91
|
-
}
|
|
92
|
-
catch (error) {
|
|
93
|
-
console.error('Error initializing Twilio Device:', error);
|
|
94
|
-
this.initializationInProgress = false;
|
|
95
|
-
// Retry after a delay
|
|
96
|
-
setTimeout(() => this.initializeTwilioDevice(), 5000);
|
|
97
|
-
}
|
|
98
|
-
},
|
|
99
|
-
error: (error) => {
|
|
100
|
-
console.error('Error getting Twilio token:', error);
|
|
101
|
-
this.initializationInProgress = false;
|
|
102
|
-
// Retry after a delay
|
|
103
|
-
setTimeout(() => this.initializeTwilioDevice(), 5000);
|
|
104
|
-
}
|
|
72
|
+
this.device = new Device(this.incomingCallToken, {
|
|
73
|
+
allowIncomingWhileBusy: true,
|
|
74
|
+
closeProtection: true
|
|
105
75
|
});
|
|
106
|
-
|
|
107
|
-
setupDeviceEventHandlers() {
|
|
108
|
-
if (!this.device)
|
|
109
|
-
return;
|
|
76
|
+
this.device.register();
|
|
110
77
|
this.device.on('incoming', (call) => {
|
|
111
|
-
|
|
78
|
+
const callSid = call.parameters?.['CallSid'];
|
|
79
|
+
const now = Date.now();
|
|
80
|
+
// Skip if this is a duplicate call within the debounce window
|
|
81
|
+
if (callSid && callSid === this.lastCallSid && (now - this.lastCallTime) < this.CALL_DEBOUNCE_MS) {
|
|
82
|
+
console.log('Skipping duplicate call:', callSid);
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
this.lastCallSid = callSid;
|
|
86
|
+
this.lastCallTime = now;
|
|
112
87
|
this.currentCall.next(call);
|
|
113
88
|
this.callType.next('INCOMING');
|
|
114
89
|
this.currentCallState.next('incoming');
|
|
@@ -116,20 +91,15 @@ export class TwilioService {
|
|
|
116
91
|
});
|
|
117
92
|
this.device.on('error', (error) => {
|
|
118
93
|
console.error('Twilio Device Error:', error);
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
if (error.code === 31201 || error.code === 20101) {
|
|
122
|
-
console.log('Authentication error, reinitializing device...');
|
|
123
|
-
this.initializeTwilioDevice();
|
|
124
|
-
}
|
|
94
|
+
// Reset initialization state on error to allow retry
|
|
95
|
+
this.tokenInitialized = false;
|
|
125
96
|
});
|
|
126
97
|
this.device.on('registered', () => {
|
|
127
|
-
console.log('Twilio Device registered');
|
|
128
|
-
this.deviceInitialized = true;
|
|
98
|
+
console.log('Twilio Device registered successfully');
|
|
129
99
|
});
|
|
130
100
|
this.device.on('unregistered', () => {
|
|
131
101
|
console.log('Twilio Device unregistered');
|
|
132
|
-
this.
|
|
102
|
+
this.tokenInitialized = false;
|
|
133
103
|
});
|
|
134
104
|
}
|
|
135
105
|
// onIncomingCall(){
|
|
@@ -232,4 +202,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
232
202
|
providedIn: 'root'
|
|
233
203
|
}]
|
|
234
204
|
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.ExtensionService }, { type: i3.NotificationService }]; } });
|
|
235
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"twilio.service.js","sourceRoot":"","sources":["../../../../../projects/dialbox/src/lib/service/twilio.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpParams } from '@angular/common/http';\nimport { HttpHeaders } from \"@angular/common/http\";\nimport { Injectable } from '@angular/core';\nimport { Call, Device } from '@twilio/voice-sdk';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { environment } from \"../environments/environments\"\nimport { ExtensionService } from './extension.service';\nimport { NotificationService } from './Notification.service';\n\nexport interface DialPayload {\n  number: string;\n  name: string;\n  img: string;\n  callerId:string;\n  isDialFromHistory:boolean;\n  from:string;\n}\n\n@Injectable({\n  providedIn: 'root'  \n})\nexport class TwilioService {\n  \n  openInProgressDialpad = new BehaviorSubject<boolean>(false);\n  currentCall = new BehaviorSubject<Call | null>(null);\n  currentCallState = new BehaviorSubject<any>('none'); //in-progress, out-progress, none\n  device:any;\n  incomingCallToken?:string;\n  outgoingCallToken?:string;\n  callType = new BehaviorSubject<string>('NIL');\n  isIncomingCallPicked = new BehaviorSubject<boolean>(false); // for both incoming and outgoing\n  token:any = localStorage.getItem('ext_token');\n  isNewContactAdded = new BehaviorSubject<boolean>(false);\n  updateRecentCall = new BehaviorSubject<boolean>(false);\n  callhandleFromNotification = new BehaviorSubject<any>({});\n\n  endCall = new BehaviorSubject<boolean>(false);\n  dialNumberFromOtherModule = new BehaviorSubject<DialPayload>({ number: '', name: '', img: '', callerId: '', isDialFromHistory: false, from: '' });\n\n  toggleCallerIdAlert = new BehaviorSubject<boolean>(true);\n  isTrialOver = new BehaviorSubject<boolean>(false);\n  isPaymentDue = new BehaviorSubject<boolean>(false);\n  isAvailableNumber = new BehaviorSubject<boolean>(false);\n  callerIdList = new BehaviorSubject<any[]>([]);\n  triggerSMSReload = new BehaviorSubject<boolean>(false);\n\n  private deviceInitialized = false;\n  private initializationInProgress = false;\n  private tokenCheckInterval: any;\n\n  constructor(private http: HttpClient, private extensionService: ExtensionService,\n      private notificationSerivce: NotificationService\n  ) {\n    // Initialize immediately if token exists\n    this.token = localStorage.getItem('ext_token');\n    if (this.token) {\n      this.initializeTwilioDevice();\n    }\n    \n    // Set up a polling mechanism to check for token availability\n    this.startTokenCheck();\n  }\n\n  private startTokenCheck() {\n    // Check every 2 seconds if we have a token\n    this.tokenCheckInterval = setInterval(() => {\n      const currentToken = localStorage.getItem('ext_token');\n      if (currentToken && currentToken !== this.token) {\n        this.token = currentToken;\n        this.initializeTwilioDevice();\n      }\n    }, 2000);\n  }\n\n  ngOnDestroy() {\n    if (this.tokenCheckInterval) {\n      clearInterval(this.tokenCheckInterval);\n    }\n    if (this.device) {\n      this.device.destroy();\n    }\n  }\n\n   public initializeTwilioDevice() {\n    // Prevent multiple initializations\n    if (this.deviceInitialized || this.initializationInProgress || !this.token) {\n      return;\n    }\n\n    this.initializationInProgress = true;\n    \n    this.extensionService.getIncomingCallToken().subscribe({\n      next: (data: any) => {\n        try {\n          this.incomingCallToken = data.token;\n          localStorage.setItem('in-token', data.token);\n          \n          // Destroy existing device if it exists\n          if (this.device) {\n            this.device.destroy();\n          }\n          \n          // Initialize Twilio Device with required options\n          const deviceOptions: any = {\n            allowIncomingWhileBusy: true,\n            closeProtection: true\n          };\n          \n          // Only add debug in development\n          if (!environment.production) {\n            deviceOptions.debug = false;\n          }\n          \n          this.device = new Device(this.incomingCallToken!, deviceOptions);\n          \n          this.device.register();\n          this.deviceInitialized = true;\n          this.initializationInProgress = false;\n          console.log('Twilio Device initialized successfully');\n\n          this.setupDeviceEventHandlers();\n        } catch (error) {\n          console.error('Error initializing Twilio Device:', error);\n          this.initializationInProgress = false;\n          // Retry after a delay\n          setTimeout(() => this.initializeTwilioDevice(), 5000);\n        }\n      },\n      error: (error) => {\n        console.error('Error getting Twilio token:', error);\n        this.initializationInProgress = false;\n        // Retry after a delay\n        setTimeout(() => this.initializeTwilioDevice(), 5000);\n      }\n    });\n  }\n\n  private setupDeviceEventHandlers() {\n    if (!this.device) return;\n\n    this.device.on('incoming', (call: Call) => {\n      console.log('Incoming call received:', call.parameters);\n      this.currentCall.next(call);\n      this.callType.next('INCOMING');\n      this.currentCallState.next('incoming');\n      this.notificationSerivce.showNotification(call);\n    });\n\n    this.device.on('error', (error: any) => {\n      console.error('Twilio Device Error:', error);\n      this.deviceInitialized = false;\n      \n      // Attempt to reinitialize on error\n      if (error.code === 31201 || error.code === 20101) {\n        console.log('Authentication error, reinitializing device...');\n        this.initializeTwilioDevice();\n      }\n    });\n\n    this.device.on('registered', () => {\n      console.log('Twilio Device registered');\n      this.deviceInitialized = true;\n    });\n\n    this.device.on('unregistered', () => {\n      console.log('Twilio Device unregistered');\n      this.deviceInitialized = false;\n    });\n  }\n\n  // onIncomingCall(){\n  //   this.device.on('incoming', (call:any) => {\n  //     console.log(call);\n     \n  //     //call.accept();\n  //   });\n  // }\n  saveContact(payload: any): Observable<[]> {\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')}) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/add/contacts/manually', payload, httpOptions);\n  }\n  updateContact(payload:any){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + this.token}) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/update/contact', payload, httpOptions);\n  }\n  \n  deleteContact(id:string){\n    let payload = {\n      phonebookid: id\n    }\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/contact/'+id,payload, httpOptions);\n  }\n  getContactList(){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/read/contacts', httpOptions);\n  }\n\n  addContactToFavourite(payload: any){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/update/favourite', payload, httpOptions);\n  }\n\n  getFavouritContacts(){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/favourite/contacts', httpOptions);\n  }\n\n  getRecentCalls(pageIndex?: string, pageSize?: string,){\n    // let params = new HttpParams();\n    // params = params.set('size', pageSize);\n    // params = params.set('page', pageIndex);\n    const headers =  { 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') };\n    const httpOptions ={headers}\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/recent/calls', httpOptions);\n  }\n  deleteRecentCallLog(id:string| string[]){\n    let payload = {\n      recordId: id\n    }\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/calls/'+id,payload, httpOptions);\n  }\n\n  uploadPhoto(payload: any){\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/upload/photo',payload, httpOptions);\n  }\n\n  deletePhoto(id:string){\n    let payload = {\n      phonebookid: id\n    }\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/photo/'+id,payload, httpOptions);\n  }\n\n  // toggleCallerIdAlertFn(val: any) {\n  //   let httpOptions = {\n  //     headers: new HttpHeaders({\n  //       'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n  //     })\n  //   };\n  //   return this.http.put<[]>(environment.apiUrl + '/utilities/softphone/callerid/alert/' + val, {}, httpOptions);\n  // }\n\n  toggleCoutryCodeToast(val:any){\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.put<[]>(environment.apiUrl + '/utilities/softphone/countrycode/alert/' + val, {}, httpOptions);\n  }\n  getToNumber(dialledNo: string, isoCode: string) {\n    const httpOptions = {\n      headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }),\n      params: new HttpParams().set('isoCode', isoCode)\n    };\n    const params = new HttpParams().set('isoCode', isoCode)\n    return this.http.get<[]>(environment.apiUrl + '/utilities/softphone/check/countryCode/' + dialledNo, httpOptions);\n  } \n\n}\n"]}
|
|
205
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"twilio.service.js","sourceRoot":"","sources":["../../../../../projects/dialbox/src/lib/service/twilio.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpHeaders, HttpParams } from '@angular/common/http';\nimport { Injectable, OnDestroy } from '@angular/core';\nimport { BehaviorSubject, Observable, of, throwError } from 'rxjs';\nimport { catchError, shareReplay, tap } from 'rxjs/operators';\nimport { Device, Call } from '@twilio/voice-sdk';\nimport { ExtensionService } from './extension.service';\nimport { NotificationService } from './Notification.service';\nimport { environment } from '../environments/environments';\n\nexport interface DialPayload {\n  number: string;\n  name: string;\n  img: string;\n  callerId:string;\n  isDialFromHistory:boolean;\n  from:string;\n}\n\n@Injectable({\n  providedIn: 'root'  \n})\nexport class TwilioService {\n  \n  openInProgressDialpad = new BehaviorSubject<boolean>(false);\n  currentCall = new BehaviorSubject<Call | null>(null);\n  currentCallState = new BehaviorSubject<any>('none'); //in-progress, out-progress, none\n  device:any;\n  incomingCallToken?:string;\n  outgoingCallToken?:string;\n  callType = new BehaviorSubject<string>('NIL');\n  isIncomingCallPicked = new BehaviorSubject<boolean>(false); // for both incoming and outgoing\n  token:any = localStorage.getItem('ext_token');\n  isNewContactAdded = new BehaviorSubject<boolean>(false);\n  updateRecentCall = new BehaviorSubject<boolean>(false);\n  callhandleFromNotification = new BehaviorSubject<any>({});\n\n  endCall = new BehaviorSubject<boolean>(false);\n  dialNumberFromOtherModule = new BehaviorSubject<DialPayload>({ number: '', name: '', img: '', callerId: '', isDialFromHistory: false, from: '' });\n\n  toggleCallerIdAlert = new BehaviorSubject<boolean>(true);\n  isTrialOver = new BehaviorSubject<boolean>(false);\n  isPaymentDue = new BehaviorSubject<boolean>(false);\n  isAvailableNumber = new BehaviorSubject<boolean>(false);\n  callerIdList = new BehaviorSubject<any[]>([]);\n  triggerSMSReload = new BehaviorSubject<boolean>(false);\n\n  private tokenInitialized = false;\n  private tokenInitialization$: Observable<any> | null = null;\n\n  constructor(\n    private http: HttpClient, \n    private extensionService: ExtensionService,\n    private notificationSerivce: NotificationService\n  ) {\n    this.initializeToken();\n  }\n\n  private initializeToken() {\n    if (this.tokenInitialized) {\n      return this.tokenInitialization$ || of(null);\n    }\n\n    this.tokenInitialization$ = this.extensionService.getIncomingCallToken().pipe(\n      tap((data: any) => {\n        this.incomingCallToken = data.token;\n        localStorage.setItem('in-token', data.token);\n        this.tokenInitialized = true;\n        this.initializeDevice();\n      }),\n      catchError(error => {\n        console.error('Error initializing Twilio token:', error);\n        return throwError(() => error);\n      }),\n      shareReplay(1) // Cache the result for subsequent subscribers\n    );\n\n    return this.tokenInitialization$;\n  }\n\n  public initializeTwilioDevice() {\n    if (this.tokenInitialized && this.incomingCallToken) {\n      this.initializeDevice();\n    } else {\n      this.initializeToken().subscribe();\n    }\n  }\n\n  private lastCallSid: string | null = null;\n  private lastCallTime: number = 0;\n  private readonly CALL_DEBOUNCE_MS = 1000; // 1 second debounce\n\n  private initializeDevice() {\n    if (this.device) {\n      this.device.destroy();\n    }\n\n    if (!this.incomingCallToken) {\n      console.error('No Twilio token available');\n      return;\n    }\n\n    this.device = new Device(this.incomingCallToken, { \n      allowIncomingWhileBusy: true,\n      closeProtection: true\n    });\n    \n    this.device.register();\n\n    this.device.on('incoming', (call: Call) => {\n      const callSid = call.parameters?.['CallSid'];\n      const now = Date.now();\n      \n      // Skip if this is a duplicate call within the debounce window\n      if (callSid && callSid === this.lastCallSid && (now - this.lastCallTime) < this.CALL_DEBOUNCE_MS) {\n        console.log('Skipping duplicate call:', callSid);\n        return;\n      }\n      \n      this.lastCallSid = callSid;\n      this.lastCallTime = now;\n      \n      this.currentCall.next(call);\n      this.callType.next('INCOMING');\n      this.currentCallState.next('incoming');\n      this.notificationSerivce.showNotification(call);\n    });\n\n    this.device.on('error', (error: any) => {\n      console.error('Twilio Device Error:', error);\n      // Reset initialization state on error to allow retry\n      this.tokenInitialized = false;\n    });\n\n    this.device.on('registered', () => {\n      console.log('Twilio Device registered successfully');\n    });\n\n    this.device.on('unregistered', () => {\n      console.log('Twilio Device unregistered');\n      this.tokenInitialized = false;\n    });\n  }\n\n  // onIncomingCall(){\n  //   this.device.on('incoming', (call:any) => {\n  //     console.log(call);\n     \n  //     //call.accept();\n  //   });\n  // }\n  saveContact(payload: any): Observable<[]> {\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')}) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/add/contacts/manually', payload, httpOptions);\n  }\n  updateContact(payload:any){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + this.token}) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/update/contact', payload, httpOptions);\n  }\n  \n  deleteContact(id:string){\n    let payload = {\n      phonebookid: id\n    }\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/contact/'+id,payload, httpOptions);\n  }\n  getContactList(){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/read/contacts', httpOptions);\n  }\n\n  addContactToFavourite(payload: any){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/update/favourite', payload, httpOptions);\n  }\n\n  getFavouritContacts(){\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/favourite/contacts', httpOptions);\n  }\n\n  getRecentCalls(pageIndex?: string, pageSize?: string,){\n    // let params = new HttpParams();\n    // params = params.set('size', pageSize);\n    // params = params.set('page', pageIndex);\n    const headers =  { 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') };\n    const httpOptions ={headers}\n    return this.http.get<[]>(environment.apiUrl + '/utilities/phonebook/recent/calls', httpOptions);\n  }\n  deleteRecentCallLog(id:string| string[]){\n    let payload = {\n      recordId: id\n    }\n    const httpOptions = { headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }) };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/calls/'+id,payload, httpOptions);\n  }\n\n  uploadPhoto(payload: any){\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/upload/photo',payload, httpOptions);\n  }\n\n  deletePhoto(id:string){\n    let payload = {\n      phonebookid: id\n    }\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.post<[]>(environment.apiUrl + '/utilities/phonebook/delete/photo/'+id,payload, httpOptions);\n  }\n\n  // toggleCallerIdAlertFn(val: any) {\n  //   let httpOptions = {\n  //     headers: new HttpHeaders({\n  //       'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n  //     })\n  //   };\n  //   return this.http.put<[]>(environment.apiUrl + '/utilities/softphone/callerid/alert/' + val, {}, httpOptions);\n  // }\n\n  toggleCoutryCodeToast(val:any){\n    let httpOptions = {\n      headers: new HttpHeaders({\n        'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token')\n      })\n    };\n    return this.http.put<[]>(environment.apiUrl + '/utilities/softphone/countrycode/alert/' + val, {}, httpOptions);\n  }\n  getToNumber(dialledNo: string, isoCode: string) {\n    const httpOptions = {\n      headers: new HttpHeaders({ 'Content-Type': 'application/json', 'Auth-Key': \"Bearer \" + localStorage.getItem('ext_token') }),\n      params: new HttpParams().set('isoCode', isoCode)\n    };\n    const params = new HttpParams().set('isoCode', isoCode)\n    return this.http.get<[]>(environment.apiUrl + '/utilities/softphone/check/countryCode/' + dialledNo, httpOptions);\n  } \n\n}\n"]}
|
|
@@ -3,11 +3,11 @@ import * as i0 from '@angular/core';
|
|
|
3
3
|
import { Injectable, EventEmitter, Component, Input, Output, ViewChild, Inject, NgModule } from '@angular/core';
|
|
4
4
|
import swal from 'sweetalert2';
|
|
5
5
|
import { AsYouType } from 'libphonenumber-js';
|
|
6
|
-
import { throwError, BehaviorSubject,
|
|
6
|
+
import { throwError, BehaviorSubject, of, Subscription, filter, interval } from 'rxjs';
|
|
7
7
|
import * as i1 from '@angular/common/http';
|
|
8
8
|
import { HttpHeaders, HttpParams, HttpClientModule } from '@angular/common/http';
|
|
9
|
+
import { catchError, switchMap, map, tap, shareReplay } from 'rxjs/operators';
|
|
9
10
|
import { Device } from '@twilio/voice-sdk';
|
|
10
|
-
import { catchError, switchMap, map } from 'rxjs/operators';
|
|
11
11
|
import * as i3$1 from '@angular/material/dialog';
|
|
12
12
|
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
13
13
|
import * as i5 from '@angular/router';
|
|
@@ -1388,85 +1388,61 @@ class TwilioService {
|
|
|
1388
1388
|
this.isAvailableNumber = new BehaviorSubject(false);
|
|
1389
1389
|
this.callerIdList = new BehaviorSubject([]);
|
|
1390
1390
|
this.triggerSMSReload = new BehaviorSubject(false);
|
|
1391
|
-
this.
|
|
1392
|
-
this.
|
|
1393
|
-
|
|
1394
|
-
this.
|
|
1395
|
-
|
|
1396
|
-
|
|
1391
|
+
this.tokenInitialized = false;
|
|
1392
|
+
this.tokenInitialization$ = null;
|
|
1393
|
+
this.lastCallSid = null;
|
|
1394
|
+
this.lastCallTime = 0;
|
|
1395
|
+
this.CALL_DEBOUNCE_MS = 1000; // 1 second debounce
|
|
1396
|
+
this.initializeToken();
|
|
1397
|
+
}
|
|
1398
|
+
initializeToken() {
|
|
1399
|
+
if (this.tokenInitialized) {
|
|
1400
|
+
return this.tokenInitialization$ || of(null);
|
|
1397
1401
|
}
|
|
1398
|
-
|
|
1399
|
-
|
|
1400
|
-
|
|
1401
|
-
|
|
1402
|
-
|
|
1403
|
-
|
|
1404
|
-
|
|
1405
|
-
|
|
1406
|
-
|
|
1407
|
-
|
|
1408
|
-
|
|
1409
|
-
}, 2000);
|
|
1402
|
+
this.tokenInitialization$ = this.extensionService.getIncomingCallToken().pipe(tap((data) => {
|
|
1403
|
+
this.incomingCallToken = data.token;
|
|
1404
|
+
localStorage.setItem('in-token', data.token);
|
|
1405
|
+
this.tokenInitialized = true;
|
|
1406
|
+
this.initializeDevice();
|
|
1407
|
+
}), catchError(error => {
|
|
1408
|
+
console.error('Error initializing Twilio token:', error);
|
|
1409
|
+
return throwError(() => error);
|
|
1410
|
+
}), shareReplay(1) // Cache the result for subsequent subscribers
|
|
1411
|
+
);
|
|
1412
|
+
return this.tokenInitialization$;
|
|
1410
1413
|
}
|
|
1411
|
-
|
|
1412
|
-
if (this.
|
|
1413
|
-
|
|
1414
|
+
initializeTwilioDevice() {
|
|
1415
|
+
if (this.tokenInitialized && this.incomingCallToken) {
|
|
1416
|
+
this.initializeDevice();
|
|
1417
|
+
}
|
|
1418
|
+
else {
|
|
1419
|
+
this.initializeToken().subscribe();
|
|
1414
1420
|
}
|
|
1421
|
+
}
|
|
1422
|
+
initializeDevice() {
|
|
1415
1423
|
if (this.device) {
|
|
1416
1424
|
this.device.destroy();
|
|
1417
1425
|
}
|
|
1418
|
-
|
|
1419
|
-
|
|
1420
|
-
// Prevent multiple initializations
|
|
1421
|
-
if (this.deviceInitialized || this.initializationInProgress || !this.token) {
|
|
1426
|
+
if (!this.incomingCallToken) {
|
|
1427
|
+
console.error('No Twilio token available');
|
|
1422
1428
|
return;
|
|
1423
1429
|
}
|
|
1424
|
-
this.
|
|
1425
|
-
|
|
1426
|
-
|
|
1427
|
-
try {
|
|
1428
|
-
this.incomingCallToken = data.token;
|
|
1429
|
-
localStorage.setItem('in-token', data.token);
|
|
1430
|
-
// Destroy existing device if it exists
|
|
1431
|
-
if (this.device) {
|
|
1432
|
-
this.device.destroy();
|
|
1433
|
-
}
|
|
1434
|
-
// Initialize Twilio Device with required options
|
|
1435
|
-
const deviceOptions = {
|
|
1436
|
-
allowIncomingWhileBusy: true,
|
|
1437
|
-
closeProtection: true
|
|
1438
|
-
};
|
|
1439
|
-
// Only add debug in development
|
|
1440
|
-
if (!environment.production) {
|
|
1441
|
-
deviceOptions.debug = false;
|
|
1442
|
-
}
|
|
1443
|
-
this.device = new Device(this.incomingCallToken, deviceOptions);
|
|
1444
|
-
this.device.register();
|
|
1445
|
-
this.deviceInitialized = true;
|
|
1446
|
-
this.initializationInProgress = false;
|
|
1447
|
-
console.log('Twilio Device initialized successfully');
|
|
1448
|
-
this.setupDeviceEventHandlers();
|
|
1449
|
-
}
|
|
1450
|
-
catch (error) {
|
|
1451
|
-
console.error('Error initializing Twilio Device:', error);
|
|
1452
|
-
this.initializationInProgress = false;
|
|
1453
|
-
// Retry after a delay
|
|
1454
|
-
setTimeout(() => this.initializeTwilioDevice(), 5000);
|
|
1455
|
-
}
|
|
1456
|
-
},
|
|
1457
|
-
error: (error) => {
|
|
1458
|
-
console.error('Error getting Twilio token:', error);
|
|
1459
|
-
this.initializationInProgress = false;
|
|
1460
|
-
// Retry after a delay
|
|
1461
|
-
setTimeout(() => this.initializeTwilioDevice(), 5000);
|
|
1462
|
-
}
|
|
1430
|
+
this.device = new Device(this.incomingCallToken, {
|
|
1431
|
+
allowIncomingWhileBusy: true,
|
|
1432
|
+
closeProtection: true
|
|
1463
1433
|
});
|
|
1464
|
-
|
|
1465
|
-
setupDeviceEventHandlers() {
|
|
1466
|
-
if (!this.device)
|
|
1467
|
-
return;
|
|
1434
|
+
this.device.register();
|
|
1468
1435
|
this.device.on('incoming', (call) => {
|
|
1469
|
-
|
|
1436
|
+
var _a;
|
|
1437
|
+
const callSid = (_a = call.parameters) === null || _a === void 0 ? void 0 : _a['CallSid'];
|
|
1438
|
+
const now = Date.now();
|
|
1439
|
+
// Skip if this is a duplicate call within the debounce window
|
|
1440
|
+
if (callSid && callSid === this.lastCallSid && (now - this.lastCallTime) < this.CALL_DEBOUNCE_MS) {
|
|
1441
|
+
console.log('Skipping duplicate call:', callSid);
|
|
1442
|
+
return;
|
|
1443
|
+
}
|
|
1444
|
+
this.lastCallSid = callSid;
|
|
1445
|
+
this.lastCallTime = now;
|
|
1470
1446
|
this.currentCall.next(call);
|
|
1471
1447
|
this.callType.next('INCOMING');
|
|
1472
1448
|
this.currentCallState.next('incoming');
|
|
@@ -1474,20 +1450,15 @@ class TwilioService {
|
|
|
1474
1450
|
});
|
|
1475
1451
|
this.device.on('error', (error) => {
|
|
1476
1452
|
console.error('Twilio Device Error:', error);
|
|
1477
|
-
|
|
1478
|
-
|
|
1479
|
-
if (error.code === 31201 || error.code === 20101) {
|
|
1480
|
-
console.log('Authentication error, reinitializing device...');
|
|
1481
|
-
this.initializeTwilioDevice();
|
|
1482
|
-
}
|
|
1453
|
+
// Reset initialization state on error to allow retry
|
|
1454
|
+
this.tokenInitialized = false;
|
|
1483
1455
|
});
|
|
1484
1456
|
this.device.on('registered', () => {
|
|
1485
|
-
console.log('Twilio Device registered');
|
|
1486
|
-
this.deviceInitialized = true;
|
|
1457
|
+
console.log('Twilio Device registered successfully');
|
|
1487
1458
|
});
|
|
1488
1459
|
this.device.on('unregistered', () => {
|
|
1489
1460
|
console.log('Twilio Device unregistered');
|
|
1490
|
-
this.
|
|
1461
|
+
this.tokenInitialized = false;
|
|
1491
1462
|
});
|
|
1492
1463
|
}
|
|
1493
1464
|
// onIncomingCall(){
|
|
@@ -1611,42 +1582,63 @@ class IncomingCallComponent {
|
|
|
1611
1582
|
this.closeIncomingCallDiv = new EventEmitter();
|
|
1612
1583
|
this.incomingCallsNewList = new EventEmitter();
|
|
1613
1584
|
this.selectedIncomingCallInfo = new EventEmitter();
|
|
1585
|
+
this.subscription = new Subscription();
|
|
1586
|
+
this.callHandled = new Set(); // Track handled calls by CallSid
|
|
1614
1587
|
}
|
|
1615
1588
|
ngOnInit() {
|
|
1616
|
-
// this.twilioService.currentCall.subscribe((call: any) => {
|
|
1617
|
-
// if (call) {
|
|
1618
|
-
// this.twilioCallData = call;
|
|
1619
|
-
// this.notificationSerivce.showNotification(call);
|
|
1620
|
-
// // Handle the call UI
|
|
1621
|
-
// }
|
|
1622
|
-
// });
|
|
1623
1589
|
try {
|
|
1624
|
-
|
|
1625
|
-
|
|
1626
|
-
|
|
1627
|
-
|
|
1628
|
-
|
|
1629
|
-
|
|
1630
|
-
|
|
1631
|
-
|
|
1632
|
-
|
|
1633
|
-
|
|
1634
|
-
|
|
1635
|
-
|
|
1636
|
-
|
|
1637
|
-
|
|
1638
|
-
|
|
1639
|
-
if (this.incomingCallData && this.incomingCallData.parameters && this.incomingCallData.parameters.CallSid) {
|
|
1640
|
-
this.newIncomingCallsList = this.newIncomingCallsList.filter((item) => item.parameters && item.parameters.CallSid !== this.incomingCallData.parameters.CallSid);
|
|
1641
|
-
}
|
|
1642
|
-
this.rejectCallFromList(call);
|
|
1643
|
-
});
|
|
1644
|
-
}
|
|
1645
|
-
});
|
|
1590
|
+
// Only handle the call if we have incomingCallData
|
|
1591
|
+
if (this.incomingCallData) {
|
|
1592
|
+
this.handleIncomingCall(this.incomingCallData);
|
|
1593
|
+
}
|
|
1594
|
+
// Subscribe to new calls
|
|
1595
|
+
this.subscription.add(this.twilioService.currentCall.pipe(
|
|
1596
|
+
// Filter out duplicate calls that we've already handled
|
|
1597
|
+
filter((call) => {
|
|
1598
|
+
if (!call)
|
|
1599
|
+
return false;
|
|
1600
|
+
const callSid = call.parameters['CallSid'];
|
|
1601
|
+
return callSid && !this.callHandled.has(callSid);
|
|
1602
|
+
})).subscribe(call => {
|
|
1603
|
+
this.handleIncomingCall(call);
|
|
1604
|
+
}));
|
|
1646
1605
|
}
|
|
1647
1606
|
catch (e) {
|
|
1648
|
-
console.
|
|
1607
|
+
console.error('Error in incoming call handling:', e);
|
|
1608
|
+
}
|
|
1609
|
+
}
|
|
1610
|
+
handleIncomingCall(call) {
|
|
1611
|
+
var _a, _b;
|
|
1612
|
+
if (!call)
|
|
1613
|
+
return;
|
|
1614
|
+
const callSid = (_a = call.parameters) === null || _a === void 0 ? void 0 : _a.CallSid;
|
|
1615
|
+
if (!callSid || this.callHandled.has(callSid)) {
|
|
1616
|
+
return; // Skip if no CallSid or already handled
|
|
1617
|
+
}
|
|
1618
|
+
this.callHandled.add(callSid);
|
|
1619
|
+
this.twilioCallData = call;
|
|
1620
|
+
this.twilioAuthId = (_b = call.customParameters) === null || _b === void 0 ? void 0 : _b.get('twilioAuthId');
|
|
1621
|
+
if (!call.parameters) {
|
|
1622
|
+
call.parameters = {};
|
|
1649
1623
|
}
|
|
1624
|
+
// Send IP for the incoming call
|
|
1625
|
+
this.sendIPforIncomingCall(this.twilioAuthId, '');
|
|
1626
|
+
// Setup call event handlers
|
|
1627
|
+
const handleCallEnd = () => {
|
|
1628
|
+
if (callSid) {
|
|
1629
|
+
this.callHandled.delete(callSid);
|
|
1630
|
+
if (this.newIncomingCallsList) {
|
|
1631
|
+
this.newIncomingCallsList = this.newIncomingCallsList.filter((item) => { var _a; return ((_a = item.parameters) === null || _a === void 0 ? void 0 : _a.CallSid) !== callSid; });
|
|
1632
|
+
}
|
|
1633
|
+
}
|
|
1634
|
+
this.rejectCallFromList(call);
|
|
1635
|
+
};
|
|
1636
|
+
call.on('cancel', handleCallEnd);
|
|
1637
|
+
call.on('disconnect', handleCallEnd);
|
|
1638
|
+
}
|
|
1639
|
+
ngOnDestroy() {
|
|
1640
|
+
this.subscription.unsubscribe();
|
|
1641
|
+
this.callHandled.clear();
|
|
1650
1642
|
}
|
|
1651
1643
|
acceptCallFromList(data) {
|
|
1652
1644
|
data.accept();
|
|
@@ -2299,28 +2291,7 @@ class DialboxComponent {
|
|
|
2299
2291
|
return;
|
|
2300
2292
|
this.token = localStorage.getItem('ext_token') || '';
|
|
2301
2293
|
if (!this.token) {
|
|
2302
|
-
console.
|
|
2303
|
-
// Clear any existing interval to prevent multiple intervals
|
|
2304
|
-
if (this.twilioInitializationInterval) {
|
|
2305
|
-
clearInterval(this.twilioInitializationInterval);
|
|
2306
|
-
}
|
|
2307
|
-
// Set up a check for when token becomes available
|
|
2308
|
-
this.twilioInitializationInterval = setInterval(() => {
|
|
2309
|
-
const newToken = localStorage.getItem('ext_token');
|
|
2310
|
-
if (newToken) {
|
|
2311
|
-
clearInterval(this.twilioInitializationInterval);
|
|
2312
|
-
this.token = newToken;
|
|
2313
|
-
this.initializeTwilio();
|
|
2314
|
-
}
|
|
2315
|
-
}, 2000);
|
|
2316
|
-
// Add cleanup for this interval
|
|
2317
|
-
this.subscriptions.add({
|
|
2318
|
-
unsubscribe: () => {
|
|
2319
|
-
if (this.twilioInitializationInterval) {
|
|
2320
|
-
clearInterval(this.twilioInitializationInterval);
|
|
2321
|
-
}
|
|
2322
|
-
}
|
|
2323
|
-
});
|
|
2294
|
+
console.error('No authentication token found');
|
|
2324
2295
|
return;
|
|
2325
2296
|
}
|
|
2326
2297
|
this.isInitialized = true;
|
|
@@ -2439,10 +2410,8 @@ class DialboxComponent {
|
|
|
2439
2410
|
// }
|
|
2440
2411
|
ngOnInit() {
|
|
2441
2412
|
try {
|
|
2442
|
-
// Initialize Twilio when component loads
|
|
2443
|
-
this.initializeTwilio();
|
|
2444
|
-
// Get token and load initial data
|
|
2445
2413
|
this.token = localStorage.getItem('ext_token') || '';
|
|
2414
|
+
//this.isCallInProgress = true;
|
|
2446
2415
|
this.getContactList();
|
|
2447
2416
|
this.getUserCallSetting();
|
|
2448
2417
|
const sub1 = this.twilioService.dialNumberFromOtherModule.subscribe((contact) => {
|
|
@@ -3308,13 +3277,6 @@ class DialboxComponent {
|
|
|
3308
3277
|
if (this.subscriptions) {
|
|
3309
3278
|
this.subscriptions.unsubscribe();
|
|
3310
3279
|
}
|
|
3311
|
-
// Clear any active intervals
|
|
3312
|
-
if (this.tokenCheckInterval) {
|
|
3313
|
-
clearInterval(this.tokenCheckInterval);
|
|
3314
|
-
}
|
|
3315
|
-
if (this.twilioInitializationInterval) {
|
|
3316
|
-
clearInterval(this.twilioInitializationInterval);
|
|
3317
|
-
}
|
|
3318
3280
|
// Clean up Twilio device when component is destroyed
|
|
3319
3281
|
if (this.twilioService['device']) {
|
|
3320
3282
|
this.twilioService['device'].destroy();
|