前言:

最近遇到 要语音转文字 的需求,语音转文字肯定要先搞定录音功能,在网上找了好久没找到具体的 RecordRTC.js 插件的使用方法,最后只能对着 github 上开源代码小试了一下,录音功能好使所以就记录一下叭

一、RecordRTC.js 源码指路

https://github.com/muaz-khan/RecordRTC

二、功能逻辑分析

需求分析:用户点击语音按钮->请求麦克风权限,语音按钮编程文本按钮,文本输入框显示为“按住说话” ->用户按住输入框说话->显示语音录入的动画,同时文本输入框显示为“松开结束”->用户松开即结束录音,将录音文件传给后端,后端进行转文字

html5 录音保存 html5录音插件_语音识别

三、使用方法

1、从下载的源码中将RecordRTC.js 文件引入放在项目js文件中,并在项目中引用

2.新建js文件 audioRTC.js书写自己的功能逻辑代码 

3、参照源码simple-demos->audio-recording.html 文件完成js代码

4、函数功能记录

captureMicrophone  获取麦克风权限

replaceAudio 释放麦克风(根据具体需求使用,我这里未释放)

recorder.startRecording() ->开始录音 (recorder 为实例对象)

recorder.stopRecording() ->结束录音

stopRecordingCallback 结束录音的回调函数

四、功能代码书写

1、准备 捕获麦克风函数

var isEdge = navigator.userAgent.indexOf('Edge') !== -1 && (!!navigator.msSaveOrOpenBlob || !!navigator.msSaveBlob);


var recorder; // globally accessible
var microphone;  // 麦克风
// 捕获麦克风
function captureMicrophone(callback) {
  console.log("捕获麦克风函数调用")
  if (microphone) {
      callback(microphone);
      return;
  }
  // 没有媒体设置告知版本低
  if (typeof navigator.mediaDevices === 'undefined' || !navigator.mediaDevices.getUserMedia) {
      var sys = isAndroidOrIOSOrPc();
      if (sys == "pc") {
          layer.msg("该浏览器不支持语音录入,<br>请使用谷歌、火狐等主流浏览器。", { area: ['320px', '80px'] });
      } else if (sys == "ios") {
          layer.msg("该浏览器不支持语音录入,<br>请将您的IOS操作系统升级最新版本,并使用safari浏览器打开使用。", { area: ['320px', '80px'] });
      } else if (sys == "android") {
          layer.msg("该浏览器不支持语音录入,<br>请使用系统自带浏览器打开使用。", { area: ['320px', '80px'] });
      } else {
          layer.msg("您当前的操作系统不支持语音录入。", { area: ['300px', '50px'] });
      }
  }
 // 获取设备的录音权限
  navigator.mediaDevices.getUserMedia({
      audio: isEdge ? true : { echoCancellation: false }
  }).then(function(mic) {
    console.log("获取麦克风成功回调",mic)
      callback(mic);
  }).catch(function(error) {
    console.log("获取麦克风失败回调",error)
    layer.msg("该浏览器不支持语音录入。<br>或您拒绝了语音授权", { area: ['300px', '60px'] });
      // 禁用麦克风走这里-》禁用语音按钮
      // 按钮背景换成语音
      $("#recordMess").css({
        "background":"url(images/noyuyin.png) no-repeat center",
        "background-size":" 100% 100%",
        })
        $("#recordMess").attr('disabled',true);
        $('#chatform').children("#messCon").show();
        $('#chatform').children("#talkmess_btn").hide();
  })
}

2、按住输入框开始录音,封装录音函数

// 开始录音
function startRec() {

  var options = {
      type: 'audio',
      numberOfAudioChannels: 1,
      checkForInactiveTracks: false,
      bufferSize: 4096,
      recorderType: StereoAudioRecorder
  };

  if (recorder) {
      recorder.destroy();
      recorder = null;
  }
  // recordre 实例
  recorder = RecordRTC(microphone, options);
  recorder.startRecording(); //开始录音
};

3、停止录音回调

//停止录音的回调
function stopRecordingCallback() {
  // 
  console.log("停止录音回调")
      var internalRecorder = recorder.getInternalRecorder();
      console.log("停止录音回调internalRecorder",internalRecorder)
      // 左声道
      var leftchannel = internalRecorder.leftchannel;
      // 右声道
      var rightchannel = internalRecorder.rightchannel;
      console.log("左声道",leftchannel)
      console.log(("啦啦啦",internalRecorder.blob,window.URL||webkitURL).createObjectURL(internalRecorder.blob))
      console.log("internalRecorderBlob",internalRecorder.blob)
        // 将录音文件 以文件对象形式传给后端
      var form = new FormData()
      form.append("upfile",internalRecorder.blob,"recorder.wav");
      console.log("form",form)
      $.ajax({
        url: '后端接口地址',
        type:'POST',
        cache: false,
        processData: false,
        contentType: false,
        data: form,
        success: function(data){
          console.log("后端返回数据对象",data)
        // 根据数据进行具体操作
         
      
        },
        error:function (err) { 
          console.log("ajaxerr",err)
         }
    })

}

4、按钮绑定监听事件,调用录音相关函数

  4.1、语音按钮绑定点击事件

var flagvoice=false;	
$("#recordMess").click(function() {
		if(flagvoice){
			$(this).css({
				"background":"url(images/yuyin.png) no-repeat center",
				"background-size":" 100% 100%",
			})
			$('#chatform').children("#messCon").show();
			$('#chatform').children("#talkmess_btn").hide();
			flagvoice=false;
		}else{
      $(this).css({
        "background":"url(images/wenben.png) no-repeat center",
        "background-size":" 100% 100%",
      })
      $('#chatform').children("#messCon").hide();
      $('#chatform').children("#talkmess_btn").show();
      flagvoice=true;
    }
    //  ---------------获取麦克风权限 START--------
    if (!microphone) {
      captureMicrophone(function(mic) {
        console.log("获取语音权限",mic)
          microphone = mic;
      });
      return;
    }
    //  ---------------获取麦克风权限 END--------
	})

  4.2、输入框绑定触摸事件,触摸按开始录音

var posStart = 0;//初始化起点坐标
var posEnd = 0;//初始化终点坐标
//长按
	var btnElem=document.getElementById("talkmess_btn");//获取ID
	btnElem.addEventListener("touchstart", function(event) {
    event.stopPropagation(); // 阻止冒泡
		event.preventDefault();//阻止浏览器默认行为
		posStart = 0;
		posStart = event.touches[0].pageY;//获取起点坐标
    // btnElem.value = '松开 结束';
    btnElem.innerText = '松开 结束';
		$("#audiobg").show();
		// $("#audiobg>p").hide();
		console.log("start");
    console.log(posStart+'---------开始坐标');
    // 开始录音
    startRec()
	});

  4.3、输入框绑定触摸事件,触摸抬起结束录音

btnElem.addEventListener("touchend", function(event) {
    event.stopPropagation();
		event.preventDefault();
		posEnd = 0;
		posEnd = event.changedTouches[0].pageY;//获取终点坐标
    // btnElem.value = '按住 说话';
    btnElem.innerText = '按住 说话';
		console.log("End");
		console.log(posEnd+'---------结束坐标');
		if(posStart - posEnd < 100 ){
			// $("#audiobg>p").hide();
			console.log("发送成功");
      // recStop()
      recorder.stopRecording(stopRecordingCallback);
		}else{
			// $("#audiobg>p").show();
			console.log("取消发送");
			console.log("Cancel");
		};
    $("#audiobg").hide();
    
	});

4.4、为测试前端是否真的拿到用户语音输入的内容,可以加一个测试按钮用于下载用户语音

// 获取文件名的随机字符串
function getRandomString() {
    if (window.crypto && window.crypto.getRandomValues && navigator.userAgent.indexOf('Safari') === -1) {
        var a = window.crypto.getRandomValues(new Uint32Array(3)),
            token = '';
        for (var i = 0, l = a.length; i < l; i++) {
            token += a[i].toString(36);
        }
        return token;
    } else {
        return (Math.random() * new Date().getTime()).toString(36).replace(/\./g, '');
    }
}
// 文件名
function getFileName(fileExtension) {
  var d = new Date();
  var year = d.getFullYear();
  var month = d.getMonth();
  var date = d.getDate();
  return 'RecordRTC-' + year + month + date + '-' + getRandomString() + '.' + fileExtension;
}
// 下载录音
function downloadRecording(){
  console.log("下载录音")
  if(!recorder || !recorder.getBlob()) return;

   
    var blob = recorder.getBlob();
    var file = new File([blob], getFileName('wav'), {
        type: 'audio/wav'
    });
    invokeSaveAsDialog(file); // 该方法在recorderRTC.js中已有
}
// 下载按钮绑定点击事件
var download = document.getElementById("download")
    download.addEventListener("click",function(){
      downloadRecording()
})

五、存在问题

完成上述代码之后,经测试发现该功能在安卓手机上正常,但是在一些较老版本的ios系统上(如 iPhone 7Plus iphone xs)存在只能录音一次的问题,解决方案:在结束录音函数中判断设备是ios即释放麦克风,在开始录音中判断麦克风是否存在并再调用一次开始录音函数,具体代码如下:

// 释放麦克风
function releaseMicrophone(){
  if(microphone) {
    microphone.stop();
    microphone = null;
  }

}
// 判断设备是否是ios
var isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
//停止录音的回调
function stopRecordingCallback() {
  // --------此处省略结束录音回调函数,代码同上,仅追加对于ios的判断--------------
  if(isSafari){
    releaseMicrophone()
  }
}
// 开始录音
function startRec() {
// -----------新增补充代码 start-------------
  if (!microphone) {
    captureMicrophone(function(mic) {
        microphone = mic;
        // click(btnStartRecording);
        startRec()
    });
    return;
  }
// -----------新增补充代码 end-------------
  var options = {
      type: 'audio',
      numberOfAudioChannels: 1,
      checkForInactiveTracks: false,
      bufferSize: 4096,
      sampleRate:48000,
      recorderType: StereoAudioRecorder
  };

  if (recorder) {
      recorder.destroy();
      recorder = null;
  }
  // recordre 实例
  recorder = RecordRTC(microphone, options);
  recorder.startRecording(); //开始录音
};