H5实现webrtc流播放

打响第一枪 2024-06-25 16:33:02 阅读 64

h5播放WebRTC,WebRTC(Web Real-Time Communication)是一种基于网页浏览器的开源项目,提供了实时音视频传输、数据共享等功能,现在各大浏览器已经逐渐加大对WebRTC技术的支持,实现WebRTC的视频推流,播放WebRTC流。

一、初始化连接

1、初始化WebRTC连接。主要涉及到获取本地媒体流、设置ICE服务器配置、创建PeerConnection对象等

this.pc = new RTCPeerConnection(null);

this.pc.ontrack = (event) => {

this._mediaElement['srcObject'] = event.streams[0];

};

this.pc.addTransceiver('audio', { direction: 'recvonly'});

this.pc.addTransceiver('video', { direction: 'recvonly'});

this.sendChannel = this.pc.createDataChannel('keepalive');

this.sendChannel.onclose = this.onChannelClose.bind(this);

this.sendChannel.onopen = this.onChannelOpen.bind(this);

this.sendChannel.onmessage = this.onChannelMessage.bind(this);

this.pc.createOffer({

offerToReceiveVideo: !0,

offerToReceiveAudio: !0

}).then((offer) => {

return this.pc.setLocalDescription(offer).then(() => {

return offer;

});

}).then((offer) => {

return new Promise((resolve, reject) => {

this.HttpPost(url, window.btoa(offer.sdp)).then((res) => {

resolve(res);

}, function (rej) {

reject(rej);

});

});

}).then((answerSdp) => {

return this.pc.setRemoteDescription(new RTCSessionDescription({

type: 'answer',

sdp: window.atob(answerSdp)

}));

}).then(() => {

this._isLoad = true;

}).catch((reason) => {

throw reason;

});

2、ontrack回调中将媒体播放地址,绑定到video上。

3、createOffer方法,这个方法返回本地会话描述。

4、setLocalDescription方法。

5、需要先与服务端建立一个连接。HttpPost(),发送:offer.sdp 到推流端,服务端收到offer.sdp,再返回回来。

HttpPost(url, data) {

return new Promise((resolve, reject) => {

var xhr = new XMLHttpRequest();

xhr.onreadystatechange = () => {

if (xhr.readyState === 4 && (xhr.status >= 200 && xhr.status < 300)) {

var respone = xhr.responseText;

xhr.onreadystatechange = new Function;

xhr = null;

resolve(respone);

}

};

xhr.open('POST', url.replace('webrtc', 'http'), true);

xhr.send(data);

});

}

6、收到应答返回的offer.sdp, 设置为你的远端连接。

this.pc.setRemoteDescription(new RTCSessionDescription({

type: 'answer',

sdp: window.atob(answerSdp)

}));

7、监听 sendChannel.onopen 连接是否建立成功。

8、前端播放的过程中需要与服务器通信保持连接,可以 sendChannel.send(msg)来保持持续拉流 。

9、服务器推流,前端开始播放。

二、完整代码

import PlayerEvents from '../FlvPlayer/flv.js/player/player-events';

import EventEmitter from 'events';

import * as common from '../common/common';

export default class WebRtcPlayer {

constructor(url, options) {

this.TAG = 'WebRtcPlayer';

this._type = 'WebRtcPlayer';

this._emitter = new EventEmitter();

// this.options = options || {};

this.pc = null;

this._mediaElement = null;

this.url = url;

this.autoplay = !!options.autoplay || false;

this.typeCallback = options.typeCallback;

this.statusCallback = options.statusCallback;

if (!url.match(/^webrtc?:\/\/||^webrtcs?:\/\//)) {

throw ('JSWebrtc just work with webrtc');

}

this.timer = null;

this.sendChannel = null;

this._mediaInfo = null;

this._statisticsInfo = null;

this.isPlaying = false;

this._isLoad = false;

this._isDestroy = false;

this.e = {

onvLoadedMetadata: this._onvLoadedMetadata.bind(this),

onvSeeking: this._onvSeeking.bind(this),

onvCanPlay: this._onvCanPlay.bind(this),

onvPlay: this._onvPlay.bind(this),

onvStalled: this._onvStalled.bind(this),

onvProgress: this._onvProgress.bind(this),

onvWaiting: this._onvWaiting.bind(this),

onvPlaying: this._onvPlaying.bind(this)

};

this.attachMediaElement(options.mediaElement, url);

}

on(event, listener) {

if (event === PlayerEvents.MEDIA_INFO) {

if (this._mediaInfo != null) {

Promise.resolve().then(() => {

this._emitter.emit(PlayerEvents.MEDIA_INFO, this.mediaInfo);

});

}

} else if (event === PlayerEvents.STATISTICS_INFO) {

if (this._statisticsInfo != null) {

Promise.resolve().then(() => {

this._emitter.emit(PlayerEvents.STATISTICS_INFO, this.statisticsInfo);

});

}

}

this._emitter.addListener(event, listener);

}

off(event, listener) {

this._emitter.removeListener(event, listener);

}

attachMediaElement(mediaElement, url) {

this._mediaElement = mediaElement;

this._mediaElement = mediaElement;

mediaElement.addEventListener('loadedmetadata', this.e.onvLoadedMetadata);

mediaElement.addEventListener('seeking', this.e.onvSeeking);

mediaElement.addEventListener('canplay', this.e.onvCanPlay);

mediaElement.addEventListener('play', this.e.onvPlay);

mediaElement.addEventListener('stalled', this.e.onvStalled);

mediaElement.addEventListener('progress', this.e.onvProgress);

mediaElement.addEventListener('waiting', this.e.onvWaiting);

mediaElement.addEventListener('playing', this.e.onvPlaying);

this.typeCallback(!false);

this.load(url);

}

load(url) {

this.statusCallback(0);

if (this.pc) {

this.pc.close();

}

this.pc = new RTCPeerConnection(null);

this.pc.ontrack = (event) => {

this._mediaElement['srcObject'] = event.streams[0];

};

this.pc.addTransceiver('audio', { direction: 'recvonly'});

this.pc.addTransceiver('video', { direction: 'recvonly'});

this.sendChannel = this.pc.createDataChannel('keepalive');

this.sendChannel.onclose = this.onChannelClose.bind(this);

this.sendChannel.onopen = this.onChannelOpen.bind(this);

this.sendChannel.onmessage = this.onChannelMessage.bind(this);

this.pc.createOffer({

offerToReceiveVideo: !0,

offerToReceiveAudio: !0

}).then((offer) => {

return this.pc.setLocalDescription(offer).then(() => {

return offer;

});

}).then((offer) => {

return new Promise((resolve, reject) => {

this.HttpPost(url, window.btoa(offer.sdp)).then((res) => {

resolve(res);

}, function (rej) {

reject(rej);

});

});

}).then((answerSdp) => {

return this.pc.setRemoteDescription(new RTCSessionDescription({

type: 'answer',

sdp: window.atob(answerSdp)

}));

}).then(() => {

this._isLoad = true;

}).catch((reason) => {

throw reason;

});

}

onChannelClose() {

clearInterval(this.timer);

if (!this._isDestroy) {

common.sleep(3000).then(() => {

this.load(this.url);

});

}

}

onChannelOpen() {

this.timer = setInterval((() => {

if (this.sendChannel) {

this.sendChannel.send('ping');

}

}), 3000);

}

onChannelMessage() {

// console.log('onmessage');

}

HttpPost(url, data) {

return new Promise((resolve, reject) => {

var xhr = new XMLHttpRequest();

xhr.onreadystatechange = () => {

if (xhr.readyState === 4 && (xhr.status >= 200 && xhr.status < 300)) {

var respone = xhr.responseText;

xhr.onreadystatechange = new Function;

xhr = null;

resolve(respone);

}

};

xhr.open('POST', url.replace('webrtc', 'http'), true);

xhr.send(data);

});

}

_onvLoadedMetadata() {

}

_onvSeeking() {

}

_onvCanPlay() {

if (this.autoplay) {

this._mediaElement.play();

}

}

_onvPlay() {

this.isPlaying = true;

this.statusCallback(100);

}

_onvStalled() {

}

_onvProgress() {

}

_onvWaiting() {

}

_onvPlaying() {

}

unload() {

}

play() {

if (!this.isPlaying) {

this._mediaElement.play();

this.isPlaying = true;

}

}

pause() {

if (this.isPlaying) {

this._mediaElement.pause();

this.isPlaying = false;

}

}

resume() {

}

openAudio() {

}

closeAudio() {

}

seek() {

}

stop() {

this.destroy();

}

destroy() {

this._isDestroy = true;

if (this.pc) {

this.pc.close();

this.pc = null;

}

if (this._mediaElement) {

this._mediaElement = null;

}

if (this.sendChannel) {

this.sendChannel = null;

}

this.autoplay = null;

this.typeCallback = null;

this.statusCallback = null;

this.isPlaying = false;

this._isLoad = false;

clearInterval(this.timer);

}

get isLoad() {

return this._isLoad;

}

}

player-events.js

const PlayerEvents = {

ERROR: 'error',

LOADING_COMPLETE: 'loading_complete',

RECOVERED_EARLY_EOF: 'recovered_early_eof',

MEDIA_INFO: 'media_info',

METADATA_ARRIVED: 'metadata_arrived',

SCRIPTDATA_ARRIVED: 'scriptdata_arrived',

STATISTICS_INFO: 'statistics_info'

};

export default PlayerEvents;

common.js

export function sleep(ms) {

return new Promise(resolve => setTimeout(resolve, ms));

}



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。