@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.
@@ -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(false);
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 stream for both Daily and waveform (single capture)
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
- // Join room; Daily handles playback of remote (bot) audio automatically.
70
- // Only pass token when it's a non-empty string (Daily rejects undefined/non-string).
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] Room connected (Daily join complete) — ${new Date().toISOString()}`);
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
- // Initial mute state: Daily starts with audio on
82
- this.micMutedSubject.next(!callObject.localAudio());
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
- const isLocal = peerId === this.localSessionId;
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
- 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()}`);
114
- 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;
115
- if (audioTrack && typeof audioTrack === 'object') {
116
- this.playRemoteTrack(audioTrack);
117
- this.monitorRemoteAudio(audioTrack);
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', (event) => {
137
- this.ngZone.run(() => {
138
- var _a;
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.onplaying = () => {
161
- console.log(`[VoiceDebug] Audio element PLAYING (browser started playback) — ${new Date().toISOString()}`);
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('DailyVoiceClient: failed to create remote audio element', err);
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 dataArray = new Uint8Array(analyser.frequencyBinCount);
197
- const THRESHOLD = 5;
198
- const SILENCE_MS = 1500;
199
- let lastSoundTime = 0;
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(dataArray);
205
- let sum = 0;
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 > THRESHOLD) {
212
- lastSoundTime = now;
213
- if (!isSpeaking) {
214
- isSpeaking = true;
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 (isSpeaking && now - lastSoundTime > SILENCE_MS) {
223
- isSpeaking = false;
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(poll);
160
+ this.remoteSpeakingRAF = requestAnimationFrame(loop);
228
161
  };
229
- this.remoteSpeakingRAF = requestAnimationFrame(poll);
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
- if (this.remoteAudioContext) {
241
- this.remoteAudioContext.close().catch(() => { });
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
- try {
248
- this.remoteAudioElement.pause();
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 (e) {
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
- if (this.callObject) {
282
- this.callObject.destroy().catch(() => { });
283
- this.callObject = null;
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":"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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"]}