interface CodecCostructor { new (codecSampleRate: number) : Codec; } class BufferChunk { buffer: AudioBuffer; index: number; constructor(buffer: AudioBuffer) { this.buffer = buffer; this.index = 0; } copyRangeTo(target: AudioBuffer, maxLength: number, offset: number) { let copy = Math.min(this.buffer.length - this.index, maxLength); for(let channel = 0; channel < this.buffer.numberOfChannels; channel++) { target.getChannelData(channel).set( this.buffer.getChannelData(channel).subarray(this.index, this.index + copy), offset ); } return copy; } } class CodecClientCache { _chunks: BufferChunk[] = []; bufferedSamples(max: number = 0) : number { let value = 0; for(let i = 0; i < this._chunks.length && value < max; i++) value += this._chunks[i].buffer.length - this._chunks[i].index; return value; } } interface Codec { on_encoded_data: (Uint8Array) => void; channelCount: number; samplesPerUnit: number; name() : string; initialise(); deinitialise(); decodeSamples(cache: CodecClientCache, data: Uint8Array) : Promise; encodeSamples(cache: CodecClientCache, pcm: AudioBuffer); reset() : boolean; }