@hivegpt/hiveai-angular 0.0.587 → 0.0.588

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.
@@ -41,20 +41,6 @@ export class DailyVoiceClientService {
41
41
  /** Emits true once when first remote audio frame starts playing. */
42
42
  this.firstRemoteAudioFrame$ = this.firstRemoteAudioFrameSubject.asObservable();
43
43
  }
44
- /**
45
- * Prompt for microphone access up front so callers can handle permission
46
- * denial before creating backend room/session resources.
47
- */
48
- ensureMicrophoneAccess() {
49
- var _a;
50
- return __awaiter(this, void 0, void 0, function* () {
51
- if (!((_a = navigator === null || navigator === void 0 ? void 0 : navigator.mediaDevices) === null || _a === void 0 ? void 0 : _a.getUserMedia)) {
52
- throw new Error('Microphone API unavailable');
53
- }
54
- const stream = yield navigator.mediaDevices.getUserMedia({ audio: true });
55
- stream.getTracks().forEach((t) => t.stop());
56
- });
57
- }
58
44
  /**
59
45
  * Connect to Daily room. Acquires mic first for waveform, then joins with audio.
60
46
  * @param roomUrl Daily room URL (from room_created)
@@ -323,4 +309,4 @@ DailyVoiceClientService.decorators = [
323
309
  DailyVoiceClientService.ctorParameters = () => [
324
310
  { type: NgZone }
325
311
  ];
326
- //# 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  private firstRemoteAudioFrameSubject = new BehaviorSubject<boolean>(false);\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  /** Emits true once when first remote audio frame starts playing. */\n  firstRemoteAudioFrame$: Observable<boolean> =\n    this.firstRemoteAudioFrameSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  /**\n   * Prompt for microphone access up front so callers can handle permission\n   * denial before creating backend room/session resources.\n   */\n  async ensureMicrophoneAccess(): Promise<void> {\n    if (!navigator?.mediaDevices?.getUserMedia) {\n      throw new Error('Microphone API unavailable');\n    }\n    const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n    stream.getTracks().forEach((t) => t.stop());\n  }\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      // Start with mic muted; VoiceAgentService auto-unmutes after first remote audio frame.\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    // 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          this.firstRemoteAudioFrameSubject.next(true);\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          this.firstRemoteAudioFrameSubject.next(true);\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    this.firstRemoteAudioFrameSubject.next(false);\n    // Keep last micMuted state; will reset on next connect\n  }\n}\n"]}
312
+ //# 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  private firstRemoteAudioFrameSubject = new BehaviorSubject<boolean>(false);\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  /** Emits true once when first remote audio frame starts playing. */\n  firstRemoteAudioFrame$: Observable<boolean> =\n    this.firstRemoteAudioFrameSubject.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      // Start with mic muted; VoiceAgentService auto-unmutes after first remote audio frame.\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    // 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          this.firstRemoteAudioFrameSubject.next(true);\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          this.firstRemoteAudioFrameSubject.next(true);\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    this.firstRemoteAudioFrameSubject.next(false);\n    // Keep last micMuted state; will reset on next connect\n  }\n}\n"]}
@@ -84,8 +84,6 @@ export class VoiceAgentService {
84
84
  try {
85
85
  this.callStateSubject.next('connecting');
86
86
  this.statusTextSubject.next('Connecting...');
87
- // Request mic permission before provisioning backend room flow.
88
- yield this.dailyClient.ensureMicrophoneAccess();
89
87
  let accessToken = token;
90
88
  // Align with chat drawer token handling: always delegate to
91
89
  // PlatformTokenRefreshService when we have a usersApiUrl, so it can
@@ -144,16 +142,10 @@ export class VoiceAgentService {
144
142
  }
145
143
  catch (err) {
146
144
  console.error('Daily join failed:', err);
147
- if (this.isMicPermissionError(err)) {
148
- yield this.cleanupMediaAndSignal();
149
- this.callStateSubject.next('idle');
150
- this.durationSubject.next('0:00');
151
- this.statusTextSubject.next('Microphone permission is required');
152
- return;
153
- }
154
145
  this.callStateSubject.next('ended');
155
146
  this.statusTextSubject.next('Connection failed');
156
147
  yield this.disconnect();
148
+ throw err;
157
149
  }
158
150
  }));
159
151
  // Forward transcripts from WebSocket
@@ -168,13 +160,6 @@ export class VoiceAgentService {
168
160
  }
169
161
  catch (error) {
170
162
  console.error('Error connecting voice agent:', error);
171
- if (this.isMicPermissionError(error)) {
172
- yield this.cleanupMediaAndSignal();
173
- this.callStateSubject.next('idle');
174
- this.durationSubject.next('0:00');
175
- this.statusTextSubject.next('Microphone permission is required');
176
- return;
177
- }
178
163
  this.callStateSubject.next('ended');
179
164
  yield this.disconnect();
180
165
  this.statusTextSubject.next('Connection failed');
@@ -268,23 +253,6 @@ export class VoiceAgentService {
268
253
  this.durationInterval = null;
269
254
  }
270
255
  }
271
- isMicPermissionError(error) {
272
- if (!error || typeof error !== 'object')
273
- return false;
274
- const maybe = error;
275
- return (maybe.name === 'NotAllowedError' ||
276
- maybe.name === 'PermissionDeniedError' ||
277
- maybe.name === 'SecurityError');
278
- }
279
- cleanupMediaAndSignal() {
280
- return __awaiter(this, void 0, void 0, function* () {
281
- this.stopDurationTimer();
282
- this.audioAnalyzer.stop();
283
- yield this.dailyClient.disconnect();
284
- this.wsClient.disconnect();
285
- this.hasAutoUnmutedAfterFirstAudio = false;
286
- });
287
- }
288
256
  }
289
257
  VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.WebSocketVoiceClientService), i0.ɵɵinject(i3.DailyVoiceClientService), i0.ɵɵinject(i4.PlatformTokenRefreshService), i0.ɵɵinject(i0.PLATFORM_ID)); }, token: VoiceAgentService, providedIn: "root" });
290
258
  VoiceAgentService.decorators = [
@@ -299,4 +267,4 @@ VoiceAgentService.ctorParameters = () => [
299
267
  { type: PlatformTokenRefreshService },
300
268
  { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
301
269
  ];
302
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AACpD,OAAO,EAAE,MAAM,EAAE,UAAU,EAAa,WAAW,EAAE,MAAM,eAAe,CAAC;AAC3E,OAAO,EACL,eAAe,EACf,aAAa,EAEb,OAAO,EACP,YAAY,GACb,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACzD,OAAO,EAAE,2BAA2B,EAAE,MAAM,kDAAkD,CAAC;AAC/F,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,2BAA2B,EAAE,MAAM,kCAAkC,CAAC;AAC/E,OAAO,EAAE,uBAAuB,EAAE,MAAM,8BAA8B,CAAC;;;;;;AAevE;;;;;;;;;;GAUG;AAIH,MAAM,OAAO,iBAAiB;IA4B5B,YACU,aAAmC,EACnC,QAAqC,EACrC,WAAoC,EACpC,oBAAiD;IACzD,8FAA8F;IACjE,UAAkB;QALvC,kBAAa,GAAb,aAAa,CAAsB;QACnC,aAAQ,GAAR,QAAQ,CAA6B;QACrC,gBAAW,GAAX,WAAW,CAAyB;QACpC,yBAAoB,GAApB,oBAAoB,CAA6B;QAE5B,eAAU,GAAV,UAAU,CAAQ;QAjCzC,qBAAgB,GAAG,IAAI,eAAe,CAAY,MAAM,CAAC,CAAC;QAC1D,sBAAiB,GAAG,IAAI,eAAe,CAAS,EAAE,CAAC,CAAC;QACpD,oBAAe,GAAG,IAAI,eAAe,CAAS,OAAO,CAAC,CAAC;QACvD,sBAAiB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACxD,0BAAqB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAC5D,uBAAkB,GAAG,IAAI,eAAe,CAAW,EAAE,CAAC,CAAC;QACvD,0BAAqB,GAAG,IAAI,OAAO,EAAkB,CAAC;QACtD,yBAAoB,GAAG,IAAI,OAAO,EAAU,CAAC;QAE7C,kBAAa,GAAG,CAAC,CAAC;QAClB,qBAAgB,GAA0C,IAAI,CAAC;QAE/D,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QACnC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAC/B,kCAA6B,GAAG,KAAK,CAAC;QAE9C,eAAU,GAA0B,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACzE,gBAAW,GAAuB,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACxE,cAAS,GAAuB,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QACpE,gBAAW,GAAwB,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACzE,oBAAe,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAC5C,iBAAY,GAAyB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC5E,oBAAe,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAC5C,mBAAc,GAAuB,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;QAU5E,8DAA8D;QAC9D,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CACnD,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,CAAC,CACrC,CACF,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;QACjC,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAED,gFAAgF;IAChF,WAAW;QACT,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM;YAAE,OAAO;QACnD,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;QAC1B,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;QAC3B,6EAA6E;QAC7E,KAAK,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;QACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACnC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAChC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAClC,IAAI,CAAC,6BAA6B,GAAG,KAAK,CAAC;IAC7C,CAAC;IAEK,OAAO,CACX,MAAc,EACd,KAAa,EACb,KAAa,EACb,cAAsB,EACtB,MAAc,EACd,UAAkB,EAClB,OAAe,EACf,QAAgB,EAChB,eAAuB,EACvB,WAAoB;;YAEpB,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM,EAAE;gBAC1C,OAAO,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;gBACzC,OAAO;aACR;YAED,IAAI;gBACF,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;gBAE7C,gEAAgE;gBAChE,MAAM,IAAI,CAAC,WAAW,CAAC,sBAAsB,EAAE,CAAC;gBAEhD,IAAI,WAAW,GAAG,KAAK,CAAC;gBACxB,4DAA4D;gBAC5D,oEAAoE;gBACpE,uEAAuE;gBACvE,IAAI,WAAW,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;oBACrD,IAAI;wBACF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,oBAAoB;6BAC5C,sBAAsB,CAAC,KAAK,EAAE,WAAW,CAAC;6BAC1C,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;6BACb,SAAS,EAAE,CAAC;wBACf,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,WAAW,EAAE;4BACxB,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;yBACnC;qBACF;oBAAC,OAAO,CAAC,EAAE;wBACV,OAAO,CAAC,IAAI,CACV,qDAAqD,EACrD,CAAC,CACF,CAAC;qBACH;iBACF;gBAED,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBAC1C,MAAM,OAAO,GAAG,GAAG,OAAO,eAAe,CAAC;gBAE1C,MAAM,OAAO,GAA2B;oBACtC,cAAc,EAAE,kBAAkB;oBAClC,aAAa,EAAE,UAAU,WAAW,EAAE;oBACtC,WAAW,EAAE,MAAM;oBACnB,aAAa,EAAE,KAAK;oBACpB,kBAAkB,EAAE,eAAe;oBACnC,QAAQ;oBACR,OAAO;oBACP,UAAU;oBACV,4BAA4B,EAAE,MAAM;iBACrC,CAAC;gBAEF,mCAAmC;gBACnC,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE;oBAC/B,MAAM,EAAE,MAAM;oBACd,OAAO;oBACP,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;wBACnB,MAAM,EAAE,KAAK;wBACb,eAAe,EAAE,cAAc;wBAC/B,KAAK,EAAE,OAAO;qBACf,CAAC;iBACH,CAAC,CAAC;gBAEH,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE;oBACX,MAAM,IAAI,KAAK,CAAC,QAAQ,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;iBACvC;gBAED,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;gBAC9B,MAAM,KAAK,GAAG,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,SAAS,CAAC;gBAC9B,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;oBACvC,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;iBAC1C;gBAED,4DAA4D;gBAC5D,IAAI,CAAC,QAAQ,CAAC,YAAY;qBACvB,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;qBACvC,SAAS,CAAC,CAAO,OAAO,EAAE,EAAE;oBAC3B,IAAI;wBACF,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;qBACnC;oBAAC,OAAO,GAAG,EAAE;wBACZ,OAAO,CAAC,KAAK,CAAC,oBAAoB,EAAE,GAAG,CAAC,CAAC;wBACzC,IAAI,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,EAAE;4BAClC,MAAM,IAAI,CAAC,qBAAqB,EAAE,CAAC;4BACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;4BACnC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;4BAClC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;4BACjE,OAAO;yBACR;wBACD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;wBACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;wBACjD,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;qBACzB;gBACH,CAAC,CAAA,CAAC,CAAC;gBAEL,qCAAqC;gBACrC,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,eAAe;qBAC1B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;qBAC9B,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CACxD,CAAC;gBACF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,cAAc;qBACzB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;qBAC9B,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CACvD,CAAC;gBAEF,iDAAiD;gBACjD,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;aAC9B;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAK,CAAC,CAAC;gBACtD,IAAI,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,EAAE;oBACpC,MAAM,IAAI,CAAC,qBAAqB,EAAE,CAAC;oBACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;oBACnC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;oBAClC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;oBACjE,OAAO;iBACR;gBACD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACpC,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;gBACxB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;gBACjD,MAAM,KAAK,CAAC;aACb;QACH,CAAC;KAAA;IAEa,aAAa,CAAC,OAAe;;YACzC,oCAAoC;YACpC,MAAM,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YACxC,IAAI,CAAC,6BAA6B,GAAG,KAAK,CAAC;YAE3C,mDAAmD;YACnD,IAAI,CAAC,WAAW,CAAC,YAAY;iBAC1B,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,EAAoB,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,EAC1C,IAAI,CAAC,CAAC,CAAC,CACR;iBACA,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;gBACpB,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YACnC,CAAC,CAAC,CAAC;YAEL,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAC7C,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,CACnC,CACF,CAAC;YACF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,aAAa,CAAC;gBACZ,IAAI,CAAC,WAAW,CAAC,SAAS;gBAC1B,IAAI,CAAC,WAAW,CAAC,aAAa;aAC/B,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,EAAE;gBAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC;gBAC5C,IAAI,OAAO,KAAK,YAAY,IAAI,CAAC,GAAG,EAAE;oBACpC,OAAO;iBACR;gBACD,IAAI,OAAO,KAAK,YAAY,IAAI,GAAG,EAAE;oBACnC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;oBAChC,IAAI,CAAC,kBAAkB,EAAE,CAAC;oBAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBACtC,OAAO;iBACR;gBACD,IAAI,IAAI,EAAE;oBACR,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;iBACzC;qBAAM,IAAI,GAAG,EAAE;oBACd,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;iBACvC;qBAAM,IAAI,OAAO,KAAK,SAAS,IAAI,OAAO,KAAK,WAAW,EAAE;oBAC3D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;iBACzC;YACH,CAAC,CAAC,CACH,CAAC;YAEF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAC7C,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CACnC,CACF,CAAC;YAEF,sEAAsE;YACtE,4FAA4F;YAC5F,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,WAAW,CAAC,sBAAsB;iBACpC,IAAI,CACH,MAAM,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,aAAa,CAAC,EACxC,IAAI,CAAC,CAAC,CAAC,CACR;iBACA,SAAS,CAAC,GAAG,EAAE;gBACd,IAAI,IAAI,CAAC,6BAA6B;oBAAE,OAAO;gBAC/C,IAAI,CAAC,6BAA6B,GAAG,IAAI,CAAC;gBAC1C,IAAI,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE;oBAChC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;iBAClC;YACH,CAAC,CAAC,CACL,CAAC;YAEF,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAC/C,CAAC;KAAA;IAEK,UAAU;;YACd,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAE1B,8BAA8B;YAC9B,MAAM,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;YACpC,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;YAE3B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC1C,IAAI,CAAC,6BAA6B,GAAG,KAAK,CAAC;QAC7C,CAAC;KAAA;IAED,SAAS;QACP,MAAM,OAAO,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;QAC7C,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,CAAC;IACtC,CAAC;IAEO,kBAAkB;QACxB,MAAM,cAAc,GAAG,GAAG,EAAE;YAC1B,IAAI,IAAI,CAAC,aAAa,GAAG,CAAC,EAAE;gBAC1B,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,IAAI,CAAC,CAAC;gBACrE,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;gBACzC,MAAM,OAAO,GAAG,OAAO,GAAG,EAAE,CAAC;gBAC7B,IAAI,CAAC,eAAe,CAAC,IAAI,CACvB,GAAG,OAAO,IAAI,MAAM,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CACjD,CAAC;aACH;QACH,CAAC,CAAC;QACF,cAAc,EAAE,CAAC;QACjB,IAAI,CAAC,gBAAgB,GAAG,WAAW,CAAC,cAAc,EAAE,IAAI,CAAC,CAAC;IAC5D,CAAC;IAEO,iBAAiB;QACvB,IAAI,IAAI,CAAC,gBAAgB,EAAE;YACzB,aAAa,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YACrC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;SAC9B;IACH,CAAC;IAEO,oBAAoB,CAAC,KAAc;QACzC,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QACtD,MAAM,KAAK,GAAG,KAA0B,CAAC;QACzC,OAAO,CACL,KAAK,CAAC,IAAI,KAAK,iBAAiB;YAChC,KAAK,CAAC,IAAI,KAAK,uBAAuB;YACtC,KAAK,CAAC,IAAI,KAAK,eAAe,CAC/B,CAAC;IACJ,CAAC;IAEa,qBAAqB;;YACjC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAC1B,MAAM,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;YACpC,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;YAC3B,IAAI,CAAC,6BAA6B,GAAG,KAAK,CAAC;QAC7C,CAAC;KAAA;;;;YAtUF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YA9BQ,oBAAoB;YACpB,2BAA2B;YAC3B,uBAAuB;YAHvB,2BAA2B;YAkES,MAAM,uBAA9C,MAAM,SAAC,WAAW","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  Observable,\n  Subject,\n  Subscription,\n} from 'rxjs';\nimport { filter, take, takeUntil } from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.service';\nimport { DailyVoiceClientService } from './daily-voice-client.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator. Coordinates WebSocket (signaling) and Daily.js (WebRTC audio).\n *\n * CRITICAL: This service must NEVER use Socket.IO or ngx-socket-io. Voice flow uses only:\n * - Native WebSocket (WebSocketVoiceClientService) for signaling (room_created, transcripts)\n * - Daily.js (DailyVoiceClientService) for WebRTC audio. Audio does NOT flow over WebSocket.\n *\n * - Maintains callState, statusText, duration, isMicMuted, isUserSpeaking, audioLevels\n * - Uses WebSocket for room_created and transcripts only (no audio)\n * - Uses Daily.js for all audio, mic, and real-time speaking detection\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n  private hasAutoUnmutedAfterFirstAudio = false;\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> =\n    this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private dailyClient: DailyVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    // Waveform visualization only - do NOT use for speaking state\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n    this.wsClient.disconnect();\n    // Fire-and-forget: Daily disconnect is async; connect() will await if needed\n    void this.dailyClient.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n    this.hasAutoUnmutedAfterFirstAudio = false;\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      // Request mic permission before provisioning backend room flow.\n      await this.dailyClient.ensureMicrophoneAccess();\n\n      let accessToken = token;\n      // Align with chat drawer token handling: always delegate to\n      // PlatformTokenRefreshService when we have a usersApiUrl, so it can\n      // fall back to stored tokens even if the caller passed an empty token.\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) {\n            accessToken = ensured.accessToken;\n          }\n        } catch (e) {\n          console.warn(\n            '[HiveGpt Voice] Token refresh before connect failed',\n            e,\n          );\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const postUrl = `${baseUrl}/ai/ask-voice`;\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n        Authorization: `Bearer ${accessToken}`,\n        'x-api-key': apiKey,\n        'hive-bot-id': botId,\n        'domain-authority': domainAuthority,\n        eventUrl,\n        eventId,\n        eventToken,\n        'ngrok-skip-browser-warning': 'true',\n      };\n\n      // POST to get ws_url for signaling\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      if (!res.ok) {\n        throw new Error(`HTTP ${res.status}`);\n      }\n\n      const json = await res.json();\n      const wsUrl = json?.rn_ws_url;\n      if (!wsUrl || typeof wsUrl !== 'string') {\n        throw new Error('No ws_url in response');\n      }\n\n      // Subscribe to room_created BEFORE connecting to avoid race\n      this.wsClient.roomCreated$\n        .pipe(take(1), takeUntil(this.destroy$))\n        .subscribe(async (roomUrl) => {\n          try {\n            await this.onRoomCreated(roomUrl);\n          } catch (err) {\n            console.error('Daily join failed:', err);\n            if (this.isMicPermissionError(err)) {\n              await this.cleanupMediaAndSignal();\n              this.callStateSubject.next('idle');\n              this.durationSubject.next('0:00');\n              this.statusTextSubject.next('Microphone permission is required');\n              return;\n            }\n            this.callStateSubject.next('ended');\n            this.statusTextSubject.next('Connection failed');\n            await this.disconnect();\n          }\n        });\n\n      // Forward transcripts from WebSocket\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      // Connect signaling WebSocket (no audio over WS)\n      this.wsClient.connect(wsUrl);\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      if (this.isMicPermissionError(error)) {\n        await this.cleanupMediaAndSignal();\n        this.callStateSubject.next('idle');\n        this.durationSubject.next('0:00');\n        this.statusTextSubject.next('Microphone permission is required');\n        return;\n      }\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private async onRoomCreated(roomUrl: string): Promise<void> {\n    // Connect Daily.js for WebRTC audio\n    await this.dailyClient.connect(roomUrl);\n    this.hasAutoUnmutedAfterFirstAudio = false;\n\n    // Waveform: use local mic stream from Daily client\n    this.dailyClient.localStream$\n      .pipe(\n        filter((s): s is MediaStream => s != null),\n        take(1),\n      )\n      .subscribe((stream) => {\n        this.audioAnalyzer.start(stream);\n      });\n\n    this.subscriptions.add(\n      this.dailyClient.userSpeaking$.subscribe((s) =>\n        this.isUserSpeakingSubject.next(s),\n      ),\n    );\n    this.subscriptions.add(\n      combineLatest([\n        this.dailyClient.speaking$,\n        this.dailyClient.userSpeaking$,\n      ]).subscribe(([bot, user]) => {\n        const current = this.callStateSubject.value;\n        if (current === 'connecting' && !bot) {\n          return;\n        }\n        if (current === 'connecting' && bot) {\n          this.callStartTime = Date.now();\n          this.startDurationTimer();\n          this.callStateSubject.next('talking');\n          return;\n        }\n        if (user) {\n          this.callStateSubject.next('listening');\n        } else if (bot) {\n          this.callStateSubject.next('talking');\n        } else if (current === 'talking' || current === 'listening') {\n          this.callStateSubject.next('connected');\n        }\n      }),\n    );\n\n    this.subscriptions.add(\n      this.dailyClient.micMuted$.subscribe((muted) =>\n        this.isMicMutedSubject.next(muted),\n      ),\n    );\n\n    // One-time auto-unmute after first remote audio frame starts playing.\n    // This keeps initial capture muted until bot audio is heard, then restores normal mic flow.\n    this.subscriptions.add(\n      this.dailyClient.firstRemoteAudioFrame$\n        .pipe(\n          filter((hasFirstFrame) => hasFirstFrame),\n          take(1),\n        )\n        .subscribe(() => {\n          if (this.hasAutoUnmutedAfterFirstAudio) return;\n          this.hasAutoUnmutedAfterFirstAudio = true;\n          if (this.isMicMutedSubject.value) {\n            this.dailyClient.setMuted(false);\n          }\n        }),\n    );\n\n    this.statusTextSubject.next('Connecting...');\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n\n    // Daily first, then WebSocket\n    await this.dailyClient.disconnect();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n    this.hasAutoUnmutedAfterFirstAudio = false;\n  }\n\n  toggleMic(): void {\n    const current = this.isMicMutedSubject.value;\n    this.dailyClient.setMuted(!current);\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`,\n        );\n      }\n    };\n    updateDuration();\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n\n  private isMicPermissionError(error: unknown): boolean {\n    if (!error || typeof error !== 'object') return false;\n    const maybe = error as { name?: string };\n    return (\n      maybe.name === 'NotAllowedError' ||\n      maybe.name === 'PermissionDeniedError' ||\n      maybe.name === 'SecurityError'\n    );\n  }\n\n  private async cleanupMediaAndSignal(): Promise<void> {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n    await this.dailyClient.disconnect();\n    this.wsClient.disconnect();\n    this.hasAutoUnmutedAfterFirstAudio = false;\n  }\n}\n"]}
270
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  Observable,\n  Subject,\n  Subscription,\n} from 'rxjs';\nimport { filter, take, takeUntil } from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.service';\nimport { DailyVoiceClientService } from './daily-voice-client.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator. Coordinates WebSocket (signaling) and Daily.js (WebRTC audio).\n *\n * CRITICAL: This service must NEVER use Socket.IO or ngx-socket-io. Voice flow uses only:\n * - Native WebSocket (WebSocketVoiceClientService) for signaling (room_created, transcripts)\n * - Daily.js (DailyVoiceClientService) for WebRTC audio. Audio does NOT flow over WebSocket.\n *\n * - Maintains callState, statusText, duration, isMicMuted, isUserSpeaking, audioLevels\n * - Uses WebSocket for room_created and transcripts only (no audio)\n * - Uses Daily.js for all audio, mic, and real-time speaking detection\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n  private hasAutoUnmutedAfterFirstAudio = false;\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> =\n    this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private dailyClient: DailyVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    // Waveform visualization only - do NOT use for speaking state\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n    this.wsClient.disconnect();\n    // Fire-and-forget: Daily disconnect is async; connect() will await if needed\n    void this.dailyClient.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n    this.hasAutoUnmutedAfterFirstAudio = false;\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      // Align with chat drawer token handling: always delegate to\n      // PlatformTokenRefreshService when we have a usersApiUrl, so it can\n      // fall back to stored tokens even if the caller passed an empty token.\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) {\n            accessToken = ensured.accessToken;\n          }\n        } catch (e) {\n          console.warn(\n            '[HiveGpt Voice] Token refresh before connect failed',\n            e,\n          );\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const postUrl = `${baseUrl}/ai/ask-voice`;\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n        Authorization: `Bearer ${accessToken}`,\n        'x-api-key': apiKey,\n        'hive-bot-id': botId,\n        'domain-authority': domainAuthority,\n        eventUrl,\n        eventId,\n        eventToken,\n        'ngrok-skip-browser-warning': 'true',\n      };\n\n      // POST to get ws_url for signaling\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      if (!res.ok) {\n        throw new Error(`HTTP ${res.status}`);\n      }\n\n      const json = await res.json();\n      const wsUrl = json?.rn_ws_url;\n      if (!wsUrl || typeof wsUrl !== 'string') {\n        throw new Error('No ws_url in response');\n      }\n\n      // Subscribe to room_created BEFORE connecting to avoid race\n      this.wsClient.roomCreated$\n        .pipe(take(1), takeUntil(this.destroy$))\n        .subscribe(async (roomUrl) => {\n          try {\n            await this.onRoomCreated(roomUrl);\n          } catch (err) {\n            console.error('Daily join failed:', err);\n            this.callStateSubject.next('ended');\n            this.statusTextSubject.next('Connection failed');\n            await this.disconnect();\n            throw err;\n          }\n        });\n\n      // Forward transcripts from WebSocket\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      // Connect signaling WebSocket (no audio over WS)\n      this.wsClient.connect(wsUrl);\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private async onRoomCreated(roomUrl: string): Promise<void> {\n    // Connect Daily.js for WebRTC audio\n    await this.dailyClient.connect(roomUrl);\n    this.hasAutoUnmutedAfterFirstAudio = false;\n\n    // Waveform: use local mic stream from Daily client\n    this.dailyClient.localStream$\n      .pipe(\n        filter((s): s is MediaStream => s != null),\n        take(1),\n      )\n      .subscribe((stream) => {\n        this.audioAnalyzer.start(stream);\n      });\n\n    this.subscriptions.add(\n      this.dailyClient.userSpeaking$.subscribe((s) =>\n        this.isUserSpeakingSubject.next(s),\n      ),\n    );\n    this.subscriptions.add(\n      combineLatest([\n        this.dailyClient.speaking$,\n        this.dailyClient.userSpeaking$,\n      ]).subscribe(([bot, user]) => {\n        const current = this.callStateSubject.value;\n        if (current === 'connecting' && !bot) {\n          return;\n        }\n        if (current === 'connecting' && bot) {\n          this.callStartTime = Date.now();\n          this.startDurationTimer();\n          this.callStateSubject.next('talking');\n          return;\n        }\n        if (user) {\n          this.callStateSubject.next('listening');\n        } else if (bot) {\n          this.callStateSubject.next('talking');\n        } else if (current === 'talking' || current === 'listening') {\n          this.callStateSubject.next('connected');\n        }\n      }),\n    );\n\n    this.subscriptions.add(\n      this.dailyClient.micMuted$.subscribe((muted) =>\n        this.isMicMutedSubject.next(muted),\n      ),\n    );\n\n    // One-time auto-unmute after first remote audio frame starts playing.\n    // This keeps initial capture muted until bot audio is heard, then restores normal mic flow.\n    this.subscriptions.add(\n      this.dailyClient.firstRemoteAudioFrame$\n        .pipe(\n          filter((hasFirstFrame) => hasFirstFrame),\n          take(1),\n        )\n        .subscribe(() => {\n          if (this.hasAutoUnmutedAfterFirstAudio) return;\n          this.hasAutoUnmutedAfterFirstAudio = true;\n          if (this.isMicMutedSubject.value) {\n            this.dailyClient.setMuted(false);\n          }\n        }),\n    );\n\n    this.statusTextSubject.next('Connecting...');\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n\n    // Daily first, then WebSocket\n    await this.dailyClient.disconnect();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n    this.hasAutoUnmutedAfterFirstAudio = false;\n  }\n\n  toggleMic(): void {\n    const current = this.isMicMutedSubject.value;\n    this.dailyClient.setMuted(!current);\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`,\n        );\n      }\n    };\n    updateDuration();\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}