javascript - 我们如何使用 mediaRecorder 将 Canvas 流与音频流混合

标签 javascript html5-canvas webrtc mediarecorder-api

<分区>

我有一个使用 canvas.captureStream() 的 Canvas 流。我有另一个来自 webrtc 视频通话的视频流。现在我想将 Canvas 流与视频流的音轨混合。我该怎么做?

最佳答案

使用 MediaStream constructor在 Firefox 和 Chrome 56 中可用,将轨道组合成一个新的流:

let stream = new MediaStream([videoTrack, audioTrack]);

以下在 Firefox 中对我有用(在 Chrome 中使用 https fiddle,尽管它在录制时出错):

navigator.mediaDevices.getUserMedia({audio: true})
  .then(stream => record(new MediaStream([stream.getTracks()[0],
                                          whiteNoise().getTracks()[0]]), 5000)
    .then(recording => {
      stop(stream);
      video.src = link.href = URL.createObjectURL(new Blob(recording));
      link.download = "recording.webm";
      link.innerHTML = "Download recording";
      log("Playing "+ recording[0].type +" recording:");
    })
    .catch(log))
  .catch(log);

var whiteNoise = () => {
  let ctx = canvas.getContext('2d');
  ctx.fillRect(0, 0, canvas.width, canvas.height);
  let p = ctx.getImageData(0, 0, canvas.width, canvas.height);
  requestAnimationFrame(function draw(){
    for (var i = 0; i < p.data.length; i++) {
      p.data[i++] = p.data[i++] = p.data[i++] = Math.random() * 255;
    }
    ctx.putImageData(p, 0, 0);
    requestAnimationFrame(draw);
  });
  return canvas.captureStream(60);
}

var record = (stream, ms) => {
  var rec = new MediaRecorder(stream), data = [];
  rec.ondataavailable = e => data.push(e.data);
  rec.start();
  log(rec.state + " for "+ (ms / 1000) +" seconds...");
  var stopped = new Promise((y, n) =>
      (rec.onstop = y, rec.onerror = e => n(e.error || e.name)));
  return Promise.all([stopped, wait(ms).then(_ => rec.stop())]).then(_ => data);
};

var stop = stream => stream.getTracks().forEach(track => track.stop());
var wait = ms => new Promise(resolve => setTimeout(resolve, ms));
var log = msg => div.innerHTML += "<br>" + msg;
<div id="div"></div><br>
<canvas id="canvas" width="160" height="120" hidden></canvas>
<video id="video" width="160" height="120" autoplay></video>
<a id="link"></a>

关于javascript - 我们如何使用 mediaRecorder 将 Canvas 流与音频流混合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39992048/

相关文章:

javascript - Ajax 调用后显示成功消息的问题

javascript - 在匿名函数中加载谷歌地图

javascript - array.map 正在改变我的原始数组

javascript - 我可以在 svg 文件上绘制形状吗

javascript - 渐变映射作为 HTML5 Canvas 元素中的不透明度?

android - 如何在 webview 中获取 webRTC 的媒体?

javascript - 无法从 JavaScript 客户端通过 https 连接到 websocket 服务器

javascript - 获取最后一个子元素之前的内部文本

javascript - 对 canvas 元素之外的元素使用 globalCompositeOperation

webrtc - WebRTC 视频在流式传输之前是否加密?