@hivegpt/hiveai-angular 0.0.576 → 0.0.578

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 (20) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +214 -356
  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/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +54 -85
  6. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +14 -22
  7. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +60 -90
  8. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +34 -81
  9. package/fesm2015/hivegpt-hiveai-angular.js +157 -272
  10. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  11. package/hivegpt-hiveai-angular.metadata.json +1 -1
  12. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +1 -7
  13. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
  14. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +3 -5
  15. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
  16. package/lib/components/voice-agent/services/voice-agent.service.d.ts +1 -3
  17. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  18. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +12 -5
  19. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
  20. package/package.json +1 -1
@@ -1,15 +1,13 @@
1
1
  import { __awaiter } from "tslib";
2
- import { isPlatformBrowser } from '@angular/common';
3
- import { Component, Inject, Input, Output, EventEmitter, Injector, PLATFORM_ID } from '@angular/core';
2
+ import { Component, Input, Output, EventEmitter, Injector } from '@angular/core';
4
3
  import { VoiceAgentService } from '../../services/voice-agent.service';
5
4
  import { AudioAnalyzerService } from '../../services/audio-analyzer.service';
6
5
  import { VOICE_MODAL_CONFIG, VOICE_MODAL_CLOSE_CALLBACK } from '../../voice-modal-tokens';
7
6
  export class VoiceAgentModalComponent {
8
- constructor(voiceAgentService, audioAnalyzer, injector, platformId) {
7
+ constructor(voiceAgentService, audioAnalyzer, injector) {
9
8
  this.voiceAgentService = voiceAgentService;
10
9
  this.audioAnalyzer = audioAnalyzer;
11
10
  this.injector = injector;
12
- this.platformId = platformId;
13
11
  this.close = new EventEmitter();
14
12
  this.apiKey = '';
15
13
  this.eventToken = '';
@@ -21,8 +19,6 @@ export class VoiceAgentModalComponent {
21
19
  this.usersApiUrl = '';
22
20
  this.injectedConfig = null;
23
21
  this.onCloseCallback = null;
24
- /** Held until destroy; passed to Daily so we do not stop/re-acquire (avoids extra prompts on some browsers). */
25
- this.warmMicStream = null;
26
22
  /** Hardcoded voice agent avatar (Nia). */
27
23
  this.displayAvatarUrl = 'https://www.jotform.com/uploads/mehmetkarakasli/form_files/1564593667676a8e85f23758.86945537_icon.png';
28
24
  this.callState = 'idle';
@@ -38,83 +34,58 @@ export class VoiceAgentModalComponent {
38
34
  this.isConnecting = false;
39
35
  }
40
36
  ngOnInit() {
41
- void this.bootstrap();
42
- }
43
- bootstrap() {
44
37
  var _a, _b, _c, _d, _e, _f, _g, _h;
45
- return __awaiter(this, void 0, void 0, function* () {
46
- this.injectedConfig = this.injector.get(VOICE_MODAL_CONFIG, null);
47
- this.onCloseCallback = this.injector.get(VOICE_MODAL_CLOSE_CALLBACK, null);
48
- if (this.injectedConfig) {
49
- this.apiUrl = this.injectedConfig.apiUrl;
50
- this.token = this.injectedConfig.token;
51
- this.botId = this.injectedConfig.botId;
52
- this.conversationId = this.injectedConfig.conversationId;
53
- this.apiKey = (_a = this.injectedConfig.apiKey) !== null && _a !== void 0 ? _a : '';
54
- this.eventToken = (_b = this.injectedConfig.eventToken) !== null && _b !== void 0 ? _b : '';
55
- this.eventId = (_c = this.injectedConfig.eventId) !== null && _c !== void 0 ? _c : '';
56
- this.eventUrl = (_d = this.injectedConfig.eventUrl) !== null && _d !== void 0 ? _d : '';
57
- this.domainAuthority = (_e = this.injectedConfig.domainAuthority) !== null && _e !== void 0 ? _e : 'prod-lite';
58
- this.agentName = (_f = this.injectedConfig.agentName) !== null && _f !== void 0 ? _f : this.agentName;
59
- this.agentRole = (_g = this.injectedConfig.agentRole) !== null && _g !== void 0 ? _g : this.agentRole;
60
- this.agentAvatar = this.injectedConfig.agentAvatar;
61
- this.usersApiUrl = (_h = this.injectedConfig.usersApiUrl) !== null && _h !== void 0 ? _h : this.usersApiUrl;
38
+ // When opened via Overlay, config is provided by injection
39
+ this.injectedConfig = this.injector.get(VOICE_MODAL_CONFIG, null);
40
+ this.onCloseCallback = this.injector.get(VOICE_MODAL_CLOSE_CALLBACK, null);
41
+ if (this.injectedConfig) {
42
+ this.apiUrl = this.injectedConfig.apiUrl;
43
+ this.token = this.injectedConfig.token;
44
+ this.botId = this.injectedConfig.botId;
45
+ this.conversationId = this.injectedConfig.conversationId;
46
+ this.apiKey = (_a = this.injectedConfig.apiKey) !== null && _a !== void 0 ? _a : '';
47
+ this.eventToken = (_b = this.injectedConfig.eventToken) !== null && _b !== void 0 ? _b : '';
48
+ this.eventId = (_c = this.injectedConfig.eventId) !== null && _c !== void 0 ? _c : '';
49
+ this.eventUrl = (_d = this.injectedConfig.eventUrl) !== null && _d !== void 0 ? _d : '';
50
+ this.domainAuthority = (_e = this.injectedConfig.domainAuthority) !== null && _e !== void 0 ? _e : 'prod-lite';
51
+ this.agentName = (_f = this.injectedConfig.agentName) !== null && _f !== void 0 ? _f : this.agentName;
52
+ this.agentRole = (_g = this.injectedConfig.agentRole) !== null && _g !== void 0 ? _g : this.agentRole;
53
+ this.agentAvatar = this.injectedConfig.agentAvatar;
54
+ this.usersApiUrl = (_h = this.injectedConfig.usersApiUrl) !== null && _h !== void 0 ? _h : this.usersApiUrl;
55
+ }
56
+ // Subscribe to observables
57
+ this.subscriptions.push(this.voiceAgentService.callState$.subscribe(state => {
58
+ this.callState = state;
59
+ this.isSpeaking = state === 'talking';
60
+ if (state === 'listening' || state === 'talking') {
61
+ this.hasLeftConnectedOnce = true;
62
62
  }
63
- // Subscribe to observables
64
- this.subscriptions.push(this.voiceAgentService.callState$.subscribe(state => {
65
- this.callState = state;
66
- this.isSpeaking = state === 'talking';
67
- if (state === 'listening' || state === 'talking') {
68
- this.hasLeftConnectedOnce = true;
69
- }
70
- if (state === 'idle' || state === 'ended') {
71
- this.hasLeftConnectedOnce = false;
72
- }
73
- }));
74
- this.subscriptions.push(this.voiceAgentService.statusText$.subscribe(text => {
75
- this.statusText = text;
76
- }));
77
- this.subscriptions.push(this.voiceAgentService.duration$.subscribe(duration => {
78
- this.duration = duration;
79
- }));
80
- this.subscriptions.push(this.voiceAgentService.isMicMuted$.subscribe(muted => {
81
- this.isMicMuted = muted;
82
- }));
83
- this.subscriptions.push(this.voiceAgentService.isUserSpeaking$.subscribe(speaking => {
84
- this.isUserSpeaking = speaking;
85
- }));
86
- this.subscriptions.push(this.voiceAgentService.audioLevels$.subscribe(levels => {
87
- this.audioLevels = levels;
88
- }));
89
- this.voiceAgentService.resetToIdle();
90
- yield this.startCall();
91
- });
63
+ if (state === 'idle' || state === 'ended') {
64
+ this.hasLeftConnectedOnce = false;
65
+ }
66
+ }));
67
+ this.subscriptions.push(this.voiceAgentService.statusText$.subscribe(text => {
68
+ this.statusText = text;
69
+ }));
70
+ this.subscriptions.push(this.voiceAgentService.duration$.subscribe(duration => {
71
+ this.duration = duration;
72
+ }));
73
+ this.subscriptions.push(this.voiceAgentService.isMicMuted$.subscribe(muted => {
74
+ this.isMicMuted = muted;
75
+ }));
76
+ this.subscriptions.push(this.voiceAgentService.isUserSpeaking$.subscribe(speaking => {
77
+ this.isUserSpeaking = speaking;
78
+ }));
79
+ this.subscriptions.push(this.voiceAgentService.audioLevels$.subscribe(levels => {
80
+ this.audioLevels = levels;
81
+ }));
82
+ // Modal opens in idle state, then immediately starts connecting.
83
+ this.voiceAgentService.resetToIdle();
84
+ void this.startCall();
92
85
  }
93
86
  ngOnDestroy() {
94
- this.subscriptions.forEach((sub) => sub.unsubscribe());
95
- void this.disconnect().finally(() => {
96
- var _a;
97
- (_a = this.warmMicStream) === null || _a === void 0 ? void 0 : _a.getTracks().forEach((t) => t.stop());
98
- this.warmMicStream = null;
99
- });
100
- }
101
- /** Ensures a live mic stream for this call (re-acquire after Daily stops tracks on hang-up). */
102
- ensureMicForCall() {
103
- var _a;
104
- return __awaiter(this, void 0, void 0, function* () {
105
- if (!isPlatformBrowser(this.platformId))
106
- return undefined;
107
- if ((_a = this.warmMicStream) === null || _a === void 0 ? void 0 : _a.getAudioTracks().some((t) => t.readyState === 'live')) {
108
- return this.warmMicStream;
109
- }
110
- try {
111
- this.warmMicStream = yield navigator.mediaDevices.getUserMedia({ audio: true });
112
- return this.warmMicStream;
113
- }
114
- catch (_b) {
115
- return undefined;
116
- }
117
- });
87
+ this.subscriptions.forEach(sub => sub.unsubscribe());
88
+ this.disconnect();
118
89
  }
119
90
  startCall() {
120
91
  return __awaiter(this, void 0, void 0, function* () {
@@ -122,8 +93,7 @@ export class VoiceAgentModalComponent {
122
93
  return;
123
94
  this.isConnecting = true;
124
95
  try {
125
- const mic = yield this.ensureMicForCall();
126
- yield this.voiceAgentService.connect(this.apiUrl, this.token, this.botId, this.conversationId, this.apiKey, this.eventToken, this.eventId, this.eventUrl, this.domainAuthority, this.usersApiUrl || undefined, mic);
96
+ yield this.voiceAgentService.connect(this.apiUrl, this.token, this.botId, this.conversationId, this.apiKey, this.eventToken, this.eventId, this.eventUrl, this.domainAuthority, this.usersApiUrl || undefined);
127
97
  }
128
98
  catch (error) {
129
99
  console.error('Failed to connect voice agent:', error);
@@ -162,7 +132,7 @@ export class VoiceAgentModalComponent {
162
132
  /** Call Again: reset to idle then start a new call. */
163
133
  callAgain() {
164
134
  this.voiceAgentService.resetToIdle();
165
- void this.startCall();
135
+ this.startCall();
166
136
  }
167
137
  /** Back to Chat: close modal and disconnect. */
168
138
  backToChat() {
@@ -189,8 +159,7 @@ VoiceAgentModalComponent.decorators = [
189
159
  VoiceAgentModalComponent.ctorParameters = () => [
190
160
  { type: VoiceAgentService },
191
161
  { type: AudioAnalyzerService },
192
- { type: Injector },
193
- { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
162
+ { type: Injector }
194
163
  ];
195
164
  VoiceAgentModalComponent.propDecorators = {
196
165
  close: [{ type: Output }],
@@ -208,4 +177,4 @@ VoiceAgentModalComponent.propDecorators = {
208
177
  agentAvatar: [{ type: Input }],
209
178
  usersApiUrl: [{ type: Input }]
210
179
  };
211
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent-modal.component.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Component, Inject, Input, OnInit, OnDestroy, Output, EventEmitter, Injector, PLATFORM_ID } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { VoiceAgentService, CallState } from '../../services/voice-agent.service';\nimport { AudioAnalyzerService } from '../../services/audio-analyzer.service';\nimport { VOICE_MODAL_CONFIG, VOICE_MODAL_CLOSE_CALLBACK, VoiceModalConfig } from '../../voice-modal-tokens';\n\n@Component({\n  selector: 'hivegpt-voice-agent-modal',\n  templateUrl: './voice-agent-modal.component.html',\n  styleUrls: ['./voice-agent-modal.component.scss']\n})\nexport class VoiceAgentModalComponent implements OnInit, OnDestroy {\n  @Output() close = new EventEmitter<void>();\n  @Input() apiUrl!: string;\n  @Input() token!: string;\n  @Input() botId!: string;\n  @Input() conversationId!: string;\n  @Input() apiKey: string = '';\n  @Input() eventToken: string = '';\n  @Input() eventId: string = '';\n  @Input() eventUrl: string = '';\n  @Input() domainAuthority: string = 'prod-lite';\n  @Input() agentName: string = 'AI Assistant';\n  @Input() agentRole: string = 'AI Agent Specialist';\n  @Input() agentAvatar?: string;\n  @Input() usersApiUrl: string = '';\n\n  private injectedConfig: VoiceModalConfig | null = null;\n  private onCloseCallback: (() => void) | null = null;\n\n  /** Held until destroy; passed to Daily so we do not stop/re-acquire (avoids extra prompts on some browsers). */\n  private warmMicStream: MediaStream | null = null;\n\n  constructor(\n    public voiceAgentService: VoiceAgentService,\n    public audioAnalyzer: AudioAnalyzerService,\n    private injector: Injector,\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {}\n\n  /** Hardcoded voice agent avatar (Nia). */\n  readonly displayAvatarUrl = 'https://www.jotform.com/uploads/mehmetkarakasli/form_files/1564593667676a8e85f23758.86945537_icon.png';\n\n  callState: CallState = 'idle';\n  statusText: string = '';\n  duration: string = '00:00';\n  isMicMuted: boolean = false;\n  isUserSpeaking: boolean = false;\n  audioLevels: number[] = [];\n  isSpeaking: boolean = false;\n\n  /** Track whether call has transitioned out of initial connected state. */\n  private hasLeftConnectedOnce: boolean = false;\n\n  private subscriptions: Subscription[] = [];\n\n  ngOnInit(): void {\n    void this.bootstrap();\n  }\n\n  private async bootstrap(): Promise<void> {\n    this.injectedConfig = this.injector.get(VOICE_MODAL_CONFIG, null);\n    this.onCloseCallback = this.injector.get(VOICE_MODAL_CLOSE_CALLBACK, null);\n    if (this.injectedConfig) {\n      this.apiUrl = this.injectedConfig.apiUrl;\n      this.token = this.injectedConfig.token;\n      this.botId = this.injectedConfig.botId;\n      this.conversationId = this.injectedConfig.conversationId;\n      this.apiKey = this.injectedConfig.apiKey ?? '';\n      this.eventToken = this.injectedConfig.eventToken ?? '';\n      this.eventId = this.injectedConfig.eventId ?? '';\n      this.eventUrl = this.injectedConfig.eventUrl ?? '';\n      this.domainAuthority = this.injectedConfig.domainAuthority ?? 'prod-lite';\n      this.agentName = this.injectedConfig.agentName ?? this.agentName;\n      this.agentRole = this.injectedConfig.agentRole ?? this.agentRole;\n      this.agentAvatar = this.injectedConfig.agentAvatar;\n      this.usersApiUrl = this.injectedConfig.usersApiUrl ?? this.usersApiUrl;\n    }\n\n    // Subscribe to observables\n    this.subscriptions.push(\n      this.voiceAgentService.callState$.subscribe(state => {\n        this.callState = state;\n        this.isSpeaking = state === 'talking';\n        if (state === 'listening' || state === 'talking') {\n          this.hasLeftConnectedOnce = true;\n        }\n        if (state === 'idle' || state === 'ended') {\n          this.hasLeftConnectedOnce = false;\n        }\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.statusText$.subscribe(text => {\n        this.statusText = text;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.duration$.subscribe(duration => {\n        this.duration = duration;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.isMicMuted$.subscribe(muted => {\n        this.isMicMuted = muted;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.isUserSpeaking$.subscribe(speaking => {\n        this.isUserSpeaking = speaking;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.audioLevels$.subscribe(levels => {\n        this.audioLevels = levels;\n      })\n    );\n\n    this.voiceAgentService.resetToIdle();\n    await this.startCall();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((sub) => sub.unsubscribe());\n    void this.disconnect().finally(() => {\n      this.warmMicStream?.getTracks().forEach((t) => t.stop());\n      this.warmMicStream = null;\n    });\n  }\n\n  isConnecting: boolean = false;\n\n  /** Ensures a live mic stream for this call (re-acquire after Daily stops tracks on hang-up). */\n  private async ensureMicForCall(): Promise<MediaStream | undefined> {\n    if (!isPlatformBrowser(this.platformId)) return undefined;\n    if (this.warmMicStream?.getAudioTracks().some((t) => t.readyState === 'live')) {\n      return this.warmMicStream;\n    }\n    try {\n      this.warmMicStream = await navigator.mediaDevices.getUserMedia({ audio: true });\n      return this.warmMicStream;\n    } catch {\n      return undefined;\n    }\n  }\n\n  async startCall(): Promise<void> {\n    if (this.isConnecting || (this.callState !== 'idle' && this.callState !== 'ended')) return;\n    this.isConnecting = true;\n    try {\n      const mic = await this.ensureMicForCall();\n      await this.voiceAgentService.connect(\n        this.apiUrl,\n        this.token,\n        this.botId,\n        this.conversationId,\n        this.apiKey,\n        this.eventToken,\n        this.eventId,\n        this.eventUrl,\n        this.domainAuthority,\n        this.usersApiUrl || undefined,\n        mic,\n      );\n    } catch (error) {\n      console.error('Failed to connect voice agent:', error);\n    } finally {\n      this.isConnecting = false;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    await this.voiceAgentService.disconnect();\n  }\n\n  toggleMic(): void {\n    this.voiceAgentService.toggleMic();\n  }\n\n  /** Map audio level (0–100) to waveform bar height in px (3–20). */\n  getWaveformHeight(level: number): number {\n    const n = Math.min(100, Math.max(0, level ?? 0));\n    return 3 + (n / 100) * 17;\n  }\n\n  /** Status label for active call. */\n  get statusLabel(): string {\n    if (this.callState === 'connecting') return this.statusText || 'Connecting...';\n    if (this.callState === 'talking') return 'Talking...';\n    if (this.callState === 'listening') return 'Listening';\n    if (this.callState === 'connected') {\n      return this.hasLeftConnectedOnce ? 'Talking...' : 'Connected';\n    }\n    return this.statusText || '';\n  }\n\n  /** Call Again: reset to idle then start a new call. */\n  callAgain(): void {\n    this.voiceAgentService.resetToIdle();\n    void this.startCall();\n  }\n\n  /** Back to Chat: close modal and disconnect. */\n  backToChat(): void {\n    this.endCall();\n  }\n\n  /** End call but keep modal open to show Call Ended + Call Again / Back to Chat. */\n  hangUp(): void {\n    this.disconnect();\n  }\n\n  endCall(): void {\n    this.disconnect();\n    this.onCloseCallback?.();\n    this.close.emit();\n  }\n}\n"]}
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent-modal.component.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAqB,MAAM,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAEpG,OAAO,EAAE,iBAAiB,EAAa,MAAM,oCAAoC,CAAC;AAClF,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,kBAAkB,EAAE,0BAA0B,EAAoB,MAAM,0BAA0B,CAAC;AAO5G,MAAM,OAAO,wBAAwB;IAmBnC,YACS,iBAAoC,EACpC,aAAmC,EAClC,QAAkB;QAFnB,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,kBAAa,GAAb,aAAa,CAAsB;QAClC,aAAQ,GAAR,QAAQ,CAAU;QArBlB,UAAK,GAAG,IAAI,YAAY,EAAQ,CAAC;QAKlC,WAAM,GAAW,EAAE,CAAC;QACpB,eAAU,GAAW,EAAE,CAAC;QACxB,YAAO,GAAW,EAAE,CAAC;QACrB,aAAQ,GAAW,EAAE,CAAC;QACtB,oBAAe,GAAW,WAAW,CAAC;QACtC,cAAS,GAAW,cAAc,CAAC;QACnC,cAAS,GAAW,qBAAqB,CAAC;QAE1C,gBAAW,GAAW,EAAE,CAAC;QAE1B,mBAAc,GAA4B,IAAI,CAAC;QAC/C,oBAAe,GAAwB,IAAI,CAAC;QAQpD,0CAA0C;QACjC,qBAAgB,GAAG,uGAAuG,CAAC;QAEpI,cAAS,GAAc,MAAM,CAAC;QAC9B,eAAU,GAAW,EAAE,CAAC;QACxB,aAAQ,GAAW,OAAO,CAAC;QAC3B,eAAU,GAAY,KAAK,CAAC;QAC5B,mBAAc,GAAY,KAAK,CAAC;QAChC,gBAAW,GAAa,EAAE,CAAC;QAC3B,eAAU,GAAY,KAAK,CAAC;QAE5B,0EAA0E;QAClE,yBAAoB,GAAY,KAAK,CAAC;QAEtC,kBAAa,GAAmB,EAAE,CAAC;QA4E3C,iBAAY,GAAY,KAAK,CAAC;IA5F3B,CAAC;IAkBJ,QAAQ;;QACN,2DAA2D;QAC3D,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC;QAClE,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,0BAA0B,EAAE,IAAI,CAAC,CAAC;QAC3E,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC;YACzC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC;YACvC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC;YACvC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC;YACzD,IAAI,CAAC,MAAM,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,MAAM,mCAAI,EAAE,CAAC;YAC/C,IAAI,CAAC,UAAU,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,UAAU,mCAAI,EAAE,CAAC;YACvD,IAAI,CAAC,OAAO,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,OAAO,mCAAI,EAAE,CAAC;YACjD,IAAI,CAAC,QAAQ,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,QAAQ,mCAAI,EAAE,CAAC;YACnD,IAAI,CAAC,eAAe,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,eAAe,mCAAI,WAAW,CAAC;YAC1E,IAAI,CAAC,SAAS,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,SAAS,mCAAI,IAAI,CAAC,SAAS,CAAC;YACjE,IAAI,CAAC,SAAS,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,SAAS,mCAAI,IAAI,CAAC,SAAS,CAAC;YACjE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC;YACnD,IAAI,CAAC,WAAW,GAAG,MAAA,IAAI,CAAC,cAAc,CAAC,WAAW,mCAAI,IAAI,CAAC,WAAW,CAAC;SACxE;QAED,2BAA2B;QAC3B,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YAClD,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACvB,IAAI,CAAC,UAAU,GAAG,KAAK,KAAK,SAAS,CAAC;YACtC,IAAI,KAAK,KAAK,WAAW,IAAI,KAAK,KAAK,SAAS,EAAE;gBAChD,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;aAClC;YACD,IAAI,KAAK,KAAK,MAAM,IAAI,KAAK,KAAK,OAAO,EAAE;gBACzC,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC;aACnC;QACH,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;YAClD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACzB,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE;YACpD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QAC3B,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACnD,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QAC1B,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE;YAC1D,IAAI,CAAC,cAAc,GAAG,QAAQ,CAAC;QACjC,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE;YACrD,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC;QAC5B,CAAC,CAAC,CACH,CAAC;QAEF,iEAAiE;QACjE,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;QACrC,KAAK,IAAI,CAAC,SAAS,EAAE,CAAC;IACxB,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC;QACrD,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAIK,SAAS;;YACb,IAAI,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,SAAS,KAAK,MAAM,IAAI,IAAI,CAAC,SAAS,KAAK,OAAO,CAAC;gBAAE,OAAO;YAC3F,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;YACzB,IAAI;gBACF,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,IAAI,CAAC,MAAM,EACX,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,MAAM,EACX,IAAI,CAAC,UAAU,EACf,IAAI,CAAC,OAAO,EACZ,IAAI,CAAC,QAAQ,EACb,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,WAAW,IAAI,SAAS,CAC9B,CAAC;aACH;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,KAAK,CAAC,gCAAgC,EAAE,KAAK,CAAC,CAAC;aACxD;oBAAS;gBACR,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;aAC3B;QACH,CAAC;KAAA;IAEK,UAAU;;YACd,MAAM,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,CAAC;QAC5C,CAAC;KAAA;IAED,SAAS;QACP,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,CAAC;IACrC,CAAC;IAED,mEAAmE;IACnE,iBAAiB,CAAC,KAAa;QAC7B,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,aAAL,KAAK,cAAL,KAAK,GAAI,CAAC,CAAC,CAAC,CAAC;QACjD,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;IAC5B,CAAC;IAED,oCAAoC;IACpC,IAAI,WAAW;QACb,IAAI,IAAI,CAAC,SAAS,KAAK,YAAY;YAAE,OAAO,IAAI,CAAC,UAAU,IAAI,eAAe,CAAC;QAC/E,IAAI,IAAI,CAAC,SAAS,KAAK,SAAS;YAAE,OAAO,YAAY,CAAC;QACtD,IAAI,IAAI,CAAC,SAAS,KAAK,WAAW;YAAE,OAAO,WAAW,CAAC;QACvD,IAAI,IAAI,CAAC,SAAS,KAAK,WAAW,EAAE;YAClC,OAAO,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,WAAW,CAAC;SAC/D;QACD,OAAO,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC;IAC/B,CAAC;IAED,uDAAuD;IACvD,SAAS;QACP,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;QACrC,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,gDAAgD;IAChD,UAAU;QACR,IAAI,CAAC,OAAO,EAAE,CAAC;IACjB,CAAC;IAED,mFAAmF;IACnF,MAAM;QACJ,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAED,OAAO;;QACL,IAAI,CAAC,UAAU,EAAE,CAAC;QAClB,MAAA,IAAI,CAAC,eAAe,+CAApB,IAAI,CAAoB,CAAC;QACzB,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;IACpB,CAAC;;;YA9LF,SAAS,SAAC;gBACT,QAAQ,EAAE,2BAA2B;gBACrC,yuUAAiD;;aAElD;;;YARQ,iBAAiB;YACjB,oBAAoB;YAHuC,QAAQ;;;oBAYzE,MAAM;qBACN,KAAK;oBACL,KAAK;oBACL,KAAK;6BACL,KAAK;qBACL,KAAK;yBACL,KAAK;sBACL,KAAK;uBACL,KAAK;8BACL,KAAK;wBACL,KAAK;wBACL,KAAK;0BACL,KAAK;0BACL,KAAK","sourcesContent":["import { Component, Input, OnInit, OnDestroy, Output, EventEmitter, Injector } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { VoiceAgentService, CallState } from '../../services/voice-agent.service';\nimport { AudioAnalyzerService } from '../../services/audio-analyzer.service';\nimport { VOICE_MODAL_CONFIG, VOICE_MODAL_CLOSE_CALLBACK, VoiceModalConfig } from '../../voice-modal-tokens';\n\n@Component({\n  selector: 'hivegpt-voice-agent-modal',\n  templateUrl: './voice-agent-modal.component.html',\n  styleUrls: ['./voice-agent-modal.component.scss']\n})\nexport class VoiceAgentModalComponent implements OnInit, OnDestroy {\n  @Output() close = new EventEmitter<void>();\n  @Input() apiUrl!: string;\n  @Input() token!: string;\n  @Input() botId!: string;\n  @Input() conversationId!: string;\n  @Input() apiKey: string = '';\n  @Input() eventToken: string = '';\n  @Input() eventId: string = '';\n  @Input() eventUrl: string = '';\n  @Input() domainAuthority: string = 'prod-lite';\n  @Input() agentName: string = 'AI Assistant';\n  @Input() agentRole: string = 'AI Agent Specialist';\n  @Input() agentAvatar?: string;\n  @Input() usersApiUrl: string = '';\n\n  private injectedConfig: VoiceModalConfig | null = null;\n  private onCloseCallback: (() => void) | null = null;\n\n  constructor(\n    public voiceAgentService: VoiceAgentService,\n    public audioAnalyzer: AudioAnalyzerService,\n    private injector: Injector\n  ) {}\n\n  /** Hardcoded voice agent avatar (Nia). */\n  readonly displayAvatarUrl = 'https://www.jotform.com/uploads/mehmetkarakasli/form_files/1564593667676a8e85f23758.86945537_icon.png';\n\n  callState: CallState = 'idle';\n  statusText: string = '';\n  duration: string = '00:00';\n  isMicMuted: boolean = false;\n  isUserSpeaking: boolean = false;\n  audioLevels: number[] = [];\n  isSpeaking: boolean = false;\n\n  /** Track whether call has transitioned out of initial connected state. */\n  private hasLeftConnectedOnce: boolean = false;\n\n  private subscriptions: Subscription[] = [];\n\n  ngOnInit(): void {\n    // When opened via Overlay, config is provided by injection\n    this.injectedConfig = this.injector.get(VOICE_MODAL_CONFIG, null);\n    this.onCloseCallback = this.injector.get(VOICE_MODAL_CLOSE_CALLBACK, null);\n    if (this.injectedConfig) {\n      this.apiUrl = this.injectedConfig.apiUrl;\n      this.token = this.injectedConfig.token;\n      this.botId = this.injectedConfig.botId;\n      this.conversationId = this.injectedConfig.conversationId;\n      this.apiKey = this.injectedConfig.apiKey ?? '';\n      this.eventToken = this.injectedConfig.eventToken ?? '';\n      this.eventId = this.injectedConfig.eventId ?? '';\n      this.eventUrl = this.injectedConfig.eventUrl ?? '';\n      this.domainAuthority = this.injectedConfig.domainAuthority ?? 'prod-lite';\n      this.agentName = this.injectedConfig.agentName ?? this.agentName;\n      this.agentRole = this.injectedConfig.agentRole ?? this.agentRole;\n      this.agentAvatar = this.injectedConfig.agentAvatar;\n      this.usersApiUrl = this.injectedConfig.usersApiUrl ?? this.usersApiUrl;\n    }\n\n    // Subscribe to observables\n    this.subscriptions.push(\n      this.voiceAgentService.callState$.subscribe(state => {\n        this.callState = state;\n        this.isSpeaking = state === 'talking';\n        if (state === 'listening' || state === 'talking') {\n          this.hasLeftConnectedOnce = true;\n        }\n        if (state === 'idle' || state === 'ended') {\n          this.hasLeftConnectedOnce = false;\n        }\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.statusText$.subscribe(text => {\n        this.statusText = text;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.duration$.subscribe(duration => {\n        this.duration = duration;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.isMicMuted$.subscribe(muted => {\n        this.isMicMuted = muted;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.isUserSpeaking$.subscribe(speaking => {\n        this.isUserSpeaking = speaking;\n      })\n    );\n\n    this.subscriptions.push(\n      this.voiceAgentService.audioLevels$.subscribe(levels => {\n        this.audioLevels = levels;\n      })\n    );\n\n    // Modal opens in idle state, then immediately starts connecting.\n    this.voiceAgentService.resetToIdle();\n    void this.startCall();\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n    this.disconnect();\n  }\n\n  isConnecting: boolean = false;\n\n  async startCall(): Promise<void> {\n    if (this.isConnecting || (this.callState !== 'idle' && this.callState !== 'ended')) return;\n    this.isConnecting = true;\n    try {\n      await this.voiceAgentService.connect(\n        this.apiUrl,\n        this.token,\n        this.botId,\n        this.conversationId,\n        this.apiKey,\n        this.eventToken,\n        this.eventId,\n        this.eventUrl,\n        this.domainAuthority,\n        this.usersApiUrl || undefined,\n      );\n    } catch (error) {\n      console.error('Failed to connect voice agent:', error);\n    } finally {\n      this.isConnecting = false;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    await this.voiceAgentService.disconnect();\n  }\n\n  toggleMic(): void {\n    this.voiceAgentService.toggleMic();\n  }\n\n  /** Map audio level (0–100) to waveform bar height in px (3–20). */\n  getWaveformHeight(level: number): number {\n    const n = Math.min(100, Math.max(0, level ?? 0));\n    return 3 + (n / 100) * 17;\n  }\n\n  /** Status label for active call. */\n  get statusLabel(): string {\n    if (this.callState === 'connecting') return this.statusText || 'Connecting...';\n    if (this.callState === 'talking') return 'Talking...';\n    if (this.callState === 'listening') return 'Listening';\n    if (this.callState === 'connected') {\n      return this.hasLeftConnectedOnce ? 'Talking...' : 'Connected';\n    }\n    return this.statusText || '';\n  }\n\n  /** Call Again: reset to idle then start a new call. */\n  callAgain(): void {\n    this.voiceAgentService.resetToIdle();\n    this.startCall();\n  }\n\n  /** Back to Chat: close modal and disconnect. */\n  backToChat(): void {\n    this.endCall();\n  }\n\n  /** End call but keep modal open to show Call Ended + Call Again / Back to Chat. */\n  hangUp(): void {\n    this.disconnect();\n  }\n\n  endCall(): void {\n    this.disconnect();\n    this.onCloseCallback?.();\n    this.close.emit();\n  }\n}\n"]}
@@ -24,8 +24,7 @@ export class DailyVoiceClientService {
24
24
  this.remoteAudioElement = null;
25
25
  /** AnalyserNode-based remote audio monitor for instant bot speaking detection. */
26
26
  this.remoteAudioContext = null;
27
- /** Poll interval id (~100ms); named historically when RAF was used. */
28
- this.remoteSpeakingPollId = null;
27
+ this.remoteSpeakingRAF = null;
29
28
  this.speakingSubject = new BehaviorSubject(false);
30
29
  this.userSpeakingSubject = new BehaviorSubject(false);
31
30
  this.micMutedSubject = new BehaviorSubject(false);
@@ -43,18 +42,15 @@ export class DailyVoiceClientService {
43
42
  * Connect to Daily room. Acquires mic first for waveform, then joins with audio.
44
43
  * @param roomUrl Daily room URL (from room_created)
45
44
  * @param token Optional meeting token
46
- * @param existingStream Optional pre-acquired mic (avoids a second getUserMedia / extra prompts on some browsers)
47
45
  */
48
- connect(roomUrl, token, existingStream) {
46
+ connect(roomUrl, token) {
49
47
  return __awaiter(this, void 0, void 0, function* () {
50
48
  if (this.callObject) {
51
49
  yield this.disconnect();
52
50
  }
53
51
  try {
54
- const hasLiveTrack = !!(existingStream === null || existingStream === void 0 ? void 0 : existingStream.getAudioTracks().some((t) => t.readyState === 'live'));
55
- const stream = hasLiveTrack
56
- ? existingStream
57
- : yield navigator.mediaDevices.getUserMedia({ audio: true });
52
+ // Get mic stream for both Daily and waveform (single capture)
53
+ const stream = yield navigator.mediaDevices.getUserMedia({ audio: true });
58
54
  const audioTrack = stream.getAudioTracks()[0];
59
55
  if (!audioTrack) {
60
56
  stream.getTracks().forEach((t) => t.stop());
@@ -186,7 +182,7 @@ export class DailyVoiceClientService {
186
182
  }
187
183
  /**
188
184
  * Monitor remote audio track energy via AnalyserNode.
189
- * Polls at ~10Hz; sufficient for speaking detection vs ~60fps RAF.
185
+ * Polls at ~60fps and flips speakingSubject based on actual audio energy.
190
186
  */
191
187
  monitorRemoteAudio(track) {
192
188
  this.stopRemoteAudioMonitor();
@@ -200,17 +196,11 @@ export class DailyVoiceClientService {
200
196
  const dataArray = new Uint8Array(analyser.frequencyBinCount);
201
197
  const THRESHOLD = 5;
202
198
  const SILENCE_MS = 1500;
203
- const POLL_MS = 100;
204
199
  let lastSoundTime = 0;
205
200
  let isSpeaking = false;
206
- this.remoteSpeakingPollId = setInterval(() => {
207
- if (!this.remoteAudioContext) {
208
- if (this.remoteSpeakingPollId) {
209
- clearInterval(this.remoteSpeakingPollId);
210
- this.remoteSpeakingPollId = null;
211
- }
201
+ const poll = () => {
202
+ if (!this.remoteAudioContext)
212
203
  return;
213
- }
214
204
  analyser.getByteFrequencyData(dataArray);
215
205
  let sum = 0;
216
206
  for (let i = 0; i < dataArray.length; i++) {
@@ -234,16 +224,18 @@ export class DailyVoiceClientService {
234
224
  console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);
235
225
  this.ngZone.run(() => this.speakingSubject.next(false));
236
226
  }
237
- }, POLL_MS);
227
+ this.remoteSpeakingRAF = requestAnimationFrame(poll);
228
+ };
229
+ this.remoteSpeakingRAF = requestAnimationFrame(poll);
238
230
  }
239
231
  catch (err) {
240
232
  console.warn('DailyVoiceClient: failed to create remote audio monitor', err);
241
233
  }
242
234
  }
243
235
  stopRemoteAudioMonitor() {
244
- if (this.remoteSpeakingPollId !== null) {
245
- clearInterval(this.remoteSpeakingPollId);
246
- this.remoteSpeakingPollId = null;
236
+ if (this.remoteSpeakingRAF) {
237
+ cancelAnimationFrame(this.remoteSpeakingRAF);
238
+ this.remoteSpeakingRAF = null;
247
239
  }
248
240
  if (this.remoteAudioContext) {
249
241
  this.remoteAudioContext.close().catch(() => { });
@@ -310,4 +302,4 @@ DailyVoiceClientService.decorators = [
310
302
  DailyVoiceClientService.ctorParameters = () => [
311
303
  { type: NgZone }
312
304
  ];
313
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"daily-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/daily-voice-client.service.ts"],"names":[],"mappings":"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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  /** Poll interval id (~100ms); named historically when RAF was used. */\n  private remoteSpeakingPollId: ReturnType<typeof setInterval> | null = null;\n\n  private speakingSubject = new BehaviorSubject<boolean>(false);\n  private userSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private micMutedSubject = new BehaviorSubject<boolean>(false);\n  private localStreamSubject = new BehaviorSubject<MediaStream | null>(null);\n\n  /** True when bot (remote participant) is the active speaker. */\n  speaking$: Observable<boolean> = this.speakingSubject.asObservable();\n\n  /** True when user (local participant) is the active speaker. */\n  userSpeaking$: Observable<boolean> = this.userSpeakingSubject.asObservable();\n\n  /** True when mic is muted. */\n  micMuted$: Observable<boolean> = this.micMutedSubject.asObservable();\n\n  /** Emits local mic stream for waveform visualization. */\n  localStream$: Observable<MediaStream | null> =\n    this.localStreamSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  /**\n   * Connect to Daily room. Acquires mic first for waveform, then joins with audio.\n   * @param roomUrl Daily room URL (from room_created)\n   * @param token Optional meeting token\n   * @param existingStream Optional pre-acquired mic (avoids a second getUserMedia / extra prompts on some browsers)\n   */\n  async connect(\n    roomUrl: string,\n    token?: string,\n    existingStream?: MediaStream,\n  ): Promise<void> {\n    if (this.callObject) {\n      await this.disconnect();\n    }\n\n    try {\n      const hasLiveTrack =\n        !!existingStream?.getAudioTracks().some((t) => t.readyState === 'live');\n      const stream = hasLiveTrack\n        ? existingStream!\n        : await navigator.mediaDevices.getUserMedia({ audio: true });\n      const audioTrack = stream.getAudioTracks()[0];\n      if (!audioTrack) {\n        stream.getTracks().forEach((t) => t.stop());\n        throw new Error('No audio track');\n      }\n\n      this.localStream = stream;\n      this.localStreamSubject.next(stream);\n\n      // Create audio-only call object\n      // videoSource: false = no camera, audioSource = our mic track\n      const callObject = Daily.createCallObject({\n        videoSource: false,\n        audioSource: audioTrack,\n      });\n\n      this.callObject = callObject;\n\n      this.setupEventHandlers(callObject);\n\n      // Join room; Daily handles playback of remote (bot) audio automatically.\n      // Only pass token when it's a non-empty string (Daily rejects undefined/non-string).\n      const joinOptions: { url: string; token?: string } = { url: roomUrl };\n      if (typeof token === 'string' && token.trim() !== '') {\n        joinOptions.token = token;\n      }\n      await callObject.join(joinOptions);\n      console.log(`[VoiceDebug] Room connected (Daily join complete) — ${new Date().toISOString()}`);\n\n      const participants = callObject.participants();\n      if (participants?.local) {\n        this.localSessionId = participants.local.session_id;\n      }\n\n      // Initial mute state: Daily starts with audio on\n      this.micMutedSubject.next(!callObject.localAudio());\n    } catch (err) {\n      this.cleanup();\n      throw err;\n    }\n  }\n\n  private setupEventHandlers(call: DailyCall): void {\n    // active-speaker-change: used ONLY for user speaking detection.\n    // Bot speaking is detected by our own AnalyserNode (instant, no debounce).\n    call.on('active-speaker-change', (event: { activeSpeaker?: { peerId?: string } }) => {\n      this.ngZone.run(() => {\n        const peerId = event?.activeSpeaker?.peerId;\n        if (!peerId || !this.localSessionId) {\n          this.userSpeakingSubject.next(false);\n          return;\n        }\n        const isLocal = peerId === this.localSessionId;\n        this.userSpeakingSubject.next(isLocal);\n      });\n    });\n\n    // track-started / track-stopped: set up remote audio playback + AnalyserNode monitor.\n    call.on('track-started', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        const track = event?.track;\n        if (p && !p.local && type === 'audio') {\n          console.log(`[VoiceDebug] Got audio track from backend (track-started) — readyState=${track?.readyState}, muted=${track?.muted} — ${new Date().toISOString()}`);\n          const audioTrack = track ?? (p as { tracks?: { audio?: { track?: MediaStreamTrack } } }).tracks?.audio?.track;\n          if (audioTrack && typeof audioTrack === 'object') {\n            this.playRemoteTrack(audioTrack);\n            this.monitorRemoteAudio(audioTrack);\n          }\n        }\n      });\n    });\n\n    call.on('track-stopped', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        if (p && !p.local && type === 'audio') {\n          this.stopRemoteAudioMonitor();\n          this.stopRemoteAudio();\n        }\n      });\n    });\n\n    call.on('left-meeting', () => {\n      this.ngZone.run(() => this.cleanup());\n    });\n\n    call.on('error', (event?: { errorMsg?: string }) => {\n      this.ngZone.run(() => {\n        console.error('DailyVoiceClient: Daily error', event?.errorMsg ?? event);\n        this.cleanup();\n      });\n    });\n  }\n\n  /**\n   * Play remote (bot) audio track via a dedicated audio element.\n   * Required in many browsers where Daily's internal playback does not output to speakers.\n   */\n  private playRemoteTrack(track: MediaStreamTrack): void {\n    this.stopRemoteAudio();\n    try {\n      console.log(`[VoiceDebug] playRemoteTrack called — track.readyState=${track.readyState}, track.muted=${track.muted} — ${new Date().toISOString()}`);\n\n      track.onunmute = () => {\n        console.log(`[VoiceDebug] Remote audio track UNMUTED (audio data arriving) — ${new Date().toISOString()}`);\n      };\n\n      const stream = new MediaStream([track]);\n      const audio = new Audio();\n      audio.autoplay = true;\n      audio.srcObject = stream;\n      this.remoteAudioElement = audio;\n\n      audio.onplaying = () => {\n        console.log(`[VoiceDebug] Audio element PLAYING (browser started playback) — ${new Date().toISOString()}`);\n      };\n\n      let firstTimeUpdate = true;\n      audio.ontimeupdate = () => {\n        if (firstTimeUpdate) {\n          firstTimeUpdate = false;\n          console.log(`[VoiceDebug] Audio element first TIMEUPDATE (actual audio output) — ${new Date().toISOString()}`);\n        }\n      };\n\n      const p = audio.play();\n      if (p && typeof p.then === 'function') {\n        p.then(() => {\n          console.log(`[VoiceDebug] audio.play() resolved — ${new Date().toISOString()}`);\n        }).catch((err) => {\n          console.warn('DailyVoiceClient: remote audio play failed (may need user gesture)', err);\n        });\n      }\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio element', err);\n    }\n  }\n\n  /**\n   * Monitor remote audio track energy via AnalyserNode.\n   * Polls at ~10Hz; sufficient for speaking detection vs ~60fps RAF.\n   */\n  private monitorRemoteAudio(track: MediaStreamTrack): void {\n    this.stopRemoteAudioMonitor();\n    try {\n      const ctx = new AudioContext();\n      const source = ctx.createMediaStreamSource(new MediaStream([track]));\n      const analyser = ctx.createAnalyser();\n      analyser.fftSize = 256;\n      source.connect(analyser);\n      this.remoteAudioContext = ctx;\n\n      const dataArray = new Uint8Array(analyser.frequencyBinCount);\n      const THRESHOLD = 5;\n      const SILENCE_MS = 1500;\n      const POLL_MS = 100;\n      let lastSoundTime = 0;\n      let isSpeaking = false;\n\n      this.remoteSpeakingPollId = setInterval(() => {\n        if (!this.remoteAudioContext) {\n          if (this.remoteSpeakingPollId) {\n            clearInterval(this.remoteSpeakingPollId);\n            this.remoteSpeakingPollId = null;\n          }\n          return;\n        }\n        analyser.getByteFrequencyData(dataArray);\n\n        let sum = 0;\n        for (let i = 0; i < dataArray.length; i++) {\n          sum += dataArray[i];\n        }\n        const avg = sum / dataArray.length;\n\n        const now = Date.now();\n        if (avg > THRESHOLD) {\n          lastSoundTime = now;\n          if (!isSpeaking) {\n            isSpeaking = true;\n            console.log(`[VoiceDebug] Bot audio energy detected (speaking=true) — avg=${avg.toFixed(1)} — ${new Date().toISOString()}`);\n            this.ngZone.run(() => {\n              this.userSpeakingSubject.next(false);\n              this.speakingSubject.next(true);\n            });\n          }\n        } else if (isSpeaking && now - lastSoundTime > SILENCE_MS) {\n          isSpeaking = false;\n          console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);\n          this.ngZone.run(() => this.speakingSubject.next(false));\n        }\n      }, POLL_MS);\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio monitor', err);\n    }\n  }\n\n  private stopRemoteAudioMonitor(): void {\n    if (this.remoteSpeakingPollId !== null) {\n      clearInterval(this.remoteSpeakingPollId);\n      this.remoteSpeakingPollId = null;\n    }\n    if (this.remoteAudioContext) {\n      this.remoteAudioContext.close().catch(() => {});\n      this.remoteAudioContext = null;\n    }\n  }\n\n  private stopRemoteAudio(): void {\n    if (this.remoteAudioElement) {\n      try {\n        this.remoteAudioElement.pause();\n        this.remoteAudioElement.srcObject = null;\n      } catch (_) {}\n      this.remoteAudioElement = null;\n    }\n  }\n\n  /** Set mic muted state. */\n  setMuted(muted: boolean): void {\n    if (!this.callObject) return;\n    this.callObject.setLocalAudio(!muted);\n    this.micMutedSubject.next(muted);\n  }\n\n  /** Disconnect and cleanup. */\n  async disconnect(): Promise<void> {\n    if (!this.callObject) {\n      this.cleanup();\n      return;\n    }\n    try {\n      await this.callObject.leave();\n    } catch (e) {\n      // ignore\n    }\n    this.cleanup();\n  }\n\n  private cleanup(): void {\n    this.stopRemoteAudioMonitor();\n    this.stopRemoteAudio();\n    if (this.callObject) {\n      this.callObject.destroy().catch(() => {});\n      this.callObject = null;\n    }\n    if (this.localStream) {\n      this.localStream.getTracks().forEach((t) => t.stop());\n      this.localStream = null;\n    }\n    this.localSessionId = null;\n    this.speakingSubject.next(false);\n    this.userSpeakingSubject.next(false);\n    this.localStreamSubject.next(null);\n    // Keep last micMuted state; will reset on next connect\n  }\n}\n"]}
305
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"daily-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/daily-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport Daily from '@daily-co/daily-js';\nimport type { DailyCall, DailyParticipant } from '@daily-co/daily-js';\n\n/**\n * Daily.js WebRTC client for voice agent audio.\n * Responsibilities:\n * - Create and manage Daily CallObject\n * - Join Daily room using room_url\n * - Handle mic capture + speaker playback\n * - Bot speaking detection via AnalyserNode on remote track (instant)\n * - User speaking detection via active-speaker-change\n * - Expose speaking$ (bot speaking), userSpeaking$ (user speaking), micMuted$\n * - Expose localStream$ for waveform visualization (AudioAnalyzerService)\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class DailyVoiceClientService {\n  private callObject: DailyCall | null = null;\n  private localStream: MediaStream | null = null;\n  private localSessionId: string | null = null;\n  /** Explicit playback of remote (bot) audio; required in some browsers. */\n  private remoteAudioElement: HTMLAudioElement | null = null;\n\n  /** AnalyserNode-based remote audio monitor for instant bot speaking detection. */\n  private remoteAudioContext: AudioContext | null = null;\n  private remoteSpeakingRAF: number | null = null;\n\n  private speakingSubject = new BehaviorSubject<boolean>(false);\n  private userSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private micMutedSubject = new BehaviorSubject<boolean>(false);\n  private localStreamSubject = new BehaviorSubject<MediaStream | null>(null);\n\n  /** True when bot (remote participant) is the active speaker. */\n  speaking$: Observable<boolean> = this.speakingSubject.asObservable();\n\n  /** True when user (local participant) is the active speaker. */\n  userSpeaking$: Observable<boolean> = this.userSpeakingSubject.asObservable();\n\n  /** True when mic is muted. */\n  micMuted$: Observable<boolean> = this.micMutedSubject.asObservable();\n\n  /** Emits local mic stream for waveform visualization. */\n  localStream$: Observable<MediaStream | null> =\n    this.localStreamSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  /**\n   * Connect to Daily room. Acquires mic first for waveform, then joins with audio.\n   * @param roomUrl Daily room URL (from room_created)\n   * @param token Optional meeting token\n   */\n  async connect(roomUrl: string, token?: string): Promise<void> {\n    if (this.callObject) {\n      await this.disconnect();\n    }\n\n    try {\n      // Get mic stream for both Daily and waveform (single capture)\n      const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n      const audioTrack = stream.getAudioTracks()[0];\n      if (!audioTrack) {\n        stream.getTracks().forEach((t) => t.stop());\n        throw new Error('No audio track');\n      }\n\n      this.localStream = stream;\n      this.localStreamSubject.next(stream);\n\n      // Create audio-only call object\n      // videoSource: false = no camera, audioSource = our mic track\n      const callObject = Daily.createCallObject({\n        videoSource: false,\n        audioSource: audioTrack,\n      });\n\n      this.callObject = callObject;\n\n      this.setupEventHandlers(callObject);\n\n      // Join room; Daily handles playback of remote (bot) audio automatically.\n      // Only pass token when it's a non-empty string (Daily rejects undefined/non-string).\n      const joinOptions: { url: string; token?: string } = { url: roomUrl };\n      if (typeof token === 'string' && token.trim() !== '') {\n        joinOptions.token = token;\n      }\n      await callObject.join(joinOptions);\n      console.log(`[VoiceDebug] Room connected (Daily join complete) — ${new Date().toISOString()}`);\n\n      const participants = callObject.participants();\n      if (participants?.local) {\n        this.localSessionId = participants.local.session_id;\n      }\n\n      // Initial mute state: Daily starts with audio on\n      this.micMutedSubject.next(!callObject.localAudio());\n    } catch (err) {\n      this.cleanup();\n      throw err;\n    }\n  }\n\n  private setupEventHandlers(call: DailyCall): void {\n    // active-speaker-change: used ONLY for user speaking detection.\n    // Bot speaking is detected by our own AnalyserNode (instant, no debounce).\n    call.on('active-speaker-change', (event: { activeSpeaker?: { peerId?: string } }) => {\n      this.ngZone.run(() => {\n        const peerId = event?.activeSpeaker?.peerId;\n        if (!peerId || !this.localSessionId) {\n          this.userSpeakingSubject.next(false);\n          return;\n        }\n        const isLocal = peerId === this.localSessionId;\n        this.userSpeakingSubject.next(isLocal);\n      });\n    });\n\n    // track-started / track-stopped: set up remote audio playback + AnalyserNode monitor.\n    call.on('track-started', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        const track = event?.track;\n        if (p && !p.local && type === 'audio') {\n          console.log(`[VoiceDebug] Got audio track from backend (track-started) — readyState=${track?.readyState}, muted=${track?.muted} — ${new Date().toISOString()}`);\n          const audioTrack = track ?? (p as { tracks?: { audio?: { track?: MediaStreamTrack } } }).tracks?.audio?.track;\n          if (audioTrack && typeof audioTrack === 'object') {\n            this.playRemoteTrack(audioTrack);\n            this.monitorRemoteAudio(audioTrack);\n          }\n        }\n      });\n    });\n\n    call.on('track-stopped', (event: { participant?: DailyParticipant | null; type?: string; track?: MediaStreamTrack }) => {\n      this.ngZone.run(() => {\n        const p = event?.participant;\n        const type = event?.type ?? event?.track?.kind;\n        if (p && !p.local && type === 'audio') {\n          this.stopRemoteAudioMonitor();\n          this.stopRemoteAudio();\n        }\n      });\n    });\n\n    call.on('left-meeting', () => {\n      this.ngZone.run(() => this.cleanup());\n    });\n\n    call.on('error', (event?: { errorMsg?: string }) => {\n      this.ngZone.run(() => {\n        console.error('DailyVoiceClient: Daily error', event?.errorMsg ?? event);\n        this.cleanup();\n      });\n    });\n  }\n\n  /**\n   * Play remote (bot) audio track via a dedicated audio element.\n   * Required in many browsers where Daily's internal playback does not output to speakers.\n   */\n  private playRemoteTrack(track: MediaStreamTrack): void {\n    this.stopRemoteAudio();\n    try {\n      console.log(`[VoiceDebug] playRemoteTrack called — track.readyState=${track.readyState}, track.muted=${track.muted} — ${new Date().toISOString()}`);\n\n      track.onunmute = () => {\n        console.log(`[VoiceDebug] Remote audio track UNMUTED (audio data arriving) — ${new Date().toISOString()}`);\n      };\n\n      const stream = new MediaStream([track]);\n      const audio = new Audio();\n      audio.autoplay = true;\n      audio.srcObject = stream;\n      this.remoteAudioElement = audio;\n\n      audio.onplaying = () => {\n        console.log(`[VoiceDebug] Audio element PLAYING (browser started playback) — ${new Date().toISOString()}`);\n      };\n\n      let firstTimeUpdate = true;\n      audio.ontimeupdate = () => {\n        if (firstTimeUpdate) {\n          firstTimeUpdate = false;\n          console.log(`[VoiceDebug] Audio element first TIMEUPDATE (actual audio output) — ${new Date().toISOString()}`);\n        }\n      };\n\n      const p = audio.play();\n      if (p && typeof p.then === 'function') {\n        p.then(() => {\n          console.log(`[VoiceDebug] audio.play() resolved — ${new Date().toISOString()}`);\n        }).catch((err) => {\n          console.warn('DailyVoiceClient: remote audio play failed (may need user gesture)', err);\n        });\n      }\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio element', err);\n    }\n  }\n\n  /**\n   * Monitor remote audio track energy via AnalyserNode.\n   * Polls at ~60fps and flips speakingSubject based on actual audio energy.\n   */\n  private monitorRemoteAudio(track: MediaStreamTrack): void {\n    this.stopRemoteAudioMonitor();\n    try {\n      const ctx = new AudioContext();\n      const source = ctx.createMediaStreamSource(new MediaStream([track]));\n      const analyser = ctx.createAnalyser();\n      analyser.fftSize = 256;\n      source.connect(analyser);\n      this.remoteAudioContext = ctx;\n\n      const dataArray = new Uint8Array(analyser.frequencyBinCount);\n      const THRESHOLD = 5;\n      const SILENCE_MS = 1500;\n      let lastSoundTime = 0;\n      let isSpeaking = false;\n\n      const poll = () => {\n        if (!this.remoteAudioContext) return;\n        analyser.getByteFrequencyData(dataArray);\n\n        let sum = 0;\n        for (let i = 0; i < dataArray.length; i++) {\n          sum += dataArray[i];\n        }\n        const avg = sum / dataArray.length;\n\n        const now = Date.now();\n        if (avg > THRESHOLD) {\n          lastSoundTime = now;\n          if (!isSpeaking) {\n            isSpeaking = true;\n            console.log(`[VoiceDebug] Bot audio energy detected (speaking=true) — avg=${avg.toFixed(1)} — ${new Date().toISOString()}`);\n            this.ngZone.run(() => {\n              this.userSpeakingSubject.next(false);\n              this.speakingSubject.next(true);\n            });\n          }\n        } else if (isSpeaking && now - lastSoundTime > SILENCE_MS) {\n          isSpeaking = false;\n          console.log(`[VoiceDebug] Bot audio silence detected (speaking=false) — ${new Date().toISOString()}`);\n          this.ngZone.run(() => this.speakingSubject.next(false));\n        }\n\n        this.remoteSpeakingRAF = requestAnimationFrame(poll);\n      };\n\n      this.remoteSpeakingRAF = requestAnimationFrame(poll);\n    } catch (err) {\n      console.warn('DailyVoiceClient: failed to create remote audio monitor', err);\n    }\n  }\n\n  private stopRemoteAudioMonitor(): void {\n    if (this.remoteSpeakingRAF) {\n      cancelAnimationFrame(this.remoteSpeakingRAF);\n      this.remoteSpeakingRAF = null;\n    }\n    if (this.remoteAudioContext) {\n      this.remoteAudioContext.close().catch(() => {});\n      this.remoteAudioContext = null;\n    }\n  }\n\n  private stopRemoteAudio(): void {\n    if (this.remoteAudioElement) {\n      try {\n        this.remoteAudioElement.pause();\n        this.remoteAudioElement.srcObject = null;\n      } catch (_) {}\n      this.remoteAudioElement = null;\n    }\n  }\n\n  /** Set mic muted state. */\n  setMuted(muted: boolean): void {\n    if (!this.callObject) return;\n    this.callObject.setLocalAudio(!muted);\n    this.micMutedSubject.next(muted);\n  }\n\n  /** Disconnect and cleanup. */\n  async disconnect(): Promise<void> {\n    if (!this.callObject) {\n      this.cleanup();\n      return;\n    }\n    try {\n      await this.callObject.leave();\n    } catch (e) {\n      // ignore\n    }\n    this.cleanup();\n  }\n\n  private cleanup(): void {\n    this.stopRemoteAudioMonitor();\n    this.stopRemoteAudio();\n    if (this.callObject) {\n      this.callObject.destroy().catch(() => {});\n      this.callObject = null;\n    }\n    if (this.localStream) {\n      this.localStream.getTracks().forEach((t) => t.stop());\n      this.localStream = null;\n    }\n    this.localSessionId = null;\n    this.speakingSubject.next(false);\n    this.userSpeakingSubject.next(false);\n    this.localStreamSubject.next(null);\n    // Keep last micMuted state; will reset on next connect\n  }\n}\n"]}