微信小程序-音视频剪辑

起因:英语配音

源码在文章尾部,可直接Copy使用

最近在做一个英语配音的小程序项目,涉及的核心技术是:音视频剪辑。其实相关的成程序产品已经有很多了,所以花了几天时间也就搞定了,讲解一下其中核心技术:

  • 1.将一段英语视频中的音轨视轨分离。
  • 2.用户进行录音,仿照英文进行朗读,并临时保存录音后的音频数据
  • 3.将视频分离后的视轨与录音音频进行合成为一个新的视频
  • 4.正常播放合成后的视频(新视频)

音视频剪辑

  1. 第一步首先创建一个用于播放视频的video标签,并设置id的值
  <view class="video-wrapper">
    <video id="myVideo" src="http://wxsnsdy.tc.qq.com/105/20210/snsdyvideodownload?filekey=30280201010421301f0201690402534804102ca905ce620b1241b726bc41dcff44e00204012882540400&bizid=1023&hy=SH&fileparam=302c020101042530230204136ffd93020457e3c4ff02024ef202031e8d7f02030f42400204045a320a0201000400" binderror="videoErrorCallback" show-center-play-btn='{{false}}' show-play-btn="{{true}}" controls picture-in-picture-mode="{{['push', 'pop']}}" bindenterpictureinpicture='bindVideoEnterPictureInPicture' bindleavepictureinpicture='bindVideoLeavePictureInPicture'></video>
  </view>
  1. 初始化(根据小程序内置API)
  1. 创建视频控制器:videoContext , 用于视频播放,暂停等控制操作
  2. 创建录音管理器对象: recorderManager,用于录音开始,终止等控制操作
  3. 创建音频文件操作对象:innerAudioContext,用于对录音后保存的音频mp3文件进行操作,用于设置音频文件地址,静音,播放,暂停等操作
  4. 创建一个音视频操作对象,保存在data上:mediaContainer。通过此属性来对媒体文件(音频/视频)进行各种操作,比如分离视频中的音频,奖音轨和视轨合并。

方法讲解

  1. 需要用到的4个对象直接在onReady生命周期中进行初始化.
  2. 录音调用并保存:startRecord(). 录音完成后,触发录音结束方法endRecord(),回调方法success中将获取临时音频文件地址。根据此地址将音频文件中的音轨数据提取出来,并保存在mediaContainer之中。提取方法为extractDataSource().
  3. 播放录音:bindPlayRecord() 播放录音方法,必带参数src,指向音频文件地址,由录音时的回调方法中获取 res.tempFilePath。同时移动端音频通常默认静音:this.innerAudioContext.obeyMuteSwitch = false;
  4. 将视频中的音轨与视轨进行分离:chooseVideo(), 其中success毁掉方法的返回值mt,mt.tracks[0]为音轨 ,mt.tracks[1]为视轨(频)。需要注意的是此时将音频暂时保存在手机中:exportVideoMedia()。

注意:使用真机调试! 音频中的音轨抽取视频中的视轨抽取,都是调用的同一个对象MediaContainer上方法 this.data.mediaContainer.extractDataSource()

源码

// pages/videosound/videosound.js
const app = getApp();

Page({
  inputValue: "",
  data: {
    savedFilePath: "",
    total: 3, // 配音总数
    step: 0, // 当前配音
    isSpeaking: false, // 是否正在说话
    recordTempFilePath: "", // 录音临时缓存地址
    recordFrameList: [], // 所有录音片段
  },
  onReady() {
    this.videoContext = wx.createVideoContext("myVideo");  // 音频控制器
    this.recorderManager = wx.getRecorderManager(); // 录音对象
    this.innerAudioContext = wx.createInnerAudioContext(); // 播放对象
    this.data.mediaContainer = wx.createMediaContainer();
  },
  destroy() {
    this.videoContext.destroy();
  },
  bindPlaySourceSound() {
    console.log("1");
    this.videoContext.play();
  },
  bindPlaySeek(numberPostion) {
    this.videoContext.seek(20);
    this.videoContext.play();
  },
  recordCurrent() {
    //  参考 https://blog.csdn.net/qq_37257212/article/details/79093470
  },
  startRecord() {
    const options = {
      duration: 5000,
      sampleRate: 16000, // 采样率,有效值 8000/16000/44100
      numberOfChannels: 1, // 录音通道数,有效值 1/2
      encodeBitRate: 96000, // 编码码率
      format: "mp3", // 音频格式,有效值 aac/mp3
      frameSize: 50, // 指定帧大小,单位 KB
    };
    //开始录音
    this.recorderManager.start(options);
    this.recorderManager.onStart(() => {
      console.log("开始录音");
    });
    this.setData({
      isSpeaking: true,
    });
    //错误回调
    this.recorderManager.onError((res) => {
      console.log(res);
    });
  },
  endRecord() {
    this.recorderManager.onStop((res) => {
      if (res.duration < 1000) {
        wx.showToast({
          title: "录音时间太短",
        });
        return;
      } else {
        this.setData({
          isSpeaking: false,
        });
        this.data.recordTempFilePath = res.tempFilePath; // 文件临时路径

        let mt = this.data.mediaContainer.extractDataSource({
          source: res.tempFilePath,
          success: (mt) => {
            this.data.audioKind = mt.tracks[0];
            this.data.recordFrameList.push(mt.tracks[0]);
            this.data.mediaContainer.addTrack(this.data.audioKind);
          },
          fail: (err) => {
            console.log(err);
          },
        });
        // this.uploadFileRecord(res);
      }
    });
    this.recorderManager.onError((res) => {
      console.log("小伙砸你录音失败了!");
    });
  },
  bindPlayRecord(e) {
    var that = this;
    this.innerAudioContext.src = this.data.recordTempFilePath;
    this.innerAudioContext.play();
    this.innerAudioContext.obeyMuteSwitch = false;
    this.innerAudioContext.onEnded((res) => {
      that.innerAudioContext.stop();
    });
  },
  // 开始合成。真机可以。跳转页面后的缓存视频已经去掉了音频通道,
  bindComposeRecord() {
    this.toNextPage();
  },
  toNextPage() {
    wx.navigateTo({
      url: "/pages/videoresult/videoresult?src=" + this.data.savedFilePath,
    });
  },
  uploadFileRecord(res) {
    wx.showLoading({
      title: "发送中...",
    });
    var tempFilePath = res.tempFilePath; // 文件临时路径
    console.log("文件临时路径", tempFilePath);

    wx.uploadFile({
      url: "", //上传服务器的地址
      filePath: tempFilePath, //临时路径
      name: "file",
      header: {
        contentType: "multipart/form-data", //按需求增加
      },
      formData: null,
      success: function (res) {
        console.log("上传成功");
        wx.hideLoading();
        that.setData({
          recordTempFilePath: tempFilePath,
        });
      },
      fail: function (err) {
        wx.hideLoading();
        console.log(err.errMsg); //上传失败
      },
    });
  },
  // wxfile://tmp_9ded76d75506015bafb1c30d49d66827f6909af54e256aac.mp4
  chooseVideo: function () {
    wx.chooseVideo({
      sourceType: ["album", "camera"],
      maxDuration: 60,
      camera: "back",
      success: (res) => {
        let videoPath = res.tempFilePath;

        let mt = this.data.mediaContainer.extractDataSource({
          source: videoPath,
          success: (mt) => {
            console.log(mt);
            this.data.videoKind = mt.tracks[1];
            // this.data.audioKind = mt.tracks[0]; // 视频中的音频抽出来
            this.data.mediaContainer.addTrack(this.data.videoKind);
            this.exportVideoMedia();
          },
          fail: (err) => {
            console.log(err);
          },
        });
      },
      fail: (err) => {
        console.log(err);
      },
    });
  },
  exportVideoMedia() {
    var that = this;
    //3.导出视频
    this.data.mediaContainer.export({
      success: (result) => {
        console.log(result);
        let tempArr1 = result.tempFilePath.split("//");
        let tempArr2 = tempArr1[1].split("/");
        let tempArr3 = tempArr2[tempArr2.length - 1].split(".");
        let tempString2 = "";
        for (let i = 0; i < tempArr2.length - 1; i++) {
          tempString2 += tempArr2[i] + "/";
        }
        let newPath =
          tempArr1[0] +
          "//" +
          tempString2 +
          new Date().getTime() +
          "." +
          tempArr3[1];
        // 导出新视频的名字每次都是一样的,估计有缓存什么的,我用时间戳重命名新导出的文件
        var filemanage = wx
          .getFileSystemManager()
          .renameSync(result.tempFilePath, newPath);

        wx.saveFile({
          tempFilePath: newPath, // 传入一个本地临时文件路径
          success(res) {
            console.log(res.savedFilePath); // res.savedFilePath 为一个本地缓存文件路径
            that.data.savedFilePath = res.savedFilePath;
          },
        });
        wx.downloadFile({
          tempFilePath: newPath, // 传入一个本地临时文件路径
          success(res) {
            console.log(res); // res.savedFilePath 为一个本地缓存文件路径
          },
        });

        // 4.移除内容,清空容器
        this.data.mediaContainer.removeTrack(this.data.videoKind);
        this.data.mediaContainer.removeTrack(this.data.audioKind);
      },
    });
  },
});

源码star

------ 如果文章对你有用,感谢右上角 >>>点赞 | 收藏 <<<

  • 4
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值