| New file |
| | |
| | | export default class Recorder { |
| | | constructor(stream, config) { |
| | | //兼容 |
| | | window.URL = window.URL || window.webkitURL |
| | | navigator.getUserMedia = |
| | | navigator.getUserMedia || |
| | | navigator.webkitGetUserMedia || |
| | | navigator.mozGetUserMedia || |
| | | navigator.msGetUserMedia |
| | | |
| | | config = config || {} |
| | | config.sampleBits = config.sampleBits || 16 //采样数位 8, 16 |
| | | config.sampleRate = config.sampleRate || 8000 //采样率(1/6 44100) |
| | | |
| | | this.context = new (window.webkitAudioContext || window.AudioContext)() |
| | | this.audioInput = this.context.createMediaStreamSource(stream) |
| | | this.createScript = |
| | | this.context.createScriptProcessor || this.context.createJavaScriptNode |
| | | this.recorder = this.createScript.apply(this.context, [4096, 1, 1]) |
| | | |
| | | this.audioData = { |
| | | size: 0, //录音文件长度 |
| | | buffer: [], //录音缓存 |
| | | inputSampleRate: this.context.sampleRate, //输入采样率 |
| | | inputSampleBits: 16, //输入采样数位 8, 16 |
| | | outputSampleRate: config.sampleRate, //输出采样率 |
| | | oututSampleBits: config.sampleBits, //输出采样数位 8, 16 |
| | | input: function(data) { |
| | | this.buffer.push(new Float32Array(data)) |
| | | this.size += data.length |
| | | }, |
| | | compress: function() { |
| | | //合并压缩 |
| | | //合并 |
| | | let data = new Float32Array(this.size) |
| | | let offset = 0 |
| | | for (let i = 0; i < this.buffer.length; i++) { |
| | | data.set(this.buffer[i], offset) |
| | | offset += this.buffer[i].length |
| | | } |
| | | //压缩 |
| | | let compression = parseInt(this.inputSampleRate / this.outputSampleRate) |
| | | let length = data.length / compression |
| | | let result = new Float32Array(length) |
| | | let index = 0, |
| | | j = 0 |
| | | while (index < length) { |
| | | result[index] = data[j] |
| | | j += compression |
| | | index++ |
| | | } |
| | | return result |
| | | }, |
| | | encodeWAV: function() { |
| | | let sampleRate = Math.min(this.inputSampleRate, this.outputSampleRate) |
| | | let sampleBits = Math.min(this.inputSampleBits, this.oututSampleBits) |
| | | let bytes = this.compress() |
| | | let dataLength = bytes.length * (sampleBits / 8) |
| | | let buffer = new ArrayBuffer(44 + dataLength) |
| | | let data = new DataView(buffer) |
| | | |
| | | let channelCount = 1 //单声道 |
| | | let offset = 0 |
| | | |
| | | let writeString = function(str) { |
| | | for (let i = 0; i < str.length; i++) { |
| | | data.setUint8(offset + i, str.charCodeAt(i)) |
| | | } |
| | | } |
| | | |
| | | // 资源交换文件标识符 |
| | | writeString('RIFF') |
| | | offset += 4 |
| | | // 下个地址开始到文件尾总字节数,即文件大小-8 |
| | | data.setUint32(offset, 36 + dataLength, true) |
| | | offset += 4 |
| | | // WAV文件标志 |
| | | writeString('WAVE') |
| | | offset += 4 |
| | | // 波形格式标志 |
| | | writeString('fmt ') |
| | | offset += 4 |
| | | // 过滤字节,一般为 0x10 = 16 |
| | | data.setUint32(offset, 16, true) |
| | | offset += 4 |
| | | // 格式类别 (PCM形式采样数据) |
| | | data.setUint16(offset, 1, true) |
| | | offset += 2 |
| | | // 通道数 |
| | | data.setUint16(offset, channelCount, true) |
| | | offset += 2 |
| | | // 采样率,每秒样本数,表示每个通道的播放速度 |
| | | data.setUint32(offset, sampleRate, true) |
| | | offset += 4 |
| | | // 波形数据传输率 (每秒平均字节数) 单声道×每秒数据位数×每样本数据位/8 |
| | | data.setUint32( |
| | | offset, |
| | | channelCount * sampleRate * (sampleBits / 8), |
| | | true |
| | | ) |
| | | offset += 4 |
| | | // 快数据调整数 采样一次占用字节数 单声道×每样本的数据位数/8 |
| | | data.setUint16(offset, channelCount * (sampleBits / 8), true) |
| | | offset += 2 |
| | | // 每样本数据位数 |
| | | data.setUint16(offset, sampleBits, true) |
| | | offset += 2 |
| | | // 数据标识符 |
| | | writeString('data') |
| | | offset += 4 |
| | | // 采样数据总数,即数据总大小-44 |
| | | data.setUint32(offset, dataLength, true) |
| | | offset += 4 |
| | | // 写入采样数据 |
| | | if (sampleBits === 8) { |
| | | for (let i = 0; i < bytes.length; i++, offset++) { |
| | | let s = Math.max(-1, Math.min(1, bytes[i])) |
| | | let val = s < 0 ? s * 0x8000 : s * 0x7fff |
| | | val = parseInt(255 / (65535 / (val + 32768))) |
| | | data.setInt8(offset, val, true) |
| | | } |
| | | } else { |
| | | for (let i = 0; i < bytes.length; i++, offset += 2) { |
| | | let s = Math.max(-1, Math.min(1, bytes[i])) |
| | | data.setInt16(offset, s < 0 ? s * 0x8000 : s * 0x7fff, true) |
| | | } |
| | | } |
| | | return new Blob([data], { |
| | | type: 'audio/wav', |
| | | }) |
| | | }, |
| | | } |
| | | } |
| | | |
| | | //开始录音 |
| | | start() { |
| | | this.audioInput.connect(this.recorder) |
| | | this.recorder.connect(this.context.destination) |
| | | |
| | | //音频采集 |
| | | let self = this |
| | | this.recorder.onaudioprocess = function(e) { |
| | | self.audioData.input(e.inputBuffer.getChannelData(0)) |
| | | } |
| | | } |
| | | |
| | | //停止 |
| | | stop() { |
| | | this.recorder.disconnect() |
| | | } |
| | | |
| | | //获取音频文件 |
| | | getBlob() { |
| | | this.stop() |
| | | return this.audioData.encodeWAV() |
| | | } |
| | | |
| | | //回放 |
| | | play(audio) { |
| | | audio.src = window.URL.createObjectURL(this.getBlob()) |
| | | } |
| | | |
| | | //清理缓存的录音数据 |
| | | clear(audio) { |
| | | this.audioData.buffer = [] |
| | | this.audioData.size = 0 |
| | | audio.src = '' |
| | | } |
| | | |
| | | static checkError(e) { |
| | | const { name } = e |
| | | let errorMsg = '' |
| | | switch (name) { |
| | | case 'AbortError': |
| | | errorMsg = '录音设备无法被使用' |
| | | break |
| | | case 'NotAllowedError': |
| | | errorMsg = '用户已禁止网页调用录音设备' |
| | | break |
| | | case 'PermissionDeniedError': |
| | | errorMsg = '用户已禁止网页调用录音设备' |
| | | break // 用户拒绝 |
| | | case 'NotFoundError': |
| | | errorMsg = '录音设备未找到' |
| | | break |
| | | case 'DevicesNotFoundError': |
| | | errorMsg = '录音设备未找到' |
| | | break |
| | | case 'NotReadableError': |
| | | errorMsg = '录音设备无法使用' |
| | | break |
| | | case 'NotSupportedError': |
| | | errorMsg = '不支持录音功能' |
| | | break |
| | | case 'MandatoryUnsatisfiedError': |
| | | errorMsg = '无法发现指定的硬件设备' |
| | | break |
| | | default: |
| | | errorMsg = '录音调用错误' |
| | | break |
| | | } |
| | | return { |
| | | error: errorMsg, |
| | | } |
| | | } |
| | | |
| | | static get(callback, config) { |
| | | if (callback) { |
| | | if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) { |
| | | navigator.mediaDevices |
| | | .getUserMedia({ |
| | | audio: true, |
| | | video: false, |
| | | }) |
| | | .then(stream => { |
| | | let rec = new Recorder(stream, config) |
| | | callback(rec) |
| | | }) |
| | | .catch(e => { |
| | | callback(Recorder.checkError(e)) |
| | | }) |
| | | } else { |
| | | navigator |
| | | .getUserMedia({ |
| | | audio: true, |
| | | video: false, |
| | | }) |
| | | .then(stream => { |
| | | let rec = new Recorder(stream, config) |
| | | callback(rec) |
| | | }) |
| | | .catch(e => { |
| | | // Recorder.checkError(e) |
| | | callback(Recorder.checkError(e)) |
| | | }) |
| | | } |
| | | } |
| | | } |
| | | } |