@hivegpt/hiveai-angular 0.0.425 → 0.0.428

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (29) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +506 -164
  2. package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
  3. package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
  4. package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
  5. package/esm2015/hivegpt-hiveai-angular.js +6 -4
  6. package/esm2015/lib/components/chat-drawer/chat-drawer.component.js +22 -7
  7. package/esm2015/lib/components/voice-agent/services/audio-analyzer.service.js +5 -1
  8. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +181 -0
  9. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +134 -140
  10. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +95 -0
  11. package/esm2015/lib/components/voice-agent/voice-agent.module.js +10 -2
  12. package/fesm2015/hivegpt-hiveai-angular.js +432 -149
  13. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  14. package/hivegpt-hiveai-angular.d.ts +5 -3
  15. package/hivegpt-hiveai-angular.d.ts.map +1 -1
  16. package/hivegpt-hiveai-angular.metadata.json +1 -1
  17. package/lib/components/chat-drawer/chat-drawer.component.d.ts +7 -0
  18. package/lib/components/chat-drawer/chat-drawer.component.d.ts.map +1 -1
  19. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +4 -0
  20. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts.map +1 -1
  21. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +48 -0
  22. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -0
  23. package/lib/components/voice-agent/services/voice-agent.service.d.ts +24 -8
  24. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  25. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +49 -0
  26. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -0
  27. package/lib/components/voice-agent/voice-agent.module.d.ts +4 -0
  28. package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
  29. package/package.json +2 -1
@@ -1,6 +1,10 @@
1
1
  import { Injectable } from '@angular/core';
2
2
  import { BehaviorSubject } from 'rxjs';
3
3
  import * as i0 from "@angular/core";
4
+ /**
5
+ * Audio analyzer for waveform visualization only.
6
+ * Do NOT use isUserSpeaking$ for call state; speaking state must come from Daily.js.
7
+ */
4
8
  export class AudioAnalyzerService {
5
9
  constructor() {
6
10
  this.audioContext = null;
@@ -116,4 +120,4 @@ AudioAnalyzerService.decorators = [
116
120
  providedIn: 'root'
117
121
  },] }
118
122
  ];
119
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;;AAKnD,MAAM,OAAO,oBAAoB;IAHjC;QAIU,iBAAY,GAAwB,IAAI,CAAC;QACzC,iBAAY,GAAwB,IAAI,CAAC;QACzC,cAAS,GAAsB,IAAI,CAAC;QACpC,qBAAgB,GAAkB,IAAI,CAAC;QACvC,cAAS,GAAG,KAAK,CAAC;QAElB,uBAAkB,GAAG,IAAI,eAAe,CAAW,EAAE,CAAC,CAAC;QACvD,0BAAqB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAEpE,iCAAiC;QACzB,eAAU,GAAG,CAAC,CAAC;QACf,sBAAiB,GAAa,EAAE,CAAC;QACxB,6BAAwB,GAAG,EAAE,CAAC;QAC9B,kCAA6B,GAAG,GAAG,CAAC;QACpC,uBAAkB,GAAG,EAAE,CAAC;QAEzC,iBAAY,GAAyB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC5E,oBAAe,GAAwB,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;KAgHlF;IA9GC,KAAK,CAAC,MAAmB;QACvB,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,IAAI,EAAE,CAAC;SACb;QAED,IAAI;YACF,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,IAAK,MAAc,CAAC,kBAAkB,CAAC,EAAE,CAAC;YACtF,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;YAEjE,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,EAAE,CAAC;YACvD,IAAI,CAAC,YAAY,CAAC,OAAO,GAAG,IAAI,CAAC;YACjC,IAAI,CAAC,YAAY,CAAC,qBAAqB,GAAG,GAAG,CAAC;YAE9C,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC;YACzD,IAAI,CAAC,SAAS,GAAG,IAAI,UAAU,CAAC,YAAY,CAAC,CAAC;YAE9C,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAElC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;YACtB,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;YAC5B,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;YAEpB,IAAI,CAAC,OAAO,EAAE,CAAC;SAChB;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,KAAK,CAAC,gCAAgC,EAAE,KAAK,CAAC,CAAC;YACvD,IAAI,CAAC,IAAI,EAAE,CAAC;SACb;IACH,CAAC;IAED,IAAI;QACF,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QAEvB,IAAI,IAAI,CAAC,gBAAgB,KAAK,IAAI,EAAE;YAClC,oBAAoB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAC5C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;SAC9B;QAED,IAAI,IAAI,CAAC,YAAY,EAAE;YACrB,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,CAAC;YAC/B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;SAC1B;QAED,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,CAAC,KAAK,KAAK,QAAQ,EAAE;YAC7D,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YAC/C,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;SAC1B;QAED,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAEO,OAAO;QACb,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YAC5D,OAAO;SACR;QAED,IAAI,CAAC,YAAY,CAAC,qBAAqB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAExD,0CAA0C;QAC1C,MAAM,GAAG,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE9C,2CAA2C;QAC3C,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,GAAG,IAAI,CAAC,wBAAwB,EAAE;YACjE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACjC,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,IAAI,CAAC,wBAAwB,EAAE;gBACnE,gCAAgC;gBAChC,MAAM,GAAG,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC9D,IAAI,CAAC,UAAU,GAAG,GAAG,GAAG,IAAI,CAAC,wBAAwB,CAAC;aACvD;SACF;aAAM;YACL,iDAAiD;YACjD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,GAAG,IAAI,GAAG,GAAG,GAAG,IAAI,CAAC;SACvD;QAED,6BAA6B;QAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,6BAA6B,CAAC;QACvE,MAAM,UAAU,GAAG,GAAG,GAAG,SAAS,CAAC;QACnC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAE5C,yBAAyB;QACzB,MAAM,IAAI,GAAG,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEnC,IAAI,CAAC,gBAAgB,GAAG,qBAAqB,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;IACtE,CAAC;IAEO,YAAY,CAAC,IAAgB;QACnC,IAAI,GAAG,GAAG,CAAC,CAAC;QACZ,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YACpC,MAAM,UAAU,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC;YACzC,GAAG,IAAI,UAAU,GAAG,UAAU,CAAC;SAChC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC;IACtC,CAAC;IAEO,oBAAoB,CAAC,IAAgB;QAC3C,MAAM,IAAI,GAAa,EAAE,CAAC;QAC1B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAE/D,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC,EAAE,EAAE;YAChD,MAAM,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC;YACvB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1B,qBAAqB;YACrB,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC;YACvD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC;SACnD;QAED,OAAO,IAAI,CAAC;IACd,CAAC;;;;YApIF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 40;\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      const source = this.audioContext.createMediaStreamSource(stream);\n      \n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.8;\n      \n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n      \n      source.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.analyze();\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n    this.isUserSpeakingSubject.next(isSpeaking);\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n    this.audioLevelsSubject.next(bars);\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const bars: number[] = [];\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    \n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const index = i * step;\n      const value = data[index];\n      // Normalize to 0-100\n      const normalized = Math.abs((value - 128) / 128) * 100;\n      bars.push(Math.min(100, Math.max(0, normalized)));\n    }\n    \n    return bars;\n  }\n}\n"]}
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Audio analyzer for waveform visualization only.\n * Do NOT use isUserSpeaking$ for call state; speaking state must come from Daily.js.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 40;\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      const source = this.audioContext.createMediaStreamSource(stream);\n      \n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.8;\n      \n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n      \n      source.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.analyze();\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n    this.isUserSpeakingSubject.next(isSpeaking);\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n    this.audioLevelsSubject.next(bars);\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const bars: number[] = [];\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    \n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const index = i * step;\n      const value = data[index];\n      // Normalize to 0-100\n      const normalized = Math.abs((value - 128) / 128) * 100;\n      bars.push(Math.min(100, Math.max(0, normalized)));\n    }\n    \n    return bars;\n  }\n}\n"]}
@@ -0,0 +1,181 @@
1
+ import { __awaiter } from "tslib";
2
+ import { Injectable, NgZone } from '@angular/core';
3
+ import { BehaviorSubject } from 'rxjs';
4
+ import Daily from '@daily-co/daily-js';
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
+ * - Provide real-time speaking detection via active-speaker-change (primary)
13
+ * and track-started/track-stopped (fallback for immediate feedback)
14
+ * - Expose speaking$ (bot speaking), userSpeaking$ (user speaking), micMuted$
15
+ * - Expose localStream$ for waveform visualization (AudioAnalyzerService)
16
+ *
17
+ * Speaking state flips immediately when agent audio starts playing.
18
+ * If user speaks while bot is talking, state switches to listening.
19
+ */
20
+ export class DailyVoiceClientService {
21
+ constructor(ngZone) {
22
+ this.ngZone = ngZone;
23
+ this.callObject = null;
24
+ this.localStream = null;
25
+ this.localSessionId = null;
26
+ this.speakingSubject = new BehaviorSubject(false);
27
+ this.userSpeakingSubject = new BehaviorSubject(false);
28
+ this.micMutedSubject = new BehaviorSubject(false);
29
+ this.localStreamSubject = new BehaviorSubject(null);
30
+ /** True when bot (remote participant) is the active speaker. */
31
+ this.speaking$ = this.speakingSubject.asObservable();
32
+ /** True when user (local participant) is the active speaker. */
33
+ this.userSpeaking$ = this.userSpeakingSubject.asObservable();
34
+ /** True when mic is muted. */
35
+ this.micMuted$ = this.micMutedSubject.asObservable();
36
+ /** Emits local mic stream for waveform visualization. */
37
+ this.localStream$ = this.localStreamSubject.asObservable();
38
+ }
39
+ /**
40
+ * Connect to Daily room. Acquires mic first for waveform, then joins with audio.
41
+ * @param roomUrl Daily room URL (from room_created)
42
+ * @param token Optional meeting token
43
+ */
44
+ connect(roomUrl, token) {
45
+ return __awaiter(this, void 0, void 0, function* () {
46
+ if (this.callObject) {
47
+ yield this.disconnect();
48
+ }
49
+ try {
50
+ // Get mic stream for both Daily and waveform (single capture)
51
+ const stream = yield navigator.mediaDevices.getUserMedia({ audio: true });
52
+ const audioTrack = stream.getAudioTracks()[0];
53
+ if (!audioTrack) {
54
+ stream.getTracks().forEach((t) => t.stop());
55
+ throw new Error('No audio track');
56
+ }
57
+ this.localStream = stream;
58
+ this.localStreamSubject.next(stream);
59
+ // Create audio-only call object
60
+ // videoSource: false = no camera, audioSource = our mic track
61
+ const callObject = Daily.createCallObject({
62
+ videoSource: false,
63
+ audioSource: audioTrack,
64
+ });
65
+ this.callObject = callObject;
66
+ this.setupEventHandlers(callObject);
67
+ // Join room; Daily handles playback of remote (bot) audio automatically
68
+ yield callObject.join({ url: roomUrl, token });
69
+ const participants = callObject.participants();
70
+ if (participants === null || participants === void 0 ? void 0 : participants.local) {
71
+ this.localSessionId = participants.local.session_id;
72
+ }
73
+ // Initial mute state: Daily starts with audio on
74
+ this.micMutedSubject.next(!callObject.localAudio());
75
+ }
76
+ catch (err) {
77
+ this.cleanup();
78
+ throw err;
79
+ }
80
+ });
81
+ }
82
+ setupEventHandlers(call) {
83
+ // active-speaker-change: primary source for real-time speaking detection.
84
+ // Emits when the loudest participant changes; bot speaking = remote is active.
85
+ call.on('active-speaker-change', (event) => {
86
+ this.ngZone.run(() => {
87
+ var _a;
88
+ const peerId = (_a = event === null || event === void 0 ? void 0 : event.activeSpeaker) === null || _a === void 0 ? void 0 : _a.peerId;
89
+ if (!peerId || !this.localSessionId) {
90
+ this.speakingSubject.next(false);
91
+ this.userSpeakingSubject.next(false);
92
+ return;
93
+ }
94
+ const isLocal = peerId === this.localSessionId;
95
+ this.userSpeakingSubject.next(isLocal);
96
+ this.speakingSubject.next(!isLocal);
97
+ });
98
+ });
99
+ // track-started / track-stopped: fallback for immediate feedback when
100
+ // remote (bot) audio track starts or stops. Ensures talking indicator
101
+ // flips as soon as agent audio begins, without waiting for active-speaker-change.
102
+ call.on('track-started', (event) => {
103
+ this.ngZone.run(() => {
104
+ var _a, _b;
105
+ const p = event === null || event === void 0 ? void 0 : event.participant;
106
+ 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;
107
+ if (p && !p.local && type === 'audio') {
108
+ this.speakingSubject.next(true);
109
+ }
110
+ });
111
+ });
112
+ call.on('track-stopped', (event) => {
113
+ this.ngZone.run(() => {
114
+ var _a, _b;
115
+ const p = event === null || event === void 0 ? void 0 : event.participant;
116
+ 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;
117
+ if (p && !p.local && type === 'audio') {
118
+ this.speakingSubject.next(false);
119
+ }
120
+ });
121
+ });
122
+ call.on('left-meeting', () => {
123
+ this.ngZone.run(() => this.cleanup());
124
+ });
125
+ call.on('error', (event) => {
126
+ this.ngZone.run(() => {
127
+ var _a;
128
+ console.error('DailyVoiceClient: Daily error', (_a = event === null || event === void 0 ? void 0 : event.errorMsg) !== null && _a !== void 0 ? _a : event);
129
+ this.cleanup();
130
+ });
131
+ });
132
+ }
133
+ /** Set mic muted state. */
134
+ setMuted(muted) {
135
+ if (!this.callObject)
136
+ return;
137
+ this.callObject.setLocalAudio(!muted);
138
+ this.micMutedSubject.next(muted);
139
+ }
140
+ /** Disconnect and cleanup. */
141
+ disconnect() {
142
+ return __awaiter(this, void 0, void 0, function* () {
143
+ if (!this.callObject) {
144
+ this.cleanup();
145
+ return;
146
+ }
147
+ try {
148
+ yield this.callObject.leave();
149
+ }
150
+ catch (e) {
151
+ // ignore
152
+ }
153
+ this.cleanup();
154
+ });
155
+ }
156
+ cleanup() {
157
+ if (this.callObject) {
158
+ this.callObject.destroy().catch(() => { });
159
+ this.callObject = null;
160
+ }
161
+ if (this.localStream) {
162
+ this.localStream.getTracks().forEach((t) => t.stop());
163
+ this.localStream = null;
164
+ }
165
+ this.localSessionId = null;
166
+ this.speakingSubject.next(false);
167
+ this.userSpeakingSubject.next(false);
168
+ this.localStreamSubject.next(null);
169
+ // Keep last micMuted state; will reset on next connect
170
+ }
171
+ }
172
+ DailyVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function DailyVoiceClientService_Factory() { return new DailyVoiceClientService(i0.ɵɵinject(i0.NgZone)); }, token: DailyVoiceClientService, providedIn: "root" });
173
+ DailyVoiceClientService.decorators = [
174
+ { type: Injectable, args: [{
175
+ providedIn: 'root',
176
+ },] }
177
+ ];
178
+ DailyVoiceClientService.ctorParameters = () => [
179
+ { type: NgZone }
180
+ ];
181
+ //# 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 * - Provide real-time speaking detection via active-speaker-change (primary)\n *   and track-started/track-stopped (fallback for immediate feedback)\n * - Expose speaking$ (bot speaking), userSpeaking$ (user speaking), micMuted$\n * - Expose localStream$ for waveform visualization (AudioAnalyzerService)\n *\n * Speaking state flips immediately when agent audio starts playing.\n * If user speaks while bot is talking, state switches to listening.\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\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      await callObject.join({ url: roomUrl, token });\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: primary source for real-time speaking detection.\n    // Emits when the loudest participant changes; bot speaking = remote is active.\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.speakingSubject.next(false);\n          this.userSpeakingSubject.next(false);\n          return;\n        }\n        const isLocal = peerId === this.localSessionId;\n        this.userSpeakingSubject.next(isLocal);\n        this.speakingSubject.next(!isLocal);\n      });\n    });\n\n    // track-started / track-stopped: fallback for immediate feedback when\n    // remote (bot) audio track starts or stops. Ensures talking indicator\n    // flips as soon as agent audio begins, without waiting for active-speaker-change.\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        if (p && !p.local && type === 'audio') {\n          this.speakingSubject.next(true);\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.speakingSubject.next(false);\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  /** 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    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"]}