!(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; } })();