You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1709 lines
73 KiB
1709 lines
73 KiB
!(function () {
|
|
/**
|
|
* @param opt
|
|
* container: DOM 容器
|
|
* contextOptions:
|
|
* videoBuffer:
|
|
* forceNoGL:
|
|
* isNotMute:
|
|
* decoder:
|
|
* @constructor
|
|
*/
|
|
function Jessibuca(opt) {
|
|
this._opt = opt;
|
|
|
|
if (typeof opt.container === "string") {
|
|
this._opt.container = document.getElementById(opt.container);
|
|
}
|
|
if (!this._opt.container) {
|
|
throw new Error('Jessibuca need container option');
|
|
return;
|
|
}
|
|
|
|
this._canvasElement = document.createElement("canvas");
|
|
this._canvasElement.style.position = "absolute";
|
|
this._canvasElement.style.top = 0;
|
|
this._canvasElement.style.left = 0;
|
|
this._opt.container.appendChild(this._canvasElement);
|
|
this._container = this._opt.container;
|
|
this._container.style.overflow = "hidden";
|
|
this._containerOldPostion = {
|
|
position: this._container.style.position,
|
|
top: this._container.style.top,
|
|
left: this._container.style.left,
|
|
width: this._container.style.width,
|
|
height: this._container.style.height
|
|
}
|
|
if (this._containerOldPostion.position != "absolute") {
|
|
this._container.style.position = "relative"
|
|
}
|
|
this._opt.videoBuffer = opt.videoBuffer || 0;
|
|
this._opt.text = opt.text || '';
|
|
//
|
|
this._opt.isResize = opt.isResize === false ? opt.isResize : true;
|
|
this._opt.isFullResize = opt.isFullResize === true ? opt.isFullResize : false;
|
|
this._opt.isDebug = opt.debug === true;
|
|
this._opt.timeout = typeof opt.timeout === 'number' ? opt.timeout : 30;
|
|
this._opt.supportDblclickFullscreen = opt.supportDblclickFullscreen === true;
|
|
this._opt.showBandwidth = opt.showBandwidth === true;
|
|
this._opt.operateBtns = Object.assign({
|
|
fullscreen: false,
|
|
screenshot: false,
|
|
play: false,
|
|
audio: false
|
|
}, opt.operateBtns || {});
|
|
this._opt.keepScreenOn = opt.keepScreenOn === true;
|
|
this._opt.rotate = typeof opt.rotate === 'number' ? opt.rotate : 0;
|
|
|
|
if (!opt.forceNoGL) this._initContextGL();
|
|
this._audioContext = new (window.AudioContext || window.webkitAudioContext)();
|
|
this._gainNode = this._audioContext.createGain();
|
|
this._audioEnabled(true);
|
|
if (!opt.isNotMute) {
|
|
this._audioEnabled(false);
|
|
}
|
|
if (this._contextGL) {
|
|
this._initProgram();
|
|
this._initBuffers();
|
|
this._initTextures();
|
|
}
|
|
this._onresize = () => this.resize();
|
|
this._onfullscreenchange = () => this._fullscreenchange();
|
|
window.addEventListener("resize", this._onresize);
|
|
document.addEventListener('fullscreenchange', this._onfullscreenchange);
|
|
this._decoderWorker = new Worker(opt.decoder || 'ff.js')
|
|
var _this = this;
|
|
this._hasLoaded = false;
|
|
this._stats = {
|
|
buf: 0,
|
|
fps: 0,
|
|
abps: '',
|
|
vbps: '',
|
|
ts: ''
|
|
};
|
|
this._audioPlayBuffers = [];
|
|
|
|
if (this._opt.supportDblclickFullscreen) {
|
|
this._canvasElement.addEventListener('dblclick', function () {
|
|
_this.fullscreen = !_this.fullscreen;
|
|
}, false);
|
|
}
|
|
this.onPlay = noop;
|
|
this.onPause = noop;
|
|
this.onRecord = noop;
|
|
this.onFullscreen = noop;
|
|
this.onMute = noop;
|
|
this.onLoad = noop;
|
|
this.onLog = noop;
|
|
this.onError = noop;
|
|
this.onTimeUpdate = noop;
|
|
this.onInitSize = noop;
|
|
this._onMessage();
|
|
this._initDom();
|
|
this._initStatus();
|
|
this._initEventListener();
|
|
this._hideBtns();
|
|
//
|
|
this._initWakeLock();
|
|
this._enableWakeLock();
|
|
};
|
|
|
|
function noop() {
|
|
|
|
}
|
|
|
|
Jessibuca.prototype._initDom = function () {
|
|
var playBase64 = '';
|
|
var pauseBase64 = '';
|
|
var screenshotBase64 = '';
|
|
var fullscreenBase64 = '';
|
|
var minScreenBase64 = '';
|
|
var quietBase64 = '';
|
|
var playAudioBase64 = '';
|
|
var recordBase64 = '';
|
|
var recordingBase64 = '';
|
|
var gifBase64 = '';
|
|
var playBigBase64 = '';
|
|
|
|
function _setStyle(dom, cssObj) {
|
|
Object.keys(cssObj).forEach(function (key) {
|
|
dom.style[key] = cssObj[key];
|
|
})
|
|
}
|
|
|
|
var doms = {};
|
|
|
|
var fragment = document.createDocumentFragment();
|
|
var btnWrap = document.createElement('div');
|
|
var control1 = document.createElement('div');
|
|
var control2 = document.createElement('div');
|
|
var textDom = document.createElement('div');
|
|
var speedDom = document.createElement('div');
|
|
var playDom = document.createElement('div');
|
|
var playBigDom = document.createElement('div');
|
|
var pauseDom = document.createElement('div');
|
|
var screenshotsDom = document.createElement('div');
|
|
var fullscreenDom = document.createElement('div');
|
|
var minScreenDom = document.createElement('div');
|
|
var loadingDom = document.createElement('div');
|
|
var loadingTextDom = document.createElement('div');
|
|
var quietAudioDom = document.createElement('div');
|
|
var playAudioDom = document.createElement('div');
|
|
var recordDom = document.createElement('div');
|
|
var recordingDom = document.createElement('div');
|
|
var bgDom = document.createElement('div');
|
|
|
|
loadingTextDom.innerText = this._opt.loadingText || '';
|
|
textDom.innerText = this._opt.text || '';
|
|
speedDom.innerText = '';
|
|
playDom.title = '播放';
|
|
pauseDom.title = '暂停';
|
|
screenshotsDom.title = '截屏';
|
|
fullscreenDom.title = '全屏';
|
|
minScreenDom.title = '退出全屏';
|
|
quietAudioDom.title = '静音';
|
|
playAudioDom.title = '取消静音';
|
|
recordDom.title = '录制';
|
|
recordingDom.title = '取消录制';
|
|
|
|
var wrapStyle = {
|
|
height: '38px',
|
|
zIndex: 11,
|
|
position: 'absolute',
|
|
left: 0,
|
|
bottom: 0,
|
|
width: '100%',
|
|
background: 'rgba(0,0,0)'
|
|
};
|
|
|
|
var bgStyle = {
|
|
position: 'absolute',
|
|
width: '100%',
|
|
height: '100%',
|
|
};
|
|
|
|
if (this._opt.background) {
|
|
bgStyle = Object.assign({}, bgStyle, {
|
|
backgroundRepeat: "no-repeat",
|
|
backgroundPosition: "center",
|
|
backgroundSize: '100%',
|
|
backgroundImage: "url('" + this._opt.background + "')"
|
|
})
|
|
}
|
|
|
|
//
|
|
var loadingStyle = {
|
|
position: 'absolute',
|
|
width: '100%',
|
|
height: '100%',
|
|
textAlign: 'center',
|
|
color: "#fff",
|
|
display: 'none',
|
|
backgroundImage: "url('" + gifBase64 + "')",
|
|
backgroundRepeat: "no-repeat",
|
|
backgroundPosition: "center",
|
|
backgroundSize: "40px 40px",
|
|
};
|
|
|
|
var playBigStyle = {
|
|
position: 'absolute',
|
|
width: '100%',
|
|
height: '100%',
|
|
display: 'none',
|
|
background: 'rgba(0,0,0,0.4)',
|
|
backgroundImage: "url('" + playBigBase64 + "')",
|
|
backgroundRepeat: "no-repeat",
|
|
backgroundPosition: "center",
|
|
backgroundSize: "48px 48px",
|
|
cursor: "pointer"
|
|
};
|
|
|
|
var loadingTextStyle = {
|
|
position: 'absolute',
|
|
width: "100%",
|
|
top: '60%',
|
|
textAlign: 'center',
|
|
}
|
|
var controlStyle = {
|
|
position: 'absolute',
|
|
top: 0,
|
|
height: '100%',
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
};
|
|
var styleObj = {
|
|
display: 'none',
|
|
position: 'relative',
|
|
fontSize: '13px',
|
|
color: '#fff',
|
|
lineHeight: '20px',
|
|
marginLeft: '5px',
|
|
marginRight: '5px',
|
|
userSelect: 'none'
|
|
};
|
|
var styleObj2 = {
|
|
display: 'none',
|
|
position: 'relative',
|
|
width: '16px',
|
|
height: '16px',
|
|
marginLeft: '8px',
|
|
marginRight: '8px',
|
|
backgroundRepeat: "no-repeat",
|
|
backgroundPosition: "center",
|
|
backgroundSize: '100%',
|
|
cursor: 'pointer',
|
|
};
|
|
_setStyle(bgDom, bgStyle);
|
|
_setStyle(btnWrap, wrapStyle);
|
|
_setStyle(loadingDom, loadingStyle);
|
|
_setStyle(playBigDom, playBigStyle);
|
|
_setStyle(loadingTextDom, loadingTextStyle);
|
|
_setStyle(control1, Object.assign({}, controlStyle, {
|
|
left: 0
|
|
}));
|
|
_setStyle(control2, Object.assign({}, controlStyle, {
|
|
right: 0
|
|
}));
|
|
_setStyle(textDom, styleObj);
|
|
_setStyle(speedDom, styleObj);
|
|
_setStyle(playDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + playBase64 + "')",
|
|
}));
|
|
|
|
_setStyle(pauseDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + pauseBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(screenshotsDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + screenshotBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(fullscreenDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + fullscreenBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(minScreenDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + minScreenBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(quietAudioDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + quietBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(playAudioDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + playAudioBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(recordDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + recordBase64 + "')"
|
|
}));
|
|
|
|
_setStyle(recordingDom, Object.assign({}, styleObj2, {
|
|
backgroundImage: "url('" + recordingBase64 + "')"
|
|
}));
|
|
|
|
loadingDom.appendChild(loadingTextDom);
|
|
if (this._opt.text) {
|
|
control1.appendChild(textDom);
|
|
doms.textDom = textDom;
|
|
}
|
|
if (this._opt.showBandwidth) {
|
|
control1.appendChild(speedDom);
|
|
doms.speedDom = speedDom;
|
|
}
|
|
|
|
// record
|
|
//control2.appendChild(recordingDom);
|
|
//control2.appendChild(recordDom);
|
|
|
|
// screenshots
|
|
if (this._opt.operateBtns.screenshot) {
|
|
control2.appendChild(screenshotsDom);
|
|
doms.screenshotsDom = screenshotsDom;
|
|
}
|
|
|
|
// play stop
|
|
if (this._opt.operateBtns.play) {
|
|
control2.appendChild(playDom);
|
|
control2.appendChild(pauseDom);
|
|
doms.playDom = playDom;
|
|
doms.pauseDom = pauseDom;
|
|
}
|
|
|
|
// audio
|
|
if (this._opt.operateBtns.audio) {
|
|
control2.appendChild(playAudioDom);
|
|
control2.appendChild(quietAudioDom);
|
|
doms.playAudioDom = playAudioDom;
|
|
doms.quietAudioDom = quietAudioDom;
|
|
}
|
|
|
|
// fullscreen
|
|
if (this._opt.operateBtns.fullscreen) {
|
|
control2.appendChild(fullscreenDom);
|
|
control2.appendChild(minScreenDom);
|
|
doms.fullscreenDom = fullscreenDom;
|
|
doms.minScreenDom = minScreenDom;
|
|
}
|
|
|
|
btnWrap.appendChild(control1);
|
|
btnWrap.appendChild(control2);
|
|
|
|
fragment.appendChild(bgDom);
|
|
doms.bgDom = bgDom;
|
|
fragment.appendChild(loadingDom);
|
|
doms.loadingDom = loadingDom;
|
|
if (this._showControl()) {
|
|
fragment.appendChild(btnWrap);
|
|
}
|
|
if (this._opt.operateBtns.play) {
|
|
fragment.appendChild(playBigDom);
|
|
doms.playBigDom = playBigDom;
|
|
}
|
|
this._container.appendChild(fragment);
|
|
this._doms = doms;
|
|
};
|
|
|
|
Jessibuca.prototype._initWakeLock = function () {
|
|
this._wakeLock = null;
|
|
var _this = this;
|
|
var handleWakeLock = () => {
|
|
if (this._wakeLock !== null && "visible" === document.visibilityState) {
|
|
_this._enableWakeLock();
|
|
}
|
|
};
|
|
|
|
document.addEventListener('visibilitychange', handleWakeLock);
|
|
document.addEventListener('fullscreenchange', handleWakeLock);
|
|
};
|
|
|
|
Jessibuca.prototype._enableWakeLock = function () {
|
|
if (this._opt.keepScreenOn) {
|
|
if ("wakeLock" in navigator) {
|
|
var _this = this;
|
|
navigator.wakeLock.request("screen").then((lock) => {
|
|
_this._wakeLock = lock;
|
|
_this._wakeLock.addEventListener('release', function () {
|
|
});
|
|
})
|
|
}
|
|
}
|
|
};
|
|
|
|
Jessibuca.prototype._showControl = function () {
|
|
var result = false;
|
|
|
|
var hasBtnShow = false;
|
|
Object.keys(this._opt.operateBtns).forEach((key) => {
|
|
if (this._opt.operateBtns[key]) {
|
|
hasBtnShow = true;
|
|
}
|
|
});
|
|
|
|
if (this._opt.showBandwidth || this._opt.text || hasBtnShow) {
|
|
result = true;
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
Jessibuca.prototype._onMessage = function () {
|
|
var _this = this;
|
|
this._decoderWorker.onmessage = function (event) {
|
|
var msg = event.data;
|
|
switch (msg.cmd) {
|
|
case "init":
|
|
_this._opt.isDebug && console.log("decoder worker init")
|
|
_this.setBufferTime(_this._opt.videoBuffer);
|
|
if (!_this._hasLoaded) {
|
|
_this._opt.isDebug && console.log("has loaded");
|
|
_this._hasLoaded = true;
|
|
_this.onLoad();
|
|
_this._trigger('load');
|
|
}
|
|
break
|
|
case "initSize":
|
|
_this._canvasElement.width = msg.w;
|
|
_this._canvasElement.height = msg.h;
|
|
_this.onInitSize();
|
|
_this.resize();
|
|
_this._trigger('videoInfo', {w: msg.w, h: msg.h});
|
|
if (_this.isWebGL()) {
|
|
|
|
} else {
|
|
_this._initRGB(msg.w, msg.h)
|
|
}
|
|
break
|
|
case "render":
|
|
if (_this.loading) {
|
|
_this.loading = false;
|
|
_this.playing = true;
|
|
_this._opt.isDebug && console.log("clear check loading timeout");
|
|
_this._clearCheckLoading();
|
|
}
|
|
if (_this.playing) {
|
|
if (_this.isWebGL()) {
|
|
_this._drawNextOutputPictureGL(msg.output);
|
|
} else {
|
|
_this._drawNextOutputPictureRGBA(msg.buffer);
|
|
}
|
|
}
|
|
_this._trigger('timeUpdate', msg.ts);
|
|
_this.onTimeUpdate(msg.ts);
|
|
_this._updateStats({bps: msg.bps, ts: msg.ts});
|
|
_this._checkHeart();
|
|
break
|
|
case "initAudio":
|
|
_this._opt.isDebug && console.log('initAudio');
|
|
_this._initAudioPlay(msg.frameCount, msg.samplerate, msg.channels)
|
|
_this._trigger('audioInfo', {
|
|
numOfChannels: msg.channels, // 声频通道
|
|
length: msg.frameCount, // 帧数
|
|
sampleRate: msg.samplerate // 采样率
|
|
});
|
|
break
|
|
case "playAudio":
|
|
if (_this.playing && !_this.quieting) {
|
|
_this._opt.isDebug && console.log('playAudio,ts', msg.ts);
|
|
_this._playAudio(msg.buffer)
|
|
}
|
|
break
|
|
case "print":
|
|
_this.onLog(msg.text)
|
|
this._trigger('log', msg.text);
|
|
_this._opt.isDebug && console.log(msg.text);
|
|
break
|
|
case "printErr":
|
|
_this.onLog(msg.text);
|
|
this._trigger('log', msg.text);
|
|
_this.onError(msg.text);
|
|
this._trigger('error', msg.text);
|
|
_this._opt.isDebug && console.error(msg.text);
|
|
break;
|
|
case "initAudioPlanar":
|
|
_this._opt.isDebug && console.log('initAudioPlanar');
|
|
_this._initAudioPlanar(msg);
|
|
_this._trigger('audioInfo', {
|
|
numOfChannels: msg.channels, // 声频通道
|
|
length: undefined, // 帧数
|
|
sampleRate: msg.samplerate // 采样率
|
|
});
|
|
break;
|
|
default:
|
|
_this._opt.isDebug && console.log(msg);
|
|
_this[msg.cmd](msg)
|
|
}
|
|
};
|
|
};
|
|
|
|
Jessibuca.prototype._initEventListener = function () {
|
|
var _this = this;
|
|
|
|
this._doms.playDom && this._doms.playDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.play();
|
|
}, false);
|
|
|
|
this._doms.playBigDom && this._doms.playBigDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.play();
|
|
}, false);
|
|
|
|
this._doms.pauseDom && this._doms.pauseDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.pause();
|
|
}, false);
|
|
|
|
// screenshots
|
|
this._doms.screenshotsDom && this._doms.screenshotsDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
var filename = _this._opt.text + '' + _now();
|
|
_this._screenshot(filename);
|
|
}, false);
|
|
//
|
|
this._doms.fullscreenDom && this._doms.fullscreenDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.fullscreen = true;
|
|
}, false);
|
|
//
|
|
this._doms.minScreenDom && this._doms.minScreenDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.fullscreen = false;
|
|
}, false);
|
|
//
|
|
this._doms.recordDom && this._doms.recordDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.recording = true;
|
|
}, false);
|
|
//
|
|
this._doms.recordingDom && this._doms.recordingDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.recording = false;
|
|
}, false);
|
|
|
|
this._doms.quietAudioDom && this._doms.quietAudioDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.cancelMute();
|
|
}, false);
|
|
|
|
this._doms.playAudioDom && this._doms.playAudioDom.addEventListener('click', function (e) {
|
|
e.stopPropagation();
|
|
_this.mute();
|
|
}, false);
|
|
};
|
|
/**
|
|
* set debug
|
|
* @param flag
|
|
*/
|
|
Jessibuca.prototype.setDebug = function (flag) {
|
|
this._opt.isDebug = !!flag;
|
|
};
|
|
/**
|
|
* mute
|
|
*/
|
|
Jessibuca.prototype.mute = function () {
|
|
this._audioEnabled(false);
|
|
this._audioPlayBuffers = [];
|
|
this.quieting = true;
|
|
};
|
|
|
|
/**
|
|
* cancel mute
|
|
*/
|
|
Jessibuca.prototype.cancelMute = function () {
|
|
this._audioEnabled(true);
|
|
this.quieting = false;
|
|
};
|
|
|
|
/**
|
|
* link to cancelMute
|
|
*/
|
|
Jessibuca.prototype.audioResume = function () {
|
|
this.cancelMute();
|
|
};
|
|
|
|
/**
|
|
* 设置旋转角度
|
|
*/
|
|
Jessibuca.prototype.setRotate = function (deg) {
|
|
deg = parseInt(deg, 10)
|
|
const list = [0, 90, 270];
|
|
if (this._opt.rotate === deg || list.indexOf(deg) === -1) {
|
|
return;
|
|
}
|
|
this._opt.rotate = deg;
|
|
this.resize();
|
|
};
|
|
|
|
Jessibuca.prototype._initStatus = function () {
|
|
this._loading = true;
|
|
this.loading = true;
|
|
this._recording = false;
|
|
this.recording = false;
|
|
this._playing = false;
|
|
this.playing = false;
|
|
this._audioPlaying = false;
|
|
this._quieting = this._opt.isNotMute ? false : true;
|
|
this.quieting = this._opt.isNotMute ? false : true;
|
|
this._fullscreen = false;
|
|
this.fullscreen = false;
|
|
}
|
|
|
|
Jessibuca.prototype._initBtns = function () {
|
|
// show
|
|
_domToggle(this._doms.pauseDom, true);
|
|
_domToggle(this._doms.screenshotsDom, true);
|
|
_domToggle(this._doms.fullscreenDom, true);
|
|
_domToggle(this._doms.quietAudioDom, true);
|
|
_domToggle(this._doms.textDom, true);
|
|
_domToggle(this._doms.speedDom, true);
|
|
_domToggle(this._doms.recordDom, true);
|
|
// hide
|
|
_domToggle(this._doms.loadingDom, false);
|
|
_domToggle(this._doms.playDom, false);
|
|
_domToggle(this._doms.playBigDom, false);
|
|
_domToggle(this._doms.bgDom, false);
|
|
};
|
|
|
|
Jessibuca.prototype._hideBtns = function () {
|
|
var _this = this;
|
|
Object.keys(this._doms).forEach(function (dom) {
|
|
if (dom !== 'bgDom') {
|
|
_domToggle(_this._doms[dom], false);
|
|
}
|
|
})
|
|
};
|
|
|
|
function _checkFull() {
|
|
var isFull = document.fullscreenElement || window.webkitFullscreenElement || document.msFullscreenElement;
|
|
if (isFull === undefined) isFull = false;
|
|
return !!isFull;
|
|
}
|
|
|
|
Jessibuca.prototype._updateStats = function (options) {
|
|
options = options || {};
|
|
|
|
if (!this._startBpsTime) {
|
|
this._startBpsTime = _now();
|
|
}
|
|
var _nowTime = _now();
|
|
var timestamp = _nowTime - this._startBpsTime;
|
|
|
|
if (timestamp < 1 * 1000) {
|
|
this._bps += (options.bps || 0);
|
|
this._stats.fps += 1;
|
|
this._stats.vbps += parseInt((options.bps || 0));
|
|
return;
|
|
}
|
|
this._stats.ts = options.ts;
|
|
this._doms.speedDom && (this._doms.speedDom.innerText = _bpsSize(this._bps));
|
|
this._trigger('bps', this._bps);
|
|
this._trigger('stats', this._stats);
|
|
this._trigger('performance', _fpsStatus(this._stats.fps));
|
|
this._bps = 0;
|
|
this._stats.fps = 0;
|
|
this._stats.vbps = 0;
|
|
this._startBpsTime = _nowTime;
|
|
};
|
|
|
|
|
|
Jessibuca.prototype._checkHeart = function () {
|
|
if (this._checkHeartTimeout) {
|
|
clearTimeout(this._checkHeartTimeout);
|
|
this._checkHeartTimeout = null;
|
|
}
|
|
var _this = this;
|
|
this._checkHeartTimeout = setTimeout(function () {
|
|
_this._opt.isDebug && console.log('check heart timeout');
|
|
_this._trigger('timeout');
|
|
_this.recording = false;
|
|
_this.playing = false;
|
|
_this._close();
|
|
}, this._opt.timeout * 1000);
|
|
};
|
|
|
|
Jessibuca.prototype._checkLoading = function () {
|
|
if (this._checkLoadingTimeout) {
|
|
clearTimeout(this._checkLoadingTimeout);
|
|
this._checkLoadingTimeout = null;
|
|
}
|
|
var _this = this;
|
|
this._checkLoadingTimeout = setTimeout(function () {
|
|
_this._opt.isDebug && console.log('check loading timeout');
|
|
_this._trigger('timeout');
|
|
_this.playing = false;
|
|
_this._close();
|
|
_domToggle(_this._doms.loadingDom, false);
|
|
}, this._opt.timeout * 1000);
|
|
};
|
|
|
|
Jessibuca.prototype._clearCheckLoading = function () {
|
|
if (this._checkLoadingTimeout) {
|
|
clearTimeout(this._checkLoadingTimeout);
|
|
this._checkLoadingTimeout = null;
|
|
}
|
|
};
|
|
|
|
Jessibuca.prototype._initCheckVariable = function () {
|
|
this._startBpsTime = '';
|
|
this._bps = 0;
|
|
if (this._checkHeartTimeout) {
|
|
clearTimeout(this._checkHeartTimeout);
|
|
this._checkHeartTimeout = null;
|
|
}
|
|
}
|
|
|
|
Jessibuca.prototype._limitAudioPlayBufferSize = function () {
|
|
if (this._audioPlayBuffers.length > 2) {
|
|
this._audioPlayBuffers.shift();
|
|
}
|
|
};
|
|
|
|
//
|
|
Jessibuca.prototype._initAudioPlanar = function (msg) {
|
|
var channels = msg.channels
|
|
var samplerate = msg.samplerate
|
|
var context = this._audioContext;
|
|
this._audioPlaying = false;
|
|
this._audioPlayBuffers = [];
|
|
if (!context) return false;
|
|
var _this = this
|
|
this._playAudio = function (buffer) {
|
|
// _this._isDebug() && console.log('_initAudioPlanar-_playAudio');
|
|
var frameCount = buffer[0][0].length
|
|
var audioBuffer = context.createBuffer(channels, frameCount * buffer.length, samplerate);
|
|
var copyToCtxBuffer = function (fromBuffer) {
|
|
for (var channel = 0; channel < channels; channel++) {
|
|
var nowBuffering = audioBuffer.getChannelData(channel);
|
|
for (var j = 0; j < buffer.length; j++) {
|
|
for (var i = 0; i < frameCount; i++) {
|
|
nowBuffering[i + j * frameCount] = fromBuffer[j][channel][i]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
var playNextBuffer = function () {
|
|
_this._audioPlaying = false;
|
|
if (_this._audioPlayBuffers.length) {
|
|
playAudio(_this._audioPlayBuffers.shift());
|
|
}
|
|
};
|
|
var playAudio = function (fromBuffer) {
|
|
if (!fromBuffer) return
|
|
if (_this._audioPlaying) {
|
|
_this._limitAudioPlayBufferSize();
|
|
_this._audioPlayBuffers.push(fromBuffer);
|
|
return;
|
|
}
|
|
_this._audioPlaying = true;
|
|
copyToCtxBuffer(fromBuffer);
|
|
var source = context.createBufferSource();
|
|
source.buffer = audioBuffer;
|
|
// _this._isDebug() && console.log('audioBuffer', audioBuffer.duration * 1000)
|
|
source.connect(_this._gainNode);
|
|
_this._gainNode.connect(context.destination);
|
|
source.start();
|
|
};
|
|
_this._playAudio = playAudio
|
|
if (!_this._audioInterval) {
|
|
_this._audioInterval = setInterval(playNextBuffer, audioBuffer.duration * 1000);
|
|
}
|
|
playAudio(buffer)
|
|
};
|
|
}
|
|
|
|
function _unlock(context) {
|
|
context.resume();
|
|
var source = context.createBufferSource();
|
|
source.buffer = context.createBuffer(1, 1, 22050);
|
|
source.connect(context.destination);
|
|
if (source.noteOn)
|
|
source.noteOn(0);
|
|
else
|
|
source.start(0);
|
|
}
|
|
|
|
function _domToggle(dom, toggle) {
|
|
if (dom) {
|
|
dom.style.display = toggle ? 'block' : "none";
|
|
}
|
|
}
|
|
|
|
function _dataURLToFile(dataURL) {
|
|
const arr = dataURL.split(",");
|
|
const bstr = atob(arr[1]);
|
|
const type = arr[0].replace("data:", "").replace(";base64", "")
|
|
let n = bstr.length, u8arr = new Uint8Array(n);
|
|
while (n--) {
|
|
u8arr[n] = bstr.charCodeAt(n);
|
|
}
|
|
return new File([u8arr], 'file', {type});
|
|
}
|
|
|
|
function _downloadImg(content, fileName) {
|
|
const aLink = document.createElement("a");
|
|
aLink.download = fileName;
|
|
aLink.href = URL.createObjectURL(content);
|
|
aLink.click();
|
|
URL.revokeObjectURL(content);
|
|
}
|
|
|
|
function _bpsSize(value) {
|
|
if (null == value || value === '') {
|
|
return "0 KB/S";
|
|
}
|
|
var srcsize = parseFloat(value);
|
|
var size = srcsize / 1024;
|
|
size = size.toFixed(2);
|
|
return size + 'KB/S';
|
|
}
|
|
|
|
function _fpsStatus(fps) {
|
|
var result = 0;
|
|
if (fps >= 24) {
|
|
result = 2;
|
|
} else if (fps >= 15) {
|
|
result = 1;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* set audio
|
|
* @param flag
|
|
*/
|
|
Jessibuca.prototype._audioEnabled = function (flag) {
|
|
if (flag) {
|
|
_unlock(this._audioContext)
|
|
this._audioEnabled = function (flag) {
|
|
if (flag) {
|
|
// 恢复
|
|
this._audioContext.resume();
|
|
|
|
} else {
|
|
// 暂停
|
|
this._audioContext.suspend();
|
|
}
|
|
}
|
|
} else {
|
|
this._audioContext.suspend();
|
|
}
|
|
}
|
|
|
|
Jessibuca.prototype._playAudio = function (data) {
|
|
this._isDebug() && console.log('_playAudio');
|
|
var context = this._audioContext;
|
|
this._audioPlaying = false;
|
|
var isDecoding = false;
|
|
if (!context) return false;
|
|
this._audioPlayBuffers = [];
|
|
var decodeQueue = []
|
|
var _this = this
|
|
var playNextBuffer = function (e) {
|
|
if (_this._audioPlayBuffers.length) {
|
|
playBuffer(_this._audioPlayBuffers.shift())
|
|
}
|
|
};
|
|
var playBuffer = function (buffer) {
|
|
_this._audioPlaying = true;
|
|
var audioBufferSouceNode = context.createBufferSource();
|
|
audioBufferSouceNode.buffer = buffer;
|
|
audioBufferSouceNode.connect(_this._gainNode);
|
|
_this._gainNode.connect(context.destination);
|
|
audioBufferSouceNode.start();
|
|
if (!_this._audioInterval) {
|
|
_this._audioInterval = setInterval(playNextBuffer, buffer.duration * 1000 - 1);
|
|
}
|
|
}
|
|
var decodeAudio = function () {
|
|
if (decodeQueue.length) {
|
|
context.decodeAudioData(decodeQueue.shift(), tryPlay, decodeAudio);
|
|
} else {
|
|
isDecoding = false
|
|
}
|
|
}
|
|
var tryPlay = function (buffer) {
|
|
decodeAudio()
|
|
if (_this._audioPlaying) {
|
|
_this._limitAudioPlayBufferSize();
|
|
_this._audioPlayBuffers.push(buffer);
|
|
} else {
|
|
playBuffer(buffer)
|
|
}
|
|
}
|
|
var playAudio = function (data) {
|
|
_this._isDebug() && console.log('_playAudio-playAudio');
|
|
decodeQueue.push(...data)
|
|
if (!isDecoding) {
|
|
isDecoding = true
|
|
decodeAudio()
|
|
}
|
|
}
|
|
this._playAudio = playAudio
|
|
playAudio(data)
|
|
}
|
|
|
|
Jessibuca.prototype._isDebug = function () {
|
|
return this._opt.isDebug;
|
|
}
|
|
Jessibuca.prototype._initAudioPlay = function (frameCount, samplerate, channels) {
|
|
var context = this._audioContext;
|
|
this._audioPlaying = false;
|
|
this._audioPlayBuffers = [];
|
|
if (!context) return false;
|
|
var _this = this
|
|
var resampled = samplerate < 22050;
|
|
if (resampled) {
|
|
_this._opt.isDebug && console.log("resampled!")
|
|
}
|
|
var audioBuffer = resampled ? context.createBuffer(channels, frameCount << 1, samplerate << 1) : context.createBuffer(channels, frameCount, samplerate);
|
|
var playNextBuffer = function () {
|
|
_this._audioPlaying = false;
|
|
_this._isDebug() && console.log("playNextBuffer:", _this._audioPlayBuffers.length)
|
|
if (_this._audioPlayBuffers.length) {
|
|
playAudio(_this._audioPlayBuffers.shift());
|
|
}
|
|
};
|
|
|
|
var copyToCtxBuffer = channels > 1 ? function (fromBuffer) {
|
|
for (var channel = 0; channel < channels; channel++) {
|
|
var nowBuffering = audioBuffer.getChannelData(channel);
|
|
if (resampled) {
|
|
for (var i = 0; i < frameCount; i++) {
|
|
nowBuffering[i * 2] = nowBuffering[i * 2 + 1] = fromBuffer[i * (channel + 1)] / 32768;
|
|
}
|
|
} else
|
|
for (var i = 0; i < frameCount; i++) {
|
|
nowBuffering[i] = fromBuffer[i * (channel + 1)] / 32768;
|
|
}
|
|
|
|
}
|
|
} : function (fromBuffer) {
|
|
var nowBuffering = audioBuffer.getChannelData(0);
|
|
for (var i = 0; i < nowBuffering.length; i++) {
|
|
nowBuffering[i] = fromBuffer[i] / 32768;
|
|
}
|
|
};
|
|
var playAudio = function (fromBuffer) {
|
|
_this._isDebug() && console.log('_initAudioPlay-playAudio,_audioPlaying', _this._audioPlaying);
|
|
if (_this._audioPlaying) {
|
|
_this._limitAudioPlayBufferSize();
|
|
_this._audioPlayBuffers.push(fromBuffer);
|
|
return;
|
|
}
|
|
_this._audioPlaying = true;
|
|
copyToCtxBuffer(fromBuffer);
|
|
var source = context.createBufferSource();
|
|
source.buffer = audioBuffer;
|
|
source.connect(_this._gainNode);
|
|
_this._gainNode.connect(context.destination);
|
|
if (!_this._audioInterval) {
|
|
_this._audioInterval = setInterval(playNextBuffer, audioBuffer.duration * 1000);
|
|
}
|
|
source.start();
|
|
};
|
|
this._playAudio = playAudio;
|
|
};
|
|
/**
|
|
* Returns true if the canvas supports WebGL
|
|
*/
|
|
Jessibuca.prototype.isWebGL = function () {
|
|
return !!this._contextGL;
|
|
};
|
|
/**
|
|
* set timeout
|
|
* @param time
|
|
*/
|
|
Jessibuca.prototype.setTimeout = function (time) {
|
|
if (typeof time === 'number') {
|
|
this._opt.timeout = Number(time);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @desc 视频缩放模式, 当视频分辨率比例与canvas显示区域比例不同时,缩放效果不同:
|
|
0 视频画面完全填充canvas区域,画面会被拉伸
|
|
1 视频画面做等比缩放后,高或宽对齐canvas区域,画面不被拉伸,但有黑边(默认)
|
|
2 视频画面做等比缩放后,完全填充canvas区域,画面不被拉伸,没有黑边,但画面显示不全
|
|
* @param type
|
|
*
|
|
*/
|
|
Jessibuca.prototype.setScaleMode = function (type) {
|
|
if (type === 0) {
|
|
this._opt.isFullResize = false;
|
|
this._opt.isResize = false;
|
|
} else if (type === 1) {
|
|
this._opt.isFullResize = false;
|
|
this._opt.isResize = true;
|
|
} else if (type === 2) {
|
|
this._opt.isFullResize = true;
|
|
}
|
|
this.resize();
|
|
};
|
|
|
|
/**
|
|
* Create the GL context from the canvas element
|
|
*/
|
|
Jessibuca.prototype._initContextGL = function () {
|
|
var canvas = this._canvasElement;
|
|
var gl = null;
|
|
|
|
var validContextNames = ["webgl", "experimental-webgl", "moz-webgl", "webkit-3d"];
|
|
var nameIndex = 0;
|
|
|
|
while (!gl && nameIndex < validContextNames.length) {
|
|
var contextName = validContextNames[nameIndex];
|
|
|
|
try {
|
|
var contextOptions = {preserveDrawingBuffer: true};
|
|
if (this._opt.contextOptions) {
|
|
contextOptions = Object.assign(contextOptions, this._opt.contextOptions);
|
|
}
|
|
|
|
gl = canvas.getContext(contextName, contextOptions);
|
|
} catch (e) {
|
|
gl = null;
|
|
}
|
|
|
|
if (!gl || typeof gl.getParameter !== "function") {
|
|
gl = null;
|
|
}
|
|
|
|
++nameIndex;
|
|
}
|
|
;
|
|
|
|
this._contextGL = gl;
|
|
};
|
|
|
|
/**
|
|
* Initialize GL shader program
|
|
*/
|
|
Jessibuca.prototype._initProgram = function () {
|
|
var gl = this._contextGL;
|
|
|
|
var vertexShaderScript = [
|
|
'attribute vec4 vertexPos;',
|
|
'attribute vec4 texturePos;',
|
|
'varying vec2 textureCoord;',
|
|
|
|
'void main()',
|
|
'{',
|
|
'gl_Position = vertexPos;',
|
|
'textureCoord = texturePos.xy;',
|
|
'}'
|
|
].join('\n');
|
|
|
|
var fragmentShaderScript = [
|
|
'precision highp float;',
|
|
'varying highp vec2 textureCoord;',
|
|
'uniform sampler2D ySampler;',
|
|
'uniform sampler2D uSampler;',
|
|
'uniform sampler2D vSampler;',
|
|
'const mat4 YUV2RGB = mat4',
|
|
'(',
|
|
'1.1643828125, 0, 1.59602734375, -.87078515625,',
|
|
'1.1643828125, -.39176171875, -.81296875, .52959375,',
|
|
'1.1643828125, 2.017234375, 0, -1.081390625,',
|
|
'0, 0, 0, 1',
|
|
');',
|
|
|
|
'void main(void) {',
|
|
'highp float y = texture2D(ySampler, textureCoord).r;',
|
|
'highp float u = texture2D(uSampler, textureCoord).r;',
|
|
'highp float v = texture2D(vSampler, textureCoord).r;',
|
|
'gl_FragColor = vec4(y, u, v, 1) * YUV2RGB;',
|
|
'}'
|
|
].join('\n');
|
|
|
|
var vertexShader = gl.createShader(gl.VERTEX_SHADER);
|
|
gl.shaderSource(vertexShader, vertexShaderScript);
|
|
gl.compileShader(vertexShader);
|
|
if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {
|
|
this._opt.isDebug && console.log('Vertex shader failed to compile: ' + gl.getShaderInfoLog(vertexShader));
|
|
}
|
|
|
|
var fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);
|
|
gl.shaderSource(fragmentShader, fragmentShaderScript);
|
|
gl.compileShader(fragmentShader);
|
|
if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {
|
|
this._opt.isDebug && console.log('Fragment shader failed to compile: ' + gl.getShaderInfoLog(fragmentShader));
|
|
}
|
|
|
|
var program = gl.createProgram();
|
|
gl.attachShader(program, vertexShader);
|
|
gl.attachShader(program, fragmentShader);
|
|
gl.linkProgram(program);
|
|
if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
|
|
this._opt.isDebug && console.log('Program failed to compile: ' + gl.getProgramInfoLog(program));
|
|
}
|
|
|
|
gl.useProgram(program);
|
|
|
|
this._shaderProgram = program;
|
|
};
|
|
|
|
/**
|
|
* Initialize vertex buffers and attach to shader program
|
|
*/
|
|
Jessibuca.prototype._initBuffers = function () {
|
|
var gl = this._contextGL;
|
|
var program = this._shaderProgram;
|
|
|
|
var vertexPosBuffer = gl.createBuffer();
|
|
gl.bindBuffer(gl.ARRAY_BUFFER, vertexPosBuffer);
|
|
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([1, 1, -1, 1, 1, -1, -1, -1]), gl.STATIC_DRAW);
|
|
|
|
var vertexPosRef = gl.getAttribLocation(program, 'vertexPos');
|
|
gl.enableVertexAttribArray(vertexPosRef);
|
|
gl.vertexAttribPointer(vertexPosRef, 2, gl.FLOAT, false, 0, 0);
|
|
|
|
var texturePosBuffer = gl.createBuffer();
|
|
gl.bindBuffer(gl.ARRAY_BUFFER, texturePosBuffer);
|
|
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([1, 0, 0, 0, 1, 1, 0, 1]), gl.STATIC_DRAW);
|
|
|
|
var texturePosRef = gl.getAttribLocation(program, 'texturePos');
|
|
gl.enableVertexAttribArray(texturePosRef);
|
|
gl.vertexAttribPointer(texturePosRef, 2, gl.FLOAT, false, 0, 0);
|
|
|
|
this._texturePosBuffer = texturePosBuffer;
|
|
};
|
|
|
|
/**
|
|
* Initialize GL textures and attach to shader program
|
|
*/
|
|
Jessibuca.prototype._initTextures = function () {
|
|
var gl = this._contextGL;
|
|
var program = this._shaderProgram;
|
|
|
|
var yTextureRef = this._initTexture();
|
|
var ySamplerRef = gl.getUniformLocation(program, 'ySampler');
|
|
gl.uniform1i(ySamplerRef, 0);
|
|
this._yTextureRef = yTextureRef;
|
|
|
|
var uTextureRef = this._initTexture();
|
|
var uSamplerRef = gl.getUniformLocation(program, 'uSampler');
|
|
gl.uniform1i(uSamplerRef, 1);
|
|
this._uTextureRef = uTextureRef;
|
|
|
|
var vTextureRef = this._initTexture();
|
|
var vSamplerRef = gl.getUniformLocation(program, 'vSampler');
|
|
gl.uniform1i(vSamplerRef, 2);
|
|
this._vTextureRef = vTextureRef;
|
|
};
|
|
|
|
/**
|
|
* Create and configure a single texture
|
|
*/
|
|
Jessibuca.prototype._initTexture = function () {
|
|
var gl = this._contextGL;
|
|
|
|
var textureRef = gl.createTexture();
|
|
gl.bindTexture(gl.TEXTURE_2D, textureRef);
|
|
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
|
|
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
|
|
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
|
|
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
|
|
gl.bindTexture(gl.TEXTURE_2D, null);
|
|
|
|
return textureRef;
|
|
};
|
|
|
|
/**
|
|
* Draw picture data to the canvas.
|
|
* If this object is using WebGL, the data must be an I420 formatted ArrayBuffer,
|
|
* Otherwise, data must be an RGBA formatted ArrayBuffer.
|
|
*/
|
|
Jessibuca.prototype._drawNextOutputPicture = function (data) {
|
|
if (this._contextGL) {
|
|
this._drawNextOutputPictureGL(data);
|
|
} else {
|
|
this._drawNextOutputPictureRGBA(data);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Draw the next output picture using WebGL
|
|
*/
|
|
Jessibuca.prototype._drawNextOutputPictureGL = function (data) {
|
|
var gl = this._contextGL;
|
|
var texturePosBuffer = this._texturePosBuffer;
|
|
var yTextureRef = this._yTextureRef;
|
|
var uTextureRef = this._uTextureRef;
|
|
var vTextureRef = this._vTextureRef;
|
|
var croppingParams = this.croppingParams
|
|
var width = this._canvasElement.width
|
|
var height = this._canvasElement.height
|
|
if (croppingParams) {
|
|
gl.viewport(0, 0, croppingParams.width, croppingParams.height);
|
|
var tTop = croppingParams.top / height;
|
|
var tLeft = croppingParams.left / width;
|
|
var tBottom = croppingParams.height / height;
|
|
var tRight = croppingParams.width / width;
|
|
var texturePosValues = new Float32Array([tRight, tTop, tLeft, tTop, tRight, tBottom, tLeft, tBottom]);
|
|
|
|
gl.bindBuffer(gl.ARRAY_BUFFER, texturePosBuffer);
|
|
gl.bufferData(gl.ARRAY_BUFFER, texturePosValues, gl.DYNAMIC_DRAW);
|
|
} else {
|
|
gl.viewport(0, 0, this._canvasElement.width, this._canvasElement.height);
|
|
}
|
|
gl.activeTexture(gl.TEXTURE0);
|
|
gl.bindTexture(gl.TEXTURE_2D, yTextureRef);
|
|
gl.texImage2D(gl.TEXTURE_2D, 0, gl.LUMINANCE, width, height, 0, gl.LUMINANCE, gl.UNSIGNED_BYTE, data[0]);
|
|
|
|
gl.activeTexture(gl.TEXTURE1);
|
|
gl.bindTexture(gl.TEXTURE_2D, uTextureRef);
|
|
gl.texImage2D(gl.TEXTURE_2D, 0, gl.LUMINANCE, width / 2, height / 2, 0, gl.LUMINANCE, gl.UNSIGNED_BYTE, data[1]);
|
|
|
|
gl.activeTexture(gl.TEXTURE2);
|
|
gl.bindTexture(gl.TEXTURE_2D, vTextureRef);
|
|
gl.texImage2D(gl.TEXTURE_2D, 0, gl.LUMINANCE, width / 2, height / 2, 0, gl.LUMINANCE, gl.UNSIGNED_BYTE, data[2]);
|
|
|
|
gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);
|
|
};
|
|
|
|
/**
|
|
* Draw next output picture using ARGB data on a 2d canvas.
|
|
*/
|
|
Jessibuca.prototype._drawNextOutputPictureRGBA = function (data) {
|
|
this.imageData.data.set(data);
|
|
var croppingParams = this.croppingParams
|
|
if (!croppingParams) {
|
|
this.ctx2d.putImageData(this.imageData, 0, 0);
|
|
} else {
|
|
this.ctx2d.putImageData(this.imageData, -croppingParams.left, -croppingParams.top, 0, 0, croppingParams.width, croppingParams.height);
|
|
}
|
|
};
|
|
Jessibuca.prototype.ctx2d = null;
|
|
Jessibuca.prototype.imageData = null;
|
|
Jessibuca.prototype._initRGB = function (width, height) {
|
|
this.ctx2d = this._canvasElement.getContext('2d');
|
|
this.imageData = this.ctx2d.getImageData(0, 0, width, height);
|
|
this.clear = function () {
|
|
this.ctx2d.clearRect(0, 0, width, height)
|
|
};
|
|
};
|
|
|
|
/**
|
|
*
|
|
*/
|
|
Jessibuca.prototype.pause = function () {
|
|
this._close();
|
|
if (this.loading) {
|
|
_domToggle(this._doms.loadingDom, false);
|
|
}
|
|
this.recording = false;
|
|
this.playing = false;
|
|
};
|
|
|
|
/**
|
|
*
|
|
* @private
|
|
*/
|
|
Jessibuca.prototype._close = function () {
|
|
if (this._audioInterval) {
|
|
clearInterval(this._audioInterval)
|
|
this._audioInterval = null;
|
|
}
|
|
this._audioPlayBuffers = [];
|
|
this._audioPlaying = false;
|
|
delete this._playAudio;
|
|
this._decoderWorker.postMessage({cmd: "close"})
|
|
|
|
if (this._wakeLock) {
|
|
this._wakeLock.release();
|
|
this._wakeLock = null;
|
|
}
|
|
|
|
// this._contextGL.clear(this._contextGL.COLOR_BUFFER_BIT);
|
|
this._initCheckVariable();
|
|
}
|
|
|
|
/**
|
|
* close
|
|
*/
|
|
Jessibuca.prototype.close = function () {
|
|
this._close();
|
|
this.clearView();
|
|
};
|
|
|
|
/**
|
|
* destroy
|
|
* @desc delete worker,
|
|
*/
|
|
Jessibuca.prototype.destroy = function () {
|
|
// destroy
|
|
this._close();
|
|
this._decoderWorker.terminate()
|
|
window.removeEventListener("resize", this._onresize);
|
|
window.removeEventListener('fullscreenchange', this._onfullscreenchange);
|
|
this._initCheckVariable();
|
|
this._clearCheckLoading();
|
|
this._off();
|
|
this._hasLoaded = false;
|
|
// remove dom
|
|
while (this._container.firstChild) {
|
|
this._container.removeChild(this._container.firstChild);
|
|
}
|
|
if (this._wakeLock) {
|
|
this._wakeLock.release();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 清理画布为黑色背景
|
|
* 用于canvas重用进行多个流切换播放时,将上一个画面清理
|
|
* 避免后一个视频播放之前出现前一个视频最后一个画面
|
|
*/
|
|
Jessibuca.prototype.clearView = function () {
|
|
this._contextGL.clear(this._contextGL.COLOR_BUFFER_BIT);
|
|
};
|
|
/**
|
|
* play
|
|
* @param url
|
|
*/
|
|
Jessibuca.prototype.play = function (url) {
|
|
if (!this.playUrl && !url) {
|
|
return;
|
|
}
|
|
var needDelay = false;
|
|
if (url) {
|
|
if (this.playUrl) {
|
|
this._close();
|
|
needDelay = true;
|
|
this.clearView();
|
|
}
|
|
this.loading = true;
|
|
_domToggle(this._doms.bgDom, false);
|
|
this._checkLoading();
|
|
this.playUrl = url;
|
|
} else if (this.playUrl) {
|
|
// retry
|
|
if (this.loading) {
|
|
this._hideBtns();
|
|
_domToggle(this._doms.fullscreenDom, true);
|
|
_domToggle(this._doms.pauseDom, true);
|
|
_domToggle(this._doms.loadingDom, true);
|
|
this._checkLoading();
|
|
} else {
|
|
this.playing = true;
|
|
}
|
|
}
|
|
this._initCheckVariable();
|
|
|
|
if (needDelay) {
|
|
var _this = this;
|
|
setTimeout(function () {
|
|
_this._decoderWorker.postMessage({cmd: "play", url: _this.playUrl, isWebGL: _this.isWebGL()})
|
|
}, 300);
|
|
} else {
|
|
this._decoderWorker.postMessage({cmd: "play", url: this.playUrl, isWebGL: this.isWebGL()})
|
|
}
|
|
};
|
|
/**
|
|
* has loaded
|
|
* @returns {boolean}
|
|
*/
|
|
Jessibuca.prototype.hasLoaded = function () {
|
|
return this._hasLoaded;
|
|
};
|
|
|
|
Object.defineProperty(Jessibuca.prototype, "fullscreen", {
|
|
set(value) {
|
|
if (value) {
|
|
if (!_checkFull()) {
|
|
this._container.requestFullscreen();
|
|
}
|
|
_domToggle(this._doms.minScreenDom, true);
|
|
_domToggle(this._doms.fullscreenDom, false);
|
|
} else {
|
|
if (_checkFull()) {
|
|
document.exitFullscreen();
|
|
}
|
|
_domToggle(this._doms.minScreenDom, false);
|
|
_domToggle(this._doms.fullscreenDom, true);
|
|
}
|
|
|
|
if (this._fullscreen !== value) {
|
|
this.onFullscreen(value);
|
|
this._trigger('fullscreen', value);
|
|
}
|
|
this._fullscreen = value;
|
|
},
|
|
get() {
|
|
return this._fullscreen;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(Jessibuca.prototype, 'playing', {
|
|
set(value) {
|
|
if (value) {
|
|
_domToggle(this._doms.playBigDom, false);
|
|
_domToggle(this._doms.playDom, false);
|
|
_domToggle(this._doms.pauseDom, true);
|
|
|
|
_domToggle(this._doms.screenshotsDom, true);
|
|
_domToggle(this._doms.recordDom, true);
|
|
if (this._quieting) {
|
|
_domToggle(this._doms.quietAudioDom, true);
|
|
_domToggle(this._doms.playAudioDom, false);
|
|
} else {
|
|
_domToggle(this._doms.quietAudioDom, false);
|
|
_domToggle(this._doms.playAudioDom, true);
|
|
}
|
|
} else {
|
|
this._doms.speedDom && (this._doms.speedDom.innerText = '');
|
|
if (this.playUrl) {
|
|
_domToggle(this._doms.playDom, true);
|
|
_domToggle(this._doms.playBigDom, true);
|
|
_domToggle(this._doms.pauseDom, false);
|
|
}
|
|
|
|
// 在停止状态下录像,截屏,音量是非激活,只有播放,最大化时可点击
|
|
_domToggle(this._doms.recordDom, false);
|
|
_domToggle(this._doms.recordingDom, false);
|
|
_domToggle(this._doms.screenshotsDom, false);
|
|
_domToggle(this._doms.quietAudioDom, false);
|
|
_domToggle(this._doms.playAudioDom, false);
|
|
}
|
|
|
|
if (this._playing !== value) {
|
|
if (value) {
|
|
this.onPlay();
|
|
this._trigger('play');
|
|
} else {
|
|
this.onPause();
|
|
this._trigger('pause');
|
|
}
|
|
}
|
|
this._playing = value;
|
|
},
|
|
get() {
|
|
return this._playing;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(Jessibuca.prototype, 'recording', {
|
|
set(value) {
|
|
if (value) {
|
|
_domToggle(this._doms.recordDom, false);
|
|
_domToggle(this._doms.recordingDom, true);
|
|
} else {
|
|
_domToggle(this._doms.recordDom, true);
|
|
_domToggle(this._doms.recordingDom, false);
|
|
|
|
}
|
|
if (this._recording !== value) {
|
|
this.onRecord(value);
|
|
this._trigger('record', value);
|
|
this._recording = value;
|
|
}
|
|
},
|
|
get() {
|
|
return this._recording;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(Jessibuca.prototype, 'quieting', {
|
|
set(value) {
|
|
if (value) {
|
|
_domToggle(this._doms.quietAudioDom, true);
|
|
_domToggle(this._doms.playAudioDom, false);
|
|
} else {
|
|
_domToggle(this._doms.quietAudioDom, false);
|
|
_domToggle(this._doms.playAudioDom, true);
|
|
}
|
|
if (this._quieting !== value) {
|
|
this.onMute(value);
|
|
this._trigger('mute', value);
|
|
}
|
|
this._quieting = value;
|
|
},
|
|
get() {
|
|
return this._quieting;
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(Jessibuca.prototype, 'loading', {
|
|
set(value) {
|
|
if (value) {
|
|
this._hideBtns();
|
|
_domToggle(this._doms.fullscreenDom, true);
|
|
_domToggle(this._doms.pauseDom, true);
|
|
_domToggle(this._doms.loadingDom, true);
|
|
} else {
|
|
this._initBtns();
|
|
}
|
|
this._loading = value;
|
|
},
|
|
get() {
|
|
return this._loading;
|
|
}
|
|
});
|
|
|
|
/**
|
|
* resize
|
|
*/
|
|
Jessibuca.prototype.resize = function () {
|
|
var width = this._container.clientWidth;
|
|
var height = this._container.clientHeight;
|
|
if (this._showControl()) {
|
|
height -= 38;
|
|
}
|
|
var resizeWidth = this._canvasElement.width;
|
|
var resizeHeight = this._canvasElement.height;
|
|
var rotate = this._opt.rotate;
|
|
var wScale = width / resizeWidth;
|
|
var hScale = height / resizeHeight;
|
|
var scale = wScale > hScale ? hScale : wScale;
|
|
if (!this._opt.isResize) {
|
|
if (wScale !== hScale) {
|
|
scale = wScale + ',' + hScale;
|
|
}
|
|
}
|
|
//
|
|
if (this._opt.isFullResize) {
|
|
scale = wScale > hScale ? wScale : hScale;
|
|
}
|
|
|
|
let transform = "scale(" + scale + ")";
|
|
|
|
if (rotate) {
|
|
transform += ' rotate(' + rotate + 'deg)'
|
|
}
|
|
|
|
this._opt.isDebug && console.log('wScale', wScale, 'hScale', hScale, 'scale', scale, 'rotate', rotate);
|
|
this._canvasElement.style.transform = transform;
|
|
this._canvasElement.style.left = ((width - resizeWidth) / 2) + "px"
|
|
this._canvasElement.style.top = ((height - resizeHeight) / 2) + "px"
|
|
}
|
|
|
|
Jessibuca.prototype._fullscreenchange = function () {
|
|
this.fullscreen = _checkFull();
|
|
}
|
|
|
|
/**
|
|
* change buffer
|
|
* @param buffer
|
|
*/
|
|
Jessibuca.prototype.changeBuffer = function (buffer) {
|
|
this._stats.buf = Number(buffer) * 1000;
|
|
this._decoderWorker.postMessage({cmd: "setVideoBuffer", time: Number(buffer)});
|
|
};
|
|
/**
|
|
* 设置最大缓冲时长,单位秒,播放器会自动消除延迟。
|
|
* @param buffer
|
|
*/
|
|
Jessibuca.prototype.setBufferTime = function (buffer) {
|
|
this.changeBuffer(buffer);
|
|
};
|
|
|
|
/**
|
|
* 设置音量大小,取值0.0 — 1.0
|
|
* 当为0.0时,完全无声
|
|
* 当为1.0时,最大音量,默认值
|
|
* @param volume
|
|
*/
|
|
Jessibuca.prototype.setVolume = function (volume) {
|
|
if (this._gainNode) {
|
|
volume = parseFloat(volume);
|
|
if (isNaN(volume)) {
|
|
return;
|
|
}
|
|
this._isDebug() && console.log('set volume:', volume);
|
|
this._gainNode.gain.setValueAtTime(volume, this._audioContext.currentTime);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 开启屏幕常亮, 在play前调用
|
|
* 在手机浏览器上, canvas标签渲染视频并不会像video标签那样保持屏幕常亮
|
|
* H5目前在chrome\edge 84, android chrome 84及以上有原生亮屏API, 需要是https页面
|
|
* 其余平台为模拟实现,此时为兼容实现,并不保证所有浏览器都支持
|
|
*/
|
|
Jessibuca.prototype.setKeepScreenOn = function () {
|
|
this._opt.keepScreenOn = true;
|
|
};
|
|
|
|
|
|
/**
|
|
* set fullscreen
|
|
* @param flag
|
|
*/
|
|
Jessibuca.prototype.setFullscreen = function (flag) {
|
|
var fullscreen = !!flag;
|
|
if (this.fullscreen !== fullscreen) {
|
|
this.fullscreen = fullscreen;
|
|
}
|
|
};
|
|
|
|
function _now() {
|
|
return new Date().getTime();
|
|
}
|
|
|
|
Jessibuca.prototype._screenshot = function (filename, format, quality) {
|
|
filename = filename || _now();
|
|
var formatType = {
|
|
png: 'image/png',
|
|
jpeg: 'image/jpeg',
|
|
webp: 'image/webp'
|
|
};
|
|
var encoderOptions = 0.92;
|
|
|
|
if (typeof quality !== 'undefined') {
|
|
encoderOptions = Number(quality);
|
|
}
|
|
|
|
var dataURL = this._canvasElement.toDataURL(formatType[format] || formatType.png, encoderOptions);
|
|
_downloadImg(_dataURLToFile(dataURL), filename);
|
|
}
|
|
|
|
/**
|
|
* 截图,调用后弹出下载框保存截图
|
|
* @param filename 保存的文件名 默认时间戳
|
|
* @param format 截图的格式,可选png或jpeg或者webp
|
|
* @param quality 可选参数,当格式是jpeg或者webp时,压缩质量,取值0.0 ~ 1.0
|
|
*/
|
|
Jessibuca.prototype.screenshot = function (filename, format, quality) {
|
|
this._screenshot(filename, format, quality);
|
|
};
|
|
|
|
|
|
var eventSplitter = /\s+/;
|
|
|
|
// Execute callbacks
|
|
function _callEach(list, args, context) {
|
|
if (list) {
|
|
for (var i = 0, len = list.length; i < len; i += 1) {
|
|
list[i].apply(context, args);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param events
|
|
* @param callback
|
|
* @returns {Jessibuca}
|
|
*/
|
|
Jessibuca.prototype.on = function (events, callback) {
|
|
var cache, event, list;
|
|
if (!callback) return this;
|
|
cache = this.__events || (this.__events = {});
|
|
events = events.split(eventSplitter);
|
|
while (event = events.shift()) {
|
|
list = cache[event] || (cache[event] = []);
|
|
list.push(callback);
|
|
}
|
|
return this;
|
|
};
|
|
/**
|
|
*
|
|
* @param events
|
|
* @param callback
|
|
* @returns {Jessibuca}
|
|
* @private
|
|
*/
|
|
Jessibuca.prototype._off = function () {
|
|
var cache;
|
|
if (!(cache = this.__events)) return this;
|
|
delete this.__events;
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
*
|
|
* @param events
|
|
* @returns {Jessibuca}
|
|
* @private
|
|
*/
|
|
Jessibuca.prototype._trigger = function (events) {
|
|
var cache, event, all, list, i, len, rest = [], args;
|
|
if (!(cache = this.__events)) return this;
|
|
events = events.split(eventSplitter);
|
|
// Fill up `rest` with the callback arguments. Since we're only copying
|
|
// the tail of `arguments`, a loop is much faster than Array#slice.
|
|
for (i = 1, len = arguments.length; i < len; i++) {
|
|
rest[i - 1] = arguments[i];
|
|
}
|
|
// For each event, walk through the list of callbacks twice, first to
|
|
// trigger the event, then to trigger any `"all"` callbacks.
|
|
while (event = events.shift()) {
|
|
if (list = cache[event]) list = list.slice();
|
|
// Execute event callbacks.
|
|
_callEach(list, rest, this);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
if (typeof define === 'function') {
|
|
define(function () {
|
|
return Jessibuca;
|
|
});
|
|
} else if (typeof exports !== 'undefined') {
|
|
module.exports = Jessibuca;
|
|
} else {
|
|
window.Jessibuca = Jessibuca;
|
|
}
|
|
})();
|