@hivegpt/hiveai-angular 0.0.569 → 0.0.570
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 +149 -273
- 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/voice-agent/services/daily-voice-client.service.js +55 -137
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +57 -102
- package/fesm2015/hivegpt-hiveai-angular.js +110 -237
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +1 -33
- 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 +5 -16
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -3,83 +3,59 @@ 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
|
-
*/
|
|
17
6
|
export class DailyVoiceClientService {
|
|
18
7
|
constructor(ngZone) {
|
|
19
8
|
this.ngZone = ngZone;
|
|
20
9
|
this.callObject = null;
|
|
21
10
|
this.localStream = null;
|
|
22
11
|
this.localSessionId = null;
|
|
23
|
-
/** Explicit playback of remote (bot) audio; required in some browsers. */
|
|
24
12
|
this.remoteAudioElement = null;
|
|
25
|
-
/** AnalyserNode-based remote audio monitor for instant bot speaking detection. */
|
|
26
13
|
this.remoteAudioContext = null;
|
|
27
14
|
this.remoteSpeakingRAF = null;
|
|
28
15
|
this.speakingSubject = new BehaviorSubject(false);
|
|
29
16
|
this.userSpeakingSubject = new BehaviorSubject(false);
|
|
30
|
-
this.micMutedSubject = new BehaviorSubject(
|
|
17
|
+
this.micMutedSubject = new BehaviorSubject(true); // 🔴 default muted
|
|
31
18
|
this.localStreamSubject = new BehaviorSubject(null);
|
|
32
|
-
/** True when bot (remote participant) is the active speaker. */
|
|
33
19
|
this.speaking$ = this.speakingSubject.asObservable();
|
|
34
|
-
/** True when user (local participant) is the active speaker. */
|
|
35
20
|
this.userSpeaking$ = this.userSpeakingSubject.asObservable();
|
|
36
|
-
/** True when mic is muted. */
|
|
37
21
|
this.micMuted$ = this.micMutedSubject.asObservable();
|
|
38
|
-
/** Emits local mic stream for waveform visualization. */
|
|
39
22
|
this.localStream$ = this.localStreamSubject.asObservable();
|
|
40
23
|
}
|
|
41
|
-
/**
|
|
42
|
-
* Connect to Daily room. Acquires mic first for waveform, then joins with audio.
|
|
43
|
-
* @param roomUrl Daily room URL (from room_created)
|
|
44
|
-
* @param token Optional meeting token
|
|
45
|
-
*/
|
|
46
24
|
connect(roomUrl, token) {
|
|
47
25
|
return __awaiter(this, void 0, void 0, function* () {
|
|
48
26
|
if (this.callObject) {
|
|
49
27
|
yield this.disconnect();
|
|
50
28
|
}
|
|
51
29
|
try {
|
|
52
|
-
// Get mic
|
|
30
|
+
// 🎤 Get mic (kept for waveform)
|
|
53
31
|
const stream = yield navigator.mediaDevices.getUserMedia({ audio: true });
|
|
54
32
|
const audioTrack = stream.getAudioTracks()[0];
|
|
55
|
-
if (!audioTrack)
|
|
56
|
-
stream.getTracks().forEach((t) => t.stop());
|
|
33
|
+
if (!audioTrack)
|
|
57
34
|
throw new Error('No audio track');
|
|
58
|
-
}
|
|
59
35
|
this.localStream = stream;
|
|
60
36
|
this.localStreamSubject.next(stream);
|
|
61
|
-
// Create audio-only call object
|
|
62
|
-
// videoSource: false = no camera, audioSource = our mic track
|
|
63
37
|
const callObject = Daily.createCallObject({
|
|
64
38
|
videoSource: false,
|
|
65
39
|
audioSource: audioTrack,
|
|
66
40
|
});
|
|
67
41
|
this.callObject = callObject;
|
|
68
42
|
this.setupEventHandlers(callObject);
|
|
69
|
-
//
|
|
70
|
-
|
|
43
|
+
// 🔴 Ensure mic is OFF before join
|
|
44
|
+
callObject.setLocalAudio(false);
|
|
45
|
+
this.micMutedSubject.next(true);
|
|
71
46
|
const joinOptions = { url: roomUrl };
|
|
72
|
-
if (typeof token === 'string' && token.trim()
|
|
47
|
+
if (typeof token === 'string' && token.trim()) {
|
|
73
48
|
joinOptions.token = token;
|
|
74
49
|
}
|
|
75
50
|
yield callObject.join(joinOptions);
|
|
76
|
-
console.log(`[VoiceDebug]
|
|
51
|
+
console.log(`[VoiceDebug] Joined room — ${new Date().toISOString()}`);
|
|
77
52
|
const participants = callObject.participants();
|
|
78
53
|
if (participants === null || participants === void 0 ? void 0 : participants.local) {
|
|
79
54
|
this.localSessionId = participants.local.session_id;
|
|
80
55
|
}
|
|
81
|
-
//
|
|
82
|
-
|
|
56
|
+
// 🔴 Force sync again (Daily sometimes overrides)
|
|
57
|
+
callObject.setLocalAudio(false);
|
|
58
|
+
this.micMutedSubject.next(true);
|
|
83
59
|
}
|
|
84
60
|
catch (err) {
|
|
85
61
|
this.cleanup();
|
|
@@ -88,8 +64,6 @@ export class DailyVoiceClientService {
|
|
|
88
64
|
});
|
|
89
65
|
}
|
|
90
66
|
setupEventHandlers(call) {
|
|
91
|
-
// active-speaker-change: used ONLY for user speaking detection.
|
|
92
|
-
// Bot speaking is detected by our own AnalyserNode (instant, no debounce).
|
|
93
67
|
call.on('active-speaker-change', (event) => {
|
|
94
68
|
this.ngZone.run(() => {
|
|
95
69
|
var _a;
|
|
@@ -98,23 +72,20 @@ export class DailyVoiceClientService {
|
|
|
98
72
|
this.userSpeakingSubject.next(false);
|
|
99
73
|
return;
|
|
100
74
|
}
|
|
101
|
-
|
|
102
|
-
this.userSpeakingSubject.next(isLocal);
|
|
75
|
+
this.userSpeakingSubject.next(peerId === this.localSessionId);
|
|
103
76
|
});
|
|
104
77
|
});
|
|
105
|
-
// track-started / track-stopped: set up remote audio playback + AnalyserNode monitor.
|
|
106
78
|
call.on('track-started', (event) => {
|
|
107
79
|
this.ngZone.run(() => {
|
|
108
|
-
var _a, _b, _c, _d;
|
|
80
|
+
var _a, _b, _c, _d, _e;
|
|
109
81
|
const p = event === null || event === void 0 ? void 0 : event.participant;
|
|
110
82
|
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;
|
|
111
|
-
const track = event === null || event === void 0 ? void 0 : event.track;
|
|
112
83
|
if (p && !p.local && type === 'audio') {
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
this.playRemoteTrack(
|
|
117
|
-
this.monitorRemoteAudio(
|
|
84
|
+
const track = (_c = event.track) !== null && _c !== void 0 ? _c : (_e = (_d = p === null || p === void 0 ? void 0 : p.tracks) === null || _d === void 0 ? void 0 : _d.audio) === null || _e === void 0 ? void 0 : _e.track;
|
|
85
|
+
if (track) {
|
|
86
|
+
console.log('[VoiceDebug] Remote audio track received');
|
|
87
|
+
this.playRemoteTrack(track);
|
|
88
|
+
this.monitorRemoteAudio(track);
|
|
118
89
|
}
|
|
119
90
|
}
|
|
120
91
|
});
|
|
@@ -133,57 +104,27 @@ export class DailyVoiceClientService {
|
|
|
133
104
|
call.on('left-meeting', () => {
|
|
134
105
|
this.ngZone.run(() => this.cleanup());
|
|
135
106
|
});
|
|
136
|
-
call.on('error', (
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
console.error('DailyVoiceClient: Daily error', (_a = event === null || event === void 0 ? void 0 : event.errorMsg) !== null && _a !== void 0 ? _a : event);
|
|
140
|
-
this.cleanup();
|
|
141
|
-
});
|
|
107
|
+
call.on('error', (e) => {
|
|
108
|
+
console.error('Daily error:', e);
|
|
109
|
+
this.cleanup();
|
|
142
110
|
});
|
|
143
111
|
}
|
|
144
|
-
/**
|
|
145
|
-
* Play remote (bot) audio track via a dedicated audio element.
|
|
146
|
-
* Required in many browsers where Daily's internal playback does not output to speakers.
|
|
147
|
-
*/
|
|
148
112
|
playRemoteTrack(track) {
|
|
149
113
|
this.stopRemoteAudio();
|
|
150
114
|
try {
|
|
151
|
-
console.log(`[VoiceDebug] playRemoteTrack called — track.readyState=${track.readyState}, track.muted=${track.muted} — ${new Date().toISOString()}`);
|
|
152
|
-
track.onunmute = () => {
|
|
153
|
-
console.log(`[VoiceDebug] Remote audio track UNMUTED (audio data arriving) — ${new Date().toISOString()}`);
|
|
154
|
-
};
|
|
155
115
|
const stream = new MediaStream([track]);
|
|
156
116
|
const audio = new Audio();
|
|
157
117
|
audio.autoplay = true;
|
|
158
118
|
audio.srcObject = stream;
|
|
159
119
|
this.remoteAudioElement = audio;
|
|
160
|
-
audio.
|
|
161
|
-
console.
|
|
162
|
-
};
|
|
163
|
-
let firstTimeUpdate = true;
|
|
164
|
-
audio.ontimeupdate = () => {
|
|
165
|
-
if (firstTimeUpdate) {
|
|
166
|
-
firstTimeUpdate = false;
|
|
167
|
-
console.log(`[VoiceDebug] Audio element first TIMEUPDATE (actual audio output) — ${new Date().toISOString()}`);
|
|
168
|
-
}
|
|
169
|
-
};
|
|
170
|
-
const p = audio.play();
|
|
171
|
-
if (p && typeof p.then === 'function') {
|
|
172
|
-
p.then(() => {
|
|
173
|
-
console.log(`[VoiceDebug] audio.play() resolved — ${new Date().toISOString()}`);
|
|
174
|
-
}).catch((err) => {
|
|
175
|
-
console.warn('DailyVoiceClient: remote audio play failed (may need user gesture)', err);
|
|
176
|
-
});
|
|
177
|
-
}
|
|
120
|
+
audio.play().catch(() => {
|
|
121
|
+
console.warn('Autoplay blocked');
|
|
122
|
+
});
|
|
178
123
|
}
|
|
179
124
|
catch (err) {
|
|
180
|
-
console.warn('
|
|
125
|
+
console.warn('Audio playback error', err);
|
|
181
126
|
}
|
|
182
127
|
}
|
|
183
|
-
/**
|
|
184
|
-
* Monitor remote audio track energy via AnalyserNode.
|
|
185
|
-
* Polls at ~60fps and flips speakingSubject based on actual audio energy.
|
|
186
|
-
*/
|
|
187
128
|
monitorRemoteAudio(track) {
|
|
188
129
|
this.stopRemoteAudioMonitor();
|
|
189
130
|
try {
|
|
@@ -193,104 +134,81 @@ export class DailyVoiceClientService {
|
|
|
193
134
|
analyser.fftSize = 256;
|
|
194
135
|
source.connect(analyser);
|
|
195
136
|
this.remoteAudioContext = ctx;
|
|
196
|
-
const
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
let isSpeaking = false;
|
|
201
|
-
const poll = () => {
|
|
137
|
+
const data = new Uint8Array(analyser.frequencyBinCount);
|
|
138
|
+
let speaking = false;
|
|
139
|
+
let lastSound = 0;
|
|
140
|
+
const loop = () => {
|
|
202
141
|
if (!this.remoteAudioContext)
|
|
203
142
|
return;
|
|
204
|
-
analyser.getByteFrequencyData(
|
|
205
|
-
|
|
206
|
-
for (let i = 0; i < dataArray.length; i++) {
|
|
207
|
-
sum += dataArray[i];
|
|
208
|
-
}
|
|
209
|
-
const avg = sum / dataArray.length;
|
|
143
|
+
analyser.getByteFrequencyData(data);
|
|
144
|
+
const avg = data.reduce((a, b) => a + b, 0) / data.length;
|
|
210
145
|
const now = Date.now();
|
|
211
|
-
if (avg >
|
|
212
|
-
|
|
213
|
-
if (!
|
|
214
|
-
|
|
215
|
-
console.log(`[VoiceDebug] Bot audio energy detected (speaking=true) — avg=${avg.toFixed(1)} — ${new Date().toISOString()}`);
|
|
146
|
+
if (avg > 5) {
|
|
147
|
+
lastSound = now;
|
|
148
|
+
if (!speaking) {
|
|
149
|
+
speaking = true;
|
|
216
150
|
this.ngZone.run(() => {
|
|
217
151
|
this.userSpeakingSubject.next(false);
|
|
218
152
|
this.speakingSubject.next(true);
|
|
219
153
|
});
|
|
220
154
|
}
|
|
221
155
|
}
|
|
222
|
-
else if (
|
|
223
|
-
|
|
224
|
-
console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);
|
|
156
|
+
else if (speaking && now - lastSound > 1500) {
|
|
157
|
+
speaking = false;
|
|
225
158
|
this.ngZone.run(() => this.speakingSubject.next(false));
|
|
226
159
|
}
|
|
227
|
-
this.remoteSpeakingRAF = requestAnimationFrame(
|
|
160
|
+
this.remoteSpeakingRAF = requestAnimationFrame(loop);
|
|
228
161
|
};
|
|
229
|
-
this.remoteSpeakingRAF = requestAnimationFrame(
|
|
230
|
-
}
|
|
231
|
-
catch (err) {
|
|
232
|
-
console.warn('DailyVoiceClient: failed to create remote audio monitor', err);
|
|
162
|
+
this.remoteSpeakingRAF = requestAnimationFrame(loop);
|
|
233
163
|
}
|
|
164
|
+
catch (_a) { }
|
|
234
165
|
}
|
|
235
166
|
stopRemoteAudioMonitor() {
|
|
167
|
+
var _a;
|
|
236
168
|
if (this.remoteSpeakingRAF) {
|
|
237
169
|
cancelAnimationFrame(this.remoteSpeakingRAF);
|
|
238
170
|
this.remoteSpeakingRAF = null;
|
|
239
171
|
}
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
this.remoteAudioContext = null;
|
|
243
|
-
}
|
|
172
|
+
(_a = this.remoteAudioContext) === null || _a === void 0 ? void 0 : _a.close().catch(() => { });
|
|
173
|
+
this.remoteAudioContext = null;
|
|
244
174
|
}
|
|
245
175
|
stopRemoteAudio() {
|
|
246
176
|
if (this.remoteAudioElement) {
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
this.remoteAudioElement.srcObject = null;
|
|
250
|
-
}
|
|
251
|
-
catch (_) { }
|
|
177
|
+
this.remoteAudioElement.pause();
|
|
178
|
+
this.remoteAudioElement.srcObject = null;
|
|
252
179
|
this.remoteAudioElement = null;
|
|
253
180
|
}
|
|
254
181
|
}
|
|
255
|
-
/** Set mic muted state. */
|
|
256
182
|
setMuted(muted) {
|
|
257
183
|
if (!this.callObject)
|
|
258
184
|
return;
|
|
259
185
|
this.callObject.setLocalAudio(!muted);
|
|
260
186
|
this.micMutedSubject.next(muted);
|
|
187
|
+
console.log(`[VoiceDebug] Mic ${muted ? 'MUTED' : 'UNMUTED'}`);
|
|
261
188
|
}
|
|
262
|
-
/** Disconnect and cleanup. */
|
|
263
189
|
disconnect() {
|
|
264
190
|
return __awaiter(this, void 0, void 0, function* () {
|
|
265
|
-
if (!this.callObject)
|
|
266
|
-
this.cleanup();
|
|
267
|
-
return;
|
|
268
|
-
}
|
|
191
|
+
if (!this.callObject)
|
|
192
|
+
return this.cleanup();
|
|
269
193
|
try {
|
|
270
194
|
yield this.callObject.leave();
|
|
271
195
|
}
|
|
272
|
-
catch (
|
|
273
|
-
// ignore
|
|
274
|
-
}
|
|
196
|
+
catch (_a) { }
|
|
275
197
|
this.cleanup();
|
|
276
198
|
});
|
|
277
199
|
}
|
|
278
200
|
cleanup() {
|
|
201
|
+
var _a, _b;
|
|
279
202
|
this.stopRemoteAudioMonitor();
|
|
280
203
|
this.stopRemoteAudio();
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
if (this.localStream) {
|
|
286
|
-
this.localStream.getTracks().forEach((t) => t.stop());
|
|
287
|
-
this.localStream = null;
|
|
288
|
-
}
|
|
204
|
+
(_a = this.callObject) === null || _a === void 0 ? void 0 : _a.destroy().catch(() => { });
|
|
205
|
+
this.callObject = null;
|
|
206
|
+
(_b = this.localStream) === null || _b === void 0 ? void 0 : _b.getTracks().forEach((t) => t.stop());
|
|
207
|
+
this.localStream = null;
|
|
289
208
|
this.localSessionId = null;
|
|
290
209
|
this.speakingSubject.next(false);
|
|
291
210
|
this.userSpeakingSubject.next(false);
|
|
292
211
|
this.localStreamSubject.next(null);
|
|
293
|
-
// Keep last micMuted state; will reset on next connect
|
|
294
212
|
}
|
|
295
213
|
}
|
|
296
214
|
DailyVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function DailyVoiceClientService_Factory() { return new DailyVoiceClientService(i0.ɵɵinject(i0.NgZone)); }, token: DailyVoiceClientService, providedIn: "root" });
|
|
@@ -302,4 +220,4 @@ DailyVoiceClientService.decorators = [
|
|
|
302
220
|
DailyVoiceClientService.ctorParameters = () => [
|
|
303
221
|
{ type: NgZone }
|
|
304
222
|
];
|
|
305
|
-
//# 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/**\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  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>(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   */\n  async connect(roomUrl: string, token?: string): Promise<void> {\n    if (this.callObject) {\n      await this.disconnect();\n    }\n\n    try {\n      // Get mic stream for both Daily and waveform (single capture)\n      const stream = 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 ~60fps and flips speakingSubject based on actual audio energy.\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      let lastSoundTime = 0;\n      let isSpeaking = false;\n\n      const poll = () => {\n        if (!this.remoteAudioContext) return;\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\n        this.remoteSpeakingRAF = requestAnimationFrame(poll);\n      };\n\n      this.remoteSpeakingRAF = requestAnimationFrame(poll);\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio monitor', err);\n    }\n  }\n\n  private stopRemoteAudioMonitor(): void {\n    if (this.remoteSpeakingRAF) {\n      cancelAnimationFrame(this.remoteSpeakingRAF);\n      this.remoteSpeakingRAF = 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"]}
|
|
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":";AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AACnD,OAAO,KAAK,MAAM,oBAAoB,CAAC;;AAMvC,MAAM,OAAO,uBAAuB;IAmBlC,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;QAlB1B,eAAU,GAAqB,IAAI,CAAC;QACpC,gBAAW,GAAuB,IAAI,CAAC;QACvC,mBAAc,GAAkB,IAAI,CAAC;QACrC,uBAAkB,GAA4B,IAAI,CAAC;QAEnD,uBAAkB,GAAwB,IAAI,CAAC;QAC/C,sBAAiB,GAAkB,IAAI,CAAC;QAExC,oBAAe,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACtD,wBAAmB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAC1D,oBAAe,GAAG,IAAI,eAAe,CAAU,IAAI,CAAC,CAAC,CAAC,mBAAmB;QACzE,uBAAkB,GAAG,IAAI,eAAe,CAAqB,IAAI,CAAC,CAAC;QAE3E,cAAS,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QAChD,kBAAa,GAAG,IAAI,CAAC,mBAAmB,CAAC,YAAY,EAAE,CAAC;QACxD,cAAS,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QAChD,iBAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;IAEjB,CAAC;IAEhC,OAAO,CAAC,OAAe,EAAE,KAAc;;YAC3C,IAAI,IAAI,CAAC,UAAU,EAAE;gBACnB,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;aACzB;YAED,IAAI;gBACF,iCAAiC;gBACjC,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,YAAY,CAAC,YAAY,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;gBAC1E,MAAM,UAAU,GAAG,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC9C,IAAI,CAAC,UAAU;oBAAE,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;gBAEnD,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC;gBAC1B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAErC,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;gBAC7B,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;gBAEpC,mCAAmC;gBACnC,UAAU,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;gBAChC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAEhC,MAAM,WAAW,GAAoC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC;gBACtE,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,IAAI,EAAE,EAAE;oBAC7C,WAAW,CAAC,KAAK,GAAG,KAAK,CAAC;iBAC3B;gBAED,MAAM,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBAEnC,OAAO,CAAC,GAAG,CAAC,8BAA8B,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;gBAEtE,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,kDAAkD;gBAClD,UAAU,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;gBAChC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aACjC;YAAC,OAAO,GAAG,EAAE;gBACZ,IAAI,CAAC,OAAO,EAAE,CAAC;gBACf,MAAM,GAAG,CAAC;aACX;QACH,CAAC;KAAA;IAEO,kBAAkB,CAAC,IAAe;QACxC,IAAI,CAAC,EAAE,CAAC,uBAAuB,EAAE,CAAC,KAAU,EAAE,EAAE;YAC9C,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,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,cAAc,CAAC,CAAC;YAChE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,KAAU,EAAE,EAAE;YACtC,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;gBAE/C,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,KAAK,OAAO,EAAE;oBACrC,MAAM,KAAK,GAAG,MAAA,KAAK,CAAC,KAAK,mCAAI,MAAA,MAAA,CAAC,aAAD,CAAC,uBAAD,CAAC,CAAE,MAAM,0CAAE,KAAK,0CAAE,KAAK,CAAC;oBAErD,IAAI,KAAK,EAAE;wBACT,OAAO,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;wBACxD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;wBAC5B,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;qBAChC;iBACF;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,KAAU,EAAE,EAAE;YACtC,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;gBAE/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,CAAM,EAAE,EAAE;YAC1B,OAAO,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;YACjC,IAAI,CAAC,OAAO,EAAE,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,eAAe,CAAC,KAAuB;QAC7C,IAAI,CAAC,eAAe,EAAE,CAAC;QAEvB,IAAI;YACF,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;YAEzB,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAEhC,KAAK,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE;gBACtB,OAAO,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;YACnC,CAAC,CAAC,CAAC;SACJ;QAAC,OAAO,GAAG,EAAE;YACZ,OAAO,CAAC,IAAI,CAAC,sBAAsB,EAAE,GAAG,CAAC,CAAC;SAC3C;IACH,CAAC;IAEO,kBAAkB,CAAC,KAAuB;QAChD,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAE9B,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;YAEtC,QAAQ,CAAC,OAAO,GAAG,GAAG,CAAC;YACvB,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAEzB,IAAI,CAAC,kBAAkB,GAAG,GAAG,CAAC;YAE9B,MAAM,IAAI,GAAG,IAAI,UAAU,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC;YAExD,IAAI,QAAQ,GAAG,KAAK,CAAC;YACrB,IAAI,SAAS,GAAG,CAAC,CAAC;YAElB,MAAM,IAAI,GAAG,GAAG,EAAE;gBAChB,IAAI,CAAC,IAAI,CAAC,kBAAkB;oBAAE,OAAO;gBAErC,QAAQ,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;gBACpC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;gBAC1D,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBAEvB,IAAI,GAAG,GAAG,CAAC,EAAE;oBACX,SAAS,GAAG,GAAG,CAAC;oBAChB,IAAI,CAAC,QAAQ,EAAE;wBACb,QAAQ,GAAG,IAAI,CAAC;wBAChB,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,QAAQ,IAAI,GAAG,GAAG,SAAS,GAAG,IAAI,EAAE;oBAC7C,QAAQ,GAAG,KAAK,CAAC;oBACjB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;iBACzD;gBAED,IAAI,CAAC,iBAAiB,GAAG,qBAAqB,CAAC,IAAI,CAAC,CAAC;YACvD,CAAC,CAAC;YAEF,IAAI,CAAC,iBAAiB,GAAG,qBAAqB,CAAC,IAAI,CAAC,CAAC;SACtD;QAAC,WAAM,GAAE;IACZ,CAAC;IAEO,sBAAsB;;QAC5B,IAAI,IAAI,CAAC,iBAAiB,EAAE;YAC1B,oBAAoB,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC7C,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;SAC/B;QACD,MAAA,IAAI,CAAC,kBAAkB,0CAAE,KAAK,GAAG,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QACjD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;IACjC,CAAC;IAEO,eAAe;QACrB,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAC3B,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,CAAC;YAChC,IAAI,CAAC,kBAAkB,CAAC,SAAS,GAAG,IAAI,CAAC;YACzC,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;SAChC;IACH,CAAC;IAED,QAAQ,CAAC,KAAc;QACrB,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO;QAE7B,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,KAAK,CAAC,CAAC;QACtC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAEjC,OAAO,CAAC,GAAG,CAAC,oBAAoB,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC;IACjE,CAAC;IAEK,UAAU;;YACd,IAAI,CAAC,IAAI,CAAC,UAAU;gBAAE,OAAO,IAAI,CAAC,OAAO,EAAE,CAAC;YAE5C,IAAI;gBACF,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;aAC/B;YAAC,WAAM,GAAE;YAEV,IAAI,CAAC,OAAO,EAAE,CAAC;QACjB,CAAC;KAAA;IAEO,OAAO;;QACb,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;QAEvB,MAAA,IAAI,CAAC,UAAU,0CAAE,OAAO,GAAG,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAC3C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QAEvB,MAAA,IAAI,CAAC,WAAW,0CAAE,SAAS,GAAG,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QACvD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QAExB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAE3B,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;IACrC,CAAC;;;;YAhPF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAPoB,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@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"]}
|