media tutorial
1.0.0
createObjectURL
错误Failed to execute 'createObjectURL' on 'URL'
window . URL . createObjectURL ( data )
var binaryData = [ ] ;
binaryData . push ( data ) ;
window . URL . createObjectURL ( new Blob ( binaryData , { type : "application/zip" } ) ) ;
navigator.mediaDevices.getDisplayMedia
屏幕源navigator.mediaDevices.getUserMedia
网络头源 <!DOCTYPE html >
< html lang =" en " >
< head >
< meta charset =" UTF-8 " >
< meta name =" viewport " content =" width=device-width, initial-scale=1.0 " >
< meta http-equiv =" X-UA-Compatible " content =" ie=edge " >
< title > Document </ title >
</ head >
< body >
< style >
video {
width: 1000px;
height: 500px
}
</ style >
< video autoplay id =" video " > Video stream not available. </ video >
< script >
let video = document . getElementById ( 'video' ) ;
navigator . mediaDevices . getDisplayMedia ( {
video : true ,
audio : true
} )
. then ( stream => {
// we have a stream, attach it to a feedback video element
console . log ( stream ) ;
video . srcObject = stream ;
} , error => {
console . log ( "Unable to acquire screen capture" , error ) ;
} ) ;
</ script >
</ body >
</ html >
使用MediaRecorder
录制视频或者音频的步骤
navigator.mediaDevices.getUserMedia(constraints).then(function(stream) {}
获取音频视频的stream
var mediaRecorder = new MediaRecorder(stream)
实例化mediaRecorder
, mediaRecorder
会不断接受navigator.mediaDevices.getUserMedia(constraints).then(function(stream) {}
所提供的stream
,并且它本身有多个方法mediaRecorder.start()
点击事件触发开始录制mediaRecorder.onstop = function(e) {}
往mediaRecorder
监听停止事件var blob = new Blob(chunks, { 'type' : 'audio/ogg; codecs=opus' });chunks = [];
这里的chunks
会从无到有,随着音视频流被监听,不断增加,注意onstop
会触发两次,一开始未录制和暂停都会触发mediaRecorder.ondataavailable = function(e) {chunks.push(e.data);}
不断监听stream
,并往chunks
备份添加音视频流的数据mediaRecorder.stop()
点击事件触发暂停录制var audioURL = URL.createObjectURL(blob);audio.src = audioURL;console.log("recorder stopped");
用blob
生成一个url
挂载到audio
或者video
标签上面去播放 var constraints = { audio : true } ;
var chunks = [ ] ;
navigator . mediaDevices . getUserMedia ( constraints )
. then ( function ( stream ) {
var mediaRecorder = new MediaRecorder ( stream ) ;
record . onclick = function ( ) {
mediaRecorder . start ( ) ;
console . log ( mediaRecorder . state ) ;
}
stop . onclick = function ( ) {
mediaRecorder . stop ( ) ;
console . log ( mediaRecorder . state ) ;
}
mediaRecorder . onstop = function ( e ) {
audio . setAttribute ( 'controls' , '' ) ;
audio . controls = true ;
var blob = new Blob ( chunks , { 'type' : 'audio/ogg; codecs=opus' } ) ;
chunks = [ ] ;
var audioURL = URL . createObjectURL ( blob ) ;
audio . src = audioURL ;
console . log ( "recorder stopped" ) ;
}
mediaRecorder . ondataavailable = function ( e ) {
chunks . push ( e . data ) ;
}
} )
. catch ( function ( err ) {
console . log ( 'The following error occurred: ' + err ) ;
} )
localPeerConnection = new RTCPeerConnection ( null ) ;
localPeerConnection . onicecandidate = function ( event ) {
if ( event . candidate ) {
remotePeerConnection . addIceCandidate ( new RTCIceCandidate ( event . candidate ) ) ; //answer方接收ICE
}
} ;
remotePeerConnection = new RTCPeerConnection ( null ) ;
remotePeerConnection . onicecandidate = function ( event ) {
if ( event . candidate ) {
localPeerConnection . addIceCandidate ( new RTCIceCandidate ( event . candidate ) ) ; //offer方接收ICE
}
} ;
remotePeerConnection . onaddstream = function gotRemoteStream ( event ) {
recordedVideo . srcObject = event . stream ;
} ;
localPeerConnection . addStream ( recordStream ) ;
localPeerConnection . createOffer ( function ( description ) { //description是offer方的 SD ==> 传输的内容
localPeerConnection . setLocalDescription ( description ) ;
remotePeerConnection . setRemoteDescription ( description ) ; //answer方接收offer的SD
remotePeerConnection . createAnswer ( function ( description ) {
remotePeerConnection . setLocalDescription ( description ) ; //answer方设置本身自己的SD
localPeerConnection . setRemoteDescription ( description ) ; //offer接收answer方的SD
} , function ( error ) {
console . log ( error )
} ) ; //answer方发送自己的SD
} , function ( error ) {
console . log ( error )
} ) ;
Blob
Binary Large Object
的缩写,代表二进制类型的大对象
new Blob ( blobParts , [ options ] ) ;
blobParts: 集群类型,集群中的每一个连接起来构成Blob对象的数据,集群中的每项元素可以是ArrayBuffer, ArrayBufferView, Blob, DOMString 。
options:可选项,字典格式类型,可以指定以下两个属性:
var data1 = "a" ;
var data2 = "b" ;
var data3 = "<div style='color:red;'>This is a blob</div>" ;
var data4 = {
"name" : "abc"
} ;
var blob1 = new Blob ( [ data1 ] ) ;
var blob2 = new Blob ( [ data1 , data2 ] ) ;
var blob3 = new Blob ( [ data3 ] ) ;
var blob4 = new Blob ( [ JSON . stringify ( data4 ) ] ) ;
var blob5 = new Blob ( [ data4 ] ) ;
var blob6 = new Blob ( [ data3 , data4 ] ) ;
console . log ( blob1 ) ; //输出:Blob {size: 1, type: ""}
console . log ( blob2 ) ; //输出:Blob {size: 2, type: ""}
console . log ( blob3 ) ; //输出:Blob {size: 44, type: ""}
console . log ( blob4 ) ; //输出:Blob {size: 14, type: ""}
console . log ( blob5 ) ; //输出:Blob {size: 15, type: ""}
console . log ( blob6 ) ; //输出:Blob {size: 59, type: ""}
Blob
对象有一个slice
方法,返回一个新的Blob
对象,包含了源Blob
对象中指定范围内的数据。
slice ( [ start ] , [ end ] , [ contentType ] )
Blob
里的下标,表示第一个会被拷贝进新的Blob
的字节起始位置。如果建立一个负数,那么这个偏移量就是从数据开始的的计算从后到前开始计算。Blob
的一个下标,这个下标-1的字节将被拷贝进新的Blob
的最后一个字节。如果你确定了一个负数,那么这个偏数移动量将从数据的消耗从后到前开始计算。Blob
赋予一个新的文档类型。这会将它的type
属性设置为确定的值。它的默认值是一个空的字符串。 var data = "abcdef" ;
var blob1 = new Blob ( [ data ] ) ;
var blob2 = blob1 . slice ( 0 , 3 ) ;
console . log ( blob1 ) ; //输出:Blob {size: 6, type: ""}
console . log ( blob2 ) ; //输出:Blob {size: 3, type: ""}
let href = URL . createObjectURL ( blob2 ) ; //浏览器可以直接打开href连接看输出
console . log ( href ) ; //abc
URL.createObjectURL()
静态方法会创建一个DOMString
,其中包含一个表示参数中给出的对象的URL。这个URL
的生命周期和创建它的窗口中的document
绑定。这个新的URL
对象表示指定的File
对象或Blob
对象。
objectURL = URL . createObjectURL ( blob ) ;
URL.revokeObjectURL()
静态方法用于释放一个之前通过调用URL.createObjectURL()
创建已经存在的URL
对象。当你结束使用某个URL
对象时,应该通过调用这个方法来让浏览器不再需要保留这个文件的引用。
window . URL . revokeObjectURL ( objectURL ) ;
目前, Blob
对象主要是运用在,处理大文件分割上传(利用Blob
中slice
方法),处理图片canvas
跨域(避免增加crossOrigin = "Anonymous"
,生成当前域名的url
,然后URL.revokeObjectURL()
释放, createjs
有用),以及隐藏视频源等等。
function upload ( blobOrFile ) {
var xhr = new XMLHttpRequest ( ) ;
xhr . open ( 'POST' , '/server' , true ) ;
xhr . onload = function ( e ) {
// ...
} ;
xhr . send ( blobOrFile ) ;
}
document . querySelector ( 'input[type="file"]' ) . addEventListener ( 'change' , function ( e ) {
var blob = this . files [ 0 ] ;
const BYTES_PER_CHUNK = 1024 * 1024 ; // 1MB chunk sizes.
const SIZE = blob . size ;
var start = 0 ;
var end = BYTES_PER_CHUNK ;
while ( start < SIZE ) {
upload ( blob . slice ( start , end ) ) ;
start = end ;
end = start + BYTES_PER_CHUNK ;
}
} , false ) ;
var xhr = new XMLHttpRequest ( ) ;
xhr . open ( 'GET' , '/path/to/image.png' , true ) ;
xhr . responseType = 'blob' ;
xhr . send ( )
xhr . onload = function ( e ) {
if ( this . status == 200 ) {
var blob = this . response ;
var img = document . createElement ( 'img' ) ;
var URL = window . URL || window . webkitURL ; //兼容处理
var objectUrl = URL . createObjectURL ( blob ) ;
img . onload = function ( e ) {
window . URL . revokeObjectURL ( img . src ) ; // 释放 url.
} ;
img . src = objectUrl ;
document . body . appendChild ( img ) ;
// ...
}
} ;
xhr . send ( ) ;
代码 | 作用 | 类型 |
---|---|---|
navigator.mediaDevices.getUserMedia(constraints).then(function(stream){}; | 根据音视频源头生成stream | 媒体流 |
mediaRecorder = new MediaRecorder(window.stream, options) | 根据音视频源头把stream 排序mediaRecorder 处理 | 媒体记录器 |
mediaRecorder.ondataavailable = function (event) {recordedBlobs.push(event.data);} | 根据stream 获取每一片段的BlobEvent 并存入recordedBlobs 数据库 | 斑点事件 |
let buffer = new Blob(recordedBlobs, {type: "video/webm"}); | 获取Blob 格式的音频视频流 | 斑点 |
test.src = window.URL.createObjectURL(buffer); | 可用window.URL.createObjectURL 方法处理Blob并殴video 或者audio 标签播放 | 源代码 |
recordStream = test.captureStream(); | 把Blob 转化为MediaStream | 媒体流 |
new RTCPeerConnection(null).addStream(recordStream); | 将recordStream 挖掘RTCPeerConnection 处理 | 媒体流 |
mediaRecorder = new MediaRecorder ( window . stream , options ) ; // 设置音频录入源、格式
mediaRecorder . ondataavailable = function ( event ) {
// 这个会不断接受BlobEvent
console . log ( event ) ;
if ( event . data && event . data . size > 0 ) {
recordedBlobs . push ( event . data ) ;
}
} ; // 存放获取的数据
let buffer = new Blob ( recordedBlobs , {
type : "video/webm"
} ) ;
console . log ( buffer ) ;
test . src = window . URL . createObjectURL ( buffer ) ;
recordStream = test . captureStream ( ) ;
RTCP对等连接
媒体记录器