@hivegpt/hiveai-angular 0.0.574 → 0.0.575
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/bundles/hivegpt-hiveai-angular.umd.js +525 -270
- package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
- package/esm2015/lib/components/chat-drawer/chat-drawer.component.js +6 -6
- package/esm2015/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +85 -54
- package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +153 -63
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +148 -84
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +79 -34
- package/fesm2015/hivegpt-hiveai-angular.js +464 -235
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +7 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +37 -3
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +19 -6
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +5 -12
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -3,59 +3,87 @@ import { Injectable, NgZone } from '@angular/core';
|
|
|
3
3
|
import { BehaviorSubject } from 'rxjs';
|
|
4
4
|
import Daily from '@daily-co/daily-js';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Daily.js WebRTC client for voice agent audio.
|
|
8
|
+
* Responsibilities:
|
|
9
|
+
* - Create and manage Daily CallObject
|
|
10
|
+
* - Join Daily room using room_url
|
|
11
|
+
* - Handle mic capture + speaker playback
|
|
12
|
+
* - Bot speaking detection via AnalyserNode on remote track (instant)
|
|
13
|
+
* - User speaking detection via active-speaker-change
|
|
14
|
+
* - Expose speaking$ (bot speaking), userSpeaking$ (user speaking), micMuted$
|
|
15
|
+
* - Expose localStream$ for waveform visualization (AudioAnalyzerService)
|
|
16
|
+
*/
|
|
6
17
|
export class DailyVoiceClientService {
|
|
7
18
|
constructor(ngZone) {
|
|
8
19
|
this.ngZone = ngZone;
|
|
9
20
|
this.callObject = null;
|
|
10
21
|
this.localStream = null;
|
|
11
22
|
this.localSessionId = null;
|
|
23
|
+
/** Explicit playback of remote (bot) audio; required in some browsers. */
|
|
12
24
|
this.remoteAudioElement = null;
|
|
25
|
+
/** AnalyserNode-based remote audio monitor for instant bot speaking detection. */
|
|
13
26
|
this.remoteAudioContext = null;
|
|
14
|
-
|
|
27
|
+
/** Poll interval id (~100ms); named historically when RAF was used. */
|
|
28
|
+
this.remoteSpeakingPollId = null;
|
|
15
29
|
this.speakingSubject = new BehaviorSubject(false);
|
|
16
30
|
this.userSpeakingSubject = new BehaviorSubject(false);
|
|
17
|
-
this.micMutedSubject = new BehaviorSubject(
|
|
31
|
+
this.micMutedSubject = new BehaviorSubject(false);
|
|
18
32
|
this.localStreamSubject = new BehaviorSubject(null);
|
|
33
|
+
/** True when bot (remote participant) is the active speaker. */
|
|
19
34
|
this.speaking$ = this.speakingSubject.asObservable();
|
|
35
|
+
/** True when user (local participant) is the active speaker. */
|
|
20
36
|
this.userSpeaking$ = this.userSpeakingSubject.asObservable();
|
|
37
|
+
/** True when mic is muted. */
|
|
21
38
|
this.micMuted$ = this.micMutedSubject.asObservable();
|
|
39
|
+
/** Emits local mic stream for waveform visualization. */
|
|
22
40
|
this.localStream$ = this.localStreamSubject.asObservable();
|
|
23
41
|
}
|
|
24
|
-
|
|
42
|
+
/**
|
|
43
|
+
* Connect to Daily room. Acquires mic first for waveform, then joins with audio.
|
|
44
|
+
* @param roomUrl Daily room URL (from room_created)
|
|
45
|
+
* @param token Optional meeting token
|
|
46
|
+
* @param existingStream Optional pre-acquired mic (avoids a second getUserMedia / extra prompts on some browsers)
|
|
47
|
+
*/
|
|
48
|
+
connect(roomUrl, token, existingStream) {
|
|
25
49
|
return __awaiter(this, void 0, void 0, function* () {
|
|
26
50
|
if (this.callObject) {
|
|
27
51
|
yield this.disconnect();
|
|
28
52
|
}
|
|
29
53
|
try {
|
|
30
|
-
|
|
31
|
-
const stream =
|
|
54
|
+
const hasLiveTrack = !!(existingStream === null || existingStream === void 0 ? void 0 : existingStream.getAudioTracks().some((t) => t.readyState === 'live'));
|
|
55
|
+
const stream = hasLiveTrack
|
|
56
|
+
? existingStream
|
|
57
|
+
: yield navigator.mediaDevices.getUserMedia({ audio: true });
|
|
32
58
|
const audioTrack = stream.getAudioTracks()[0];
|
|
33
|
-
if (!audioTrack)
|
|
59
|
+
if (!audioTrack) {
|
|
60
|
+
stream.getTracks().forEach((t) => t.stop());
|
|
34
61
|
throw new Error('No audio track');
|
|
62
|
+
}
|
|
35
63
|
this.localStream = stream;
|
|
36
64
|
this.localStreamSubject.next(stream);
|
|
65
|
+
// Create audio-only call object
|
|
66
|
+
// videoSource: false = no camera, audioSource = our mic track
|
|
37
67
|
const callObject = Daily.createCallObject({
|
|
38
68
|
videoSource: false,
|
|
39
69
|
audioSource: audioTrack,
|
|
40
70
|
});
|
|
41
71
|
this.callObject = callObject;
|
|
42
72
|
this.setupEventHandlers(callObject);
|
|
43
|
-
//
|
|
44
|
-
|
|
45
|
-
this.micMutedSubject.next(true);
|
|
73
|
+
// Join room; Daily handles playback of remote (bot) audio automatically.
|
|
74
|
+
// Only pass token when it's a non-empty string (Daily rejects undefined/non-string).
|
|
46
75
|
const joinOptions = { url: roomUrl };
|
|
47
|
-
if (typeof token === 'string' && token.trim()) {
|
|
76
|
+
if (typeof token === 'string' && token.trim() !== '') {
|
|
48
77
|
joinOptions.token = token;
|
|
49
78
|
}
|
|
50
79
|
yield callObject.join(joinOptions);
|
|
51
|
-
console.log(`[VoiceDebug]
|
|
80
|
+
console.log(`[VoiceDebug] Room connected (Daily join complete) — ${new Date().toISOString()}`);
|
|
52
81
|
const participants = callObject.participants();
|
|
53
82
|
if (participants === null || participants === void 0 ? void 0 : participants.local) {
|
|
54
83
|
this.localSessionId = participants.local.session_id;
|
|
55
84
|
}
|
|
56
|
-
//
|
|
57
|
-
callObject.
|
|
58
|
-
this.micMutedSubject.next(true);
|
|
85
|
+
// Initial mute state: Daily starts with audio on
|
|
86
|
+
this.micMutedSubject.next(!callObject.localAudio());
|
|
59
87
|
}
|
|
60
88
|
catch (err) {
|
|
61
89
|
this.cleanup();
|
|
@@ -64,6 +92,8 @@ export class DailyVoiceClientService {
|
|
|
64
92
|
});
|
|
65
93
|
}
|
|
66
94
|
setupEventHandlers(call) {
|
|
95
|
+
// active-speaker-change: used ONLY for user speaking detection.
|
|
96
|
+
// Bot speaking is detected by our own AnalyserNode (instant, no debounce).
|
|
67
97
|
call.on('active-speaker-change', (event) => {
|
|
68
98
|
this.ngZone.run(() => {
|
|
69
99
|
var _a;
|
|
@@ -72,20 +102,23 @@ export class DailyVoiceClientService {
|
|
|
72
102
|
this.userSpeakingSubject.next(false);
|
|
73
103
|
return;
|
|
74
104
|
}
|
|
75
|
-
|
|
105
|
+
const isLocal = peerId === this.localSessionId;
|
|
106
|
+
this.userSpeakingSubject.next(isLocal);
|
|
76
107
|
});
|
|
77
108
|
});
|
|
109
|
+
// track-started / track-stopped: set up remote audio playback + AnalyserNode monitor.
|
|
78
110
|
call.on('track-started', (event) => {
|
|
79
111
|
this.ngZone.run(() => {
|
|
80
|
-
var _a, _b, _c, _d
|
|
112
|
+
var _a, _b, _c, _d;
|
|
81
113
|
const p = event === null || event === void 0 ? void 0 : event.participant;
|
|
82
114
|
const type = (_a = event === null || event === void 0 ? void 0 : event.type) !== null && _a !== void 0 ? _a : (_b = event === null || event === void 0 ? void 0 : event.track) === null || _b === void 0 ? void 0 : _b.kind;
|
|
115
|
+
const track = event === null || event === void 0 ? void 0 : event.track;
|
|
83
116
|
if (p && !p.local && type === 'audio') {
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
this.playRemoteTrack(
|
|
88
|
-
this.monitorRemoteAudio(
|
|
117
|
+
console.log(`[VoiceDebug] Got audio track from backend (track-started) — readyState=${track === null || track === void 0 ? void 0 : track.readyState}, muted=${track === null || track === void 0 ? void 0 : track.muted} — ${new Date().toISOString()}`);
|
|
118
|
+
const audioTrack = track !== null && track !== void 0 ? track : (_d = (_c = p.tracks) === null || _c === void 0 ? void 0 : _c.audio) === null || _d === void 0 ? void 0 : _d.track;
|
|
119
|
+
if (audioTrack && typeof audioTrack === 'object') {
|
|
120
|
+
this.playRemoteTrack(audioTrack);
|
|
121
|
+
this.monitorRemoteAudio(audioTrack);
|
|
89
122
|
}
|
|
90
123
|
}
|
|
91
124
|
});
|
|
@@ -104,27 +137,57 @@ export class DailyVoiceClientService {
|
|
|
104
137
|
call.on('left-meeting', () => {
|
|
105
138
|
this.ngZone.run(() => this.cleanup());
|
|
106
139
|
});
|
|
107
|
-
call.on('error', (
|
|
108
|
-
|
|
109
|
-
|
|
140
|
+
call.on('error', (event) => {
|
|
141
|
+
this.ngZone.run(() => {
|
|
142
|
+
var _a;
|
|
143
|
+
console.error('DailyVoiceClient: Daily error', (_a = event === null || event === void 0 ? void 0 : event.errorMsg) !== null && _a !== void 0 ? _a : event);
|
|
144
|
+
this.cleanup();
|
|
145
|
+
});
|
|
110
146
|
});
|
|
111
147
|
}
|
|
148
|
+
/**
|
|
149
|
+
* Play remote (bot) audio track via a dedicated audio element.
|
|
150
|
+
* Required in many browsers where Daily's internal playback does not output to speakers.
|
|
151
|
+
*/
|
|
112
152
|
playRemoteTrack(track) {
|
|
113
153
|
this.stopRemoteAudio();
|
|
114
154
|
try {
|
|
155
|
+
console.log(`[VoiceDebug] playRemoteTrack called — track.readyState=${track.readyState}, track.muted=${track.muted} — ${new Date().toISOString()}`);
|
|
156
|
+
track.onunmute = () => {
|
|
157
|
+
console.log(`[VoiceDebug] Remote audio track UNMUTED (audio data arriving) — ${new Date().toISOString()}`);
|
|
158
|
+
};
|
|
115
159
|
const stream = new MediaStream([track]);
|
|
116
160
|
const audio = new Audio();
|
|
117
161
|
audio.autoplay = true;
|
|
118
162
|
audio.srcObject = stream;
|
|
119
163
|
this.remoteAudioElement = audio;
|
|
120
|
-
audio.
|
|
121
|
-
console.
|
|
122
|
-
}
|
|
164
|
+
audio.onplaying = () => {
|
|
165
|
+
console.log(`[VoiceDebug] Audio element PLAYING (browser started playback) — ${new Date().toISOString()}`);
|
|
166
|
+
};
|
|
167
|
+
let firstTimeUpdate = true;
|
|
168
|
+
audio.ontimeupdate = () => {
|
|
169
|
+
if (firstTimeUpdate) {
|
|
170
|
+
firstTimeUpdate = false;
|
|
171
|
+
console.log(`[VoiceDebug] Audio element first TIMEUPDATE (actual audio output) — ${new Date().toISOString()}`);
|
|
172
|
+
}
|
|
173
|
+
};
|
|
174
|
+
const p = audio.play();
|
|
175
|
+
if (p && typeof p.then === 'function') {
|
|
176
|
+
p.then(() => {
|
|
177
|
+
console.log(`[VoiceDebug] audio.play() resolved — ${new Date().toISOString()}`);
|
|
178
|
+
}).catch((err) => {
|
|
179
|
+
console.warn('DailyVoiceClient: remote audio play failed (may need user gesture)', err);
|
|
180
|
+
});
|
|
181
|
+
}
|
|
123
182
|
}
|
|
124
183
|
catch (err) {
|
|
125
|
-
console.warn('
|
|
184
|
+
console.warn('DailyVoiceClient: failed to create remote audio element', err);
|
|
126
185
|
}
|
|
127
186
|
}
|
|
187
|
+
/**
|
|
188
|
+
* Monitor remote audio track energy via AnalyserNode.
|
|
189
|
+
* Polls at ~10Hz; sufficient for speaking detection vs ~60fps RAF.
|
|
190
|
+
*/
|
|
128
191
|
monitorRemoteAudio(track) {
|
|
129
192
|
this.stopRemoteAudioMonitor();
|
|
130
193
|
try {
|
|
@@ -134,81 +197,108 @@ export class DailyVoiceClientService {
|
|
|
134
197
|
analyser.fftSize = 256;
|
|
135
198
|
source.connect(analyser);
|
|
136
199
|
this.remoteAudioContext = ctx;
|
|
137
|
-
const
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
const
|
|
141
|
-
|
|
200
|
+
const dataArray = new Uint8Array(analyser.frequencyBinCount);
|
|
201
|
+
const THRESHOLD = 5;
|
|
202
|
+
const SILENCE_MS = 1500;
|
|
203
|
+
const POLL_MS = 100;
|
|
204
|
+
let lastSoundTime = 0;
|
|
205
|
+
let isSpeaking = false;
|
|
206
|
+
this.remoteSpeakingPollId = setInterval(() => {
|
|
207
|
+
if (!this.remoteAudioContext) {
|
|
208
|
+
if (this.remoteSpeakingPollId) {
|
|
209
|
+
clearInterval(this.remoteSpeakingPollId);
|
|
210
|
+
this.remoteSpeakingPollId = null;
|
|
211
|
+
}
|
|
142
212
|
return;
|
|
143
|
-
|
|
144
|
-
|
|
213
|
+
}
|
|
214
|
+
analyser.getByteFrequencyData(dataArray);
|
|
215
|
+
let sum = 0;
|
|
216
|
+
for (let i = 0; i < dataArray.length; i++) {
|
|
217
|
+
sum += dataArray[i];
|
|
218
|
+
}
|
|
219
|
+
const avg = sum / dataArray.length;
|
|
145
220
|
const now = Date.now();
|
|
146
|
-
if (avg >
|
|
147
|
-
|
|
148
|
-
if (!
|
|
149
|
-
|
|
221
|
+
if (avg > THRESHOLD) {
|
|
222
|
+
lastSoundTime = now;
|
|
223
|
+
if (!isSpeaking) {
|
|
224
|
+
isSpeaking = true;
|
|
225
|
+
console.log(`[VoiceDebug] Bot audio energy detected (speaking=true) — avg=${avg.toFixed(1)} — ${new Date().toISOString()}`);
|
|
150
226
|
this.ngZone.run(() => {
|
|
151
227
|
this.userSpeakingSubject.next(false);
|
|
152
228
|
this.speakingSubject.next(true);
|
|
153
229
|
});
|
|
154
230
|
}
|
|
155
231
|
}
|
|
156
|
-
else if (
|
|
157
|
-
|
|
232
|
+
else if (isSpeaking && now - lastSoundTime > SILENCE_MS) {
|
|
233
|
+
isSpeaking = false;
|
|
234
|
+
console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);
|
|
158
235
|
this.ngZone.run(() => this.speakingSubject.next(false));
|
|
159
236
|
}
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
237
|
+
}, POLL_MS);
|
|
238
|
+
}
|
|
239
|
+
catch (err) {
|
|
240
|
+
console.warn('DailyVoiceClient: failed to create remote audio monitor', err);
|
|
163
241
|
}
|
|
164
|
-
catch (_a) { }
|
|
165
242
|
}
|
|
166
243
|
stopRemoteAudioMonitor() {
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
244
|
+
if (this.remoteSpeakingPollId !== null) {
|
|
245
|
+
clearInterval(this.remoteSpeakingPollId);
|
|
246
|
+
this.remoteSpeakingPollId = null;
|
|
247
|
+
}
|
|
248
|
+
if (this.remoteAudioContext) {
|
|
249
|
+
this.remoteAudioContext.close().catch(() => { });
|
|
250
|
+
this.remoteAudioContext = null;
|
|
171
251
|
}
|
|
172
|
-
(_a = this.remoteAudioContext) === null || _a === void 0 ? void 0 : _a.close().catch(() => { });
|
|
173
|
-
this.remoteAudioContext = null;
|
|
174
252
|
}
|
|
175
253
|
stopRemoteAudio() {
|
|
176
254
|
if (this.remoteAudioElement) {
|
|
177
|
-
|
|
178
|
-
|
|
255
|
+
try {
|
|
256
|
+
this.remoteAudioElement.pause();
|
|
257
|
+
this.remoteAudioElement.srcObject = null;
|
|
258
|
+
}
|
|
259
|
+
catch (_) { }
|
|
179
260
|
this.remoteAudioElement = null;
|
|
180
261
|
}
|
|
181
262
|
}
|
|
263
|
+
/** Set mic muted state. */
|
|
182
264
|
setMuted(muted) {
|
|
183
265
|
if (!this.callObject)
|
|
184
266
|
return;
|
|
185
267
|
this.callObject.setLocalAudio(!muted);
|
|
186
268
|
this.micMutedSubject.next(muted);
|
|
187
|
-
console.log(`[VoiceDebug] Mic ${muted ? 'MUTED' : 'UNMUTED'}`);
|
|
188
269
|
}
|
|
270
|
+
/** Disconnect and cleanup. */
|
|
189
271
|
disconnect() {
|
|
190
272
|
return __awaiter(this, void 0, void 0, function* () {
|
|
191
|
-
if (!this.callObject)
|
|
192
|
-
|
|
273
|
+
if (!this.callObject) {
|
|
274
|
+
this.cleanup();
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
193
277
|
try {
|
|
194
278
|
yield this.callObject.leave();
|
|
195
279
|
}
|
|
196
|
-
catch (
|
|
280
|
+
catch (e) {
|
|
281
|
+
// ignore
|
|
282
|
+
}
|
|
197
283
|
this.cleanup();
|
|
198
284
|
});
|
|
199
285
|
}
|
|
200
286
|
cleanup() {
|
|
201
|
-
var _a, _b;
|
|
202
287
|
this.stopRemoteAudioMonitor();
|
|
203
288
|
this.stopRemoteAudio();
|
|
204
|
-
(
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
289
|
+
if (this.callObject) {
|
|
290
|
+
this.callObject.destroy().catch(() => { });
|
|
291
|
+
this.callObject = null;
|
|
292
|
+
}
|
|
293
|
+
if (this.localStream) {
|
|
294
|
+
this.localStream.getTracks().forEach((t) => t.stop());
|
|
295
|
+
this.localStream = null;
|
|
296
|
+
}
|
|
208
297
|
this.localSessionId = null;
|
|
209
298
|
this.speakingSubject.next(false);
|
|
210
299
|
this.userSpeakingSubject.next(false);
|
|
211
300
|
this.localStreamSubject.next(null);
|
|
301
|
+
// Keep last micMuted state; will reset on next connect
|
|
212
302
|
}
|
|
213
303
|
}
|
|
214
304
|
DailyVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function DailyVoiceClientService_Factory() { return new DailyVoiceClientService(i0.ɵɵinject(i0.NgZone)); }, token: DailyVoiceClientService, providedIn: "root" });
|
|
@@ -220,4 +310,4 @@ DailyVoiceClientService.decorators = [
|
|
|
220
310
|
DailyVoiceClientService.ctorParameters = () => [
|
|
221
311
|
{ type: NgZone }
|
|
222
312
|
];
|
|
223
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"daily-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/daily-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport Daily from '@daily-co/daily-js';\nimport type { DailyCall, DailyParticipant } from '@daily-co/daily-js';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class DailyVoiceClientService {\n  private callObject: DailyCall | null = null;\n  private localStream: MediaStream | null = null;\n  private localSessionId: string | null = null;\n  private remoteAudioElement: HTMLAudioElement | null = null;\n\n  private remoteAudioContext: AudioContext | null = null;\n  private remoteSpeakingRAF: number | null = null;\n\n  private speakingSubject = new BehaviorSubject<boolean>(false);\n  private userSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private micMutedSubject = new BehaviorSubject<boolean>(true); // 🔴 default muted\n  private localStreamSubject = new BehaviorSubject<MediaStream | null>(null);\n\n  speaking$ = this.speakingSubject.asObservable();\n  userSpeaking$ = this.userSpeakingSubject.asObservable();\n  micMuted$ = this.micMutedSubject.asObservable();\n  localStream$ = this.localStreamSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  async connect(roomUrl: string, token?: string): Promise<void> {\n    if (this.callObject) {\n      await this.disconnect();\n    }\n\n    try {\n      // 🎤 Get mic (kept for waveform)\n      const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n      const audioTrack = stream.getAudioTracks()[0];\n      if (!audioTrack) throw new Error('No audio track');\n\n      this.localStream = stream;\n      this.localStreamSubject.next(stream);\n\n      const callObject = Daily.createCallObject({\n        videoSource: false,\n        audioSource: audioTrack,\n      });\n\n      this.callObject = callObject;\n      this.setupEventHandlers(callObject);\n\n      // 🔴 Ensure mic is OFF before join\n      callObject.setLocalAudio(false);\n      this.micMutedSubject.next(true);\n\n      const joinOptions: { url: string; token?: string } = { url: roomUrl };\n      if (typeof token === 'string' && token.trim()) {\n        joinOptions.token = token;\n      }\n\n      await callObject.join(joinOptions);\n\n      console.log(`[VoiceDebug] Joined room — ${new Date().toISOString()}`);\n\n      const participants = callObject.participants();\n      if (participants?.local) {\n        this.localSessionId = participants.local.session_id;\n      }\n\n      // 🔴 Force sync again (Daily sometimes overrides)\n      callObject.setLocalAudio(false);\n      this.micMutedSubject.next(true);\n    } catch (err) {\n      this.cleanup();\n      throw err;\n    }\n  }\n\n  private setupEventHandlers(call: DailyCall): void {\n    call.on('active-speaker-change', (event: any) => {\n      this.ngZone.run(() => {\n        const peerId = event?.activeSpeaker?.peerId;\n        if (!peerId || !this.localSessionId) {\n          this.userSpeakingSubject.next(false);\n          return;\n        }\n        this.userSpeakingSubject.next(peerId === this.localSessionId);\n      });\n    });\n\n    call.on('track-started', (event: any) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n\n        if (p && !p.local && type === 'audio') {\n          const track = event.track ?? p?.tracks?.audio?.track;\n\n          if (track) {\n            console.log('[VoiceDebug] Remote audio track received');\n            this.playRemoteTrack(track);\n            this.monitorRemoteAudio(track);\n          }\n        }\n      });\n    });\n\n    call.on('track-stopped', (event: any) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n\n        if (p && !p.local && type === 'audio') {\n          this.stopRemoteAudioMonitor();\n          this.stopRemoteAudio();\n        }\n      });\n    });\n\n    call.on('left-meeting', () => {\n      this.ngZone.run(() => this.cleanup());\n    });\n\n    call.on('error', (e: any) => {\n      console.error('Daily error:', e);\n      this.cleanup();\n    });\n  }\n\n  private playRemoteTrack(track: MediaStreamTrack): void {\n    this.stopRemoteAudio();\n\n    try {\n      const stream = new MediaStream([track]);\n      const audio = new Audio();\n      audio.autoplay = true;\n      audio.srcObject = stream;\n\n      this.remoteAudioElement = audio;\n\n      audio.play().catch(() => {\n        console.warn('Autoplay blocked');\n      });\n    } catch (err) {\n      console.warn('Audio playback error', err);\n    }\n  }\n\n  private monitorRemoteAudio(track: MediaStreamTrack): void {\n    this.stopRemoteAudioMonitor();\n\n    try {\n      const ctx = new AudioContext();\n      const source = ctx.createMediaStreamSource(new MediaStream([track]));\n      const analyser = ctx.createAnalyser();\n\n      analyser.fftSize = 256;\n      source.connect(analyser);\n\n      this.remoteAudioContext = ctx;\n\n      const data = new Uint8Array(analyser.frequencyBinCount);\n\n      let speaking = false;\n      let lastSound = 0;\n\n      const loop = () => {\n        if (!this.remoteAudioContext) return;\n\n        analyser.getByteFrequencyData(data);\n        const avg = data.reduce((a, b) => a + b, 0) / data.length;\n        const now = Date.now();\n\n        if (avg > 5) {\n          lastSound = now;\n          if (!speaking) {\n            speaking = true;\n            this.ngZone.run(() => {\n              this.userSpeakingSubject.next(false);\n              this.speakingSubject.next(true);\n            });\n          }\n        } else if (speaking && now - lastSound > 1500) {\n          speaking = false;\n          this.ngZone.run(() => this.speakingSubject.next(false));\n        }\n\n        this.remoteSpeakingRAF = requestAnimationFrame(loop);\n      };\n\n      this.remoteSpeakingRAF = requestAnimationFrame(loop);\n    } catch {}\n  }\n\n  private stopRemoteAudioMonitor(): void {\n    if (this.remoteSpeakingRAF) {\n      cancelAnimationFrame(this.remoteSpeakingRAF);\n      this.remoteSpeakingRAF = null;\n    }\n    this.remoteAudioContext?.close().catch(() => {});\n    this.remoteAudioContext = null;\n  }\n\n  private stopRemoteAudio(): void {\n    if (this.remoteAudioElement) {\n      this.remoteAudioElement.pause();\n      this.remoteAudioElement.srcObject = null;\n      this.remoteAudioElement = null;\n    }\n  }\n\n  setMuted(muted: boolean): void {\n    if (!this.callObject) return;\n\n    this.callObject.setLocalAudio(!muted);\n    this.micMutedSubject.next(muted);\n\n    console.log(`[VoiceDebug] Mic ${muted ? 'MUTED' : 'UNMUTED'}`);\n  }\n\n  async disconnect(): Promise<void> {\n    if (!this.callObject) return this.cleanup();\n\n    try {\n      await this.callObject.leave();\n    } catch {}\n\n    this.cleanup();\n  }\n\n  private cleanup(): void {\n    this.stopRemoteAudioMonitor();\n    this.stopRemoteAudio();\n\n    this.callObject?.destroy().catch(() => {});\n    this.callObject = null;\n\n    this.localStream?.getTracks().forEach((t) => t.stop());\n    this.localStream = null;\n\n    this.localSessionId = null;\n\n    this.speakingSubject.next(false);\n    this.userSpeakingSubject.next(false);\n    this.localStreamSubject.next(null);\n  }\n}\n"]}
|
|
313
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"daily-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/daily-voice-client.service.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AACnD,OAAO,KAAK,MAAM,oBAAoB,CAAC;;AAGvC;;;;;;;;;;GAUG;AAIH,MAAM,OAAO,uBAAuB;IA8BlC,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;QA7B1B,eAAU,GAAqB,IAAI,CAAC;QACpC,gBAAW,GAAuB,IAAI,CAAC;QACvC,mBAAc,GAAkB,IAAI,CAAC;QAC7C,0EAA0E;QAClE,uBAAkB,GAA4B,IAAI,CAAC;QAE3D,kFAAkF;QAC1E,uBAAkB,GAAwB,IAAI,CAAC;QACvD,uEAAuE;QAC/D,yBAAoB,GAA0C,IAAI,CAAC;QAEnE,oBAAe,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACtD,wBAAmB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAC1D,oBAAe,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACtD,uBAAkB,GAAG,IAAI,eAAe,CAAqB,IAAI,CAAC,CAAC;QAE3E,gEAAgE;QAChE,cAAS,GAAwB,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QAErE,gEAAgE;QAChE,kBAAa,GAAwB,IAAI,CAAC,mBAAmB,CAAC,YAAY,EAAE,CAAC;QAE7E,8BAA8B;QAC9B,cAAS,GAAwB,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QAErE,yDAAyD;QACzD,iBAAY,GACV,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;IAEJ,CAAC;IAEtC;;;;;OAKG;IACG,OAAO,CACX,OAAe,EACf,KAAc,EACd,cAA4B;;YAE5B,IAAI,IAAI,CAAC,UAAU,EAAE;gBACnB,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;aACzB;YAED,IAAI;gBACF,MAAM,YAAY,GAChB,CAAC,CAAC,CAAA,cAAc,aAAd,cAAc,uBAAd,cAAc,CAAE,cAAc,GAAG,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,KAAK,MAAM,CAAC,CAAA,CAAC;gBAC1E,MAAM,MAAM,GAAG,YAAY;oBACzB,CAAC,CAAC,cAAe;oBACjB,CAAC,CAAC,MAAM,SAAS,CAAC,YAAY,CAAC,YAAY,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;gBAC/D,MAAM,UAAU,GAAG,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC9C,IAAI,CAAC,UAAU,EAAE;oBACf,MAAM,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;oBAC5C,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;iBACnC;gBAED,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC;gBAC1B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAErC,gCAAgC;gBAChC,8DAA8D;gBAC9D,MAAM,UAAU,GAAG,KAAK,CAAC,gBAAgB,CAAC;oBACxC,WAAW,EAAE,KAAK;oBAClB,WAAW,EAAE,UAAU;iBACxB,CAAC,CAAC;gBAEH,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;gBAE7B,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;gBAEpC,yEAAyE;gBACzE,qFAAqF;gBACrF,MAAM,WAAW,GAAoC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC;gBACtE,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE;oBACpD,WAAW,CAAC,KAAK,GAAG,KAAK,CAAC;iBAC3B;gBACD,MAAM,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACnC,OAAO,CAAC,GAAG,CAAC,uDAAuD,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;gBAE/F,MAAM,YAAY,GAAG,UAAU,CAAC,YAAY,EAAE,CAAC;gBAC/C,IAAI,YAAY,aAAZ,YAAY,uBAAZ,YAAY,CAAE,KAAK,EAAE;oBACvB,IAAI,CAAC,cAAc,GAAG,YAAY,CAAC,KAAK,CAAC,UAAU,CAAC;iBACrD;gBAED,iDAAiD;gBACjD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC,CAAC;aACrD;YAAC,OAAO,GAAG,EAAE;gBACZ,IAAI,CAAC,OAAO,EAAE,CAAC;gBACf,MAAM,GAAG,CAAC;aACX;QACH,CAAC;KAAA;IAEO,kBAAkB,CAAC,IAAe;QACxC,gEAAgE;QAChE,2EAA2E;QAC3E,IAAI,CAAC,EAAE,CAAC,uBAAuB,EAAE,CAAC,KAA8C,EAAE,EAAE;YAClF,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;;gBACnB,MAAM,MAAM,GAAG,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,aAAa,0CAAE,MAAM,CAAC;gBAC5C,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;oBACnC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;oBACrC,OAAO;iBACR;gBACD,MAAM,OAAO,GAAG,MAAM,KAAK,IAAI,CAAC,cAAc,CAAC;gBAC/C,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACzC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,sFAAsF;QACtF,IAAI,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,KAAyF,EAAE,EAAE;YACrH,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;;gBACnB,MAAM,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,WAAW,CAAC;gBAC7B,MAAM,IAAI,GAAG,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,IAAI,mCAAI,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,0CAAE,IAAI,CAAC;gBAC/C,MAAM,KAAK,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,CAAC;gBAC3B,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,KAAK,OAAO,EAAE;oBACrC,OAAO,CAAC,GAAG,CAAC,0EAA0E,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,UAAU,WAAW,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,MAAM,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;oBAChK,MAAM,UAAU,GAAG,KAAK,aAAL,KAAK,cAAL,KAAK,GAAI,MAAA,MAAC,CAA2D,CAAC,MAAM,0CAAE,KAAK,0CAAE,KAAK,CAAC;oBAC9G,IAAI,UAAU,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE;wBAChD,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;wBACjC,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;qBACrC;iBACF;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,KAAyF,EAAE,EAAE;YACrH,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;;gBACnB,MAAM,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,WAAW,CAAC;gBAC7B,MAAM,IAAI,GAAG,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,IAAI,mCAAI,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,0CAAE,IAAI,CAAC;gBAC/C,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,KAAK,OAAO,EAAE;oBACrC,IAAI,CAAC,sBAAsB,EAAE,CAAC;oBAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;iBACxB;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,cAAc,EAAE,GAAG,EAAE;YAC3B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,KAA6B,EAAE,EAAE;YACjD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;;gBACnB,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,QAAQ,mCAAI,KAAK,CAAC,CAAC;gBACzE,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACK,eAAe,CAAC,KAAuB;QAC7C,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI;YACF,OAAO,CAAC,GAAG,CAAC,0DAA0D,KAAK,CAAC,UAAU,iBAAiB,KAAK,CAAC,KAAK,MAAM,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;YAEpJ,KAAK,CAAC,QAAQ,GAAG,GAAG,EAAE;gBACpB,OAAO,CAAC,GAAG,CAAC,mEAAmE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;YAC7G,CAAC,CAAC;YAEF,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;YACxC,MAAM,KAAK,GAAG,IAAI,KAAK,EAAE,CAAC;YAC1B,KAAK,CAAC,QAAQ,GAAG,IAAI,CAAC;YACtB,KAAK,CAAC,SAAS,GAAG,MAAM,CAAC;YACzB,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAEhC,KAAK,CAAC,SAAS,GAAG,GAAG,EAAE;gBACrB,OAAO,CAAC,GAAG,CAAC,mEAAmE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;YAC7G,CAAC,CAAC;YAEF,IAAI,eAAe,GAAG,IAAI,CAAC;YAC3B,KAAK,CAAC,YAAY,GAAG,GAAG,EAAE;gBACxB,IAAI,eAAe,EAAE;oBACnB,eAAe,GAAG,KAAK,CAAC;oBACxB,OAAO,CAAC,GAAG,CAAC,uEAAuE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;iBAChH;YACH,CAAC,CAAC;YAEF,MAAM,CAAC,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;YACvB,IAAI,CAAC,IAAI,OAAO,CAAC,CAAC,IAAI,KAAK,UAAU,EAAE;gBACrC,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE;oBACV,OAAO,CAAC,GAAG,CAAC,wCAAwC,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;gBAClF,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;oBACf,OAAO,CAAC,IAAI,CAAC,oEAAoE,EAAE,GAAG,CAAC,CAAC;gBAC1F,CAAC,CAAC,CAAC;aACJ;SACF;QAAC,OAAO,GAAG,EAAE;YACZ,OAAO,CAAC,IAAI,CAAC,yDAAyD,EAAE,GAAG,CAAC,CAAC;SAC9E;IACH,CAAC;IAED;;;OAGG;IACK,kBAAkB,CAAC,KAAuB;QAChD,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAC9B,IAAI;YACF,MAAM,GAAG,GAAG,IAAI,YAAY,EAAE,CAAC;YAC/B,MAAM,MAAM,GAAG,GAAG,CAAC,uBAAuB,CAAC,IAAI,WAAW,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACrE,MAAM,QAAQ,GAAG,GAAG,CAAC,cAAc,EAAE,CAAC;YACtC,QAAQ,CAAC,OAAO,GAAG,GAAG,CAAC;YACvB,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YACzB,IAAI,CAAC,kBAAkB,GAAG,GAAG,CAAC;YAE9B,MAAM,SAAS,GAAG,IAAI,UAAU,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC;YAC7D,MAAM,SAAS,GAAG,CAAC,CAAC;YACpB,MAAM,UAAU,GAAG,IAAI,CAAC;YACxB,MAAM,OAAO,GAAG,GAAG,CAAC;YACpB,IAAI,aAAa,GAAG,CAAC,CAAC;YACtB,IAAI,UAAU,GAAG,KAAK,CAAC;YAEvB,IAAI,CAAC,oBAAoB,GAAG,WAAW,CAAC,GAAG,EAAE;gBAC3C,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE;oBAC5B,IAAI,IAAI,CAAC,oBAAoB,EAAE;wBAC7B,aAAa,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;wBACzC,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;qBAClC;oBACD,OAAO;iBACR;gBACD,QAAQ,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;gBAEzC,IAAI,GAAG,GAAG,CAAC,CAAC;gBACZ,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;oBACzC,GAAG,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC;iBACrB;gBACD,MAAM,GAAG,GAAG,GAAG,GAAG,SAAS,CAAC,MAAM,CAAC;gBAEnC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACvB,IAAI,GAAG,GAAG,SAAS,EAAE;oBACnB,aAAa,GAAG,GAAG,CAAC;oBACpB,IAAI,CAAC,UAAU,EAAE;wBACf,UAAU,GAAG,IAAI,CAAC;wBAClB,OAAO,CAAC,GAAG,CAAC,gEAAgE,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;wBAC5H,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;4BACnB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;4BACrC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;wBAClC,CAAC,CAAC,CAAC;qBACJ;iBACF;qBAAM,IAAI,UAAU,IAAI,GAAG,GAAG,aAAa,GAAG,UAAU,EAAE;oBACzD,UAAU,GAAG,KAAK,CAAC;oBACnB,OAAO,CAAC,GAAG,CAAC,8DAA8D,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;oBACtG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;iBACzD;YACH,CAAC,EAAE,OAAO,CAAC,CAAC;SACb;QAAC,OAAO,GAAG,EAAE;YACZ,OAAO,CAAC,IAAI,CAAC,yDAAyD,EAAE,GAAG,CAAC,CAAC;SAC9E;IACH,CAAC;IAEO,sBAAsB;QAC5B,IAAI,IAAI,CAAC,oBAAoB,KAAK,IAAI,EAAE;YACtC,aAAa,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;YACzC,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;SAClC;QACD,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YAChD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;SAChC;IACH,CAAC;IAEO,eAAe;QACrB,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI;gBACF,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,CAAC;gBAChC,IAAI,CAAC,kBAAkB,CAAC,SAAS,GAAG,IAAI,CAAC;aAC1C;YAAC,OAAO,CAAC,EAAE,GAAE;YACd,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;SAChC;IACH,CAAC;IAED,2BAA2B;IAC3B,QAAQ,CAAC,KAAc;QACrB,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAC7B,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,KAAK,CAAC,CAAC;QACtC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACnC,CAAC;IAED,8BAA8B;IACxB,UAAU;;YACd,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;gBACpB,IAAI,CAAC,OAAO,EAAE,CAAC;gBACf,OAAO;aACR;YACD,IAAI;gBACF,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;aAC/B;YAAC,OAAO,CAAC,EAAE;gBACV,SAAS;aACV;YACD,IAAI,CAAC,OAAO,EAAE,CAAC;QACjB,CAAC;KAAA;IAEO,OAAO;QACb,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YAC1C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;SACxB;QACD,IAAI,IAAI,CAAC,WAAW,EAAE;YACpB,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;YACtD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;SACzB;QACD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC3B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACjC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnC,uDAAuD;IACzD,CAAC;;;;YA1TF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAlBoB,MAAM","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport Daily from '@daily-co/daily-js';\nimport type { DailyCall, DailyParticipant } from '@daily-co/daily-js';\n\n/**\n * Daily.js WebRTC client for voice agent audio.\n * Responsibilities:\n * - Create and manage Daily CallObject\n * - Join Daily room using room_url\n * - Handle mic capture + speaker playback\n * - Bot speaking detection via AnalyserNode on remote track (instant)\n * - User speaking detection via active-speaker-change\n * - Expose speaking$ (bot speaking), userSpeaking$ (user speaking), micMuted$\n * - Expose localStream$ for waveform visualization (AudioAnalyzerService)\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class DailyVoiceClientService {\n  private callObject: DailyCall | null = null;\n  private localStream: MediaStream | null = null;\n  private localSessionId: string | null = null;\n  /** Explicit playback of remote (bot) audio; required in some browsers. */\n  private remoteAudioElement: HTMLAudioElement | null = null;\n\n  /** AnalyserNode-based remote audio monitor for instant bot speaking detection. */\n  private remoteAudioContext: AudioContext | null = null;\n  /** Poll interval id (~100ms); named historically when RAF was used. */\n  private remoteSpeakingPollId: ReturnType<typeof setInterval> | null = null;\n\n  private speakingSubject = new BehaviorSubject<boolean>(false);\n  private userSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private micMutedSubject = new BehaviorSubject<boolean>(false);\n  private localStreamSubject = new BehaviorSubject<MediaStream | null>(null);\n\n  /** True when bot (remote participant) is the active speaker. */\n  speaking$: Observable<boolean> = this.speakingSubject.asObservable();\n\n  /** True when user (local participant) is the active speaker. */\n  userSpeaking$: Observable<boolean> = this.userSpeakingSubject.asObservable();\n\n  /** True when mic is muted. */\n  micMuted$: Observable<boolean> = this.micMutedSubject.asObservable();\n\n  /** Emits local mic stream for waveform visualization. */\n  localStream$: Observable<MediaStream | null> =\n    this.localStreamSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  /**\n   * Connect to Daily room. Acquires mic first for waveform, then joins with audio.\n   * @param roomUrl Daily room URL (from room_created)\n   * @param token Optional meeting token\n   * @param existingStream Optional pre-acquired mic (avoids a second getUserMedia / extra prompts on some browsers)\n   */\n  async connect(\n    roomUrl: string,\n    token?: string,\n    existingStream?: MediaStream,\n  ): Promise<void> {\n    if (this.callObject) {\n      await this.disconnect();\n    }\n\n    try {\n      const hasLiveTrack =\n        !!existingStream?.getAudioTracks().some((t) => t.readyState === 'live');\n      const stream = hasLiveTrack\n        ? existingStream!\n        : await navigator.mediaDevices.getUserMedia({ audio: true });\n      const audioTrack = stream.getAudioTracks()[0];\n      if (!audioTrack) {\n        stream.getTracks().forEach((t) => t.stop());\n        throw new Error('No audio track');\n      }\n\n      this.localStream = stream;\n      this.localStreamSubject.next(stream);\n\n      // Create audio-only call object\n      // videoSource: false = no camera, audioSource = our mic track\n      const callObject = Daily.createCallObject({\n        videoSource: false,\n        audioSource: audioTrack,\n      });\n\n      this.callObject = callObject;\n\n      this.setupEventHandlers(callObject);\n\n      // Join room; Daily handles playback of remote (bot) audio automatically.\n      // Only pass token when it's a non-empty string (Daily rejects undefined/non-string).\n      const joinOptions: { url: string; token?: string } = { url: roomUrl };\n      if (typeof token === 'string' && token.trim() !== '') {\n        joinOptions.token = token;\n      }\n      await callObject.join(joinOptions);\n      console.log(`[VoiceDebug] Room connected (Daily join complete) — ${new Date().toISOString()}`);\n\n      const participants = callObject.participants();\n      if (participants?.local) {\n        this.localSessionId = participants.local.session_id;\n      }\n\n      // Initial mute state: Daily starts with audio on\n      this.micMutedSubject.next(!callObject.localAudio());\n    } catch (err) {\n      this.cleanup();\n      throw err;\n    }\n  }\n\n  private setupEventHandlers(call: DailyCall): void {\n    // active-speaker-change: used ONLY for user speaking detection.\n    // Bot speaking is detected by our own AnalyserNode (instant, no debounce).\n    call.on('active-speaker-change', (event: { activeSpeaker?: { peerId?: string } }) => {\n      this.ngZone.run(() => {\n        const peerId = event?.activeSpeaker?.peerId;\n        if (!peerId || !this.localSessionId) {\n          this.userSpeakingSubject.next(false);\n          return;\n        }\n        const isLocal = peerId === this.localSessionId;\n        this.userSpeakingSubject.next(isLocal);\n      });\n    });\n\n    // track-started / track-stopped: set up remote audio playback + AnalyserNode monitor.\n    call.on('track-started', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        const track = event?.track;\n        if (p && !p.local && type === 'audio') {\n          console.log(`[VoiceDebug] Got audio track from backend (track-started) — readyState=${track?.readyState}, muted=${track?.muted} — ${new Date().toISOString()}`);\n          const audioTrack = track ?? (p as { tracks?: { audio?: { track?: MediaStreamTrack } } }).tracks?.audio?.track;\n          if (audioTrack && typeof audioTrack === 'object') {\n            this.playRemoteTrack(audioTrack);\n            this.monitorRemoteAudio(audioTrack);\n          }\n        }\n      });\n    });\n\n    call.on('track-stopped', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        if (p && !p.local && type === 'audio') {\n          this.stopRemoteAudioMonitor();\n          this.stopRemoteAudio();\n        }\n      });\n    });\n\n    call.on('left-meeting', () => {\n      this.ngZone.run(() => this.cleanup());\n    });\n\n    call.on('error', (event?: { errorMsg?: string }) => {\n      this.ngZone.run(() => {\n        console.error('DailyVoiceClient: Daily error', event?.errorMsg ?? event);\n        this.cleanup();\n      });\n    });\n  }\n\n  /**\n   * Play remote (bot) audio track via a dedicated audio element.\n   * Required in many browsers where Daily's internal playback does not output to speakers.\n   */\n  private playRemoteTrack(track: MediaStreamTrack): void {\n    this.stopRemoteAudio();\n    try {\n      console.log(`[VoiceDebug] playRemoteTrack called — track.readyState=${track.readyState}, track.muted=${track.muted} — ${new Date().toISOString()}`);\n\n      track.onunmute = () => {\n        console.log(`[VoiceDebug] Remote audio track UNMUTED (audio data arriving) — ${new Date().toISOString()}`);\n      };\n\n      const stream = new MediaStream([track]);\n      const audio = new Audio();\n      audio.autoplay = true;\n      audio.srcObject = stream;\n      this.remoteAudioElement = audio;\n\n      audio.onplaying = () => {\n        console.log(`[VoiceDebug] Audio element PLAYING (browser started playback) — ${new Date().toISOString()}`);\n      };\n\n      let firstTimeUpdate = true;\n      audio.ontimeupdate = () => {\n        if (firstTimeUpdate) {\n          firstTimeUpdate = false;\n          console.log(`[VoiceDebug] Audio element first TIMEUPDATE (actual audio output) — ${new Date().toISOString()}`);\n        }\n      };\n\n      const p = audio.play();\n      if (p && typeof p.then === 'function') {\n        p.then(() => {\n          console.log(`[VoiceDebug] audio.play() resolved — ${new Date().toISOString()}`);\n        }).catch((err) => {\n          console.warn('DailyVoiceClient: remote audio play failed (may need user gesture)', err);\n        });\n      }\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio element', err);\n    }\n  }\n\n  /**\n   * Monitor remote audio track energy via AnalyserNode.\n   * Polls at ~10Hz; sufficient for speaking detection vs ~60fps RAF.\n   */\n  private monitorRemoteAudio(track: MediaStreamTrack): void {\n    this.stopRemoteAudioMonitor();\n    try {\n      const ctx = new AudioContext();\n      const source = ctx.createMediaStreamSource(new MediaStream([track]));\n      const analyser = ctx.createAnalyser();\n      analyser.fftSize = 256;\n      source.connect(analyser);\n      this.remoteAudioContext = ctx;\n\n      const dataArray = new Uint8Array(analyser.frequencyBinCount);\n      const THRESHOLD = 5;\n      const SILENCE_MS = 1500;\n      const POLL_MS = 100;\n      let lastSoundTime = 0;\n      let isSpeaking = false;\n\n      this.remoteSpeakingPollId = setInterval(() => {\n        if (!this.remoteAudioContext) {\n          if (this.remoteSpeakingPollId) {\n            clearInterval(this.remoteSpeakingPollId);\n            this.remoteSpeakingPollId = null;\n          }\n          return;\n        }\n        analyser.getByteFrequencyData(dataArray);\n\n        let sum = 0;\n        for (let i = 0; i < dataArray.length; i++) {\n          sum += dataArray[i];\n        }\n        const avg = sum / dataArray.length;\n\n        const now = Date.now();\n        if (avg > THRESHOLD) {\n          lastSoundTime = now;\n          if (!isSpeaking) {\n            isSpeaking = true;\n            console.log(`[VoiceDebug] Bot audio energy detected (speaking=true) — avg=${avg.toFixed(1)} — ${new Date().toISOString()}`);\n            this.ngZone.run(() => {\n              this.userSpeakingSubject.next(false);\n              this.speakingSubject.next(true);\n            });\n          }\n        } else if (isSpeaking && now - lastSoundTime > SILENCE_MS) {\n          isSpeaking = false;\n          console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);\n          this.ngZone.run(() => this.speakingSubject.next(false));\n        }\n      }, POLL_MS);\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio monitor', err);\n    }\n  }\n\n  private stopRemoteAudioMonitor(): void {\n    if (this.remoteSpeakingPollId !== null) {\n      clearInterval(this.remoteSpeakingPollId);\n      this.remoteSpeakingPollId = null;\n    }\n    if (this.remoteAudioContext) {\n      this.remoteAudioContext.close().catch(() => {});\n      this.remoteAudioContext = null;\n    }\n  }\n\n  private stopRemoteAudio(): void {\n    if (this.remoteAudioElement) {\n      try {\n        this.remoteAudioElement.pause();\n        this.remoteAudioElement.srcObject = null;\n      } catch (_) {}\n      this.remoteAudioElement = null;\n    }\n  }\n\n  /** Set mic muted state. */\n  setMuted(muted: boolean): void {\n    if (!this.callObject) return;\n    this.callObject.setLocalAudio(!muted);\n    this.micMutedSubject.next(muted);\n  }\n\n  /** Disconnect and cleanup. */\n  async disconnect(): Promise<void> {\n    if (!this.callObject) {\n      this.cleanup();\n      return;\n    }\n    try {\n      await this.callObject.leave();\n    } catch (e) {\n      // ignore\n    }\n    this.cleanup();\n  }\n\n  private cleanup(): void {\n    this.stopRemoteAudioMonitor();\n    this.stopRemoteAudio();\n    if (this.callObject) {\n      this.callObject.destroy().catch(() => {});\n      this.callObject = null;\n    }\n    if (this.localStream) {\n      this.localStream.getTracks().forEach((t) => t.stop());\n      this.localStream = null;\n    }\n    this.localSessionId = null;\n    this.speakingSubject.next(false);\n    this.userSpeakingSubject.next(false);\n    this.localStreamSubject.next(null);\n    // Keep last micMuted state; will reset on next connect\n  }\n}\n"]}
|