vue实现PC端录音功能的实例代码


Posted in Javascript onJune 05, 2019

录音功能一般来说在移动端比较常见,但是在pc端也要实现按住说话的功能呢?项目需求:按住说话,时长不超过60秒,生成语音文件并上传,我这里用的是recorder.js

1.项目中新建一个recorder.js文件,内容如下,也可在百度上直接搜一个

// 兼容
window.URL = window.URL || window.webkitURL
navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia
let HZRecorder = function (stream, config) {
 config = config || {}
 config.sampleBits = config.sampleBits || 8 // 采样数位 8, 16
 config.sampleRate = config.sampleRate || (44100 / 6) // 采样率(1/6 44100)
 let context = new (window.webkitAudioContext || window.AudioContext)()
 let audioInput = context.createMediaStreamSource(stream)
 let createScript = context.createScriptProcessor || context.createJavaScriptNode
 let recorder = createScript.apply(context, [4096, 1, 1])
 let audioData = {
  size: 0, // 录音文件长度
  buffer: [], // 录音缓存
  inputSampleRate: 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; let 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/mp3' })
  }
 }
 // 开始录音
 this.start = function () {
  audioInput.connect(recorder)
  recorder.connect(context.destination)
 }
 // 停止
 this.stop = function () {
  recorder.disconnect()
 }
 // 获取音频文件
 this.getBlob = function () {
  this.stop()
  return audioData.encodeWAV()
 }
 // 回放
 this.play = function (audio) {
  let downRec = document.getElementById('downloadRec')
  downRec.href = window.URL.createObjectURL(this.getBlob())
  downRec.download = new Date().toLocaleString() + '.mp3'
  audio.src = window.URL.createObjectURL(this.getBlob())
 }
 // 上传
 this.upload = function (url, callback) {
  let fd = new FormData()
  fd.append('audioData', this.getBlob())
  let xhr = new XMLHttpRequest()
  /* eslint-disable */
  if (callback) {
   xhr.upload.addEventListener('progress', function (e) {
    callback('uploading', e)
   }, false)
   xhr.addEventListener('load', function (e) {
    callback('ok', e)
   }, false)
   xhr.addEventListener('error', function (e) {
    callback('error', e)
   }, false)
   xhr.addEventListener('abort', function (e) {
    callback('cancel', e)
   }, false)
  }
  /* eslint-disable */
  xhr.open('POST', url)
  xhr.send(fd)
 }
 // 音频采集
 recorder.onaudioprocess = function (e) {
  audioData.input(e.inputBuffer.getChannelData(0))
  // record(e.inputBuffer.getChannelData(0));
 }
}
// 抛出异常
HZRecorder.throwError = function (message) {
 alert(message)
 throw new function () { this.toString = function () { return message } }()
}
// 是否支持录音
HZRecorder.canRecording = (navigator.getUserMedia != null)
// 获取录音机
HZRecorder.get = function (callback, config) {
 if (callback) {
  if (navigator.getUserMedia) {
   navigator.getUserMedia(
    { audio: true } // 只启用音频
    , function (stream) {
     let rec = new HZRecorder(stream, config)
     callback(rec)
    }
    , function (error) {
     switch (error.code || error.name) {
      case 'PERMISSION_DENIED':
      case 'PermissionDeniedError':
       HZRecorder.throwError('用户拒绝提供信息。')
       break
      case 'NOT_SUPPORTED_ERROR':
      case 'NotSupportedError':
       HZRecorder.throwError('浏览器不支持硬件设备。')
       break
      case 'MANDATORY_UNSATISFIED_ERROR':
      case 'MandatoryUnsatisfiedError':
       HZRecorder.throwError('无法发现指定的硬件设备。')
       break
      default:
       HZRecorder.throwError('无法打开麦克风。异常信息:' + (error.code || error.name))
       break
     }
    })
  } else {
   HZRecorder.throwErr('当前浏览器不支持录音功能。'); return
  }
 }
}
export default HZRecorder

2.页面中使用,具体如下

<template>
 <div class="wrap">
  <el-form v-model="form">
   <el-form-item>
    <input type="button" class="btn-record-voice" @mousedown.prevent="mouseStart" @mouseup.prevent="mouseEnd" v-model="form.time"/>
    <audio v-if="form.audioUrl" :src="form.audioUrl" controls="controls" class="content-audio" style="display: block;">语音</audio>
   </el-form-item>
  <el-form>
 </div>
</template>
<script>
// 引入recorder.js
import recording from '@/js/recorder/recorder.js'
export default {
 data() {
  return {
   form: {
    time: '按住说话(60秒)',
    audioUrl: ''
   },
   num: 60, // 按住说话时间
   recorder: null,
   interval: '',
   audioFileList: [], // 上传语音列表
   startTime: '', // 语音开始时间
   endTime: '', // 语音结束
  }
 },
 methods: {
  // 清除定时器
  clearTimer () {
   if (this.interval) {
    this.num = 60
    clearInterval(this.interval)
   }
  },
  // 长按说话
  mouseStart () {
   this.clearTimer()
   this.startTime = new Date().getTime()
   recording.get((rec) => {
    // 当首次按下时,要获取浏览器的麦克风权限,所以这时要做一个判断处理
    if (rec) {
     // 首次按下,只调用一次
     if (this.flag) {
      this.mouseEnd()
      this.flag = false
     } else {
      this.recorder = rec
      this.interval = setInterval(() => {
       if (this.num <= 0) {
        this.recorder.stop()
        this.num = 60
        this.clearTimer()
       } else {
        this.num--
        this.time = '松开结束(' + this.num + '秒)'
        this.recorder.start()
       }
      }, 1000)
     }
    }
   })
  },
  // 松开时上传语音
  mouseEnd () {
   this.clearTimer()
   this.endTime = new Date().getTime()
   if (this.recorder) {
    this.recorder.stop()
    // 重置说话时间
    this.num = 60
    this.time = '按住说话(' + this.num + '秒)'
    // 获取语音二进制文件
    let bold = this.recorder.getBlob()
    // 将获取的二进制对象转为二进制文件流
    let files = new File([bold], 'test.mp3', {type: 'audio/mp3', lastModified: Date.now()})
    let fd = new FormData()
    fd.append('file', files)
    fd.append('tenantId', 3) // 额外参数,可根据选择填写
    // 这里是通过上传语音文件的接口,获取接口返回的路径作为语音路径
    this.uploadFile(fd)
   }
  }
 }
}
</script>
<style scoped>
</style>

3.除了上述代码中的注释外,还有一些地方需要注意

  • 上传语音时,一般会有两个参数,一个是语音的路径,一个是语音的时长,路径直接就是 this.form.audioUrl ,不过时长这里需要注意的是,由于我们一开始设置了定时器是有一秒的延迟,所以,要在获取到的时长基础上在减去一秒
  • 初次按住说话一定要做判断,不然就会报错啦
  • 第三点也是很重要的一点,因为我是在本地项目中测试的,可以实现录音功能,但是打包到测试环境后,就无法访问麦克风,经过多方尝试后,发现是由于我们测试环境的地址是http://***,而在谷歌浏览器中有这样一种安全策略,只允许在localhost下及https下才可以访问 ,因此换一下就完美的解决了这个问题了
  • 在使用过程中,针对不同的浏览器可能会有些兼容性的问题,如果遇到了还需自己单独处理下

总结

以上所述是小编给大家介绍的vue实现PC端录音功能的实例代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Javascript 相关文章推荐
小议Function.apply()之二------利用Apply的参数数组化来提高 JavaScript程序性能
Nov 30 Javascript
Jquery选择器 $实现原理
Dec 02 Javascript
DOM Scripting中的图片切换[兼容Firefox]
Jun 12 Javascript
JavaScript中的常见问题解决方法(乱码,IE缓存,代理)
Nov 28 Javascript
js实现绿白相间竖向网页百叶窗动画切换效果
Mar 02 Javascript
Node.js实现文件上传
Jul 05 Javascript
浅析JavaScript中作用域和作用域链
Dec 06 Javascript
jQuery中的siblings()是什么意思(推荐)
Dec 29 Javascript
jQuery实现的事件绑定功能基本示例
Oct 11 jQuery
D3.js实现简洁实用的动态仪表盘的示例
Apr 04 Javascript
使用vue重构资讯页面的实例代码解析
Nov 26 Javascript
vuex Module将 store 分割成模块的操作
Dec 07 Vue.js
vue-cli3添加模式配置多环境变量的方法
Jun 05 #Javascript
Vue+axios+WebApi+NPOI导出Excel文件实例方法
Jun 05 #Javascript
js实现随机8位验证码
Jul 24 #Javascript
Vue中全局变量的定义和使用
Jun 05 #Javascript
详解express使用vue-router的history踩坑
Jun 05 #Javascript
laravel-admin 与 vue 结合使用实例代码详解
Jun 04 #Javascript
用webpack4开发小程序的实现方法
Jun 04 #Javascript
You might like
PHP 和 MySQL 基础教程(三)
2006/10/09 PHP
php生成随机密码的几种方法
2011/01/17 PHP
php设计模式 Mediator (中介者模式)
2011/06/26 PHP
使用php自动备份数据库表的实现方法
2017/07/28 PHP
PHP的mysqli_thread_id()函数讲解
2019/01/24 PHP
php解析非标准json、非规范json的方式实例
2020/12/10 PHP
javascript 写类方式之五
2009/07/05 Javascript
深入理解JSON数据源格式
2014/01/10 Javascript
js实现的标题栏新消息闪烁提示效果
2014/06/06 Javascript
ECMAScript6的新特性箭头函数(Arrow Function)详细介绍
2014/06/07 Javascript
js验证真实姓名与身份证号是否匹配
2015/10/13 Javascript
jquery实现表单验证简单实例演示
2015/11/23 Javascript
仿百度换肤功能的简单实例代码
2016/07/11 Javascript
JavaScript实现左右下拉框动态增删示例
2017/03/09 Javascript
详解Vue中状态管理Vuex
2017/05/11 Javascript
JScript实现表格的简单操作
2017/08/15 Javascript
JQuery选中select组件被选中的值方法
2018/03/08 jQuery
Vue+Node实现商品列表的分页、排序、筛选,添加购物车功能详解
2019/12/07 Javascript
[44:30]完美世界DOTA2联赛PWL S2 GXR vs Magma 第一场 11.25
2020/11/26 DOTA
python去掉空白行的多种实现代码
2018/03/19 Python
python3连接MySQL数据库实例详解
2018/05/24 Python
python3.6.3安装图文教程 TensorFlow安装配置方法
2020/06/24 Python
通过python将大量文件按修改时间分类的方法
2018/10/17 Python
Python for循环与range函数的使用详解
2019/03/23 Python
什么是Python变量作用域
2020/06/03 Python
企业演讲稿范文
2013/12/28 职场文书
高中生期末评语
2014/01/28 职场文书
父亲的菜园教学反思
2014/02/13 职场文书
2014年公司植树节活动方案
2014/03/04 职场文书
小学生常见病防治方案
2014/06/06 职场文书
在职党员进社区活动总结
2014/07/05 职场文书
经营目标管理责任书
2014/07/25 职场文书
追讨欠款律师函
2015/05/27 职场文书
初婚未育证明样本
2015/06/18 职场文书
Python机器学习之逻辑回归
2021/05/11 Python
JAVA长虹键法之建造者Builder模式实现
2022/04/10 Java/Android