@hivegpt/hiveai-angular 0.0.590 → 0.0.591

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.
@@ -74,9 +74,8 @@ export class VoiceAgentModalComponent {
74
74
  this.subscriptions.push(this.voiceAgentService.audioLevels$.subscribe(levels => {
75
75
  this.audioLevels = levels;
76
76
  }));
77
- // Modal opens in idle state, then immediately starts connecting.
78
- this.voiceAgentService.resetToIdle();
79
- void this.startCall();
77
+ // Always start from a fresh voice session to avoid stale/disconnected state flashes.
78
+ void this.openFreshCallSession();
80
79
  }
81
80
  ngOnDestroy() {
82
81
  this.subscriptions.forEach(sub => sub.unsubscribe());
@@ -103,6 +102,12 @@ export class VoiceAgentModalComponent {
103
102
  yield this.voiceAgentService.disconnect();
104
103
  });
105
104
  }
105
+ openFreshCallSession() {
106
+ return __awaiter(this, void 0, void 0, function* () {
107
+ yield this.voiceAgentService.prepareFreshSession();
108
+ yield this.startCall();
109
+ });
110
+ }
106
111
  toggleMic() {
107
112
  this.voiceAgentService.toggleMic();
108
113
  }
@@ -135,8 +140,9 @@ export class VoiceAgentModalComponent {
135
140
  }
136
141
  /** Call Again: reset to idle then start a new call. */
137
142
  callAgain() {
138
- this.voiceAgentService.resetToIdle();
139
- this.startCall();
143
+ return __awaiter(this, void 0, void 0, function* () {
144
+ yield this.openFreshCallSession();
145
+ });
140
146
  }
141
147
  /** Back to Chat: close modal and disconnect. */
142
148
  backToChat() {
@@ -148,9 +154,11 @@ export class VoiceAgentModalComponent {
148
154
  }
149
155
  endCall() {
150
156
  var _a;
151
- this.disconnect();
152
- (_a = this.onCloseCallback) === null || _a === void 0 ? void 0 : _a.call(this);
153
- this.close.emit();
157
+ return __awaiter(this, void 0, void 0, function* () {
158
+ yield this.disconnect();
159
+ (_a = this.onCloseCallback) === null || _a === void 0 ? void 0 : _a.call(this);
160
+ this.close.emit();
161
+ });
154
162
  }
155
163
  }
156
164
  VoiceAgentModalComponent.decorators = [
@@ -181,4 +189,4 @@ VoiceAgentModalComponent.propDecorators = {
181
189
  agentAvatar: [{ type: Input }],
182
190
  usersApiUrl: [{ type: Input }]
183
191
  };
184
- //# 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 { 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\n  /** True while the bot is speaking — drives avatar pulse animation and voice visualizer. */\n  get isBotTalking(): boolean { return this.callState === 'talking'; }\n\n  /** True while the user is actively speaking — drives waveform active color. */\n  get isUserActive(): boolean { return this.callState === 'listening' && this.isUserSpeaking && !this.isMicMuted; }\n\n  /** True during the brief processing pause between user speech and bot response. */\n  get isProcessing(): boolean { return this.callState === 'connected' && this.statusText === 'Processing...'; }\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    this.subscriptions.push(\n      this.voiceAgentService.callState$.subscribe(state => {\n        this.callState = state;\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  /**\n   * Map audio level (0–100) to waveform bar height in px with a gaussian\n   * bell-curve envelope so centre bars are tallest and edge bars appear\n   * as tiny dots — matching the audio-waveform reference design.\n   */\n  getWaveformHeight(level: number, index: number): number {\n    const n = Math.min(100, Math.max(0, level ?? 0));\n    const total = this.audioLevels.length || 60;\n    const center = (total - 1) / 2;\n    const sigma = total / 5;\n    const envelope = Math.exp(-Math.pow(index - center, 2) / (2 * sigma * sigma));\n    // Minimum height scales with envelope so edges show as dots even in silence\n    const minHeight = 2 + envelope * 3;\n    const maxHeight = 4 + envelope * 38;\n    return minHeight + (n / 100) * (maxHeight - minHeight);\n  }\n\n  /** Status label for active call — driven by callState + service statusText. */\n  get statusLabel(): string {\n    switch (this.callState) {\n      case 'connecting': return 'Connecting...';\n      case 'talking':    return 'Talking...';\n      case 'listening':  return 'Listening...';\n      // 'connected' covers: initial connect, between turns (Listening / Processing)\n      case 'connected':  return this.statusText || 'Connected';\n      default:           return this.statusText || '';\n    }\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"]}
192
+ //# 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 { 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\n  /** True while the bot is speaking — drives avatar pulse animation and voice visualizer. */\n  get isBotTalking(): boolean { return this.callState === 'talking'; }\n\n  /** True while the user is actively speaking — drives waveform active color. */\n  get isUserActive(): boolean { return this.callState === 'listening' && this.isUserSpeaking && !this.isMicMuted; }\n\n  /** True during the brief processing pause between user speech and bot response. */\n  get isProcessing(): boolean { return this.callState === 'connected' && this.statusText === 'Processing...'; }\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    this.subscriptions.push(\n      this.voiceAgentService.callState$.subscribe(state => {\n        this.callState = state;\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    // Always start from a fresh voice session to avoid stale/disconnected state flashes.\n    void this.openFreshCallSession();\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  private async openFreshCallSession(): Promise<void> {\n    await this.voiceAgentService.prepareFreshSession();\n    await this.startCall();\n  }\n\n  toggleMic(): void {\n    this.voiceAgentService.toggleMic();\n  }\n\n  /**\n   * Map audio level (0–100) to waveform bar height in px with a gaussian\n   * bell-curve envelope so centre bars are tallest and edge bars appear\n   * as tiny dots — matching the audio-waveform reference design.\n   */\n  getWaveformHeight(level: number, index: number): number {\n    const n = Math.min(100, Math.max(0, level ?? 0));\n    const total = this.audioLevels.length || 60;\n    const center = (total - 1) / 2;\n    const sigma = total / 5;\n    const envelope = Math.exp(-Math.pow(index - center, 2) / (2 * sigma * sigma));\n    // Minimum height scales with envelope so edges show as dots even in silence\n    const minHeight = 2 + envelope * 3;\n    const maxHeight = 4 + envelope * 38;\n    return minHeight + (n / 100) * (maxHeight - minHeight);\n  }\n\n  /** Status label for active call — driven by callState + service statusText. */\n  get statusLabel(): string {\n    switch (this.callState) {\n      case 'connecting': return 'Connecting...';\n      case 'talking':    return 'Talking...';\n      case 'listening':  return 'Listening...';\n      // 'connected' covers: initial connect, between turns (Listening / Processing)\n      case 'connected':  return this.statusText || 'Connected';\n      default:           return this.statusText || '';\n    }\n  }\n\n  /** Call Again: reset to idle then start a new call. */\n  async callAgain(): Promise<void> {\n    await this.openFreshCallSession();\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  async endCall(): Promise<void> {\n    await this.disconnect();\n    this.onCloseCallback?.();\n    this.close.emit();\n  }\n}\n"]}
@@ -41,6 +41,7 @@ export class VoiceAgentService {
41
41
  this.durationInterval = null;
42
42
  this.pcClient = null;
43
43
  this.botAudioElement = null;
44
+ this.lifecycleToken = 0;
44
45
  this.subscriptions = new Subscription();
45
46
  this.destroy$ = new Subject();
46
47
  this.callState$ = this.callStateSubject.asObservable();
@@ -60,12 +61,26 @@ export class VoiceAgentService {
60
61
  }
61
62
  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */
62
63
  resetToIdle() {
63
- if (this.callStateSubject.value === 'idle')
64
- return;
65
- void this.disconnect();
66
- this.callStateSubject.next('idle');
67
- this.statusTextSubject.next('');
68
- this.durationSubject.next('0:00');
64
+ void this.prepareFreshSession();
65
+ }
66
+ /**
67
+ * Hard reset voice session so each modal open starts from a clean state.
68
+ * Closes any previous client and prevents stale callbacks from mutating UI state.
69
+ */
70
+ prepareFreshSession() {
71
+ return __awaiter(this, void 0, void 0, function* () {
72
+ this.lifecycleToken += 1;
73
+ this.stopDurationTimer();
74
+ this.callStartTime = 0;
75
+ this.audioAnalyzer.stop();
76
+ this.stopBotAudio();
77
+ this.isUserSpeakingSubject.next(false);
78
+ this.isMicMutedSubject.next(false);
79
+ this.durationSubject.next('00:00');
80
+ this.statusTextSubject.next('');
81
+ this.callStateSubject.next('idle');
82
+ yield this.cleanupPipecatClient();
83
+ });
69
84
  }
70
85
  connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
71
86
  return __awaiter(this, void 0, void 0, function* () {
@@ -73,6 +88,7 @@ export class VoiceAgentService {
73
88
  console.warn('[HiveGpt Voice] Call already in progress');
74
89
  return;
75
90
  }
91
+ const connectToken = ++this.lifecycleToken;
76
92
  try {
77
93
  this.callStateSubject.next('connecting');
78
94
  this.statusTextSubject.next('Connecting...');
@@ -96,13 +112,32 @@ export class VoiceAgentService {
96
112
  enableMic: true,
97
113
  enableCam: false,
98
114
  callbacks: {
99
- onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),
100
- onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),
101
- onBotReady: () => this.ngZone.run(() => this.onBotReady()),
102
- onUserTranscript: (data) => this.ngZone.run(() => this.userTranscriptSubject.next({ text: data.text, final: !!data.final })),
103
- onBotTranscript: (data) => this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),
115
+ onConnected: () => this.ngZone.run(() => {
116
+ if (!this.isLifecycleActive(connectToken))
117
+ return;
118
+ this.onPipecatConnected();
119
+ }),
120
+ onDisconnected: () => this.ngZone.run(() => {
121
+ if (!this.isLifecycleActive(connectToken))
122
+ return;
123
+ this.onPipecatDisconnected();
124
+ }),
125
+ onBotReady: () => this.ngZone.run(() => {
126
+ if (!this.isLifecycleActive(connectToken))
127
+ return;
128
+ this.onBotReady();
129
+ }),
130
+ onUserTranscript: (data) => this.ngZone.run(() => this.isLifecycleActive(connectToken) &&
131
+ this.userTranscriptSubject.next({ text: data.text, final: !!data.final })),
132
+ onBotTranscript: (data) => this.ngZone.run(() => {
133
+ if (!this.isLifecycleActive(connectToken))
134
+ return;
135
+ this.botTranscriptSubject.next(data.text);
136
+ }),
104
137
  onError: (err) => {
105
138
  this.ngZone.run(() => {
139
+ if (!this.isLifecycleActive(connectToken))
140
+ return;
106
141
  console.error('[HiveGpt Voice] PipecatClient error', err);
107
142
  this.callStateSubject.next('ended');
108
143
  this.statusTextSubject.next('Connection failed');
@@ -113,19 +148,33 @@ export class VoiceAgentService {
113
148
  this.pcClient = pcClient;
114
149
  // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element
115
150
  pcClient.on(RTVIEvent.TrackStarted, (track, participant) => {
151
+ if (!this.isLifecycleActive(connectToken))
152
+ return;
116
153
  if (!(participant === null || participant === void 0 ? void 0 : participant.local) && track.kind === 'audio') {
117
154
  this.ngZone.run(() => this.setupBotAudioTrack(track));
118
155
  }
119
156
  });
120
157
  // Speaking state comes straight from RTVI events
121
- pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() => this.onBotStartedSpeaking()));
122
- pcClient.on(RTVIEvent.BotStoppedSpeaking, () => this.ngZone.run(() => this.onBotStoppedSpeaking()));
158
+ pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() => {
159
+ if (!this.isLifecycleActive(connectToken))
160
+ return;
161
+ this.onBotStartedSpeaking();
162
+ }));
163
+ pcClient.on(RTVIEvent.BotStoppedSpeaking, () => this.ngZone.run(() => {
164
+ if (!this.isLifecycleActive(connectToken))
165
+ return;
166
+ this.onBotStoppedSpeaking();
167
+ }));
123
168
  pcClient.on(RTVIEvent.UserStartedSpeaking, () => this.ngZone.run(() => {
169
+ if (!this.isLifecycleActive(connectToken))
170
+ return;
124
171
  this.isUserSpeakingSubject.next(true);
125
172
  this.callStateSubject.next('listening');
126
173
  this.statusTextSubject.next('Listening...');
127
174
  }));
128
175
  pcClient.on(RTVIEvent.UserStoppedSpeaking, () => this.ngZone.run(() => {
176
+ if (!this.isLifecycleActive(connectToken))
177
+ return;
129
178
  this.isUserSpeakingSubject.next(false);
130
179
  if (this.callStateSubject.value === 'listening') {
131
180
  // Brief 'Processing...' while we wait for the bot to respond.
@@ -157,6 +206,8 @@ export class VoiceAgentService {
157
206
  });
158
207
  }
159
208
  catch (error) {
209
+ if (!this.isLifecycleActive(connectToken))
210
+ return;
160
211
  console.error('[HiveGpt Voice] connect failed', error);
161
212
  this.callStateSubject.next('ended');
162
213
  yield this.cleanupPipecatClient();
@@ -239,6 +290,7 @@ export class VoiceAgentService {
239
290
  }
240
291
  disconnect() {
241
292
  return __awaiter(this, void 0, void 0, function* () {
293
+ this.lifecycleToken += 1;
242
294
  this.stopDurationTimer();
243
295
  this.callStartTime = 0;
244
296
  this.audioAnalyzer.stop();
@@ -248,6 +300,9 @@ export class VoiceAgentService {
248
300
  this.statusTextSubject.next('Call Ended');
249
301
  });
250
302
  }
303
+ isLifecycleActive(token) {
304
+ return token === this.lifecycleToken;
305
+ }
251
306
  cleanupPipecatClient() {
252
307
  return __awaiter(this, void 0, void 0, function* () {
253
308
  if (this.pcClient) {
@@ -301,4 +356,4 @@ VoiceAgentService.ctorParameters = () => [
301
356
  { type: NgZone },
302
357
  { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
303
358
  ];
304
- //# 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,EAAE,MAAM,EAAa,WAAW,EAAE,MAAM,eAAe,CAAC;AACnF,OAAO,EAAE,eAAe,EAAc,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAC1E,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AACjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,iCAAiC,CAAC;AACrE,OAAO,EAAE,2BAA2B,EAAE,MAAM,kDAAkD,CAAC;AAC/F,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;;;;AAehE;;;;;;;;;;GAUG;AAIH,MAAM,OAAO,iBAAiB;IA4B5B,YACU,aAAmC,EACnC,oBAAiD,EACjD,MAAc;IACtB,8FAA8F;IACjE,UAAkB;QAJvC,kBAAa,GAAb,aAAa,CAAsB;QACnC,yBAAoB,GAApB,oBAAoB,CAA6B;QACjD,WAAM,GAAN,MAAM,CAAQ;QAEO,eAAU,GAAV,UAAU,CAAQ;QAhCzC,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,aAAQ,GAAyB,IAAI,CAAC;QACtC,oBAAe,GAA4B,IAAI,CAAC;QAEhD,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QACnC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAEvC,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,GAAwB,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QACjF,iBAAY,GAAyB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC5E,oBAAe,GAA+B,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QACxF,mBAAc,GAAuB,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;QAS5E,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,KAAK,IAAI,CAAC,UAAU,EAAE,CAAC;IACzB,CAAC;IAED,6EAA6E;IAC7E,WAAW;QACT,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM;YAAE,OAAO;QACnD,KAAK,IAAI,CAAC,UAAU,EAAE,CAAC;QACvB,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;IACpC,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,0CAA0C,CAAC,CAAC;gBACzD,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,IAAI,WAAW,GAAG,KAAK,CAAC;gBACxB,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;4BAAE,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;qBAC7D;oBAAC,OAAO,CAAC,EAAE;wBACV,OAAO,CAAC,IAAI,CAAC,sCAAsC,EAAE,CAAC,CAAC,CAAC;qBACzD;iBACF;gBAED,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBAE1C,MAAM,QAAQ,GAAG,IAAI,aAAa,CAAC;oBACjC,SAAS,EAAE,IAAI,kBAAkB,EAAE;oBACnC,SAAS,EAAE,IAAI;oBACf,SAAS,EAAE,KAAK;oBAChB,SAAS,EAAE;wBACT,WAAW,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC;wBACnE,cAAc,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC;wBACzE,UAAU,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;wBAC1D,gBAAgB,EAAE,CAAC,IAAI,EAAE,EAAE,CACzB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAC1E;wBACH,eAAe,EAAE,CAAC,IAAI,EAAE,EAAE,CACxB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;wBAClE,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;4BACf,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gCACnB,OAAO,CAAC,KAAK,CAAC,qCAAqC,EAAE,GAAG,CAAC,CAAC;gCAC1D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gCACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;4BACnD,CAAC,CAAC,CAAC;wBACL,CAAC;qBACF;iBACF,CAAC,CAAC;gBAEH,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;gBAEzB,6EAA6E;gBAC7E,QAAQ,CAAC,EAAE,CACT,SAAS,CAAC,YAAY,EACtB,CAAC,KAAuB,EAAE,WAAiC,EAAE,EAAE;oBAC7D,IAAI,CAAC,CAAA,WAAW,aAAX,WAAW,uBAAX,WAAW,CAAE,KAAK,CAAA,IAAI,KAAK,CAAC,IAAI,KAAK,OAAO,EAAE;wBACjD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC;qBACvD;gBACH,CAAC,CACF,CAAC;gBAEF,iDAAiD;gBACjD,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,kBAAkB,EAAE,GAAG,EAAE,CAC7C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CACnD,CAAC;gBACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,kBAAkB,EAAE,GAAG,EAAE,CAC7C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CACnD,CAAC;gBACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,mBAAmB,EAAE,GAAG,EAAE,CAC9C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBACtC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;oBACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;gBAC9C,CAAC,CAAC,CACH,CAAC;gBACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,mBAAmB,EAAE,GAAG,EAAE,CAC9C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;oBACvC,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,WAAW,EAAE;wBAC/C,8DAA8D;wBAC9D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;wBACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;qBAC9C;gBACH,CAAC,CAAC,CACH,CAAC;gBAEF,mDAAmD;gBACnD,MAAM,QAAQ,CAAC,WAAW,EAAE,CAAC;gBAE7B,sFAAsF;gBACtF,MAAM,cAAc,GAAG,IAAI,OAAO,EAAE,CAAC;gBACrC,cAAc,CAAC,MAAM,CAAC,eAAe,EAAE,UAAU,WAAW,EAAE,CAAC,CAAC;gBAChE,cAAc,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;gBAC3C,cAAc,CAAC,MAAM,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;gBAC5C,cAAc,CAAC,MAAM,CAAC,kBAAkB,EAAE,eAAe,CAAC,CAAC;gBAC3D,cAAc,CAAC,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAC5C,cAAc,CAAC,MAAM,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;gBAC1C,cAAc,CAAC,MAAM,CAAC,YAAY,EAAE,UAAU,CAAC,CAAC;gBAChD,cAAc,CAAC,MAAM,CAAC,4BAA4B,EAAE,MAAM,CAAC,CAAC;gBAE5D,mFAAmF;gBACnF,MAAM,QAAQ,CAAC,kBAAkB,CAAC;oBAChC,QAAQ,EAAE,GAAG,OAAO,sBAAsB;oBAC1C,OAAO,EAAE,cAAc;oBACvB,WAAW,EAAE;wBACX,MAAM,EAAE,KAAK;wBACb,eAAe,EAAE,cAAc;wBAC/B,KAAK,EAAE,OAAO;qBACf;iBACF,CAAC,CAAC;aACJ;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,KAAK,CAAC,gCAAgC,EAAE,KAAK,CAAC,CAAC;gBACvD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACpC,MAAM,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAClC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;gBACjD,MAAM,KAAK,CAAC;aACb;QACH,CAAC;KAAA;IAEO,kBAAkB;QACxB,gEAAgE;QAChE,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAChC,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACnC,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;IAEO,qBAAqB;QAC3B,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;QACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;QAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC5C,CAAC;IAEO,UAAU;;QAChB,kEAAkE;QAClE,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAG,MAAA,MAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,MAAM,EAA6C,0CAChF,GAAG,0CAAE,KAAK,CAAC;QACf,IAAI,QAAQ;YAAE,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAChD,qEAAqE;QACrE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IAC9C,CAAC;IAEO,qBAAqB;;QAC3B,MAAM,UAAU,GAAG,MAAA,MAAC,MAAA,IAAI,CAAC,QAAQ,0CAAE,MAAM,EAA+C,0CACpF,KAAK,0CAAE,KAAK,CAAC;QACjB,IAAI,UAAU,EAAE;YACd,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,WAAW,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;SACzD;IACH,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACtC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC1C,2DAA2D;QAC3D,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAEO,oBAAoB;QAC1B,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,SAAS,EAAE;YAC7C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;SAC7C;IACH,CAAC;IAEO,kBAAkB,CAAC,KAAuB;;QAChD,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE;YACzB,IAAI,CAAC,eAAe,GAAG,IAAI,KAAK,EAAE,CAAC;YACnC,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;SACtC;QACD,MAAM,QAAQ,GAAG,MAAC,IAAI,CAAC,eAAe,CAAC,SAAgC,0CACnE,cAAc,GAAG,CAAC,CAAC,CAAC;QACxB,IAAI,CAAA,QAAQ,aAAR,QAAQ,uBAAR,QAAQ,CAAE,EAAE,MAAK,KAAK,CAAC,EAAE;YAAE,OAAO;QAEtC,IAAI,CAAC,eAAe,CAAC,SAAS,GAAG,IAAI,WAAW,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CACxC,OAAO,CAAC,IAAI,CAAC,wCAAwC,EAAE,GAAG,CAAC,CAC5D,CAAC;IACJ,CAAC;IAEO,YAAY;;QAClB,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI;gBACF,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;gBAC7B,MAAC,IAAI,CAAC,eAAe,CAAC,SAAgC,0CAClD,cAAc,GACf,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;gBAC5B,IAAI,CAAC,eAAe,CAAC,SAAS,GAAG,IAAI,CAAC;aACvC;YAAC,WAAM;gBACN,SAAS;aACV;YACD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;SAC7B;IACH,CAAC;IAEK,UAAU;;YACd,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;YACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,MAAM,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAClC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC5C,CAAC;KAAA;IAEa,oBAAoB;;YAChC,IAAI,IAAI,CAAC,QAAQ,EAAE;gBACjB,IAAI;oBACF,MAAM,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;iBAClC;gBAAC,WAAM;oBACN,SAAS;iBACV;gBACD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;aACtB;QACH,CAAC;KAAA;IAED,SAAS;QACP,IAAI,CAAC,IAAI,CAAC,QAAQ;YAAE,OAAO;QAC3B,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;QAChD,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,IAAI,SAAS;YAAE,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACxD,CAAC;IAEO,kBAAkB;QACxB,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,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,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;gBACnC,MAAM,CAAC,GAAG,OAAO,GAAG,EAAE,CAAC;gBACvB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;aACjE;QACH,CAAC,CAAC;QACF,IAAI,EAAE,CAAC;QACP,IAAI,CAAC,gBAAgB,GAAG,WAAW,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAClD,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;;;;YA9TF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YA5BQ,oBAAoB;YADpB,2BAA2B;YALP,MAAM;YAoEU,MAAM,uBAA9C,MAAM,SAAC,WAAW","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.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 using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\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 pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\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> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\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    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    void this.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\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('[HiveGpt Voice] 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      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) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),\n          onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),\n          onBotReady: () => this.ngZone.run(() => this.onBotReady()),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStartedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStoppedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n          this.statusTextSubject.next('Listening...');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            // Brief 'Processing...' while we wait for the bot to respond.\n            this.callStateSubject.next('connected');\n            this.statusTextSubject.next('Processing...');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    // Start the duration timer from the moment the session is live.\n    this.callStartTime = Date.now();\n    this.startDurationTimer();\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    this.isMicMutedSubject.next(false);\n    this.startLocalMicAnalyzer();\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected.\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n    // Bot is initialised — signal that we're now waiting for user input.\n    this.statusTextSubject.next('Listening...');\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    this.callStateSubject.next('talking');\n    this.statusTextSubject.next('Talking...');\n    // Mark user as no longer speaking when bot takes the turn.\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Listening...');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.pcClient) return;\n    const nextMuted = !this.isMicMutedSubject.value;\n    this.pcClient.enableMic(!nextMuted);\n    this.isMicMutedSubject.next(nextMuted);\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const m = Math.floor(elapsed / 60);\n        const s = elapsed % 60;\n        this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);\n      }\n    };\n    tick();\n    this.durationInterval = setInterval(tick, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
359
+ //# 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, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.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 using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\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 pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n  private lifecycleToken = 0;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\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> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\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    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    void this.prepareFreshSession();\n  }\n\n  /**\n   * Hard reset voice session so each modal open starts from a clean state.\n   * Closes any previous client and prevents stale callbacks from mutating UI state.\n   */\n  async prepareFreshSession(): Promise<void> {\n    this.lifecycleToken += 1;\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.isUserSpeakingSubject.next(false);\n    this.isMicMutedSubject.next(false);\n    this.durationSubject.next('00:00');\n    this.statusTextSubject.next('');\n    this.callStateSubject.next('idle');\n    await this.cleanupPipecatClient();\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('[HiveGpt Voice] Call already in progress');\n      return;\n    }\n\n    const connectToken = ++this.lifecycleToken;\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = 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) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onPipecatConnected();\n            }),\n          onDisconnected: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onPipecatDisconnected();\n            }),\n          onBotReady: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onBotReady();\n            }),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.isLifecycleActive(connectToken) &&\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.botTranscriptSubject.next(data.text);\n            }),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStartedSpeaking();\n        }),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStoppedSpeaking();\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n          this.statusTextSubject.next('Listening...');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            // Brief 'Processing...' while we wait for the bot to respond.\n            this.callStateSubject.next('connected');\n            this.statusTextSubject.next('Processing...');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      if (!this.isLifecycleActive(connectToken)) return;\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    // Start the duration timer from the moment the session is live.\n    this.callStartTime = Date.now();\n    this.startDurationTimer();\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    this.isMicMutedSubject.next(false);\n    this.startLocalMicAnalyzer();\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected.\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n    // Bot is initialised — signal that we're now waiting for user input.\n    this.statusTextSubject.next('Listening...');\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    this.callStateSubject.next('talking');\n    this.statusTextSubject.next('Talking...');\n    // Mark user as no longer speaking when bot takes the turn.\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Listening...');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.lifecycleToken += 1;\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private isLifecycleActive(token: number): boolean {\n    return token === this.lifecycleToken;\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.pcClient) return;\n    const nextMuted = !this.isMicMutedSubject.value;\n    this.pcClient.enableMic(!nextMuted);\n    this.isMicMutedSubject.next(nextMuted);\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const m = Math.floor(elapsed / 60);\n        const s = elapsed % 60;\n        this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);\n      }\n    };\n    tick();\n    this.durationInterval = setInterval(tick, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}