@hivegpt/hiveai-angular 0.0.598 → 0.0.599
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.
- package/esm2020/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.mjs +41 -22
- package/esm2020/lib/components/voice-agent/services/audio-analyzer.service.mjs +18 -14
- package/esm2020/lib/components/voice-agent/services/voice-agent.service.mjs +29 -18
- package/fesm2015/hivegpt-hiveai-angular.mjs +85 -51
- package/fesm2015/hivegpt-hiveai-angular.mjs.map +1 -1
- package/fesm2020/hivegpt-hiveai-angular.mjs +85 -51
- package/fesm2020/hivegpt-hiveai-angular.mjs.map +1 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +10 -6
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
- package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +2 -0
- package/lib/components/voice-agent/services/audio-analyzer.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +3 -4
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -10,6 +10,7 @@ export class AudioAnalyzerService {
|
|
|
10
10
|
this.ngZone = ngZone;
|
|
11
11
|
this.audioContext = null;
|
|
12
12
|
this.analyserNode = null;
|
|
13
|
+
this.sourceNode = null;
|
|
13
14
|
this.dataArray = null;
|
|
14
15
|
this.animationFrameId = null;
|
|
15
16
|
this.isRunning = false;
|
|
@@ -23,6 +24,7 @@ export class AudioAnalyzerService {
|
|
|
23
24
|
this.WAVEFORM_BAR_COUNT = 60;
|
|
24
25
|
// Amplify raw amplitude so normal speech (±10–20 units) maps to visible levels
|
|
25
26
|
this.SENSITIVITY_MULTIPLIER = 5;
|
|
27
|
+
this.barsBuffer = new Array(this.WAVEFORM_BAR_COUNT).fill(0);
|
|
26
28
|
this.audioLevels$ = this.audioLevelsSubject.asObservable();
|
|
27
29
|
this.isUserSpeaking$ = this.isUserSpeakingSubject.asObservable();
|
|
28
30
|
}
|
|
@@ -35,13 +37,13 @@ export class AudioAnalyzerService {
|
|
|
35
37
|
if (this.audioContext.state === 'suspended') {
|
|
36
38
|
this.audioContext.resume().catch(console.warn);
|
|
37
39
|
}
|
|
38
|
-
|
|
40
|
+
this.sourceNode = this.audioContext.createMediaStreamSource(stream);
|
|
39
41
|
this.analyserNode = this.audioContext.createAnalyser();
|
|
40
42
|
this.analyserNode.fftSize = 2048;
|
|
41
43
|
this.analyserNode.smoothingTimeConstant = 0.4;
|
|
42
44
|
const bufferLength = this.analyserNode.frequencyBinCount;
|
|
43
45
|
this.dataArray = new Uint8Array(bufferLength);
|
|
44
|
-
|
|
46
|
+
this.sourceNode.connect(this.analyserNode);
|
|
45
47
|
this.isRunning = true;
|
|
46
48
|
this.noiseFloorSamples = [];
|
|
47
49
|
this.noiseFloor = 0;
|
|
@@ -58,6 +60,10 @@ export class AudioAnalyzerService {
|
|
|
58
60
|
cancelAnimationFrame(this.animationFrameId);
|
|
59
61
|
this.animationFrameId = null;
|
|
60
62
|
}
|
|
63
|
+
if (this.sourceNode) {
|
|
64
|
+
this.sourceNode.disconnect();
|
|
65
|
+
this.sourceNode = null;
|
|
66
|
+
}
|
|
61
67
|
if (this.analyserNode) {
|
|
62
68
|
this.analyserNode.disconnect();
|
|
63
69
|
this.analyserNode = null;
|
|
@@ -95,11 +101,12 @@ export class AudioAnalyzerService {
|
|
|
95
101
|
const isSpeaking = rms > threshold;
|
|
96
102
|
// Generate waveform bars
|
|
97
103
|
const bars = this.generateWaveformBars(this.dataArray);
|
|
98
|
-
//
|
|
99
|
-
this.
|
|
100
|
-
this.isUserSpeakingSubject.next(isSpeaking);
|
|
101
|
-
|
|
102
|
-
|
|
104
|
+
// Only trigger change detection when speaking state actually changes
|
|
105
|
+
if (isSpeaking !== this.isUserSpeakingSubject.value) {
|
|
106
|
+
this.ngZone.run(() => this.isUserSpeakingSubject.next(isSpeaking));
|
|
107
|
+
}
|
|
108
|
+
// Emit audio levels outside Angular's zone — no CD triggered here
|
|
109
|
+
this.audioLevelsSubject.next(bars);
|
|
103
110
|
this.animationFrameId = requestAnimationFrame(() => this.analyze());
|
|
104
111
|
}
|
|
105
112
|
calculateRMS(data) {
|
|
@@ -111,16 +118,13 @@ export class AudioAnalyzerService {
|
|
|
111
118
|
return Math.sqrt(sum / data.length);
|
|
112
119
|
}
|
|
113
120
|
generateWaveformBars(data) {
|
|
114
|
-
const bars = [];
|
|
115
121
|
const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);
|
|
116
122
|
for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {
|
|
117
|
-
const
|
|
118
|
-
const value = data[index];
|
|
119
|
-
// Normalize and amplify so quiet speech produces visible movement
|
|
123
|
+
const value = data[i * step];
|
|
120
124
|
const normalized = Math.abs((value - 128) / 128) * 100 * this.SENSITIVITY_MULTIPLIER;
|
|
121
|
-
|
|
125
|
+
this.barsBuffer[i] = Math.min(100, Math.max(0, normalized));
|
|
122
126
|
}
|
|
123
|
-
return
|
|
127
|
+
return this.barsBuffer;
|
|
124
128
|
}
|
|
125
129
|
}
|
|
126
130
|
AudioAnalyzerService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: AudioAnalyzerService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
@@ -131,4 +135,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
131
135
|
providedIn: 'root'
|
|
132
136
|
}]
|
|
133
137
|
}], ctorParameters: function () { return [{ type: i0.NgZone }]; } });
|
|
134
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Audio analyzer for waveform visualization and local (mic) speaking detection.\n * VoiceAgentService may combine this with WebSocket server events for call state.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 60;\n  // Amplify raw amplitude so normal speech (±10–20 units) maps to visible levels\n  private readonly SENSITIVITY_MULTIPLIER = 5;\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      if (this.audioContext.state === 'suspended') {\n        this.audioContext.resume().catch(console.warn);\n      }\n      const source = this.audioContext.createMediaStreamSource(stream);\n      \n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.4;\n      \n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n      \n      source.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.ngZone.runOutsideAngular(() => this.analyze());\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n\n    // Emit inside Angular's zone so change detection picks up UI updates\n    this.ngZone.run(() => {\n      this.isUserSpeakingSubject.next(isSpeaking);\n      this.audioLevelsSubject.next(bars);\n    });\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const bars: number[] = [];\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    \n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const index = i * step;\n      const value = data[index];\n      // Normalize and amplify so quiet speech produces visible movement\n      const normalized = Math.abs((value - 128) / 128) * 100 * this.SENSITIVITY_MULTIPLIER;\n      bars.push(Math.min(100, Math.max(0, normalized)));\n    }\n    \n    return bars;\n  }\n}\n"]}
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Audio analyzer for waveform visualization and local (mic) speaking detection.\n * VoiceAgentService may combine this with WebSocket server events for call state.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private sourceNode: MediaStreamAudioSourceNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 60;\n  // Amplify raw amplitude so normal speech (±10–20 units) maps to visible levels\n  private readonly SENSITIVITY_MULTIPLIER = 5;\n  private barsBuffer: number[] = new Array(this.WAVEFORM_BAR_COUNT).fill(0);\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      if (this.audioContext.state === 'suspended') {\n        this.audioContext.resume().catch(console.warn);\n      }\n      this.sourceNode = this.audioContext.createMediaStreamSource(stream);\n\n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.4;\n\n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n\n      this.sourceNode.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.ngZone.runOutsideAngular(() => this.analyze());\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.sourceNode) {\n      this.sourceNode.disconnect();\n      this.sourceNode = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n\n    // Only trigger change detection when speaking state actually changes\n    if (isSpeaking !== this.isUserSpeakingSubject.value) {\n      this.ngZone.run(() => this.isUserSpeakingSubject.next(isSpeaking));\n    }\n    // Emit audio levels outside Angular's zone — no CD triggered here\n    this.audioLevelsSubject.next(bars);\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const value = data[i * step];\n      const normalized = Math.abs((value - 128) / 128) * 100 * this.SENSITIVITY_MULTIPLIER;\n      this.barsBuffer[i] = Math.min(100, Math.max(0, normalized));\n    }\n    return this.barsBuffer;\n  }\n}\n"]}
|
|
@@ -40,7 +40,7 @@ export class VoiceAgentService {
|
|
|
40
40
|
this.botAudioElement = null;
|
|
41
41
|
this.lifecycleToken = 0;
|
|
42
42
|
this.subscriptions = new Subscription();
|
|
43
|
-
this.
|
|
43
|
+
this.pcEventCleanup = [];
|
|
44
44
|
this.callState$ = this.callStateSubject.asObservable();
|
|
45
45
|
this.statusText$ = this.statusTextSubject.asObservable();
|
|
46
46
|
this.duration$ = this.durationSubject.asObservable();
|
|
@@ -49,12 +49,7 @@ export class VoiceAgentService {
|
|
|
49
49
|
this.audioLevels$ = this.audioLevelsSubject.asObservable();
|
|
50
50
|
this.userTranscript$ = this.userTranscriptSubject.asObservable();
|
|
51
51
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
52
|
-
this.subscriptions.add(this.audioAnalyzer.audioLevels$.pipe(throttleTime(50)).subscribe((levels) => this.audioLevelsSubject.next(levels)));
|
|
53
|
-
}
|
|
54
|
-
ngOnDestroy() {
|
|
55
|
-
this.destroy$.next();
|
|
56
|
-
this.subscriptions.unsubscribe();
|
|
57
|
-
void this.disconnect();
|
|
52
|
+
this.subscriptions.add(this.audioAnalyzer.audioLevels$.pipe(throttleTime(50)).subscribe((levels) => this.ngZone.run(() => this.audioLevelsSubject.next(levels))));
|
|
58
53
|
}
|
|
59
54
|
/** Reset to idle (e.g. when modal re-opens so user can click Start Call). */
|
|
60
55
|
resetToIdle() {
|
|
@@ -141,32 +136,36 @@ export class VoiceAgentService {
|
|
|
141
136
|
});
|
|
142
137
|
this.pcClient = pcClient;
|
|
143
138
|
// Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element
|
|
144
|
-
|
|
139
|
+
const onTrackStarted = (track, participant) => {
|
|
145
140
|
if (!this.isLifecycleActive(connectToken))
|
|
146
141
|
return;
|
|
147
142
|
if (!participant?.local && track.kind === 'audio') {
|
|
148
143
|
this.ngZone.run(() => this.setupBotAudioTrack(track));
|
|
149
144
|
}
|
|
150
|
-
}
|
|
145
|
+
};
|
|
146
|
+
pcClient.on(RTVIEvent.TrackStarted, onTrackStarted);
|
|
151
147
|
// Speaking state comes straight from RTVI events
|
|
152
|
-
|
|
148
|
+
const onBotStarted = () => this.ngZone.run(() => {
|
|
153
149
|
if (!this.isLifecycleActive(connectToken))
|
|
154
150
|
return;
|
|
155
151
|
this.onBotStartedSpeaking();
|
|
156
|
-
})
|
|
157
|
-
pcClient.on(RTVIEvent.
|
|
152
|
+
});
|
|
153
|
+
pcClient.on(RTVIEvent.BotStartedSpeaking, onBotStarted);
|
|
154
|
+
const onBotStopped = () => this.ngZone.run(() => {
|
|
158
155
|
if (!this.isLifecycleActive(connectToken))
|
|
159
156
|
return;
|
|
160
157
|
this.onBotStoppedSpeaking();
|
|
161
|
-
})
|
|
162
|
-
pcClient.on(RTVIEvent.
|
|
158
|
+
});
|
|
159
|
+
pcClient.on(RTVIEvent.BotStoppedSpeaking, onBotStopped);
|
|
160
|
+
const onUserStarted = () => this.ngZone.run(() => {
|
|
163
161
|
if (!this.isLifecycleActive(connectToken))
|
|
164
162
|
return;
|
|
165
163
|
this.isUserSpeakingSubject.next(true);
|
|
166
164
|
this.callStateSubject.next('listening');
|
|
167
165
|
this.statusTextSubject.next('Listening...');
|
|
168
|
-
})
|
|
169
|
-
pcClient.on(RTVIEvent.
|
|
166
|
+
});
|
|
167
|
+
pcClient.on(RTVIEvent.UserStartedSpeaking, onUserStarted);
|
|
168
|
+
const onUserStopped = () => this.ngZone.run(() => {
|
|
170
169
|
if (!this.isLifecycleActive(connectToken))
|
|
171
170
|
return;
|
|
172
171
|
this.isUserSpeakingSubject.next(false);
|
|
@@ -175,7 +174,16 @@ export class VoiceAgentService {
|
|
|
175
174
|
this.callStateSubject.next('connected');
|
|
176
175
|
this.statusTextSubject.next('Processing...');
|
|
177
176
|
}
|
|
178
|
-
})
|
|
177
|
+
});
|
|
178
|
+
pcClient.on(RTVIEvent.UserStoppedSpeaking, onUserStopped);
|
|
179
|
+
// Store cleanup fns so listeners are removed on disconnect
|
|
180
|
+
this.pcEventCleanup = [
|
|
181
|
+
() => pcClient.off(RTVIEvent.TrackStarted, onTrackStarted),
|
|
182
|
+
() => pcClient.off(RTVIEvent.BotStartedSpeaking, onBotStarted),
|
|
183
|
+
() => pcClient.off(RTVIEvent.BotStoppedSpeaking, onBotStopped),
|
|
184
|
+
() => pcClient.off(RTVIEvent.UserStartedSpeaking, onUserStarted),
|
|
185
|
+
() => pcClient.off(RTVIEvent.UserStoppedSpeaking, onUserStopped),
|
|
186
|
+
];
|
|
179
187
|
// Acquire mic (triggers browser permission prompt)
|
|
180
188
|
await pcClient.initDevices();
|
|
181
189
|
// Build headers using the browser Headers API (required by pipecat's APIRequest type)
|
|
@@ -303,6 +311,9 @@ export class VoiceAgentService {
|
|
|
303
311
|
async cleanupPipecatClient() {
|
|
304
312
|
if (this.pcClient) {
|
|
305
313
|
try {
|
|
314
|
+
// Remove all event listeners before disconnecting
|
|
315
|
+
this.pcEventCleanup.forEach(fn => fn());
|
|
316
|
+
this.pcEventCleanup = [];
|
|
306
317
|
await this.pcClient.disconnect();
|
|
307
318
|
}
|
|
308
319
|
catch {
|
|
@@ -353,4 +364,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
353
364
|
type: Inject,
|
|
354
365
|
args: [PLATFORM_ID]
|
|
355
366
|
}] }]; } });
|
|
356
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/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 { distinctUntilChanged, take, throttleTime } 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().pipe(distinctUntilChanged());\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$.pipe(throttleTime(50)).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    // Preserve any mute action the user made during \"connecting\".\n    const shouldBeMuted = this.isMicMutedSubject.value;\n    this.pcClient?.enableMic(!shouldBeMuted);\n    this.isMicMutedSubject.next(shouldBeMuted);\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      (this.botAudioElement as any).playsInline = true;\n      this.botAudioElement.setAttribute('playsinline', '');\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.pause();\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    const nextMuted = !this.isMicMutedSubject.value;\n    // Reflect the user's intent immediately, even if connect is still in progress.\n    this.isMicMutedSubject.next(nextMuted);\n    if (this.pcClient) {\n      this.pcClient.enableMic(!nextMuted);\n    }\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.ngZone.run(() => this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`));\n      }\n    };\n    tick();\n    this.ngZone.runOutsideAngular(() => {\n      this.durationInterval = setInterval(tick, 1000);\n    });\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|
|
367
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/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, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { distinctUntilChanged, take, throttleTime } 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 {\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 pcEventCleanup: (() => 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().pipe(distinctUntilChanged());\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$.pipe(throttleTime(50)).subscribe((levels) =>\n        this.ngZone.run(() => this.audioLevelsSubject.next(levels)),\n      ),\n    );\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      const onTrackStarted = (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      pcClient.on(RTVIEvent.TrackStarted, onTrackStarted);\n\n      // Speaking state comes straight from RTVI events\n      const onBotStarted = () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStartedSpeaking();\n        });\n      pcClient.on(RTVIEvent.BotStartedSpeaking, onBotStarted);\n\n      const onBotStopped = () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStoppedSpeaking();\n        });\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, onBotStopped);\n\n      const onUserStarted = () =>\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      pcClient.on(RTVIEvent.UserStartedSpeaking, onUserStarted);\n\n      const onUserStopped = () =>\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      pcClient.on(RTVIEvent.UserStoppedSpeaking, onUserStopped);\n\n      // Store cleanup fns so listeners are removed on disconnect\n      this.pcEventCleanup = [\n        () => pcClient.off(RTVIEvent.TrackStarted, onTrackStarted),\n        () => pcClient.off(RTVIEvent.BotStartedSpeaking, onBotStarted),\n        () => pcClient.off(RTVIEvent.BotStoppedSpeaking, onBotStopped),\n        () => pcClient.off(RTVIEvent.UserStartedSpeaking, onUserStarted),\n        () => pcClient.off(RTVIEvent.UserStoppedSpeaking, onUserStopped),\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    // Preserve any mute action the user made during \"connecting\".\n    const shouldBeMuted = this.isMicMutedSubject.value;\n    this.pcClient?.enableMic(!shouldBeMuted);\n    this.isMicMutedSubject.next(shouldBeMuted);\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      (this.botAudioElement as any).playsInline = true;\n      this.botAudioElement.setAttribute('playsinline', '');\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.pause();\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        // Remove all event listeners before disconnecting\n        this.pcEventCleanup.forEach(fn => fn());\n        this.pcEventCleanup = [];\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    const nextMuted = !this.isMicMutedSubject.value;\n    // Reflect the user's intent immediately, even if connect is still in progress.\n    this.isMicMutedSubject.next(nextMuted);\n    if (this.pcClient) {\n      this.pcClient.enableMic(!nextMuted);\n    }\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.ngZone.run(() => this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`));\n      }\n    };\n    tick();\n    this.ngZone.runOutsideAngular(() => {\n      this.durationInterval = setInterval(tick, 1000);\n    });\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|