@hivegpt/hiveai-angular 0.0.589 → 0.0.590

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (59) hide show
  1. package/README.md +26 -0
  2. package/bundles/hivegpt-hiveai-angular.umd.js +3146 -534
  3. package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
  4. package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
  5. package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
  6. package/esm2015/hivegpt-hiveai-angular.js +6 -5
  7. package/esm2015/lib/components/NotificationSocket.js +29 -22
  8. package/esm2015/lib/components/bot.service.js +18 -12
  9. package/esm2015/lib/components/chat-drawer/chat-drawer.component.js +2429 -214
  10. package/esm2015/lib/components/chatbot/chatbot.component.js +9 -5
  11. package/esm2015/lib/components/conversation.service.js +13 -5
  12. package/esm2015/lib/components/socket-service.service.js +21 -11
  13. package/esm2015/lib/components/translations/translation.service.js +38 -36
  14. package/esm2015/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +46 -35
  15. package/esm2015/lib/components/voice-agent/services/audio-analyzer.service.js +11 -5
  16. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +226 -151
  17. package/esm2015/lib/components/voice-agent/voice-agent.module.js +6 -2
  18. package/esm2015/lib/components/voice-agent/voice-modal-tokens.js +1 -1
  19. package/esm2015/lib/services/platform-token-refresh.service.js +173 -0
  20. package/esm2015/lib/utils/utils.js +4 -2
  21. package/esm2015/public-api.js +2 -3
  22. package/fesm2015/hivegpt-hiveai-angular.js +2996 -486
  23. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  24. package/hivegpt-hiveai-angular.d.ts +5 -4
  25. package/hivegpt-hiveai-angular.d.ts.map +1 -1
  26. package/hivegpt-hiveai-angular.metadata.json +1 -1
  27. package/lib/components/NotificationSocket.d.ts +1 -3
  28. package/lib/components/NotificationSocket.d.ts.map +1 -1
  29. package/lib/components/bot.service.d.ts +1 -2
  30. package/lib/components/bot.service.d.ts.map +1 -1
  31. package/lib/components/chat-drawer/chat-drawer.component.d.ts +285 -52
  32. package/lib/components/chat-drawer/chat-drawer.component.d.ts.map +1 -1
  33. package/lib/components/chatbot/chatbot.component.d.ts +8 -1
  34. package/lib/components/chatbot/chatbot.component.d.ts.map +1 -1
  35. package/lib/components/conversation.service.d.ts +8 -2
  36. package/lib/components/conversation.service.d.ts.map +1 -1
  37. package/lib/components/socket-service.service.d.ts +3 -4
  38. package/lib/components/socket-service.service.d.ts.map +1 -1
  39. package/lib/components/translations/translation.service.d.ts.map +1 -1
  40. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +15 -6
  41. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
  42. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +5 -0
  43. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts.map +1 -1
  44. package/lib/components/voice-agent/services/voice-agent.service.d.ts +39 -10
  45. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  46. package/lib/components/voice-agent/voice-agent.module.d.ts +4 -0
  47. package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
  48. package/lib/components/voice-agent/voice-modal-tokens.d.ts +4 -0
  49. package/lib/components/voice-agent/voice-modal-tokens.d.ts.map +1 -1
  50. package/lib/services/platform-token-refresh.service.d.ts +59 -0
  51. package/lib/services/platform-token-refresh.service.d.ts.map +1 -0
  52. package/lib/utils/utils.d.ts +1 -1
  53. package/lib/utils/utils.d.ts.map +1 -1
  54. package/package.json +4 -2
  55. package/public-api.d.ts +1 -2
  56. package/public-api.d.ts.map +1 -1
  57. package/esm2015/lib/models/chat.js +0 -2
  58. package/lib/models/chat.d.ts +0 -134
  59. package/lib/models/chat.d.ts.map +0 -1
@@ -1,18 +1,34 @@
1
1
  import { __awaiter } from "tslib";
2
- import { Injectable } from '@angular/core';
3
- import { BehaviorSubject, Subject } from 'rxjs';
4
- import { PipecatClient, RTVIEvent, } from '@pipecat-ai/client-js';
2
+ import { isPlatformBrowser } from '@angular/common';
3
+ import { Inject, Injectable, NgZone, PLATFORM_ID } from '@angular/core';
4
+ import { BehaviorSubject, Subject, Subscription } from 'rxjs';
5
+ import { take } from 'rxjs/operators';
6
+ import { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';
5
7
  import { WebSocketTransport } from '@pipecat-ai/websocket-transport';
8
+ import { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';
6
9
  import { AudioAnalyzerService } from './audio-analyzer.service';
7
10
  import * as i0 from "@angular/core";
8
11
  import * as i1 from "./audio-analyzer.service";
12
+ import * as i2 from "../../../services/platform-token-refresh.service";
13
+ /**
14
+ * Voice agent orchestrator using the official PipecatClient SDK.
15
+ *
16
+ * Audio flow (mirrors the React reference implementation):
17
+ * - Local mic: acquired by PipecatClient.initDevices(); local track fed to
18
+ * AudioAnalyzerService for waveform visualisation.
19
+ * - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,
20
+ * played through a hidden <audio> element.
21
+ * - All binary protobuf framing / RTVI protocol handled by
22
+ * @pipecat-ai/client-js + @pipecat-ai/websocket-transport.
23
+ */
9
24
  export class VoiceAgentService {
10
- constructor(audioAnalyzer) {
25
+ constructor(audioAnalyzer, platformTokenRefresh, ngZone,
26
+ /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */
27
+ platformId) {
11
28
  this.audioAnalyzer = audioAnalyzer;
12
- this.client = null;
13
- this.transport = null;
14
- this.userMediaStream = null;
15
- this.botAudioElement = null;
29
+ this.platformTokenRefresh = platformTokenRefresh;
30
+ this.ngZone = ngZone;
31
+ this.platformId = platformId;
16
32
  this.callStateSubject = new BehaviorSubject('idle');
17
33
  this.statusTextSubject = new BehaviorSubject('');
18
34
  this.durationSubject = new BehaviorSubject('00:00');
@@ -23,6 +39,10 @@ export class VoiceAgentService {
23
39
  this.botTranscriptSubject = new Subject();
24
40
  this.callStartTime = 0;
25
41
  this.durationInterval = null;
42
+ this.pcClient = null;
43
+ this.botAudioElement = null;
44
+ this.subscriptions = new Subscription();
45
+ this.destroy$ = new Subject();
26
46
  this.callState$ = this.callStateSubject.asObservable();
27
47
  this.statusText$ = this.statusTextSubject.asObservable();
28
48
  this.duration$ = this.durationSubject.asObservable();
@@ -31,199 +51,254 @@ export class VoiceAgentService {
31
51
  this.audioLevels$ = this.audioLevelsSubject.asObservable();
32
52
  this.userTranscript$ = this.userTranscriptSubject.asObservable();
33
53
  this.botTranscript$ = this.botTranscriptSubject.asObservable();
34
- this.audioAnalyzer.audioLevels$.subscribe(levels => {
35
- this.audioLevelsSubject.next(levels);
36
- });
37
- this.audioAnalyzer.isUserSpeaking$.subscribe(isSpeaking => {
38
- this.isUserSpeakingSubject.next(isSpeaking);
39
- if (isSpeaking && this.callStateSubject.value === 'connected') {
40
- this.callStateSubject.next('listening');
41
- }
42
- else if (!isSpeaking && this.callStateSubject.value === 'listening') {
43
- this.callStateSubject.next('connected');
44
- }
45
- });
54
+ this.subscriptions.add(this.audioAnalyzer.audioLevels$.subscribe((levels) => this.audioLevelsSubject.next(levels)));
55
+ }
56
+ ngOnDestroy() {
57
+ this.destroy$.next();
58
+ this.subscriptions.unsubscribe();
59
+ void this.disconnect();
46
60
  }
47
- /** Reset to idle state (e.g. when modal opens so user can click Start Call). */
61
+ /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */
48
62
  resetToIdle() {
49
63
  if (this.callStateSubject.value === 'idle')
50
64
  return;
51
- if (this.durationInterval) {
52
- clearInterval(this.durationInterval);
53
- this.durationInterval = null;
54
- }
55
- this.audioAnalyzer.stop();
56
- this.client = null;
57
- this.transport = null;
58
- if (this.userMediaStream) {
59
- this.userMediaStream.getTracks().forEach(track => track.stop());
60
- this.userMediaStream = null;
61
- }
62
- if (this.botAudioElement) {
63
- this.botAudioElement.pause();
64
- this.botAudioElement.srcObject = null;
65
- this.botAudioElement = null;
66
- }
65
+ void this.disconnect();
67
66
  this.callStateSubject.next('idle');
68
67
  this.statusTextSubject.next('');
69
68
  this.durationSubject.next('0:00');
70
69
  }
71
- connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventUrl, domainAuthority) {
72
- var _a;
70
+ connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
73
71
  return __awaiter(this, void 0, void 0, function* () {
74
72
  if (this.callStateSubject.value !== 'idle') {
75
- console.warn('Call already in progress');
73
+ console.warn('[HiveGpt Voice] Call already in progress');
76
74
  return;
77
75
  }
78
76
  try {
79
77
  this.callStateSubject.next('connecting');
80
78
  this.statusTextSubject.next('Connecting...');
79
+ let accessToken = token;
80
+ if (usersApiUrl && isPlatformBrowser(this.platformId)) {
81
+ try {
82
+ const ensured = yield this.platformTokenRefresh
83
+ .ensureValidAccessToken(token, usersApiUrl)
84
+ .pipe(take(1))
85
+ .toPromise();
86
+ if (ensured === null || ensured === void 0 ? void 0 : ensured.accessToken)
87
+ accessToken = ensured.accessToken;
88
+ }
89
+ catch (e) {
90
+ console.warn('[HiveGpt Voice] Token refresh failed', e);
91
+ }
92
+ }
81
93
  const baseUrl = apiUrl.replace(/\/$/, '');
82
- const connectUrl = `${baseUrl}/ai/ask-voice`;
83
- const headers = new Headers();
84
- headers.set('Authorization', `Bearer ${token}`);
85
- // Do not set Content-Type here — the Pipecat client already sets "application/json" once; adding it again produces "application/json, application/json"
86
- headers.set('domain-authority', domainAuthority);
87
- headers.set('eventtoken', eventToken);
88
- headers.set('eventurl', eventUrl);
89
- headers.set('hive-bot-id', botId);
90
- headers.set('x-api-key', apiKey);
91
- const startBotParams = {
92
- endpoint: connectUrl,
93
- headers,
94
+ const pcClient = new PipecatClient({
95
+ transport: new WebSocketTransport(),
96
+ enableMic: true,
97
+ enableCam: false,
98
+ callbacks: {
99
+ onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),
100
+ onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),
101
+ onBotReady: () => this.ngZone.run(() => this.onBotReady()),
102
+ onUserTranscript: (data) => this.ngZone.run(() => this.userTranscriptSubject.next({ text: data.text, final: !!data.final })),
103
+ onBotTranscript: (data) => this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),
104
+ onError: (err) => {
105
+ this.ngZone.run(() => {
106
+ console.error('[HiveGpt Voice] PipecatClient error', err);
107
+ this.callStateSubject.next('ended');
108
+ this.statusTextSubject.next('Connection failed');
109
+ });
110
+ },
111
+ },
112
+ });
113
+ this.pcClient = pcClient;
114
+ // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element
115
+ pcClient.on(RTVIEvent.TrackStarted, (track, participant) => {
116
+ if (!(participant === null || participant === void 0 ? void 0 : participant.local) && track.kind === 'audio') {
117
+ this.ngZone.run(() => this.setupBotAudioTrack(track));
118
+ }
119
+ });
120
+ // Speaking state comes straight from RTVI events
121
+ pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() => this.onBotStartedSpeaking()));
122
+ pcClient.on(RTVIEvent.BotStoppedSpeaking, () => this.ngZone.run(() => this.onBotStoppedSpeaking()));
123
+ pcClient.on(RTVIEvent.UserStartedSpeaking, () => this.ngZone.run(() => {
124
+ this.isUserSpeakingSubject.next(true);
125
+ this.callStateSubject.next('listening');
126
+ this.statusTextSubject.next('Listening...');
127
+ }));
128
+ pcClient.on(RTVIEvent.UserStoppedSpeaking, () => this.ngZone.run(() => {
129
+ this.isUserSpeakingSubject.next(false);
130
+ if (this.callStateSubject.value === 'listening') {
131
+ // Brief 'Processing...' while we wait for the bot to respond.
132
+ this.callStateSubject.next('connected');
133
+ this.statusTextSubject.next('Processing...');
134
+ }
135
+ }));
136
+ // Acquire mic (triggers browser permission prompt)
137
+ yield pcClient.initDevices();
138
+ // Build headers using the browser Headers API (required by pipecat's APIRequest type)
139
+ const requestHeaders = new Headers();
140
+ requestHeaders.append('Authorization', `Bearer ${accessToken}`);
141
+ requestHeaders.append('x-api-key', apiKey);
142
+ requestHeaders.append('hive-bot-id', botId);
143
+ requestHeaders.append('domain-authority', domainAuthority);
144
+ requestHeaders.append('eventUrl', eventUrl);
145
+ requestHeaders.append('eventId', eventId);
146
+ requestHeaders.append('eventToken', eventToken);
147
+ requestHeaders.append('ngrok-skip-browser-warning', 'true');
148
+ // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects
149
+ yield pcClient.startBotAndConnect({
150
+ endpoint: `${baseUrl}/ai/ask-voice-socket`,
151
+ headers: requestHeaders,
94
152
  requestData: {
95
153
  bot_id: botId,
96
154
  conversation_id: conversationId,
97
155
  voice: 'alloy',
98
156
  },
99
- };
100
- this.transport = new WebSocketTransport();
101
- this.client = new PipecatClient({
102
- transport: this.transport,
103
- enableMic: true,
104
- enableCam: false,
105
157
  });
106
- this.setupEventHandlers();
107
- this.botAudioElement = new Audio();
108
- this.botAudioElement.autoplay = true;
109
- yield this.client.startBotAndConnect(startBotParams);
110
- const tracks = this.client.tracks();
111
- if ((_a = tracks === null || tracks === void 0 ? void 0 : tracks.local) === null || _a === void 0 ? void 0 : _a.audio) {
112
- this.userMediaStream = new MediaStream([tracks.local.audio]);
113
- this.audioAnalyzer.start(this.userMediaStream);
114
- }
115
- this.isMicMutedSubject.next(!this.client.isMicEnabled);
116
- this.callStateSubject.next('connected');
117
- this.statusTextSubject.next('Connected');
118
- this.callStartTime = Date.now();
119
- this.startDurationTimer();
120
158
  }
121
159
  catch (error) {
122
- console.error('Error connecting voice agent:', error);
160
+ console.error('[HiveGpt Voice] connect failed', error);
123
161
  this.callStateSubject.next('ended');
124
- yield this.disconnect();
162
+ yield this.cleanupPipecatClient();
125
163
  this.statusTextSubject.next('Connection failed');
126
164
  throw error;
127
165
  }
128
166
  });
129
167
  }
168
+ onPipecatConnected() {
169
+ // Start the duration timer from the moment the session is live.
170
+ this.callStartTime = Date.now();
171
+ this.startDurationTimer();
172
+ this.callStateSubject.next('connected');
173
+ this.statusTextSubject.next('Connected');
174
+ this.isMicMutedSubject.next(false);
175
+ this.startLocalMicAnalyzer();
176
+ }
177
+ onPipecatDisconnected() {
178
+ this.stopDurationTimer();
179
+ this.callStartTime = 0;
180
+ this.audioAnalyzer.stop();
181
+ this.stopBotAudio();
182
+ this.callStateSubject.next('ended');
183
+ this.statusTextSubject.next('Call Ended');
184
+ }
185
+ onBotReady() {
186
+ var _a, _b, _c;
187
+ // Retry track wiring in case tracks weren't ready at onConnected.
188
+ this.startLocalMicAnalyzer();
189
+ const botTrack = (_c = (_b = (_a = this.pcClient) === null || _a === void 0 ? void 0 : _a.tracks()) === null || _b === void 0 ? void 0 : _b.bot) === null || _c === void 0 ? void 0 : _c.audio;
190
+ if (botTrack)
191
+ this.setupBotAudioTrack(botTrack);
192
+ // Bot is initialised — signal that we're now waiting for user input.
193
+ this.statusTextSubject.next('Listening...');
194
+ }
195
+ startLocalMicAnalyzer() {
196
+ var _a, _b, _c;
197
+ const localTrack = (_c = (_b = (_a = this.pcClient) === null || _a === void 0 ? void 0 : _a.tracks()) === null || _b === void 0 ? void 0 : _b.local) === null || _c === void 0 ? void 0 : _c.audio;
198
+ if (localTrack) {
199
+ this.audioAnalyzer.start(new MediaStream([localTrack]));
200
+ }
201
+ }
202
+ onBotStartedSpeaking() {
203
+ this.callStateSubject.next('talking');
204
+ this.statusTextSubject.next('Talking...');
205
+ // Mark user as no longer speaking when bot takes the turn.
206
+ this.isUserSpeakingSubject.next(false);
207
+ }
208
+ onBotStoppedSpeaking() {
209
+ if (this.callStateSubject.value === 'talking') {
210
+ this.callStateSubject.next('connected');
211
+ this.statusTextSubject.next('Listening...');
212
+ }
213
+ }
214
+ setupBotAudioTrack(track) {
215
+ var _a;
216
+ if (!this.botAudioElement) {
217
+ this.botAudioElement = new Audio();
218
+ this.botAudioElement.autoplay = true;
219
+ }
220
+ const existing = (_a = this.botAudioElement.srcObject) === null || _a === void 0 ? void 0 : _a.getAudioTracks()[0];
221
+ if ((existing === null || existing === void 0 ? void 0 : existing.id) === track.id)
222
+ return;
223
+ this.botAudioElement.srcObject = new MediaStream([track]);
224
+ this.botAudioElement.play().catch((err) => console.warn('[HiveGpt Voice] Bot audio play blocked', err));
225
+ }
226
+ stopBotAudio() {
227
+ var _a;
228
+ if (this.botAudioElement) {
229
+ try {
230
+ this.botAudioElement.pause();
231
+ (_a = this.botAudioElement.srcObject) === null || _a === void 0 ? void 0 : _a.getAudioTracks().forEach((t) => t.stop());
232
+ this.botAudioElement.srcObject = null;
233
+ }
234
+ catch (_b) {
235
+ // ignore
236
+ }
237
+ this.botAudioElement = null;
238
+ }
239
+ }
130
240
  disconnect() {
131
241
  return __awaiter(this, void 0, void 0, function* () {
132
- try {
133
- if (this.durationInterval) {
134
- clearInterval(this.durationInterval);
135
- this.durationInterval = null;
136
- }
137
- this.audioAnalyzer.stop();
138
- if (this.client) {
139
- yield this.client.disconnect();
140
- this.client = null;
141
- }
142
- this.transport = null;
143
- if (this.userMediaStream) {
144
- this.userMediaStream.getTracks().forEach(track => track.stop());
145
- this.userMediaStream = null;
242
+ this.stopDurationTimer();
243
+ this.callStartTime = 0;
244
+ this.audioAnalyzer.stop();
245
+ this.stopBotAudio();
246
+ yield this.cleanupPipecatClient();
247
+ this.callStateSubject.next('ended');
248
+ this.statusTextSubject.next('Call Ended');
249
+ });
250
+ }
251
+ cleanupPipecatClient() {
252
+ return __awaiter(this, void 0, void 0, function* () {
253
+ if (this.pcClient) {
254
+ try {
255
+ yield this.pcClient.disconnect();
146
256
  }
147
- if (this.botAudioElement) {
148
- this.botAudioElement.pause();
149
- this.botAudioElement.srcObject = null;
150
- this.botAudioElement = null;
257
+ catch (_a) {
258
+ // ignore
151
259
  }
152
- this.callStateSubject.next('ended');
153
- this.statusTextSubject.next('Call Ended');
154
- // Keep current duration so "Call Ended" state can show it (e.g. "Call Ended 1:04")
155
- }
156
- catch (error) {
157
- console.error('Error disconnecting voice agent:', error);
260
+ this.pcClient = null;
158
261
  }
159
262
  });
160
263
  }
161
264
  toggleMic() {
162
- if (!this.client)
265
+ if (!this.pcClient)
163
266
  return;
164
- const newState = !this.client.isMicEnabled;
165
- this.client.enableMic(newState);
166
- this.isMicMutedSubject.next(!newState);
167
- }
168
- setupEventHandlers() {
169
- if (!this.client)
170
- return;
171
- this.client.on(RTVIEvent.BotStartedSpeaking, () => {
172
- this.callStateSubject.next('talking');
173
- });
174
- this.client.on(RTVIEvent.BotStoppedSpeaking, () => {
175
- if (this.callStateSubject.value === 'talking') {
176
- this.callStateSubject.next('connected');
177
- }
178
- });
179
- this.client.on(RTVIEvent.TrackStarted, (track, participant) => {
180
- if (this.botAudioElement && participant && !participant.local && track.kind === 'audio') {
181
- const stream = new MediaStream([track]);
182
- this.botAudioElement.srcObject = stream;
183
- this.botAudioElement.play().catch(console.error);
184
- }
185
- });
186
- this.client.on(RTVIEvent.UserTranscript, (data) => {
187
- var _a;
188
- this.userTranscriptSubject.next({
189
- text: data.text,
190
- final: (_a = data.final) !== null && _a !== void 0 ? _a : false,
191
- });
192
- });
193
- this.client.on(RTVIEvent.BotTranscript, (data) => {
194
- if (data === null || data === void 0 ? void 0 : data.text) {
195
- this.botTranscriptSubject.next(data.text);
196
- }
197
- });
198
- this.client.on(RTVIEvent.Error, () => {
199
- this.statusTextSubject.next('Error occurred');
200
- });
201
- this.client.on(RTVIEvent.Disconnected, () => {
202
- this.callStateSubject.next('ended');
203
- this.statusTextSubject.next('Disconnected');
204
- this.disconnect();
205
- });
267
+ const nextMuted = !this.isMicMutedSubject.value;
268
+ this.pcClient.enableMic(!nextMuted);
269
+ this.isMicMutedSubject.next(nextMuted);
270
+ if (nextMuted)
271
+ this.isUserSpeakingSubject.next(false);
206
272
  }
207
273
  startDurationTimer() {
208
- const updateDuration = () => {
274
+ const tick = () => {
209
275
  if (this.callStartTime > 0) {
210
276
  const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
211
- const minutes = Math.floor(elapsed / 60);
212
- const seconds = elapsed % 60;
213
- this.durationSubject.next(`${minutes}:${String(seconds).padStart(2, '0')}`);
277
+ const m = Math.floor(elapsed / 60);
278
+ const s = elapsed % 60;
279
+ this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);
214
280
  }
215
281
  };
216
- updateDuration(); // show 0:00 immediately
217
- this.durationInterval = setInterval(updateDuration, 1000);
282
+ tick();
283
+ this.durationInterval = setInterval(tick, 1000);
284
+ }
285
+ stopDurationTimer() {
286
+ if (this.durationInterval) {
287
+ clearInterval(this.durationInterval);
288
+ this.durationInterval = null;
289
+ }
218
290
  }
219
291
  }
220
- VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService)); }, token: VoiceAgentService, providedIn: "root" });
292
+ VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.PlatformTokenRefreshService), i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i0.PLATFORM_ID)); }, token: VoiceAgentService, providedIn: "root" });
221
293
  VoiceAgentService.decorators = [
222
294
  { type: Injectable, args: [{
223
- providedIn: 'root'
295
+ providedIn: 'root',
224
296
  },] }
225
297
  ];
226
298
  VoiceAgentService.ctorParameters = () => [
227
- { type: AudioAnalyzerService }
299
+ { type: AudioAnalyzerService },
300
+ { type: PlatformTokenRefreshService },
301
+ { type: NgZone },
302
+ { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
228
303
  ];
229
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport {\n  PipecatClient,\n  RTVIEvent,\n  APIRequest,\n} from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\n\nexport type CallState = 'idle' | 'connecting' | 'connected' | 'talking' | 'listening' | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class VoiceAgentService {\n  private client: PipecatClient | null = null;\n  private transport: InstanceType<typeof WebSocketTransport> | null = null;\n  private userMediaStream: MediaStream | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime: number = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n    this.audioAnalyzer.stop();\n    this.client = null;\n    this.transport = null;\n    if (this.userMediaStream) {\n      this.userMediaStream.getTracks().forEach(track => track.stop());\n      this.userMediaStream = null;\n    }\n    if (this.botAudioElement) {\n      this.botAudioElement.pause();\n      this.botAudioElement.srcObject = null;\n      this.botAudioElement = null;\n    }\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n  }\n\n  constructor(private audioAnalyzer: AudioAnalyzerService) {\n    this.audioAnalyzer.audioLevels$.subscribe(levels => {\n      this.audioLevelsSubject.next(levels);\n    });\n\n    this.audioAnalyzer.isUserSpeaking$.subscribe(isSpeaking => {\n      this.isUserSpeakingSubject.next(isSpeaking);\n      if (isSpeaking && this.callStateSubject.value === 'connected') {\n        this.callStateSubject.next('listening');\n      } else if (!isSpeaking && this.callStateSubject.value === 'listening') {\n        this.callStateSubject.next('connected');\n      }\n    });\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventUrl: string,\n    domainAuthority: string\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const connectUrl = `${baseUrl}/ai/ask-voice`;\n\n      const headers = new Headers();\n      headers.set('Authorization', `Bearer ${token}`);\n      // Do not set Content-Type here — the Pipecat client already sets \"application/json\" once; adding it again produces \"application/json, application/json\"\n      headers.set('domain-authority', domainAuthority);\n      headers.set('eventtoken', eventToken);\n      headers.set('eventurl', eventUrl);\n      headers.set('hive-bot-id', botId);\n      headers.set('x-api-key', apiKey);\n\n      const startBotParams: APIRequest = {\n        endpoint: connectUrl,\n        headers,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      };\n\n      this.transport = new WebSocketTransport();\n      this.client = new PipecatClient({\n        transport: this.transport,\n        enableMic: true,\n        enableCam: false,\n      });\n\n      this.setupEventHandlers();\n\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n\n      await this.client.startBotAndConnect(startBotParams);\n\n      const tracks = this.client.tracks();\n      if (tracks?.local?.audio) {\n        this.userMediaStream = new MediaStream([tracks.local.audio]);\n        this.audioAnalyzer.start(this.userMediaStream);\n      }\n\n      this.isMicMutedSubject.next(!this.client.isMicEnabled);\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Connected');\n      this.callStartTime = Date.now();\n      this.startDurationTimer();\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    try {\n      if (this.durationInterval) {\n        clearInterval(this.durationInterval);\n        this.durationInterval = null;\n      }\n\n      this.audioAnalyzer.stop();\n\n      if (this.client) {\n        await this.client.disconnect();\n        this.client = null;\n      }\n\n      this.transport = null;\n\n      if (this.userMediaStream) {\n        this.userMediaStream.getTracks().forEach(track => track.stop());\n        this.userMediaStream = null;\n      }\n\n      if (this.botAudioElement) {\n        this.botAudioElement.pause();\n        this.botAudioElement.srcObject = null;\n        this.botAudioElement = null;\n      }\n\n      this.callStateSubject.next('ended');\n      this.statusTextSubject.next('Call Ended');\n      // Keep current duration so \"Call Ended\" state can show it (e.g. \"Call Ended 1:04\")\n    } catch (error) {\n      console.error('Error disconnecting voice agent:', error);\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.client) return;\n    const newState = !this.client.isMicEnabled;\n    this.client.enableMic(newState);\n    this.isMicMutedSubject.next(!newState);\n  }\n\n  private setupEventHandlers(): void {\n    if (!this.client) return;\n\n    this.client.on(RTVIEvent.BotStartedSpeaking, () => {\n      this.callStateSubject.next('talking');\n    });\n\n    this.client.on(RTVIEvent.BotStoppedSpeaking, () => {\n      if (this.callStateSubject.value === 'talking') {\n        this.callStateSubject.next('connected');\n      }\n    });\n\n    this.client.on(RTVIEvent.TrackStarted, (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n      if (this.botAudioElement && participant && !participant.local && track.kind === 'audio') {\n        const stream = new MediaStream([track]);\n        this.botAudioElement.srcObject = stream;\n        this.botAudioElement.play().catch(console.error);\n      }\n    });\n\n    this.client.on(RTVIEvent.UserTranscript, (data: { text: string; final?: boolean }) => {\n      this.userTranscriptSubject.next({\n        text: data.text,\n        final: data.final ?? false,\n      });\n    });\n\n    this.client.on(RTVIEvent.BotTranscript, (data: { text?: string }) => {\n      if (data?.text) {\n        this.botTranscriptSubject.next(data.text);\n      }\n    });\n\n    this.client.on(RTVIEvent.Error, () => {\n      this.statusTextSubject.next('Error occurred');\n    });\n\n    this.client.on(RTVIEvent.Disconnected, () => {\n      this.callStateSubject.next('ended');\n      this.statusTextSubject.next('Disconnected');\n      this.disconnect();\n    });\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`\n        );\n      }\n    };\n    updateDuration(); // show 0:00 immediately\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n}\n"]}
304
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    void this.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('[HiveGpt Voice] Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),\n          onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),\n          onBotReady: () => this.ngZone.run(() => this.onBotReady()),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStartedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStoppedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n          this.statusTextSubject.next('Listening...');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            // Brief 'Processing...' while we wait for the bot to respond.\n            this.callStateSubject.next('connected');\n            this.statusTextSubject.next('Processing...');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    // Start the duration timer from the moment the session is live.\n    this.callStartTime = Date.now();\n    this.startDurationTimer();\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    this.isMicMutedSubject.next(false);\n    this.startLocalMicAnalyzer();\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected.\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n    // Bot is initialised — signal that we're now waiting for user input.\n    this.statusTextSubject.next('Listening...');\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    this.callStateSubject.next('talking');\n    this.statusTextSubject.next('Talking...');\n    // Mark user as no longer speaking when bot takes the turn.\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Listening...');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.pcClient) return;\n    const nextMuted = !this.isMicMutedSubject.value;\n    this.pcClient.enableMic(!nextMuted);\n    this.isMicMutedSubject.next(nextMuted);\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const m = Math.floor(elapsed / 60);\n        const s = elapsed % 60;\n        this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);\n      }\n    };\n    tick();\n    this.durationInterval = setInterval(tick, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
@@ -3,6 +3,10 @@ import { CommonModule } from '@angular/common';
3
3
  import { VoiceAgentModalComponent } from './components/voice-agent-modal/voice-agent-modal.component';
4
4
  import { VoiceAgentService } from './services/voice-agent.service';
5
5
  import { AudioAnalyzerService } from './services/audio-analyzer.service';
6
+ /**
7
+ * Voice agent module. Uses @pipecat-ai/client-js + @pipecat-ai/websocket-transport
8
+ * (peer dependencies) for WebSocket transport, RTVI protocol, and audio.
9
+ */
6
10
  export class VoiceAgentModule {
7
11
  }
8
12
  VoiceAgentModule.decorators = [
@@ -15,11 +19,11 @@ VoiceAgentModule.decorators = [
15
19
  ],
16
20
  providers: [
17
21
  VoiceAgentService,
18
- AudioAnalyzerService
22
+ AudioAnalyzerService,
19
23
  ],
20
24
  exports: [
21
25
  VoiceAgentModalComponent
22
26
  ]
23
27
  },] }
24
28
  ];
25
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtYWdlbnQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLWFnZW50Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQztBQUN0RyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQWlCekUsTUFBTSxPQUFPLGdCQUFnQjs7O1lBZjVCLFFBQVEsU0FBQztnQkFDUixZQUFZLEVBQUU7b0JBQ1osd0JBQXdCO2lCQUN6QjtnQkFDRCxPQUFPLEVBQUU7b0JBQ1AsWUFBWTtpQkFDYjtnQkFDRCxTQUFTLEVBQUU7b0JBQ1QsaUJBQWlCO29CQUNqQixvQkFBb0I7aUJBQ3JCO2dCQUNELE9BQU8sRUFBRTtvQkFDUCx3QkFBd0I7aUJBQ3pCO2FBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFZvaWNlQWdlbnRNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vY29tcG9uZW50cy92b2ljZS1hZ2VudC1tb2RhbC92b2ljZS1hZ2VudC1tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VBZ2VudFNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3ZvaWNlLWFnZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgQXVkaW9BbmFseXplclNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2F1ZGlvLWFuYWx5emVyLnNlcnZpY2UnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBWb2ljZUFnZW50TW9kYWxDb21wb25lbnRcbiAgXSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZVxuICBdLFxuICBwcm92aWRlcnM6IFtcbiAgICBWb2ljZUFnZW50U2VydmljZSxcbiAgICBBdWRpb0FuYWx5emVyU2VydmljZVxuICBdLFxuICBleHBvcnRzOiBbXG4gICAgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50XG4gIF1cbn0pXG5leHBvcnQgY2xhc3MgVm9pY2VBZ2VudE1vZHVsZSB7IH1cbiJdfQ==
29
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtYWdlbnQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLWFnZW50Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQztBQUN0RyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUV6RTs7O0dBR0c7QUFnQkgsTUFBTSxPQUFPLGdCQUFnQjs7O1lBZjVCLFFBQVEsU0FBQztnQkFDUixZQUFZLEVBQUU7b0JBQ1osd0JBQXdCO2lCQUN6QjtnQkFDRCxPQUFPLEVBQUU7b0JBQ1AsWUFBWTtpQkFDYjtnQkFDRCxTQUFTLEVBQUU7b0JBQ1QsaUJBQWlCO29CQUNqQixvQkFBb0I7aUJBQ3JCO2dCQUNELE9BQU8sRUFBRTtvQkFDUCx3QkFBd0I7aUJBQ3pCO2FBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFZvaWNlQWdlbnRNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vY29tcG9uZW50cy92b2ljZS1hZ2VudC1tb2RhbC92b2ljZS1hZ2VudC1tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VBZ2VudFNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3ZvaWNlLWFnZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgQXVkaW9BbmFseXplclNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2F1ZGlvLWFuYWx5emVyLnNlcnZpY2UnO1xuXG4vKipcbiAqIFZvaWNlIGFnZW50IG1vZHVsZS4gVXNlcyBAcGlwZWNhdC1haS9jbGllbnQtanMgKyBAcGlwZWNhdC1haS93ZWJzb2NrZXQtdHJhbnNwb3J0XG4gKiAocGVlciBkZXBlbmRlbmNpZXMpIGZvciBXZWJTb2NrZXQgdHJhbnNwb3J0LCBSVFZJIHByb3RvY29sLCBhbmQgYXVkaW8uXG4gKi9cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1xuICAgIFZvaWNlQWdlbnRNb2RhbENvbXBvbmVudFxuICBdLFxuICBpbXBvcnRzOiBbXG4gICAgQ29tbW9uTW9kdWxlXG4gIF0sXG4gIHByb3ZpZGVyczogW1xuICAgIFZvaWNlQWdlbnRTZXJ2aWNlLFxuICAgIEF1ZGlvQW5hbHl6ZXJTZXJ2aWNlLFxuICBdLFxuICBleHBvcnRzOiBbXG4gICAgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50XG4gIF1cbn0pXG5leHBvcnQgY2xhc3MgVm9pY2VBZ2VudE1vZHVsZSB7IH1cbiJdfQ==
@@ -1,4 +1,4 @@
1
1
  import { InjectionToken } from '@angular/core';
2
2
  export const VOICE_MODAL_CONFIG = new InjectionToken('VOICE_MODAL_CONFIG');
3
3
  export const VOICE_MODAL_CLOSE_CALLBACK = new InjectionToken('VOICE_MODAL_CLOSE_CALLBACK');
4
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtbW9kYWwtdG9rZW5zLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLW1vZGFsLXRva2Vucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBZ0IvQyxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLGNBQWMsQ0FBbUIsb0JBQW9CLENBQUMsQ0FBQztBQUM3RixNQUFNLENBQUMsTUFBTSwwQkFBMEIsR0FBRyxJQUFJLGNBQWMsQ0FBYSw0QkFBNEIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGludGVyZmFjZSBWb2ljZU1vZGFsQ29uZmlnIHtcbiAgYXBpVXJsOiBzdHJpbmc7XG4gIHRva2VuOiBzdHJpbmc7XG4gIGJvdElkOiBzdHJpbmc7XG4gIGNvbnZlcnNhdGlvbklkOiBzdHJpbmc7XG4gIGFwaUtleTogc3RyaW5nO1xuICBldmVudFRva2VuOiBzdHJpbmc7XG4gIGV2ZW50VXJsOiBzdHJpbmc7XG4gIGRvbWFpbkF1dGhvcml0eTogc3RyaW5nO1xuICBhZ2VudE5hbWU6IHN0cmluZztcbiAgYWdlbnRSb2xlOiBzdHJpbmc7XG4gIGFnZW50QXZhdGFyPzogc3RyaW5nO1xufVxuXG5leHBvcnQgY29uc3QgVk9JQ0VfTU9EQUxfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPFZvaWNlTW9kYWxDb25maWc+KCdWT0lDRV9NT0RBTF9DT05GSUcnKTtcbmV4cG9ydCBjb25zdCBWT0lDRV9NT0RBTF9DTE9TRV9DQUxMQkFDSyA9IG5ldyBJbmplY3Rpb25Ub2tlbjwoKSA9PiB2b2lkPignVk9JQ0VfTU9EQUxfQ0xPU0VfQ0FMTEJBQ0snKTtcbiJdfQ==
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtbW9kYWwtdG9rZW5zLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLW1vZGFsLXRva2Vucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBb0IvQyxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLGNBQWMsQ0FBbUIsb0JBQW9CLENBQUMsQ0FBQztBQUM3RixNQUFNLENBQUMsTUFBTSwwQkFBMEIsR0FBRyxJQUFJLGNBQWMsQ0FBYSw0QkFBNEIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGludGVyZmFjZSBWb2ljZU1vZGFsQ29uZmlnIHtcbiAgYXBpVXJsOiBzdHJpbmc7XG4gIHRva2VuOiBzdHJpbmc7XG4gIGJvdElkOiBzdHJpbmc7XG4gIGNvbnZlcnNhdGlvbklkOiBzdHJpbmc7XG4gIGFwaUtleTogc3RyaW5nO1xuICBldmVudFRva2VuOiBzdHJpbmc7XG4gIC8qKiBFdmVudCBzY29wZSBpZCAoc2FtZSBhcyBob3N0IGBldmVudElkYCAvIEFQSSBgZXZlbnRzLzpldmVudElkLy4uLmApLiAqL1xuICBldmVudElkOiBzdHJpbmc7XG4gIGV2ZW50VXJsOiBzdHJpbmc7XG4gIGRvbWFpbkF1dGhvcml0eTogc3RyaW5nO1xuICBhZ2VudE5hbWU6IHN0cmluZztcbiAgYWdlbnRSb2xlOiBzdHJpbmc7XG4gIGFnZW50QXZhdGFyPzogc3RyaW5nO1xuICAvKiogU2FtZSBiYXNlIGFzIGhvc3QgYFVTRVJTX0FQSWAgKGUuZy4gYGh0dHBzOi8vZXMtdXNlci5zb2NpYWwyNy5jb20vYXBpYCkgZm9yIHByb2FjdGl2ZSB0b2tlbiByZWZyZXNoLiAqL1xuICB1c2Vyc0FwaVVybD86IHN0cmluZztcbn1cblxuZXhwb3J0IGNvbnN0IFZPSUNFX01PREFMX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxWb2ljZU1vZGFsQ29uZmlnPignVk9JQ0VfTU9EQUxfQ09ORklHJyk7XG5leHBvcnQgY29uc3QgVk9JQ0VfTU9EQUxfQ0xPU0VfQ0FMTEJBQ0sgPSBuZXcgSW5qZWN0aW9uVG9rZW48KCkgPT4gdm9pZD4oJ1ZPSUNFX01PREFMX0NMT1NFX0NBTExCQUNLJyk7XG4iXX0=