|
|
@@ -8,6 +8,7 @@ export class AudioHandler {
|
|
|
private nextPlayTime: number = 0;
|
|
|
private isPlaying: boolean = false;
|
|
|
private playbackQueue: AudioBufferSourceNode[] = [];
|
|
|
+ private playBuffer: Int16Array[] = [];
|
|
|
|
|
|
constructor() {
|
|
|
this.context = new AudioContext({ sampleRate: this.sampleRate });
|
|
|
@@ -84,12 +85,14 @@ export class AudioHandler {
|
|
|
this.isPlaying = false;
|
|
|
this.playbackQueue.forEach((source) => source.stop());
|
|
|
this.playbackQueue = [];
|
|
|
+ this.playBuffer = [];
|
|
|
}
|
|
|
|
|
|
playChunk(chunk: Uint8Array) {
|
|
|
if (!this.isPlaying) return;
|
|
|
|
|
|
const int16Data = new Int16Array(chunk.buffer);
|
|
|
+ this.playBuffer.push.apply(this.playBuffer, int16Data); // save playBuffer
|
|
|
|
|
|
const float32Data = new Float32Array(int16Data.length);
|
|
|
for (let i = 0; i < int16Data.length; i++) {
|
|
|
@@ -125,6 +128,33 @@ export class AudioHandler {
|
|
|
this.nextPlayTime = this.context.currentTime;
|
|
|
}
|
|
|
}
|
|
|
+ _saveData(data: Int16Array, bytesPerSample = 16): Blob {
|
|
|
+ const headerLength = 44;
|
|
|
+ const numberOfChannels = 1;
|
|
|
+ const dataLength = data.length;
|
|
|
+ const wav = new Uint8Array(headerLength + dataLength * 2);
|
|
|
+ const view = new DataView(wav.buffer);
|
|
|
+ view.setUint32(0, 1380533830, false); // RIFF identifier 'RIFF'
|
|
|
+ view.setUint32(4, 36 + dataLength * 2, true); // file length minus RIFF identifier length and file description length
|
|
|
+ view.setUint32(8, 1463899717, false); // RIFF type 'WAVE'
|
|
|
+ view.setUint32(12, 1718449184, false); // format chunk identifier 'fmt '
|
|
|
+ view.setUint32(16, 16, true); // format chunk length
|
|
|
+ view.setUint16(20, 1, true); // sample format (raw)
|
|
|
+ view.setUint16(22, numberOfChannels, true); // channel count
|
|
|
+ view.setUint32(24, this.sampleRate, true); // sample rate
|
|
|
+ view.setUint32(28, this.sampleRate * 4, true); // byte rate (sample rate * block align)
|
|
|
+ view.setUint16(32, numberOfChannels * 2, true); // block align (channel count * bytes per sample)
|
|
|
+ view.setUint16(34, bytesPerSample, true); // bits per sample
|
|
|
+ view.setUint32(36, 1684108385, false); // data chunk identifier 'data'
|
|
|
+ view.setUint32(40, dataLength * 2, true); // data chunk length
|
|
|
+ for (let i = 0; i < dataLength; i++) {
|
|
|
+ view.setInt16(44 + i * 2, data[i], true);
|
|
|
+ }
|
|
|
+ return new Blob([view], { type: "audio/mpeg" });
|
|
|
+ }
|
|
|
+ savePlayFile() {
|
|
|
+ return this._saveData(this.playBuffer);
|
|
|
+ }
|
|
|
async close() {
|
|
|
this.workletNode?.disconnect();
|
|
|
this.source?.disconnect();
|