// ==UserScript== // @description 基于 flv.js 的斗鱼HTML5播放器. // @icon https://ojiju7xvu.qnssl.com/d5hp/icon.png // @name 斗鱼HTML5播放器 // @namespace http://imspace.cn/gms // @run-at document_end // @version 0.3.7 // @grant GM_xmlhttpRequest // @match *://*.douyu.com/* // @downloadURL none // ==/UserScript== class GMXMLHttpRequest { constructor () { this.config = { headers: {} } this.xhr = null } open (method, url) { this.config.method = method this.config.url = url } send () { for (let key of Object.keys(this)) { if (key === 'config') continue if (key.substr(0, 2) === 'on') { this.config[key] = this.wrapper(this[key]) } else { this.config[key] = this[key] } } this.xhr = GM_xmlhttpRequest(this.config) } setRequestHeader (key, value) { this.config.headers[key] = value } abort () { this.xhr && this.xhr.abort() } wrapper (func) { return e => { e.target = this.xhr if (e.response) { e.target.response = e.response } func(e) } } get status () { return this.xhr ? this.xhr.status : 0 } get readyState () { return this.xhr ? this.xhr.readyState : 0 } } window.fetch = function (url, config) { let conf = {} Object.assign(conf, config || { method: 'GET' }) conf.url = url conf.data = config ? config.body : null return new Promise((resolve, reject) => { conf.onload = (response) => { if (response.status === 200) { resolve({ json () { return Promise.resolve(JSON.parse(response.responseText)) } }) } else { reject(response) } } GM_xmlhttpRequest(conf) }) } window.XMLHttpRequest = GMXMLHttpRequest window.__space_inject = {script: "(function (global, factory) {\n"+ " typeof exports === 'object' && typeof module !== 'undefined' ? factory() :\n"+ " typeof define === 'function' && define.amd ? define(factory) :\n"+ " (factory());\n"+ "}(this, (function () { 'use strict';\n"+ "\n"+ "/*\r\n"+ " * Jsocket - Socket on Javascript\r\n"+ " * Author: Masahiro Chiba \r\n"+ " * Depends:\r\n"+ " * - jQuery: http://jquery.com/\r\n"+ " * - jQuery TOOLS - Flashembed: http://flowplayer.org/tools/flashembed.html\r\n"+ " * SYNOPSIS:\r\n"+ " * JSocket.init('/static/JSocket.swf', function () {\r\n"+ " * socket = new JSocket({\r\n"+ " * connectHandler: connectHandler,\r\n"+ " * dataHandler: dataHandler,\r\n"+ " * closeHandler: closeHandler,\r\n"+ " * errorHandler: errorHandler\r\n"+ " * });\r\n"+ " * socket.connect(location.hostname, location.port || 80);\r\n"+ " * });\r\n"+ " * function connectHandler() {\r\n"+ " * socket.writeFlush(\"GET / HTTP/1.0\\x0D\\x0A\");\r\n"+ " * socket.write(\"Host: \" + location.hostname + \"\\x0D\\x0A\\x0D\\x0A\");\r\n"+ " * socket.flush();\r\n"+ " * }\r\n"+ " * function dataHandler(data) {\r\n"+ " * alert(data);\r\n"+ " * socket.close();\r\n"+ " * }\r\n"+ " * function closeHandler() {\r\n"+ " * alert('lost connection')\r\n"+ " * }\r\n"+ " * function errorHandler(errorstr) {\r\n"+ " * alert(errorstr);\r\n"+ " * }\r\n"+ " * \r\n"+ " * */\r\n"+ "function JSocket() {\r\n"+ " this.initialize.apply(this, arguments);\r\n"+ "}\r\n"+ "JSocket.VERSION = '0.05';\r\n"+ "JSocket.init = function(src, swfloadedcb) {\r\n"+ " var flash = ['', '', '', '', '', '', '', '', '', \"\"].join(\"\");\r\n"+ " var div = document.createElement('div');\r\n"+ " div.style.width = '1px';\r\n"+ " div.style.height = '1px';\r\n"+ " document.body.appendChild(div);\r\n"+ " div.innerHTML = flash;\r\n"+ " var api = document.querySelector('#jsocket');\r\n"+ " console.log(div, api);\r\n"+ " JSocket.flashapi = api;\r\n"+ "\r\n"+ " if ( JSocket.flashapi.newsocket ) {\r\n"+ " // for IE(because already construct)\r\n"+ " swfloadedcb();\r\n"+ " }\r\n"+ " else {\r\n"+ " JSocket.swfloadedcb = swfloadedcb;\r\n"+ " }\r\n"+ "};\r\n"+ "JSocket.swfloaded = function() {\r\n"+ " if ( JSocket.swfloadedcb ) {\r\n"+ " setTimeout(JSocket.swfloadedcb, 0);\r\n"+ " }\r\n"+ "};\r\n"+ "JSocket.handlers = new Array();\r\n"+ "JSocket.defaultHandlers = {\r\n"+ " connectHandler: function () {},\r\n"+ " dataHandler: function () {},\r\n"+ " closeHandler: function () {},\r\n"+ " errorHandler: function () {}\r\n"+ "};\r\n"+ "JSocket.connectHandler = function(socid) {\r\n"+ " JSocket.handlers[socid].connectHandler();\r\n"+ "};\r\n"+ "JSocket.dataHandler = function(socid, data) {\r\n"+ " // setTimeout(() => JSocket.handlers[socid].dataHandler(atob(data)), 0)\r\n"+ " try {\r\n"+ " JSocket.handlers[socid].dataHandler(atob(data));\r\n"+ " } catch (e) {\r\n"+ " console.error(e);\r\n"+ " }\r\n"+ "};\r\n"+ "JSocket.closeHandler = function(socid) {\r\n"+ " JSocket.handlers[socid].closeHandler();\r\n"+ "};\r\n"+ "JSocket.errorHandler = function(socid, str) {\r\n"+ " JSocket.handlers[socid].errorHandler(str);\r\n"+ "};\r\n"+ "JSocket.prototype = {\r\n"+ " initialize: function(handlers, newsocketopt) {\r\n"+ " this.socid = JSocket.flashapi.newsocket(newsocketopt);\r\n"+ " JSocket.handlers[this.socid] = handlers;\r\n"+ " },\r\n"+ " connect: function(host, port) {\r\n"+ " JSocket.flashapi.connect(this.socid, host, port);\r\n"+ " },\r\n"+ " write: function(data) {\r\n"+ " JSocket.flashapi.write(this.socid, btoa(data));\r\n"+ " },\r\n"+ " writeFlush: function(data) {\r\n"+ " JSocket.flashapi.writeFlush(this.socid, btoa(data));\r\n"+ " },\r\n"+ " close: function() {\r\n"+ " JSocket.flashapi.close(this.socid);\r\n"+ " },\r\n"+ " flush: function() {\r\n"+ " JSocket.flashapi.flush(this.socid);\r\n"+ " }\r\n"+ "};\r\n"+ "window.JSocket = JSocket;\n"+ "\n"+ "function utf8ToUtf16(utf8_bytes) {\r\n"+ " var unicode_codes = [];\r\n"+ " var unicode_code = 0;\r\n"+ " var num_followed = 0;\r\n"+ " for (var i = 0; i < utf8_bytes.length; ++i) {\r\n"+ " var utf8_byte = utf8_bytes[i];\r\n"+ " if (utf8_byte >= 0x100) {\r\n"+ " // Malformed utf8 byte ignored.\r\n"+ " } else if ((utf8_byte & 0xC0) == 0x80) {\r\n"+ " if (num_followed > 0) {\r\n"+ " unicode_code = (unicode_code << 6) | (utf8_byte & 0x3f);\r\n"+ " num_followed -= 1;\r\n"+ " } else {\r\n"+ " // Malformed UTF-8 sequence ignored.\r\n"+ " }\r\n"+ " } else {\r\n"+ " if (num_followed == 0) {\r\n"+ " unicode_codes.push(unicode_code);\r\n"+ " } else {\r\n"+ " // Malformed UTF-8 sequence ignored.\r\n"+ " }\r\n"+ " if (utf8_byte < 0x80){ // 1-byte\r\n"+ " unicode_code = utf8_byte;\r\n"+ " num_followed = 0;\r\n"+ " } else if ((utf8_byte & 0xE0) == 0xC0) { // 2-byte\r\n"+ " unicode_code = utf8_byte & 0x1f;\r\n"+ " num_followed = 1;\r\n"+ " } else if ((utf8_byte & 0xF0) == 0xE0) { // 3-byte\r\n"+ " unicode_code = utf8_byte & 0x0f;\r\n"+ " num_followed = 2;\r\n"+ " } else if ((utf8_byte & 0xF8) == 0xF0) { // 4-byte\r\n"+ " unicode_code = utf8_byte & 0x07;\r\n"+ " num_followed = 3;\r\n"+ " } else {\r\n"+ " // Malformed UTF-8 sequence ignored.\r\n"+ " }\r\n"+ " }\r\n"+ " }\r\n"+ " if (num_followed == 0) {\r\n"+ " unicode_codes.push(unicode_code);\r\n"+ " } else {\r\n"+ " // Malformed UTF-8 sequence ignored.\r\n"+ " }\r\n"+ " unicode_codes.shift(); // Trim the first element.\r\n"+ "\r\n"+ " var utf16_codes = [];\r\n"+ " for (var i = 0; i < unicode_codes.length; ++i) {\r\n"+ " var unicode_code = unicode_codes[i];\r\n"+ " if (unicode_code < (1 << 16)) {\r\n"+ " utf16_codes.push(unicode_code);\r\n"+ " } else {\r\n"+ " var first = ((unicode_code - (1 << 16)) / (1 << 10)) + 0xD800;\r\n"+ " var second = (unicode_code % (1 << 10)) + 0xDC00;\r\n"+ " utf16_codes.push(first);\r\n"+ " utf16_codes.push(second);\r\n"+ " }\r\n"+ " }\r\n"+ " return utf16_codes;\r\n"+ "}\r\n"+ "\r\n"+ "function utf8_to_ascii( str ) {\r\n"+ " // return unescape(encodeURIComponent(str))\r\n"+ " var char2bytes = function (unicode_code) {\r\n"+ " var utf8_bytes = [];\r\n"+ " if (unicode_code < 0x80) { // 1-byte\r\n"+ " utf8_bytes.push(unicode_code);\r\n"+ " } else if (unicode_code < (1 << 11)) { // 2-byte\r\n"+ " utf8_bytes.push((unicode_code >>> 6) | 0xC0);\r\n"+ " utf8_bytes.push((unicode_code & 0x3F) | 0x80);\r\n"+ " } else if (unicode_code < (1 << 16)) { // 3-byte\r\n"+ " utf8_bytes.push((unicode_code >>> 12) | 0xE0);\r\n"+ " utf8_bytes.push(((unicode_code >> 6) & 0x3f) | 0x80);\r\n"+ " utf8_bytes.push((unicode_code & 0x3F) | 0x80);\r\n"+ " } else if (unicode_code < (1 << 21)) { // 4-byte\r\n"+ " utf8_bytes.push((unicode_code >>> 18) | 0xF0);\r\n"+ " utf8_bytes.push(((unicode_code >> 12) & 0x3F) | 0x80);\r\n"+ " utf8_bytes.push(((unicode_code >> 6) & 0x3F) | 0x80);\r\n"+ " utf8_bytes.push((unicode_code & 0x3F) | 0x80);\r\n"+ " }\r\n"+ " return utf8_bytes;\r\n"+ " };\r\n"+ " var o = [];\r\n"+ " for (var i = 0; i < str.length; i++) {\r\n"+ " o = o.concat(char2bytes(str.charCodeAt(i)));\r\n"+ " }\r\n"+ " return o.map(function (i) { return String.fromCharCode(i); }).join('')\r\n"+ "}\r\n"+ "function ascii_to_utf8( str ) {\r\n"+ " // return decodeURIComponent(escape(str))\r\n"+ " var bytes = str.split('').map(function (i) { return i.charCodeAt(0); });\r\n"+ " return utf8ToUtf16(bytes).map(function (i) { return String.fromCharCode(i); }).join('')\r\n"+ "}\n"+ "\n"+ "function createCommonjsModule(fn, module) {\n"+ " return module = { exports: {} }, fn(module, module.exports), module.exports;\n"+ "}\n"+ "\n"+ "var crypt = createCommonjsModule(function (module) {\n"+ "(function() {\n"+ " var base64map\n"+ " = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',\n"+ "\n"+ " crypt = {\n"+ " // Bit-wise rotation left\n"+ " rotl: function(n, b) {\n"+ " return (n << b) | (n >>> (32 - b));\n"+ " },\n"+ "\n"+ " // Bit-wise rotation right\n"+ " rotr: function(n, b) {\n"+ " return (n << (32 - b)) | (n >>> b);\n"+ " },\n"+ "\n"+ " // Swap big-endian to little-endian and vice versa\n"+ " endian: function(n) {\n"+ " // If number given, swap endian\n"+ " if (n.constructor == Number) {\n"+ " return crypt.rotl(n, 8) & 0x00FF00FF | crypt.rotl(n, 24) & 0xFF00FF00;\n"+ " }\n"+ "\n"+ " // Else, assume array and swap all items\n"+ " for (var i = 0; i < n.length; i++)\n"+ " { n[i] = crypt.endian(n[i]); }\n"+ " return n;\n"+ " },\n"+ "\n"+ " // Generate an array of any length of random bytes\n"+ " randomBytes: function(n) {\n"+ " for (var bytes = []; n > 0; n--)\n"+ " { bytes.push(Math.floor(Math.random() * 256)); }\n"+ " return bytes;\n"+ " },\n"+ "\n"+ " // Convert a byte array to big-endian 32-bit words\n"+ " bytesToWords: function(bytes) {\n"+ " for (var words = [], i = 0, b = 0; i < bytes.length; i++, b += 8)\n"+ " { words[b >>> 5] |= bytes[i] << (24 - b % 32); }\n"+ " return words;\n"+ " },\n"+ "\n"+ " // Convert big-endian 32-bit words to a byte array\n"+ " wordsToBytes: function(words) {\n"+ " for (var bytes = [], b = 0; b < words.length * 32; b += 8)\n"+ " { bytes.push((words[b >>> 5] >>> (24 - b % 32)) & 0xFF); }\n"+ " return bytes;\n"+ " },\n"+ "\n"+ " // Convert a byte array to a hex string\n"+ " bytesToHex: function(bytes) {\n"+ " for (var hex = [], i = 0; i < bytes.length; i++) {\n"+ " hex.push((bytes[i] >>> 4).toString(16));\n"+ " hex.push((bytes[i] & 0xF).toString(16));\n"+ " }\n"+ " return hex.join('');\n"+ " },\n"+ "\n"+ " // Convert a hex string to a byte array\n"+ " hexToBytes: function(hex) {\n"+ " for (var bytes = [], c = 0; c < hex.length; c += 2)\n"+ " { bytes.push(parseInt(hex.substr(c, 2), 16)); }\n"+ " return bytes;\n"+ " },\n"+ "\n"+ " // Convert a byte array to a base-64 string\n"+ " bytesToBase64: function(bytes) {\n"+ " for (var base64 = [], i = 0; i < bytes.length; i += 3) {\n"+ " var triplet = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2];\n"+ " for (var j = 0; j < 4; j++)\n"+ " { if (i * 8 + j * 6 <= bytes.length * 8)\n"+ " { base64.push(base64map.charAt((triplet >>> 6 * (3 - j)) & 0x3F)); }\n"+ " else\n"+ " { base64.push('='); } }\n"+ " }\n"+ " return base64.join('');\n"+ " },\n"+ "\n"+ " // Convert a base-64 string to a byte array\n"+ " base64ToBytes: function(base64) {\n"+ " // Remove non-base-64 characters\n"+ " base64 = base64.replace(/[^A-Z0-9+\\/]/ig, '');\n"+ "\n"+ " for (var bytes = [], i = 0, imod4 = 0; i < base64.length;\n"+ " imod4 = ++i % 4) {\n"+ " if (imod4 == 0) { continue; }\n"+ " bytes.push(((base64map.indexOf(base64.charAt(i - 1))\n"+ " & (Math.pow(2, -2 * imod4 + 8) - 1)) << (imod4 * 2))\n"+ " | (base64map.indexOf(base64.charAt(i)) >>> (6 - imod4 * 2)));\n"+ " }\n"+ " return bytes;\n"+ " }\n"+ " };\n"+ "\n"+ " module.exports = crypt;\n"+ "})();\n"+ "});\n"+ "\n"+ "/*!\n"+ " * Determine if an object is a Buffer\n"+ " *\n"+ " * @author Feross Aboukhadijeh \n"+ " * @license MIT\n"+ " */\n"+ "\n"+ "// The _isBuffer check is for Safari 5-7 support, because it's missing\n"+ "// Object.prototype.constructor. Remove this eventually\n"+ "var index = function (obj) {\n"+ " return obj != null && (isBuffer(obj) || isSlowBuffer(obj) || !!obj._isBuffer)\n"+ "};\n"+ "\n"+ "function isBuffer (obj) {\n"+ " return !!obj.constructor && typeof obj.constructor.isBuffer === 'function' && obj.constructor.isBuffer(obj)\n"+ "}\n"+ "\n"+ "// For Node v0.10 support. Remove this eventually.\n"+ "function isSlowBuffer (obj) {\n"+ " return typeof obj.readFloatLE === 'function' && typeof obj.slice === 'function' && isBuffer(obj.slice(0, 0))\n"+ "}\n"+ "\n"+ "var charenc = {\n"+ " // UTF-8 encoding\n"+ " utf8: {\n"+ " // Convert a string to a byte array\n"+ " stringToBytes: function(str) {\n"+ " return charenc.bin.stringToBytes(unescape(encodeURIComponent(str)));\n"+ " },\n"+ "\n"+ " // Convert a byte array to a string\n"+ " bytesToString: function(bytes) {\n"+ " return decodeURIComponent(escape(charenc.bin.bytesToString(bytes)));\n"+ " }\n"+ " },\n"+ "\n"+ " // Binary encoding\n"+ " bin: {\n"+ " // Convert a string to a byte array\n"+ " stringToBytes: function(str) {\n"+ " for (var bytes = [], i = 0; i < str.length; i++)\n"+ " { bytes.push(str.charCodeAt(i) & 0xFF); }\n"+ " return bytes;\n"+ " },\n"+ "\n"+ " // Convert a byte array to a string\n"+ " bytesToString: function(bytes) {\n"+ " for (var str = [], i = 0; i < bytes.length; i++)\n"+ " { str.push(String.fromCharCode(bytes[i])); }\n"+ " return str.join('');\n"+ " }\n"+ " }\n"+ "};\n"+ "\n"+ "var charenc_1 = charenc;\n"+ "\n"+ "var md5 = createCommonjsModule(function (module) {\n"+ "(function(){\r\n"+ " var crypt$$1 = crypt,\r\n"+ " utf8 = charenc_1.utf8,\r\n"+ " isBuffer = index,\r\n"+ " bin = charenc_1.bin,\r\n"+ "\r\n"+ " // The core\r\n"+ " md5 = function (message, options) {\r\n"+ " // Convert to byte array\r\n"+ " if (message.constructor == String)\r\n"+ " { if (options && options.encoding === 'binary')\r\n"+ " { message = bin.stringToBytes(message); }\r\n"+ " else\r\n"+ " { message = utf8.stringToBytes(message); } }\r\n"+ " else if (isBuffer(message))\r\n"+ " { message = Array.prototype.slice.call(message, 0); }\r\n"+ " else if (!Array.isArray(message))\r\n"+ " { message = message.toString(); }\r\n"+ " // else, assume byte array already\r\n"+ "\r\n"+ " var m = crypt$$1.bytesToWords(message),\r\n"+ " l = message.length * 8,\r\n"+ " a = 1732584193,\r\n"+ " b = -271733879,\r\n"+ " c = -1732584194,\r\n"+ " d = 271733878;\r\n"+ "\r\n"+ " // Swap endian\r\n"+ " for (var i = 0; i < m.length; i++) {\r\n"+ " m[i] = ((m[i] << 8) | (m[i] >>> 24)) & 0x00FF00FF |\r\n"+ " ((m[i] << 24) | (m[i] >>> 8)) & 0xFF00FF00;\r\n"+ " }\r\n"+ "\r\n"+ " // Padding\r\n"+ " m[l >>> 5] |= 0x80 << (l % 32);\r\n"+ " m[(((l + 64) >>> 9) << 4) + 14] = l;\r\n"+ "\r\n"+ " // Method shortcuts\r\n"+ " var FF = md5._ff,\r\n"+ " GG = md5._gg,\r\n"+ " HH = md5._hh,\r\n"+ " II = md5._ii;\r\n"+ "\r\n"+ " for (var i = 0; i < m.length; i += 16) {\r\n"+ "\r\n"+ " var aa = a,\r\n"+ " bb = b,\r\n"+ " cc = c,\r\n"+ " dd = d;\r\n"+ "\r\n"+ " a = FF(a, b, c, d, m[i+ 0], 7, -680876936);\r\n"+ " d = FF(d, a, b, c, m[i+ 1], 12, -389564586);\r\n"+ " c = FF(c, d, a, b, m[i+ 2], 17, 606105819);\r\n"+ " b = FF(b, c, d, a, m[i+ 3], 22, -1044525330);\r\n"+ " a = FF(a, b, c, d, m[i+ 4], 7, -176418897);\r\n"+ " d = FF(d, a, b, c, m[i+ 5], 12, 1200080426);\r\n"+ " c = FF(c, d, a, b, m[i+ 6], 17, -1473231341);\r\n"+ " b = FF(b, c, d, a, m[i+ 7], 22, -45705983);\r\n"+ " a = FF(a, b, c, d, m[i+ 8], 7, 1770035416);\r\n"+ " d = FF(d, a, b, c, m[i+ 9], 12, -1958414417);\r\n"+ " c = FF(c, d, a, b, m[i+10], 17, -42063);\r\n"+ " b = FF(b, c, d, a, m[i+11], 22, -1990404162);\r\n"+ " a = FF(a, b, c, d, m[i+12], 7, 1804603682);\r\n"+ " d = FF(d, a, b, c, m[i+13], 12, -40341101);\r\n"+ " c = FF(c, d, a, b, m[i+14], 17, -1502002290);\r\n"+ " b = FF(b, c, d, a, m[i+15], 22, 1236535329);\r\n"+ "\r\n"+ " a = GG(a, b, c, d, m[i+ 1], 5, -165796510);\r\n"+ " d = GG(d, a, b, c, m[i+ 6], 9, -1069501632);\r\n"+ " c = GG(c, d, a, b, m[i+11], 14, 643717713);\r\n"+ " b = GG(b, c, d, a, m[i+ 0], 20, -373897302);\r\n"+ " a = GG(a, b, c, d, m[i+ 5], 5, -701558691);\r\n"+ " d = GG(d, a, b, c, m[i+10], 9, 38016083);\r\n"+ " c = GG(c, d, a, b, m[i+15], 14, -660478335);\r\n"+ " b = GG(b, c, d, a, m[i+ 4], 20, -405537848);\r\n"+ " a = GG(a, b, c, d, m[i+ 9], 5, 568446438);\r\n"+ " d = GG(d, a, b, c, m[i+14], 9, -1019803690);\r\n"+ " c = GG(c, d, a, b, m[i+ 3], 14, -187363961);\r\n"+ " b = GG(b, c, d, a, m[i+ 8], 20, 1163531501);\r\n"+ " a = GG(a, b, c, d, m[i+13], 5, -1444681467);\r\n"+ " d = GG(d, a, b, c, m[i+ 2], 9, -51403784);\r\n"+ " c = GG(c, d, a, b, m[i+ 7], 14, 1735328473);\r\n"+ " b = GG(b, c, d, a, m[i+12], 20, -1926607734);\r\n"+ "\r\n"+ " a = HH(a, b, c, d, m[i+ 5], 4, -378558);\r\n"+ " d = HH(d, a, b, c, m[i+ 8], 11, -2022574463);\r\n"+ " c = HH(c, d, a, b, m[i+11], 16, 1839030562);\r\n"+ " b = HH(b, c, d, a, m[i+14], 23, -35309556);\r\n"+ " a = HH(a, b, c, d, m[i+ 1], 4, -1530992060);\r\n"+ " d = HH(d, a, b, c, m[i+ 4], 11, 1272893353);\r\n"+ " c = HH(c, d, a, b, m[i+ 7], 16, -155497632);\r\n"+ " b = HH(b, c, d, a, m[i+10], 23, -1094730640);\r\n"+ " a = HH(a, b, c, d, m[i+13], 4, 681279174);\r\n"+ " d = HH(d, a, b, c, m[i+ 0], 11, -358537222);\r\n"+ " c = HH(c, d, a, b, m[i+ 3], 16, -722521979);\r\n"+ " b = HH(b, c, d, a, m[i+ 6], 23, 76029189);\r\n"+ " a = HH(a, b, c, d, m[i+ 9], 4, -640364487);\r\n"+ " d = HH(d, a, b, c, m[i+12], 11, -421815835);\r\n"+ " c = HH(c, d, a, b, m[i+15], 16, 530742520);\r\n"+ " b = HH(b, c, d, a, m[i+ 2], 23, -995338651);\r\n"+ "\r\n"+ " a = II(a, b, c, d, m[i+ 0], 6, -198630844);\r\n"+ " d = II(d, a, b, c, m[i+ 7], 10, 1126891415);\r\n"+ " c = II(c, d, a, b, m[i+14], 15, -1416354905);\r\n"+ " b = II(b, c, d, a, m[i+ 5], 21, -57434055);\r\n"+ " a = II(a, b, c, d, m[i+12], 6, 1700485571);\r\n"+ " d = II(d, a, b, c, m[i+ 3], 10, -1894986606);\r\n"+ " c = II(c, d, a, b, m[i+10], 15, -1051523);\r\n"+ " b = II(b, c, d, a, m[i+ 1], 21, -2054922799);\r\n"+ " a = II(a, b, c, d, m[i+ 8], 6, 1873313359);\r\n"+ " d = II(d, a, b, c, m[i+15], 10, -30611744);\r\n"+ " c = II(c, d, a, b, m[i+ 6], 15, -1560198380);\r\n"+ " b = II(b, c, d, a, m[i+13], 21, 1309151649);\r\n"+ " a = II(a, b, c, d, m[i+ 4], 6, -145523070);\r\n"+ " d = II(d, a, b, c, m[i+11], 10, -1120210379);\r\n"+ " c = II(c, d, a, b, m[i+ 2], 15, 718787259);\r\n"+ " b = II(b, c, d, a, m[i+ 9], 21, -343485551);\r\n"+ "\r\n"+ " a = (a + aa) >>> 0;\r\n"+ " b = (b + bb) >>> 0;\r\n"+ " c = (c + cc) >>> 0;\r\n"+ " d = (d + dd) >>> 0;\r\n"+ " }\r\n"+ "\r\n"+ " return crypt$$1.endian([a, b, c, d]);\r\n"+ " };\r\n"+ "\r\n"+ " // Auxiliary functions\r\n"+ " md5._ff = function (a, b, c, d, x, s, t) {\r\n"+ " var n = a + (b & c | ~b & d) + (x >>> 0) + t;\r\n"+ " return ((n << s) | (n >>> (32 - s))) + b;\r\n"+ " };\r\n"+ " md5._gg = function (a, b, c, d, x, s, t) {\r\n"+ " var n = a + (b & d | c & ~d) + (x >>> 0) + t;\r\n"+ " return ((n << s) | (n >>> (32 - s))) + b;\r\n"+ " };\r\n"+ " md5._hh = function (a, b, c, d, x, s, t) {\r\n"+ " var n = a + (b ^ c ^ d) + (x >>> 0) + t;\r\n"+ " return ((n << s) | (n >>> (32 - s))) + b;\r\n"+ " };\r\n"+ " md5._ii = function (a, b, c, d, x, s, t) {\r\n"+ " var n = a + (c ^ (b | ~d)) + (x >>> 0) + t;\r\n"+ " return ((n << s) | (n >>> (32 - s))) + b;\r\n"+ " };\r\n"+ "\r\n"+ " // Package private blocksize\r\n"+ " md5._blocksize = 16;\r\n"+ " md5._digestsize = 16;\r\n"+ "\r\n"+ " module.exports = function (message, options) {\r\n"+ " if (message === undefined || message === null)\r\n"+ " { throw new Error('Illegal argument ' + message); }\r\n"+ "\r\n"+ " var digestbytes = crypt$$1.wordsToBytes(md5(message, options));\r\n"+ " return options && options.asBytes ? digestbytes :\r\n"+ " options && options.asString ? bin.bytesToString(digestbytes) :\r\n"+ " crypt$$1.bytesToHex(digestbytes);\r\n"+ " };\r\n"+ "\r\n"+ "})();\n"+ "});\n"+ "\n"+ "var p32 = function (i) { return [i, i / 256, i / 65536, i / 16777216].map(function (i) { return String.fromCharCode(Math.floor(i) % 256); }).join(''); };\r\n"+ "var u32 = function (s) { return s.split('').map(function (i) { return i.charCodeAt(0); }).reduce(function (a, b) { return b * 256 + a; }); };\r\n"+ "\r\n"+ "function ACJ (id, data) {\r\n"+ " if (typeof data == 'object') {\r\n"+ " data = douyuClient.douyuEncode(data);\r\n"+ " }\r\n"+ " try {\r\n"+ " _ACJ_([id, data]);\r\n"+ " } catch (e) {\r\n"+ " console.error(id, data, e);\r\n"+ " }\r\n"+ "}\r\n"+ "var getACF = function (key) {\r\n"+ " try {\r\n"+ " return new RegExp((\"acf_\" + key + \"=(.*?);\")).exec(document.cookie)[1]\r\n"+ " } catch (e) {\r\n"+ " return ''\r\n"+ " }\r\n"+ "};\r\n"+ "\r\n"+ "\r\n"+ "var douyuClient = function douyuClient () {\n"+ " var this$1 = this;\n"+ "\r\n"+ " this.map = {};\r\n"+ " this.buffer = '';\r\n"+ " this.socket = new JSocket({\r\n"+ " connectHandler: function () { return this$1.connectHandler(); },\r\n"+ " dataHandler: function (data) { return this$1.dataHandler(data); },\r\n"+ " closeHandler: function () { return this$1.closeHandler(); },\r\n"+ " errorHandler: function () { return this$1.errorHandler(); }\r\n"+ " });\r\n"+ " // let oldSend = this.send\r\n"+ " // this.send = data => oldSend(data)\r\n"+ "};\r\n"+ "douyuClient.prototype.connectHandler = function connectHandler () {};\r\n"+ "douyuClient.prototype.dataHandler = function dataHandler (data) {\n"+ " var this$1 = this;\n"+ "\r\n"+ " this.buffer += data;\r\n"+ " var buffer = this.buffer;\r\n"+ " var map = this.map;\r\n"+ " while (buffer.length >= 4) {\r\n"+ " var size = u32(buffer.substr(0, 4));\r\n"+ " if (buffer.length >= size) {\r\n"+ " var pkg = '';\r\n"+ " try {\r\n"+ " pkg = ascii_to_utf8(buffer.substr(12, size-8));\r\n"+ " } catch (e) {\r\n"+ " console.log('deocde fail', escape(buffer.substr(12, size-8)));\r\n"+ " }\r\n"+ " this$1.buffer = buffer = buffer.substr(size+4);\r\n"+ " if (pkg.length === 0) { continue }\r\n"+ " try {\r\n"+ " var rawString = pkg;\r\n"+ " var caller = function (func) { return func(pkg, function (data) { return this$1.send(data); }, {\r\n"+ " ACJ: ACJ,\r\n"+ " rawString: rawString,\r\n"+ " decode: douyuClient.douyuDecode,\r\n"+ " encode: douyuClient.douyuEncode\r\n"+ " }); };\r\n"+ " pkg = douyuClient.douyuDecode(pkg);\r\n"+ " if (map) {\r\n"+ " var cb = map[pkg.type];\r\n"+ " if (cb) {\r\n"+ " if (typeof cb == 'string') {\r\n"+ " ACJ(cb, pkg);\r\n"+ " } else {\r\n"+ " caller(map[pkg.type]);\r\n"+ " }\r\n"+ " } else {\r\n"+ " map.default && caller(map.default);\r\n"+ " }\r\n"+ " }\r\n"+ " } catch (e) {\r\n"+ " console.error('call map', e);\r\n"+ " }\r\n"+ " } else {\r\n"+ " break\r\n"+ " }\r\n"+ " }\r\n"+ "};\n"+ "\n"+ "douyuClient.prototype.connect = function connect (ip, port) {\r\n"+ " this.socket.connect(ip, port);\r\n"+ "};\r\n"+ "douyuClient.filterEnc = function filterEnc (s) {\r\n"+ " s = s.toString();\r\n"+ " s = s.replace(/@/g, '@A');\r\n"+ " return s.replace(/\\//g, '@S')\r\n"+ "};\r\n"+ "douyuClient.filterDec = function filterDec (s) {\r\n"+ " s = s.toString();\r\n"+ " s = s.replace(/@S/g, '/');\r\n"+ " return s.replace(/@A/g, '@')\r\n"+ "};\r\n"+ "douyuClient.douyuEncode = function douyuEncode (data) {\r\n"+ " return Object.keys(data).map(function (key) { return (key + \"@=\" + (douyuClient.filterEnc(data[key]))); }).join('/') + '/'\r\n"+ "};\r\n"+ "douyuClient.douyuDecode = function douyuDecode (data) {\r\n"+ " var out = {};\r\n"+ " data.split('/').filter(function (i) { return i.length > 2; }).some(function (i) {\r\n"+ " var e = i.split('@=');\r\n"+ " out[e[0]] = douyuClient.filterDec(e[1]);\r\n"+ " });\r\n"+ " return out\r\n"+ "};\r\n"+ "douyuClient.encode = function encode (data) {\r\n"+ " return douyuClient.douyuEncode(data)\r\n"+ "};\r\n"+ "douyuClient.decode = function decode (data) {\r\n"+ " return douyuClient.douyuDecode(data)\r\n"+ "};\r\n"+ "douyuClient.decodeList = function decodeList (list) {\r\n"+ " return list = list.split('/').filter(function (i) { return i.length > 2; }).map(douyuClient.filterDec).map(douyuClient.douyuDecode)\r\n"+ "};\r\n"+ "douyuClient.prototype.closeHandler = function closeHandler () {\r\n"+ " console.error('lost connection');\r\n"+ "};\r\n"+ "douyuClient.prototype.errorHandler = function errorHandler (errorstr) {\r\n"+ " console.error(errorstr);\r\n"+ "};\r\n"+ "douyuClient.prototype.send = function send (data) {\r\n"+ " var msg = douyuClient.douyuEncode(data) + '\\0';\r\n"+ " msg = utf8_to_ascii(msg);\r\n"+ " msg = p32(msg.length+8) + p32(msg.length+8) + p32(689) + msg;\r\n"+ " this.socket.writeFlush(msg);\r\n"+ "};\r\n"+ "\r\n"+ "var douyuApi = function (roomId) {\r\n"+ " console.log('douyu api', roomId);\r\n"+ " var _room_args = null;\r\n"+ " var blacklist = [];\r\n"+ " var danmuServer = new douyuClient();\r\n"+ " var miscServer = new douyuClient();\r\n"+ "\r\n"+ " var getRoomArgs = function () {\r\n"+ " if (_room_args) { return _room_args }\r\n"+ " if (window.room_args) {\r\n"+ " return window.room_args\r\n"+ " } else {\r\n"+ " return $ROOM.args\r\n"+ " }\r\n"+ " };\r\n"+ " var randServer = function () {\r\n"+ " var servers = JSON.parse(decodeURIComponent(getRoomArgs().server_config));\r\n"+ " var i = Math.floor(Math.random() * servers.length);\r\n"+ " return servers[i]\r\n"+ " };\r\n"+ " var randDanmuServer = function () {\r\n"+ " var ports = [8601, 8602, 12601, 12602];\r\n"+ " var i = Math.floor(Math.random() * ports.length);\r\n"+ " return {\r\n"+ " ip: 'danmu.douyu.com',\r\n"+ " // ip: '211.91.140.131',\r\n"+ " port: ports[i]\r\n"+ " }\r\n"+ " };\r\n"+ "\r\n"+ " var loginreq = function () {\r\n"+ " var rt = Math.round(new Date().getTime() / 1000);\r\n"+ " var devid = getACF('devid'); // md5(Math.random()).toUpperCase()\r\n"+ " var username = getACF('username');\r\n"+ " console.log('username', username, devid);\r\n"+ " return {\r\n"+ " type: 'loginreq',\r\n"+ " username: username,\r\n"+ " ct: 0,\r\n"+ " password: '',\r\n"+ " roomid: roomId,\r\n"+ " devid: devid,\r\n"+ " rt: rt,\r\n"+ " vk: md5((rt + \"r5*^5;}2#${XF[h+;'./.Q'1;,-]f'p[\" + devid)),\r\n"+ " ver: '20150929',\r\n"+ " aver: '2017012111',\r\n"+ " biz: getACF('biz'),\r\n"+ " stk: getACF('stk'),\r\n"+ " ltkid: getACF('ltkid')\r\n"+ " }\r\n"+ " };\r\n"+ " var keepalive = function () {\r\n"+ " return {\r\n"+ " type: 'keeplive',\r\n"+ " tick: Math.round(new Date().getTime() / 1000)\r\n"+ " }\r\n"+ " };\r\n"+ " var reqOnlineGift = function (loginres) {\r\n"+ " return {\r\n"+ " type: 'reqog',\r\n"+ " uid: loginres.userid\r\n"+ " }\r\n"+ " };\r\n"+ " var onchatmsg = function (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ " var encode = ref.encode;\n"+ "\r\n"+ " if (blacklist.includes(data.uid)) {\r\n"+ " console.log('black');\r\n"+ " return\r\n"+ " }\r\n"+ " try {\r\n"+ " window.postMsg({\r\n"+ " type: \"DANMU\",\r\n"+ " data: data\r\n"+ " }, \"*\");\r\n"+ " } catch (e) {\r\n"+ " console.error('wtf', e);\r\n"+ " }\r\n"+ " ACJ('room_data_chat2', data);\r\n"+ " if (window.BarrageReturn) {\r\n"+ " window.BarrageReturn(encode(data));\r\n"+ " }\r\n"+ " };\r\n"+ "\r\n"+ " miscServer.map = {\r\n"+ " chatmsg: onchatmsg,\r\n"+ " qtlr: 'room_data_tasklis',\r\n"+ " initcl: 'room_data_chatinit',\r\n"+ " memberinfores: 'room_data_info',\r\n"+ " ranklist: 'room_data_cqrank',\r\n"+ " rsm: 'room_data_brocast',\r\n"+ " qausrespond: 'data_rank_score',\r\n"+ " resog: function resog (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ "\r\n"+ " // lev@=1/lack_time[t]@=0/dl@=2/\r\n"+ " ACJ('room_data_chest', {\r\n"+ " lev: data.lv,\r\n"+ " lack_time: data.t,\r\n"+ " dl: data.dl\r\n"+ " });\r\n"+ " },\r\n"+ " loginres: function loginres (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ "\r\n"+ " console.log('loginres', data);\r\n"+ " send(reqOnlineGift(data));\r\n"+ " send(keepalive());\r\n"+ " setInterval(function () { return send(keepalive()); }, 30*1000);\r\n"+ " ACJ('room_data_login', data);\r\n"+ " ACJ('room_data_getdid', {\r\n"+ " devid: getACF('devid')\r\n"+ " });\r\n"+ " },\r\n"+ " keeplive: function keeplive (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ " var rawString = ref.rawString;\n"+ "\r\n"+ " ACJ('room_data_userc', data.uc);\r\n"+ " ACJ('room_data_tbredpacket', rawString);\r\n"+ " },\r\n"+ " setmsggroup: function setmsggroup (data, send) {\r\n"+ " danmuServer.send({\r\n"+ " type: 'joingroup',\r\n"+ " rid: data.rid,\r\n"+ " gid: data.gid\r\n"+ " });\r\n"+ " },\r\n"+ " default: function default$1 (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ "\r\n"+ " ACJ('room_data_handler', data);\r\n"+ " console.log('ms', data);\r\n"+ " }\r\n"+ " };\r\n"+ " \r\n"+ " danmuServer.map = {\r\n"+ " chatmsg: onchatmsg,\r\n"+ " chatres: 'room_data_chat2',\r\n"+ " initcl: 'room_data_chatinit',\r\n"+ " dgb: 'room_data_giftbat1',\r\n"+ " dgn: 'room_data_giftbat1',\r\n"+ " spbc: 'room_data_giftbat1',\r\n"+ " uenter: 'room_data_nstip2',\r\n"+ " upgrade: 'room_data_ulgrow',\r\n"+ " newblackres: 'room_data_sys',\r\n"+ " ranklist: 'room_data_cqrank',\r\n"+ " rankup: 'room_data_ulgrow',\r\n"+ " gift_title: 'room_data_schat',\r\n"+ " rss: 'room_data_state',\r\n"+ " srres: 'room_data_wbsharesuc',\r\n"+ " onlinegift: 'room_data_olyw',\r\n"+ " // ggbr: '',\r\n"+ " default: function default$2 (data, send, ref) {\n"+ " var ACJ = ref.ACJ;\n"+ "\r\n"+ " ACJ('room_data_handler', data);\r\n"+ " console.log('dm', data);\r\n"+ " }\r\n"+ " };\r\n"+ "\r\n"+ " miscServer.connectHandler = function () {\r\n"+ " miscServer.send(loginreq());\r\n"+ " };\r\n"+ " danmuServer.connectHandler = function () {\r\n"+ " danmuServer.send(loginreq());\r\n"+ " setInterval(function () { return danmuServer.send(keepalive()); }, 30*1000);\r\n"+ " };\r\n"+ "\r\n"+ " return fetch('/swf_api/get_room_args').then(function (r) { return r.json(); }).then(function (args) {\r\n"+ " _room_args = args;\r\n"+ " }).then(function () {\r\n"+ " var server = randServer();\r\n"+ " miscServer.connect(server.ip, server.port);\r\n"+ " server = randDanmuServer();\r\n"+ " danmuServer.connect(server.ip, server.port);\r\n"+ " var repeatPacket = function (text) { return douyuClient.decode(text); };\r\n"+ " var jsMap = {\r\n"+ " js_rewardList: {\r\n"+ " type: 'qrl',\r\n"+ " rid: roomId\r\n"+ " },\r\n"+ " js_queryTask: {\r\n"+ " type: 'qtlnq'\r\n"+ " },\r\n"+ " js_newQueryTask: {\r\n"+ " type: 'qtlq'\r\n"+ " },\r\n"+ " js_getRankScore: repeatPacket,\r\n"+ " js_sendmsg: function js_sendmsg (msg) {\r\n"+ " msg = douyuClient.decode(msg);\r\n"+ " msg.type = 'chatmessage';\r\n"+ " return msg\r\n"+ " },\r\n"+ " js_giveGift: function js_giveGift (gift) {\r\n"+ " gift = douyuClient.decode(gift);\r\n"+ " if (gift.type === 'dn_s_gf') {\r\n"+ " gift.type = 'sgq';\r\n"+ " gift.bat = 0;\r\n"+ " }\r\n"+ " console.log('giveGift', gift);\r\n"+ " return gift\r\n"+ " },\r\n"+ " js_GetHongbao: repeatPacket,\r\n"+ " js_UserHaveHandle: function js_UserHaveHandle () {},\r\n"+ " js_myblacklist: function js_myblacklist (list) {\r\n"+ " console.log('add blacklist', list);\r\n"+ " blacklist = list.split('|');\r\n"+ " }\r\n"+ " };\r\n"+ "\r\n"+ "\r\n"+ " return {\r\n"+ " hookExe: function hookExe () {\r\n"+ " var api = require('douyu/page/room/base/api');\r\n"+ " var hookd = function hookd () {\n"+ " var args = [], len = arguments.length;\n"+ " while ( len-- ) args[ len ] = arguments[ len ];\n"+ "\r\n"+ " var req = jsMap[args[0]];\r\n"+ " if (req) {\r\n"+ " if (typeof req == 'function') {\r\n"+ " req = req.apply(null, args.slice(1));\r\n"+ " }\r\n"+ " req && miscServer.send(req);\r\n"+ " } else {\r\n"+ " console.log('exe', args);\r\n"+ " try {\r\n"+ " return oldExe.apply(api, args)\r\n"+ " } catch (e) {}\r\n"+ " }\r\n"+ " };\r\n"+ " if (api) {\r\n"+ " var oldExe$1 = api.exe;\r\n"+ " if (oldExe$1 !== hookd) {\r\n"+ " api.exe = hookd;\r\n"+ " }\r\n"+ " } else if (window.thisMovie) {\r\n"+ " window.thisMovie = function () { return new Proxy({}, {\r\n"+ " get: function get (target, key, receiver) {\r\n"+ " return function () {\n"+ " var args = [], len = arguments.length;\n"+ " while ( len-- ) args[ len ] = arguments[ len ];\n"+ "\n"+ " return hookd.apply(null, [key].concat(args));\n"+ " }\r\n"+ " },\r\n"+ " set: function set (target, key, receiver) {\r\n"+ " }\r\n"+ " }); };\r\n"+ " }\r\n"+ " },\r\n"+ " sendDanmu: function sendDanmu (content) {\r\n"+ " miscServer.send({\r\n"+ " col: '0',\r\n"+ " content: content,\r\n"+ " dy: '',\r\n"+ " pid: '',\r\n"+ " sender: '702735', //TODO uid\r\n"+ " type: 'chatmessage'\r\n"+ " });\r\n"+ " },\r\n"+ " serverSend: function serverSend (c) {\r\n"+ " return miscServer.send(c)\r\n"+ " },\r\n"+ " roomId: roomId\r\n"+ " }\r\n"+ " })\r\n"+ "};\n"+ "\n"+ "window.postMsg = window.postMessage;\r\n"+ "function hookFunc (obj, funcName, newFunc) {\r\n"+ " var old = obj[funcName];\r\n"+ " obj[funcName] = function () {\r\n"+ " return newFunc.call(this, old.bind(this), Array.from(arguments))\r\n"+ " };\r\n"+ "}\r\n"+ "function getParam(flash, name) {\r\n"+ " var children = flash.children;\r\n"+ " for (var i=0; i div {\n"+ " float: left;\n"+ " cursor: pointer;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-btn {\n"+ " box-sizing: border-box;\n"+ " display: inline-block;\n"+ " height: 30px;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-mute {\n"+ " float: right;\n"+ " width: 30px;\n"+ " height: 30px;\n"+ " padding: 5px;\n"+ " transition: all .3s ease;\n"+ " background: url(%2FPg0KPCFET0NUWVBFIHN2ZyBQVUJMSUMgIi0vL1czQy8vRFREIFNWRyAxLjEvL0VOIiANCiJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPg0KDQo8c3ZnIHdpZHRoPSIyMCIgaGVpZ2h0PSIyMCIgdmVyc2lvbj0iMS4xIg0KeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBzdHlsZT0ic3Ryb2tlOiAjNWE1YTVhOyBmaWxsOnRyYW5zcGFyZW50OyBzdHJva2Utd2lkdGg6MiI%2BDQogIDxwYXRoIGQ9Ik05LDQgTDUsOCBMMiw4IEwyLDEyIEw1LDEyIEw5LDE2IFoiIHN0eWxlPSJmaWxsOiAjNWE1YTVhIiAvPg0KICA8cGF0aCBkPSJNMTIsMTIgQTMsMyAwIDAgMCAxMiw4IFoiIHN0eWxlPSJmaWxsOiAjNWE1YTVhIiAvPg0KICA8cGF0aCBkPSJNMTIsMiBBIDksOSAwIDAgMSAxMiwxOCIgc3R5bGU9InN0cm9rZS13aWR0aDoyIi8%2BDQo8L3N2Zz4%3D) no-repeat center;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-mute[muted] {\n"+ " background: url(%2FPg0KPCFET0NUWVBFIHN2ZyBQVUJMSUMgIi0vL1czQy8vRFREIFNWRyAxLjEvL0VOIiANCiJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPg0KDQo8c3ZnIHdpZHRoPSIyMCIgaGVpZ2h0PSIyMCIgdmVyc2lvbj0iMS4xIg0KeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBzdHlsZT0ic3Ryb2tlOiAjNWE1YTVhOyBmaWxsOnRyYW5zcGFyZW50OyBzdHJva2Utd2lkdGg6MiI%2BDQogIDxwYXRoIGQ9Ik05LDQgTDUsOCBMMiw4IEwyLDEyIEw1LDEyIEw5LDE2IFoiIHN0eWxlPSJmaWxsOiAjNWE1YTVhIiAvPg0KICA8cGF0aCBkPSJNMTIsMTMgTDE4LDciIC8%2BDQogIDxwYXRoIGQ9Ik0xMiw3IEwxOCwxMyIgLz4NCjwvc3ZnPg%3D%3D) no-repeat center;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume {\n"+ " box-sizing: border-box;\n"+ " margin: 5px;\n"+ " height: 20px;\n"+ " float: right;\n"+ " position: relative;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume .progress {\n"+ " position: absolute;\n"+ " pointer-events: none;\n"+ " top: 9px;\n"+ " left: 0;\n"+ " width: 0;\n"+ " height: 2px;\n"+ " background-color: #4285f4;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"] {\n"+ " cursor: pointer;\n"+ " height: 20px;\n"+ " outline: none;\n"+ " background-color: transparent;\n"+ " -webkit-appearance: none;\n"+ " -moz-appearance: none;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"] .thumb {\n"+ " -webkit-appearance: none;\n"+ " -moz-appearance: none;\n"+ " height: 12px;\n"+ " width: 12px;\n"+ " margin-top: -5px;\n"+ " border-radius: 50%;\n"+ " background-color: #4285f4;\n"+ " position: relative;\n"+ " box-shadow: 0px 0px 0px #000000, 0px 0px 0px #0d0d0d;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"]::-webkit-slider-thumb {\n"+ " -webkit-appearance: none;\n"+ " -moz-appearance: none;\n"+ " height: 12px;\n"+ " width: 12px;\n"+ " margin-top: -5px;\n"+ " border-radius: 50%;\n"+ " background-color: #4285f4;\n"+ " position: relative;\n"+ " box-shadow: 0px 0px 0px #000000, 0px 0px 0px #0d0d0d;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"]::-moz-range-thumb {\n"+ " -webkit-appearance: none;\n"+ " -moz-appearance: none;\n"+ " height: 12px;\n"+ " width: 12px;\n"+ " margin-top: -5px;\n"+ " border-radius: 50%;\n"+ " background-color: #4285f4;\n"+ " position: relative;\n"+ " box-shadow: 0px 0px 0px #000000, 0px 0px 0px #0d0d0d;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"] .track {\n"+ " height: 2px;\n"+ " background-color: #9f9f9f;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"]::-webkit-slider-runnable-track {\n"+ " height: 2px;\n"+ " background-color: #9f9f9f;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-volume > input[type=\"range\"]::-moz-range-track {\n"+ " height: 2px;\n"+ " background-color: #9f9f9f;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-fullpage {\n"+ " float: right;\n"+ " width: 30px;\n"+ " height: 30px;\n"+ " padding: 5px;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-fullpage::before {\n"+ " content: \" \";\n"+ " display: block;\n"+ " width: 20px;\n"+ " height: 16px;\n"+ " border: 2px solid #5a5a5a;\n"+ " box-sizing: border-box;\n"+ " margin-top: 2px;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-fullscreen {\n"+ " float: right;\n"+ " width: 30px;\n"+ " height: 30px;\n"+ " padding: 5px;\n"+ " background: url(%2FPg0KPCFET0NUWVBFIHN2ZyBQVUJMSUMgIi0vL1czQy8vRFREIFNWRyAxLjEvL0VOIiANCiJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPg0KDQo8c3ZnIHdpZHRoPSIyMCIgaGVpZ2h0PSIyMCIgdmVyc2lvbj0iMS4xIg0KeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBzdHlsZT0ic3Ryb2tlOiAjNWE1YTVhOyBmaWxsOnRyYW5zcGFyZW50OyBzdHJva2Utd2lkdGg6MiI%2BDQogIDxwb2x5bGluZSBwb2ludHM9IjEsOCAxLDEgOCwxIiAvPg0KICA8cG9seWxpbmUgcG9pbnRzPSIxOSw4IDE5LDEgMTIsMSIgLz4NCiAgPHBvbHlsaW5lIHBvaW50cz0iMSwxMiAxLDE5IDgsMTkiIC8%2BDQogIDxwb2x5bGluZSBwb2ludHM9IjE5LDEyIDE5LDE5IDEyLDE5IiAvPg0KPC9zdmc%2B);\n"+ " background-repeat: no-repeat;\n"+ " background-position: center;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-switch {\n"+ " float: right;\n"+ " display: inline-block;\n"+ " height: 30px;\n"+ " line-height: 30px;\n"+ " padding: 0 5px;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-tip {\n"+ " float: right;\n"+ " display: inline-block;\n"+ " height: 30px;\n"+ " line-height: 30px;\n"+ " padding: 0 5px;\n"+ " cursor: default;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-reload {\n"+ " width: 30px;\n"+ " height: 30px;\n"+ " padding: 5px;\n"+ " background: url(%2FPg0KPCFET0NUWVBFIHN2ZyBQVUJMSUMgIi0vL1czQy8vRFREIFNWRyAxLjEvL0VOIiANCiJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPg0KDQo8c3ZnIHdpZHRoPSIyMCIgaGVpZ2h0PSIyMCIgdmVyc2lvbj0iMS4xIg0KeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBzdHlsZT0ic3Ryb2tlOiAjNWE1YTVhOyBmaWxsOnRyYW5zcGFyZW50OyBzdHJva2Utd2lkdGg6Mi41Ij4NCiAgPHBhdGggZD0iTSAxOC42OSwxMi4zMyBBOCA4IDAgMSAxIDE4LjY5LDcuNjciIC8%2BDQogIDxwYXRoIGQ9Ik0gMTYsNyBMIDE5LDcgTCAxOSw0IFoiIC8%2BDQo8L3N2Zz4NCg%3D%3D);\n"+ " background-repeat: no-repeat;\n"+ " background-position: center;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-playpause {\n"+ " width: 30px;\n"+ " height: 30px;\n"+ " padding: 5px;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-playpause::before {\n"+ " transition: all .3s ease;\n"+ " border-color: transparent;\n"+ " content: \" \";\n"+ " display: block;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-playpause:not([pause])::before {\n"+ " width: 0;\n"+ " height: 0;\n"+ " border-top: 10px solid transparent;\n"+ " border-left: 20px solid #5a5a5a;\n"+ " border-bottom: 10px solid transparent;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl .danmu-playpause[pause]::before {\n"+ " box-sizing: border-box;\n"+ " width: 15px;\n"+ " height: 20px;\n"+ " margin-left: 2.5px;\n"+ " border-left: 5px solid #5a5a5a;\n"+ " border-right: 5px solid #5a5a5a;\n"+ "}\n"+ ".danmu-wrap .danmu-layout {\n"+ " position: absolute;\n"+ " top: 0;\n"+ " left: 0;\n"+ " right: 0;\n"+ " bottom: 0;\n"+ " pointer-events: none;\n"+ " color: #fff;\n"+ " font-size: 25px;\n"+ " font-family: SimHei, \"Microsoft JhengHei\", Arial, Helvetica, sans-serif;\n"+ " text-shadow: #000000 1px 0px 1px, #000000 0px 1px 1px, #000000 0px -1px 1px, #000000 -1px 0px 1px;\n"+ " line-height: 1.25;\n"+ " font-weight: bold;\n"+ " overflow: hidden;\n"+ " opacity: 0.5;\n"+ "}\n"+ ".danmu-wrap .danmu-layout > div {\n"+ " display: none;\n"+ " position: absolute;\n"+ " white-space: pre;\n"+ "}\n"+ ".danmu-wrap .danmu-layout .danmu-self {\n"+ " outline: 2px solid #fff;\n"+ "}\n"+ ".danmu-wrap[fullpage] {\n"+ " top: 0;\n"+ " left: 0;\n"+ " width: 100%;\n"+ " height: 100%;\n"+ " position: fixed;\n"+ " z-index: 100000;\n"+ " cursor: none;\n"+ "}\n"+ ".danmu-wrap[fullpage] .danmu-input {\n"+ " position: absolute;\n"+ " top: 75%;\n"+ " width: 100%;\n"+ " display: block;\n"+ " transition: all .3s ease;\n"+ " transform: translateY(50px);\n"+ " opacity: 0;\n"+ "}\n"+ ".danmu-wrap[fullpage] .danmu-input > input {\n"+ " outline: 0;\n"+ " box-shadow: 0 0 10px 1px rgba(255, 255, 255, 0.8);\n"+ " width: 300px;\n"+ " margin: 0 auto;\n"+ " display: block;\n"+ " border: 0;\n"+ " background: rgba(255, 255, 255, 0.8);\n"+ " padding: 5px;\n"+ " color: #000;\n"+ " cursor: default;\n"+ "}\n"+ ".danmu-wrap[fullpage] .danmu-input > input::-webkit-input-placeholder {\n"+ " color: #888;\n"+ "}\n"+ ".danmu-wrap[fullpage][inputing] .danmu-input {\n"+ " transform: translateY(0);\n"+ " opacity: 1;\n"+ "}\n"+ ".danmu-wrap[fullpage][inputing] .danmu-input > input {\n"+ " cursor: text;\n"+ "}\n"+ ".danmu-wrap[fullpage] .danmu-video {\n"+ " height: 100%;\n"+ " transition: all .3s ease;\n"+ "}\n"+ ".danmu-wrap[fullpage] .danmu-ctrl {\n"+ " position: absolute;\n"+ " bottom: 0;\n"+ " opacity: 0;\n"+ " transition: all .3s ease;\n"+ "}\n"+ ".danmu-wrap[fullpage][hover] {\n"+ " cursor: default;\n"+ "}\n"+ ".danmu-wrap[fullpage][hover] .danmu-ctrl {\n"+ " cursor: default;\n"+ " opacity: 0.75;\n"+ "}\n"+ "\n"}; (function (global, factory) { typeof exports === 'object' && typeof module !== 'undefined' ? factory() : typeof define === 'function' && define.amd ? define(factory) : (factory()); }(this, (function () { 'use strict'; function hookFetchCode () { var self = this; var convertHeader = function convertHeader(headers) { var out = new Headers(); for (var i = 0, list = Object.keys(headers); i < list.length; i += 1) { var key = list[i]; out.set(key, headers[key]); } return out }; var hideHookStack = function (stack) { return stack.replace(/^\s*at\s.*?hookfetch\.js:\d.*$\n/mg, '') }; var wrapPort = function wrapPort (port) { var curMethod = ''; var curResolve = null; var curReject = null; var stack = new Error().stack; port.onMessage.addListener(function (msg) { if (msg.method === curMethod) { if (msg.err) { // TODO 潜在安全性问题= = var ctor = new Function('return ' + msg.err.name)(); var err = ctor(msg.err.message); err.stack = hideHookStack(stack); // console.log('fetch err', err) curReject(err); } else { curResolve.apply(null, msg.args); } } else { console.error('wtf?'); } }); return function (method, args) { return new Promise(function (resolve, reject) { curMethod = method; curResolve = resolve; curReject = reject; port.postMessage({ method: method, args: args }); }) } }; var bgFetch = function bgFetch() { var args = [], len = arguments.length; while ( len-- ) args[ len ] = arguments[ len ]; var port = wrapPort(chrome.runtime.connect({name: "fetch"})); return port('fetch', args).then(function (r) { console.log(r); var hasReader = false; var requireReader = function (after) { if (hasReader) { return Promise.resolve().then(after) } else { return port('body.getReader').then(function () { return hasReader = true; }).then(after) } }; r.json = function () { return port('json'); }; r.headers = convertHeader(r.headers); r.body = { getReader: function getReader () { return { read: function read () { return requireReader(function () { return port('reader.read'); }).then(function (r) { if (r.done == false) { r.value = new Uint8Array(r.value); } return r }) }, cancel: function cancel () { return requireReader(function () { return port('reader.cancel'); }) } } } }; return r }) }; function hookFetch () { if (fetch !== bgFetch) { fetch = bgFetch; } } var oldBlob = Blob; var newBlob = function newBlob(a, b) { a[0] = "(" + hookFetchCode + ")();" + (a[0]); console.log('new blob', a, b); return new oldBlob(a, b) }; // if(self.document !== undefined) { // if (self.Blob !== newBlob) { // self.Blob = newBlob // } // } hookFetch(); } if (typeof chrome !== 'undefined') { hookFetchCode(); } var commonjsGlobal = typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {}; function commonjsRequire () { throw new Error('Dynamic requires are not currently supported by rollup-plugin-commonjs'); } function unwrapExports (x) { return x && x.__esModule ? x['default'] : x; } function createCommonjsModule(fn, module) { return module = { exports: {} }, fn(module, module.exports), module.exports; } var flv = createCommonjsModule(function (module, exports) { (function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f();}else if(typeof define==="function"&&define.amd){define([],f);}else{var g;if(typeof window!=="undefined"){g=window;}else if(typeof commonjsGlobal!=="undefined"){g=commonjsGlobal;}else if(typeof self!=="undefined"){g=self;}else{g=this;}g.flvjs = f();}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof commonjsRequire=="function"&&commonjsRequire;if(!u&&a){ return a(o,!0); }if(i){ return i(o,!0); }var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r);}return n[o].exports}var i=typeof commonjsRequire=="function"&&commonjsRequire;for(var o=0;o postsJSON values[1] // => commentsJSON return values; }); ``` @class Promise @param {function} resolver Useful for tooling. @constructor */ function Promise(resolver) { this[PROMISE_ID] = nextId(); this._result = this._state = undefined; this._subscribers = []; if (noop !== resolver) { typeof resolver !== 'function' && needsResolver(); this instanceof Promise ? initializePromise(this, resolver) : needsNew(); } } Promise.all = all; Promise.race = race; Promise.resolve = resolve; Promise.reject = reject; Promise._setScheduler = setScheduler; Promise._setAsap = setAsap; Promise._asap = asap; Promise.prototype = { constructor: Promise, /** The primary way of interacting with a promise is through its `then` method, which registers callbacks to receive either a promise's eventual value or the reason why the promise cannot be fulfilled. ```js findUser().then(function(user){ // user is available }, function(reason){ // user is unavailable, and you are given the reason why }); ``` Chaining -------- The return value of `then` is itself a promise. This second, 'downstream' promise is resolved with the return value of the first promise's fulfillment or rejection handler, or rejected if the handler throws an exception. ```js findUser().then(function (user) { return user.name; }, function (reason) { return 'default name'; }).then(function (userName) { // If `findUser` fulfilled, `userName` will be the user's name, otherwise it // will be `'default name'` }); findUser().then(function (user) { throw new Error('Found user, but still unhappy'); }, function (reason) { throw new Error('`findUser` rejected and we're unhappy'); }).then(function (value) { // never reached }, function (reason) { // if `findUser` fulfilled, `reason` will be 'Found user, but still unhappy'. // If `findUser` rejected, `reason` will be '`findUser` rejected and we're unhappy'. }); ``` If the downstream promise does not specify a rejection handler, rejection reasons will be propagated further downstream. ```js findUser().then(function (user) { throw new PedagogicalException('Upstream error'); }).then(function (value) { // never reached }).then(function (value) { // never reached }, function (reason) { // The `PedgagocialException` is propagated all the way down to here }); ``` Assimilation ------------ Sometimes the value you want to propagate to a downstream promise can only be retrieved asynchronously. This can be achieved by returning a promise in the fulfillment or rejection handler. The downstream promise will then be pending until the returned promise is settled. This is called *assimilation*. ```js findUser().then(function (user) { return findCommentsByAuthor(user); }).then(function (comments) { // The user's comments are now available }); ``` If the assimliated promise rejects, then the downstream promise will also reject. ```js findUser().then(function (user) { return findCommentsByAuthor(user); }).then(function (comments) { // If `findCommentsByAuthor` fulfills, we'll have the value here }, function (reason) { // If `findCommentsByAuthor` rejects, we'll have the reason here }); ``` Simple Example -------------- Synchronous Example ```javascript let result; try { result = findResult(); // success } catch(reason) { // failure } ``` Errback Example ```js findResult(function(result, err){ if (err) { // failure } else { // success } }); ``` Promise Example; ```javascript findResult().then(function(result){ // success }, function(reason){ // failure }); ``` Advanced Example -------------- Synchronous Example ```javascript let author, books; try { author = findAuthor(); books = findBooksByAuthor(author); // success } catch(reason) { // failure } ``` Errback Example ```js function foundBooks(books) { } function failure(reason) { } findAuthor(function(author, err){ if (err) { failure(err); // failure } else { try { findBoooksByAuthor(author, function(books, err) { if (err) { failure(err); } else { try { foundBooks(books); } catch(reason) { failure(reason); } } }); } catch(error) { failure(err); } // success } }); ``` Promise Example; ```javascript findAuthor(). then(findBooksByAuthor). then(function(books){ // found books }).catch(function(reason){ // something went wrong }); ``` @method then @param {Function} onFulfilled @param {Function} onRejected Useful for tooling. @return {Promise} */ then: then, /** `catch` is simply sugar for `then(undefined, onRejection)` which makes it the same as the catch block of a try/catch statement. ```js function findAuthor(){ throw new Error('couldn't find that author'); } // synchronous try { findAuthor(); } catch(reason) { // something went wrong } // async with promises findAuthor().catch(function(reason){ // something went wrong }); ``` @method catch @param {Function} onRejection Useful for tooling. @return {Promise} */ 'catch': function _catch(onRejection) { return this.then(null, onRejection); } }; function polyfill() { var local = undefined; if (typeof global !== 'undefined') { local = global; } else if (typeof self !== 'undefined') { local = self; } else { try { local = Function('return this')(); } catch (e) { throw new Error('polyfill failed because global object is unavailable in this environment'); } } var P = local.Promise; if (P) { var promiseToString = null; try { promiseToString = Object.prototype.toString.call(P.resolve()); } catch (e) { // silently ignored } if (promiseToString === '[object Promise]' && !P.cast) { return; } } local.Promise = Promise; } // Strange compat.. Promise.polyfill = polyfill; Promise.Promise = Promise; return Promise; }))); }).call(this,_dereq_('_process'),typeof commonjsGlobal !== "undefined" ? commonjsGlobal : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}); },{"_process":3}],2:[function(_dereq_,module,exports){ // Copyright Joyent, Inc. and other Node contributors. // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to permit // persons to whom the Software is furnished to do so, subject to the // following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE // USE OR OTHER DEALINGS IN THE SOFTWARE. function EventEmitter() { this._events = this._events || {}; this._maxListeners = this._maxListeners || undefined; } module.exports = EventEmitter; // Backwards-compat with node 0.10.x EventEmitter.EventEmitter = EventEmitter; EventEmitter.prototype._events = undefined; EventEmitter.prototype._maxListeners = undefined; // By default EventEmitters will print a warning if more than 10 listeners are // added to it. This is a useful default which helps finding memory leaks. EventEmitter.defaultMaxListeners = 10; // Obviously not all Emitters should be limited to 10. This function allows // that to be increased. Set to zero for unlimited. EventEmitter.prototype.setMaxListeners = function(n) { if (!isNumber(n) || n < 0 || isNaN(n)) { throw TypeError('n must be a positive number'); } this._maxListeners = n; return this; }; EventEmitter.prototype.emit = function(type) { var this$1 = this; var er, handler, len, args, i, listeners; if (!this._events) { this._events = {}; } // If there is no 'error' event listener then throw. if (type === 'error') { if (!this._events.error || (isObject(this._events.error) && !this._events.error.length)) { er = arguments[1]; if (er instanceof Error) { throw er; // Unhandled 'error' event } else { // At least give some kind of context to the user var err = new Error('Uncaught, unspecified "error" event. (' + er + ')'); err.context = er; throw err; } } } handler = this._events[type]; if (isUndefined(handler)) { return false; } if (isFunction(handler)) { switch (arguments.length) { // fast cases case 1: handler.call(this); break; case 2: handler.call(this, arguments[1]); break; case 3: handler.call(this, arguments[1], arguments[2]); break; // slower default: args = Array.prototype.slice.call(arguments, 1); handler.apply(this, args); } } else if (isObject(handler)) { args = Array.prototype.slice.call(arguments, 1); listeners = handler.slice(); len = listeners.length; for (i = 0; i < len; i++) { listeners[i].apply(this$1, args); } } return true; }; EventEmitter.prototype.addListener = function(type, listener) { var m; if (!isFunction(listener)) { throw TypeError('listener must be a function'); } if (!this._events) { this._events = {}; } // To avoid recursion in the case that type === "newListener"! Before // adding it to the listeners, first emit "newListener". if (this._events.newListener) { this.emit('newListener', type, isFunction(listener.listener) ? listener.listener : listener); } if (!this._events[type]) // Optimize the case of one listener. Don't need the extra array object. { this._events[type] = listener; } else if (isObject(this._events[type])) // If we've already got an array, just append. { this._events[type].push(listener); } else // Adding the second element, need to change to array. { this._events[type] = [this._events[type], listener]; } // Check for listener leak if (isObject(this._events[type]) && !this._events[type].warned) { if (!isUndefined(this._maxListeners)) { m = this._maxListeners; } else { m = EventEmitter.defaultMaxListeners; } if (m && m > 0 && this._events[type].length > m) { this._events[type].warned = true; console.error('(node) warning: possible EventEmitter memory ' + 'leak detected. %d listeners added. ' + 'Use emitter.setMaxListeners() to increase limit.', this._events[type].length); if (typeof console.trace === 'function') { // not supported in IE 10 console.trace(); } } } return this; }; EventEmitter.prototype.on = EventEmitter.prototype.addListener; EventEmitter.prototype.once = function(type, listener) { if (!isFunction(listener)) { throw TypeError('listener must be a function'); } var fired = false; function g() { this.removeListener(type, g); if (!fired) { fired = true; listener.apply(this, arguments); } } g.listener = listener; this.on(type, g); return this; }; // emits a 'removeListener' event iff the listener was removed EventEmitter.prototype.removeListener = function(type, listener) { var list, position, length, i; if (!isFunction(listener)) { throw TypeError('listener must be a function'); } if (!this._events || !this._events[type]) { return this; } list = this._events[type]; length = list.length; position = -1; if (list === listener || (isFunction(list.listener) && list.listener === listener)) { delete this._events[type]; if (this._events.removeListener) { this.emit('removeListener', type, listener); } } else if (isObject(list)) { for (i = length; i-- > 0;) { if (list[i] === listener || (list[i].listener && list[i].listener === listener)) { position = i; break; } } if (position < 0) { return this; } if (list.length === 1) { list.length = 0; delete this._events[type]; } else { list.splice(position, 1); } if (this._events.removeListener) { this.emit('removeListener', type, listener); } } return this; }; EventEmitter.prototype.removeAllListeners = function(type) { var this$1 = this; var key, listeners; if (!this._events) { return this; } // not listening for removeListener, no need to emit if (!this._events.removeListener) { if (arguments.length === 0) { this._events = {}; } else if (this._events[type]) { delete this._events[type]; } return this; } // emit removeListener for all listeners on all events if (arguments.length === 0) { for (key in this._events) { if (key === 'removeListener') { continue; } this$1.removeAllListeners(key); } this.removeAllListeners('removeListener'); this._events = {}; return this; } listeners = this._events[type]; if (isFunction(listeners)) { this.removeListener(type, listeners); } else if (listeners) { // LIFO order while (listeners.length) { this$1.removeListener(type, listeners[listeners.length - 1]); } } delete this._events[type]; return this; }; EventEmitter.prototype.listeners = function(type) { var ret; if (!this._events || !this._events[type]) { ret = []; } else if (isFunction(this._events[type])) { ret = [this._events[type]]; } else { ret = this._events[type].slice(); } return ret; }; EventEmitter.prototype.listenerCount = function(type) { if (this._events) { var evlistener = this._events[type]; if (isFunction(evlistener)) { return 1; } else if (evlistener) { return evlistener.length; } } return 0; }; EventEmitter.listenerCount = function(emitter, type) { return emitter.listenerCount(type); }; function isFunction(arg) { return typeof arg === 'function'; } function isNumber(arg) { return typeof arg === 'number'; } function isObject(arg) { return typeof arg === 'object' && arg !== null; } function isUndefined(arg) { return arg === void 0; } },{}],3:[function(_dereq_,module,exports){ // shim for using process in browser var process = module.exports = {}; // cached from whatever global is present so that test runners that stub it // don't break things. But we need to wrap it in a try catch in case it is // wrapped in strict mode code which doesn't define any globals. It's inside a // function because try/catches deoptimize in certain engines. var cachedSetTimeout; var cachedClearTimeout; function defaultSetTimout() { throw new Error('setTimeout has not been defined'); } function defaultClearTimeout () { throw new Error('clearTimeout has not been defined'); } (function () { try { if (typeof setTimeout === 'function') { cachedSetTimeout = setTimeout; } else { cachedSetTimeout = defaultSetTimout; } } catch (e) { cachedSetTimeout = defaultSetTimout; } try { if (typeof clearTimeout === 'function') { cachedClearTimeout = clearTimeout; } else { cachedClearTimeout = defaultClearTimeout; } } catch (e) { cachedClearTimeout = defaultClearTimeout; } } ()); function runTimeout(fun) { if (cachedSetTimeout === setTimeout) { //normal enviroments in sane situations return setTimeout(fun, 0); } // if setTimeout wasn't available but was latter defined if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) { cachedSetTimeout = setTimeout; return setTimeout(fun, 0); } try { // when when somebody has screwed with setTimeout but no I.E. maddness return cachedSetTimeout(fun, 0); } catch(e){ try { // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally return cachedSetTimeout.call(null, fun, 0); } catch(e){ // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error return cachedSetTimeout.call(this, fun, 0); } } } function runClearTimeout(marker) { if (cachedClearTimeout === clearTimeout) { //normal enviroments in sane situations return clearTimeout(marker); } // if clearTimeout wasn't available but was latter defined if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) { cachedClearTimeout = clearTimeout; return clearTimeout(marker); } try { // when when somebody has screwed with setTimeout but no I.E. maddness return cachedClearTimeout(marker); } catch (e){ try { // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally return cachedClearTimeout.call(null, marker); } catch (e){ // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error. // Some versions of I.E. have different rules for clearTimeout vs setTimeout return cachedClearTimeout.call(this, marker); } } } var queue = []; var draining = false; var currentQueue; var queueIndex = -1; function cleanUpNextTick() { if (!draining || !currentQueue) { return; } draining = false; if (currentQueue.length) { queue = currentQueue.concat(queue); } else { queueIndex = -1; } if (queue.length) { drainQueue(); } } function drainQueue() { if (draining) { return; } var timeout = runTimeout(cleanUpNextTick); draining = true; var len = queue.length; while(len) { currentQueue = queue; queue = []; while (++queueIndex < len) { if (currentQueue) { currentQueue[queueIndex].run(); } } queueIndex = -1; len = queue.length; } currentQueue = null; draining = false; runClearTimeout(timeout); } process.nextTick = function (fun) { var arguments$1 = arguments; var args = new Array(arguments.length - 1); if (arguments.length > 1) { for (var i = 1; i < arguments.length; i++) { args[i - 1] = arguments$1[i]; } } queue.push(new Item(fun, args)); if (queue.length === 1 && !draining) { runTimeout(drainQueue); } }; // v8 likes predictible objects function Item(fun, array) { this.fun = fun; this.array = array; } Item.prototype.run = function () { this.fun.apply(null, this.array); }; process.title = 'browser'; process.browser = true; process.env = {}; process.argv = []; process.version = ''; // empty string to avoid regexp issues process.versions = {}; function noop() {} process.on = noop; process.addListener = noop; process.once = noop; process.off = noop; process.removeListener = noop; process.removeAllListeners = noop; process.emit = noop; process.binding = function (name) { throw new Error('process.binding is not supported'); }; process.cwd = function () { return '/' }; process.chdir = function (dir) { throw new Error('process.chdir is not supported'); }; process.umask = function() { return 0; }; },{}],4:[function(_dereq_,module,exports){ var bundleFn = arguments[3]; var sources = arguments[4]; var cache = arguments[5]; var stringify = JSON.stringify; module.exports = function (fn, options) { var wkey; var cacheKeys = Object.keys(cache); for (var i = 0, l = cacheKeys.length; i < l; i++) { var key = cacheKeys[i]; var exp = cache[key].exports; // Using babel as a transpiler to use esmodule, the export will always // be an object with the default export as a property of it. To ensure // the existing api and babel esmodule exports are both supported we // check for both if (exp === fn || exp && exp.default === fn) { wkey = key; break; } } if (!wkey) { wkey = Math.floor(Math.pow(16, 8) * Math.random()).toString(16); var wcache = {}; for (var i = 0, l = cacheKeys.length; i < l; i++) { var key = cacheKeys[i]; wcache[key] = key; } sources[wkey] = [ Function(['require','module','exports'], '(' + fn + ')(self)'), wcache ]; } var skey = Math.floor(Math.pow(16, 8) * Math.random()).toString(16); var scache = {}; scache[wkey] = wkey; sources[skey] = [ Function(['require'], ( // try to call default if defined to also support babel esmodule // exports 'var f = require(' + stringify(wkey) + ');' + '(f.default ? f.default : f)(self);' )), scache ]; var workerSources = {}; resolveSources(skey); function resolveSources(key) { workerSources[key] = true; for (var depPath in sources[key][1]) { var depKey = sources[key][1][depPath]; if (!workerSources[depKey]) { resolveSources(depKey); } } } var src = '(' + bundleFn + ')({' + Object.keys(workerSources).map(function (key) { return stringify(key) + ':[' + sources[key][0] + ',' + stringify(sources[key][1]) + ']' ; }).join(',') + '},{},[' + stringify(skey) + '])'; var URL = window.URL || window.webkitURL || window.mozURL || window.msURL; var blob = new Blob([src], { type: 'text/javascript' }); if (options && options.bare) { return blob; } var workerUrl = URL.createObjectURL(blob); var worker = new Worker(workerUrl); worker.objectURL = workerUrl; return worker; }; },{}],5:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.createDefaultConfig = createDefaultConfig; /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var defaultConfig = exports.defaultConfig = { enableWorker: false, enableStashBuffer: true, stashInitialSize: undefined, isLive: false, lazyLoad: true, lazyLoadMaxDuration: 3 * 60, deferLoadAfterSourceOpen: true, statisticsInfoReportInterval: 600, accurateSeek: false, seekType: 'range', // [range, param, custom] seekParamStart: 'bstart', seekParamEnd: 'bend', rangeLoadZeroStart: false, customSeekHandler: undefined }; function createDefaultConfig() { return Object.assign({}, defaultConfig); } },{}],6:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _ioController = _dereq_('../io/io-controller.js'); var _ioController2 = _interopRequireDefault(_ioController); var _config = _dereq_('../config.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var Features = function () { function Features() { _classCallCheck(this, Features); } _createClass(Features, null, [{ key: 'supportMSEH264Playback', value: function supportMSEH264Playback() { return window.MediaSource && window.MediaSource.isTypeSupported('video/mp4; codecs="avc1.42E01E,mp4a.40.2"'); } }, { key: 'supportNetworkStreamIO', value: function supportNetworkStreamIO() { var ioctl = new _ioController2.default({}, (0, _config.createDefaultConfig)()); var loaderType = ioctl.loaderType; ioctl.destroy(); return loaderType == 'fetch-stream-loader' || loaderType == 'xhr-moz-chunked-loader'; } }, { key: 'getNetworkLoaderTypeName', value: function getNetworkLoaderTypeName() { var ioctl = new _ioController2.default({}, (0, _config.createDefaultConfig)()); var loaderType = ioctl.loaderType; ioctl.destroy(); return loaderType; } }, { key: 'supportNativeMediaPlayback', value: function supportNativeMediaPlayback(mimeType) { if (Features.videoElement == undefined) { Features.videoElement = window.document.createElement('video'); } var canPlay = Features.videoElement.canPlayType(mimeType); return canPlay === 'probably' || canPlay == 'maybe'; } }, { key: 'getFeatureList', value: function getFeatureList() { var features = { mseFlvPlayback: false, mseLiveFlvPlayback: false, networkStreamIO: false, networkLoaderName: '', nativeMP4H264Playback: false, nativeWebmVP8Playback: false, nativeWebmVP9Playback: false }; features.mseFlvPlayback = Features.supportMSEH264Playback(); features.networkStreamIO = Features.supportNetworkStreamIO(); features.networkLoaderName = Features.getNetworkLoaderTypeName(); features.mseLiveFlvPlayback = features.mseFlvPlayback && features.networkStreamIO; features.nativeMP4H264Playback = Features.supportNativeMediaPlayback('video/mp4; codecs="avc1.42001E, mp4a.40.2"'); features.nativeWebmVP8Playback = Features.supportNativeMediaPlayback('video/webm; codecs="vp8.0, vorbis"'); features.nativeWebmVP9Playback = Features.supportNativeMediaPlayback('video/webm; codecs="vp9"'); return features; } }]); return Features; }(); exports.default = Features; },{"../config.js":5,"../io/io-controller.js":23}],7:[function(_dereq_,module,exports){ "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var MediaInfo = function () { function MediaInfo() { _classCallCheck(this, MediaInfo); this.mimeType = null; this.duration = null; this.hasAudio = null; this.hasVideo = null; this.audioCodec = null; this.videoCodec = null; this.audioDataRate = null; this.videoDataRate = null; this.audioSampleRate = null; this.audioChannelCount = null; this.width = null; this.height = null; this.fps = null; this.profile = null; this.level = null; this.chromaFormat = null; this.sarNum = null; this.sarDen = null; this.metadata = null; this.segments = null; // MediaInfo[] this.segmentCount = null; this.hasKeyframesIndex = null; this.keyframesIndex = null; } _createClass(MediaInfo, [{ key: "isComplete", value: function isComplete() { var audioInfoComplete = this.hasAudio === false || this.hasAudio === true && this.audioCodec != null && this.audioSampleRate != null && this.audioChannelCount != null; var videoInfoComplete = this.hasVideo === false || this.hasVideo === true && this.videoCodec != null && this.width != null && this.height != null && this.fps != null && this.profile != null && this.level != null && this.chromaFormat != null && this.sarNum != null && this.sarDen != null; // keyframesIndex may not be present return this.mimeType != null && this.duration != null && this.metadata != null && this.hasKeyframesIndex != null && audioInfoComplete && videoInfoComplete; } }, { key: "isSeekable", value: function isSeekable() { return this.hasKeyframesIndex === true; } }, { key: "getNearestKeyframe", value: function getNearestKeyframe(milliseconds) { if (this.keyframesIndex == null) { return null; } var table = this.keyframesIndex; var keyframeIdx = this._search(table.times, milliseconds); return { index: keyframeIdx, milliseconds: table.times[keyframeIdx], fileposition: table.filepositions[keyframeIdx] }; } }, { key: "_search", value: function _search(list, value) { var idx = 0; var last = list.length - 1; var mid = 0; var lbound = 0; var ubound = last; if (value < list[0]) { idx = 0; lbound = ubound + 1; // skip search } while (lbound <= ubound) { mid = lbound + Math.floor((ubound - lbound) / 2); if (mid === last || value >= list[mid] && value < list[mid + 1]) { idx = mid; break; } else if (list[mid] < value) { lbound = mid + 1; } else { ubound = mid - 1; } } return idx; } }]); return MediaInfo; }(); exports.default = MediaInfo; },{}],8:[function(_dereq_,module,exports){ "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // Represents an media sample (audio / video) var SampleInfo = exports.SampleInfo = function SampleInfo(dts, pts, duration, originalDts, isSync) { _classCallCheck(this, SampleInfo); this.dts = dts; this.pts = pts; this.duration = duration; this.originalDts = originalDts; this.isSyncPoint = isSync; this.fileposition = null; }; // Media Segment concept is defined in Media Source Extensions spec. // Particularly in ISO BMFF format, an Media Segment contains a moof box followed by a mdat box. var MediaSegmentInfo = exports.MediaSegmentInfo = function () { function MediaSegmentInfo() { _classCallCheck(this, MediaSegmentInfo); this.beginDts = 0; this.endDts = 0; this.beginPts = 0; this.endPts = 0; this.originalBeginDts = 0; this.originalEndDts = 0; this.syncPoints = []; // SampleInfo[n], for video IDR frames only this.firstSample = null; // SampleInfo this.lastSample = null; // SampleInfo } _createClass(MediaSegmentInfo, [{ key: "appendSyncPoint", value: function appendSyncPoint(sampleInfo) { // also called Random Access Point sampleInfo.isSyncPoint = true; this.syncPoints.push(sampleInfo); } }]); return MediaSegmentInfo; }(); // Ordered list for recording video IDR frames, sorted by originalDts var IDRSampleList = exports.IDRSampleList = function () { function IDRSampleList() { _classCallCheck(this, IDRSampleList); this._list = []; } _createClass(IDRSampleList, [{ key: "clear", value: function clear() { this._list = []; } }, { key: "appendArray", value: function appendArray(syncPoints) { var list = this._list; if (syncPoints.length === 0) { return; } if (list.length > 0 && syncPoints[0].originalDts < list[list.length - 1].originalDts) { this.clear(); } Array.prototype.push.apply(list, syncPoints); } }, { key: "getLastSyncPointBeforeDts", value: function getLastSyncPointBeforeDts(dts) { if (this._list.length == 0) { return null; } var list = this._list; var idx = 0; var last = list.length - 1; var mid = 0; var lbound = 0; var ubound = last; if (dts < list[0].dts) { idx = 0; lbound = ubound + 1; } while (lbound <= ubound) { mid = lbound + Math.floor((ubound - lbound) / 2); if (mid === last || dts >= list[mid].dts && dts < list[mid + 1].dts) { idx = mid; break; } else if (list[mid].dts < dts) { lbound = mid + 1; } else { ubound = mid - 1; } } return this._list[idx]; } }]); return IDRSampleList; }(); // Data structure for recording information of media segments in single track. var MediaSegmentInfoList = exports.MediaSegmentInfoList = function () { function MediaSegmentInfoList(type) { _classCallCheck(this, MediaSegmentInfoList); this._type = type; this._list = []; this._lastAppendLocation = -1; // cached last insert location } _createClass(MediaSegmentInfoList, [{ key: "isEmpty", value: function isEmpty() { return this._list.length === 0; } }, { key: "clear", value: function clear() { this._list = []; this._lastAppendLocation = -1; } }, { key: "_searchNearestSegmentBefore", value: function _searchNearestSegmentBefore(originalBeginDts) { var list = this._list; if (list.length === 0) { return -2; } var last = list.length - 1; var mid = 0; var lbound = 0; var ubound = last; var idx = 0; if (originalBeginDts < list[0].originalBeginDts) { idx = -1; return idx; } while (lbound <= ubound) { mid = lbound + Math.floor((ubound - lbound) / 2); if (mid === last || originalBeginDts > list[mid].lastSample.originalDts && originalBeginDts < list[mid + 1].originalBeginDts) { idx = mid; break; } else if (list[mid].originalBeginDts < originalBeginDts) { lbound = mid + 1; } else { ubound = mid - 1; } } return idx; } }, { key: "_searchNearestSegmentAfter", value: function _searchNearestSegmentAfter(originalBeginDts) { return this._searchNearestSegmentBefore(originalBeginDts) + 1; } }, { key: "append", value: function append(mediaSegmentInfo) { var list = this._list; var msi = mediaSegmentInfo; var lastAppendIdx = this._lastAppendLocation; var insertIdx = 0; if (lastAppendIdx !== -1 && lastAppendIdx < list.length && msi.originalBeginDts >= list[lastAppendIdx].lastSample.originalDts && (lastAppendIdx === list.length - 1 || lastAppendIdx < list.length - 1 && msi.originalBeginDts < list[lastAppendIdx + 1].originalBeginDts)) { insertIdx = lastAppendIdx + 1; // use cached location idx } else { if (list.length > 0) { insertIdx = this._searchNearestSegmentBefore(msi.originalBeginDts) + 1; } } this._lastAppendLocation = insertIdx; this._list.splice(insertIdx, 0, msi); } }, { key: "getLastSegmentBefore", value: function getLastSegmentBefore(originalBeginDts) { var idx = this._searchNearestSegmentBefore(originalBeginDts); if (idx >= 0) { return this._list[idx]; } else { // -1 return null; } } }, { key: "getLastSampleBefore", value: function getLastSampleBefore(originalBeginDts) { var segment = this.getLastSegmentBefore(originalBeginDts); if (segment != null) { return segment.lastSample; } else { return null; } } }, { key: "getLastSyncPointBefore", value: function getLastSyncPointBefore(originalBeginDts) { var this$1 = this; var segmentIdx = this._searchNearestSegmentBefore(originalBeginDts); var syncPoints = this._list[segmentIdx].syncPoints; while (syncPoints.length === 0 && segmentIdx > 0) { segmentIdx--; syncPoints = this$1._list[segmentIdx].syncPoints; } if (syncPoints.length > 0) { return syncPoints[syncPoints.length - 1]; } else { return null; } } }, { key: "type", get: function get() { return this._type; } }, { key: "length", get: function get() { return this._list.length; } }]); return MediaSegmentInfoList; }(); },{}],9:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _browser = _dereq_('../utils/browser.js'); var _browser2 = _interopRequireDefault(_browser); var _mseEvents = _dereq_('./mse-events.js'); var _mseEvents2 = _interopRequireDefault(_mseEvents); var _mediaSegmentInfo = _dereq_('./media-segment-info.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // Media Source Extensions controller var MSEController = function () { function MSEController() { _classCallCheck(this, MSEController); this.TAG = 'MSEController'; this._emitter = new _events2.default(); this.e = { onSourceOpen: this._onSourceOpen.bind(this), onSourceEnded: this._onSourceEnded.bind(this), onSourceClose: this._onSourceClose.bind(this), onSourceBufferError: this._onSourceBufferError.bind(this), onSourceBufferUpdateEnd: this._onSourceBufferUpdateEnd.bind(this) }; this._mediaSource = null; this._mediaSourceObjectURL = null; this._mediaElement = null; this._isBufferFull = false; this._hasPendingEos = false; this._pendingSourceBufferInit = []; this._mimeTypes = { video: null, audio: null }; this._sourceBuffers = { video: null, audio: null }; this._lastInitSegments = { video: null, audio: null }; this._pendingSegments = { video: [], audio: [] }; this._pendingRemoveRanges = { video: [], audio: [] }; this._idrList = new _mediaSegmentInfo.IDRSampleList(); } _createClass(MSEController, [{ key: 'destroy', value: function destroy() { if (this._mediaElement || this._mediaSource) { this.detachMediaElement(); } this.e = null; this._emitter.removeAllListeners(); this._emitter = null; } }, { key: 'on', value: function on(event, listener) { this._emitter.addListener(event, listener); } }, { key: 'off', value: function off(event, listener) { this._emitter.removeListener(event, listener); } }, { key: 'attachMediaElement', value: function attachMediaElement(mediaElement) { if (this._mediaSource) { throw new _exception.IllegalStateException('MediaSource has been attached to an HTMLMediaElement!'); } var ms = this._mediaSource = new window.MediaSource(); ms.addEventListener('sourceopen', this.e.onSourceOpen); ms.addEventListener('sourceended', this.e.onSourceEnded); ms.addEventListener('sourceclose', this.e.onSourceClose); this._mediaElement = mediaElement; this._mediaSourceObjectURL = window.URL.createObjectURL(this._mediaSource); mediaElement.src = this._mediaSourceObjectURL; } }, { key: 'detachMediaElement', value: function detachMediaElement() { var this$1 = this; if (this._mediaSource) { var ms = this._mediaSource; for (var type in this._sourceBuffers) { // pending segments should be discard var ps = this$1._pendingSegments[type]; ps.splice(0, ps.length); this$1._pendingSegments[type] = null; this$1._pendingRemoveRanges[type] = null; this$1._lastInitSegments[type] = null; // remove all sourcebuffers var sb = this$1._sourceBuffers[type]; if (sb) { if (ms.readyState !== 'closed') { ms.removeSourceBuffer(sb); sb.removeEventListener('error', this$1.e.onSourceBufferError); sb.removeEventListener('updateend', this$1.e.onSourceBufferUpdateEnd); } this$1._mimeTypes[type] = null; this$1._sourceBuffers[type] = null; } } if (ms.readyState === 'open') { try { ms.endOfStream(); } catch (error) { _logger2.default.e(this.TAG, error.message); } } ms.removeEventListener('sourceopen', this.e.onSourceOpen); ms.removeEventListener('sourceended', this.e.onSourceEnded); ms.removeEventListener('sourceclose', this.e.onSourceClose); this._pendingSourceBufferInit = []; this._isBufferFull = false; this._idrList.clear(); this._mediaSource = null; } if (this._mediaElement) { this._mediaElement.src = ''; this._mediaElement.removeAttribute('src'); this._mediaElement = null; } if (this._mediaSourceObjectURL) { window.URL.revokeObjectURL(this._mediaSourceObjectURL); this._mediaSourceObjectURL = null; } } }, { key: 'appendInitSegment', value: function appendInitSegment(initSegment, deferred) { if (!this._mediaSource || this._mediaSource.readyState !== 'open') { // sourcebuffer creation requires mediaSource.readyState === 'open' // so we defer the sourcebuffer creation, until sourceopen event triggered this._pendingSourceBufferInit.push(initSegment); // make sure that this InitSegment is in the front of pending segments queue this._pendingSegments[initSegment.type].push(initSegment); return; } var is = initSegment; var mimeType = is.container + ';codecs=' + is.codec; var firstInitSegment = false; _logger2.default.v(this.TAG, 'Received Initialization Segment, mimeType: ' + mimeType); this._lastInitSegments[is.type] = is; if (mimeType !== this._mimeTypes[is.type]) { if (!this._mimeTypes[is.type]) { // empty, first chance create sourcebuffer firstInitSegment = true; try { var sb = this._sourceBuffers[is.type] = this._mediaSource.addSourceBuffer(mimeType); sb.addEventListener('error', this.e.onSourceBufferError); sb.addEventListener('updateend', this.e.onSourceBufferUpdateEnd); } catch (error) { _logger2.default.e(this.TAG, error.message); this._emitter.emit(_mseEvents2.default.ERROR, { code: error.code, msg: error.message }); return; } } else { _logger2.default.v(this.TAG, 'Notice: ' + is.type + ' mimeType changed, origin: ' + this._mimeTypes[is.type] + ', target: ' + mimeType); } this._mimeTypes[is.type] = mimeType; } if (!deferred) { // deferred means this InitSegment has been pushed to pendingSegments queue this._pendingSegments[is.type].push(is); } if (!firstInitSegment) { // append immediately only if init segment in subsequence if (this._sourceBuffers[is.type] && !this._sourceBuffers[is.type].updating) { this._doAppendSegments(); } } } }, { key: 'appendMediaSegment', value: function appendMediaSegment(mediaSegment) { var ms = mediaSegment; this._pendingSegments[ms.type].push(ms); var sb = this._sourceBuffers[ms.type]; if (sb && !sb.updating && !this._hasPendingRemoveRanges()) { this._doAppendSegments(); } } }, { key: 'seek', value: function seek(seconds) { var this$1 = this; // remove all appended buffers for (var type in this._sourceBuffers) { if (!this$1._sourceBuffers[type]) { continue; } // abort current buffer append algorithm var sb = this$1._sourceBuffers[type]; if (this$1._mediaSource.readyState === 'open') { try { // If range removal algorithm is running, InvalidStateError will be throwed // Ignore it. sb.abort(); } catch (error) { _logger2.default.e(this$1.TAG, error.message); } } // IDRList should be clear this$1._idrList.clear(); // pending segments should be discard var ps = this$1._pendingSegments[type]; ps.splice(0, ps.length); if (this$1._mediaSource.readyState === 'closed') { // Parent MediaSource object has been detached from HTMLMediaElement continue; } // record ranges to be remove from SourceBuffer for (var i = 0; i < sb.buffered.length; i++) { var start = sb.buffered.start(i); var end = sb.buffered.end(i); this$1._pendingRemoveRanges[type].push({ start: start, end: end }); } // if sb is not updating, let's remove ranges now! if (!sb.updating) { this$1._doRemoveRanges(); } // Safari 10 may get InvalidStateError in the later appendBuffer() after SourceBuffer.remove() call // Internal parser's state may be invalid at this time. Re-append last InitSegment to workaround. // Related issue: https://bugs.webkit.org/show_bug.cgi?id=159230 if (_browser2.default.safari) { var lastInitSegment = this$1._lastInitSegments[type]; if (lastInitSegment) { this$1._pendingSegments[type].push(lastInitSegment); if (!sb.updating) { this$1._doAppendSegments(); } } } } } }, { key: 'endOfStream', value: function endOfStream() { var ms = this._mediaSource; var sb = this._sourceBuffers; if (!ms || ms.readyState !== 'open') { if (ms && ms.readyState === 'closed' && this._hasPendingSegments()) { // If MediaSource hasn't turned into open state, and there're pending segments // Mark pending endOfStream, defer call until all pending segments appended complete this._hasPendingEos = true; } return; } if (sb.video && sb.video.updating || sb.audio && sb.audio.updating) { // If any sourcebuffer is updating, defer endOfStream operation // See _onSourceBufferUpdateEnd() this._hasPendingEos = true; } else { this._hasPendingEos = false; // Notify media data loading complete // This is helpful for correcting total duration to match last media segment // Otherwise MediaElement's ended event may not be triggered ms.endOfStream(); } } }, { key: 'getNearestKeyframe', value: function getNearestKeyframe(dts) { return this._idrList.getLastSyncPointBeforeDts(dts); } }, { key: '_doRemoveRanges', value: function _doRemoveRanges() { var this$1 = this; for (var type in this._pendingRemoveRanges) { if (!this$1._sourceBuffers[type] || this$1._sourceBuffers[type].updating) { continue; } var sb = this$1._sourceBuffers[type]; var ranges = this$1._pendingRemoveRanges[type]; while (ranges.length && !sb.updating) { var range = ranges.shift(); sb.remove(range.start, range.end); } } } }, { key: '_doAppendSegments', value: function _doAppendSegments() { var this$1 = this; var pendingSegments = this._pendingSegments; for (var type in pendingSegments) { if (!this$1._sourceBuffers[type] || this$1._sourceBuffers[type].updating) { continue; } if (pendingSegments[type].length > 0) { var segment = pendingSegments[type].shift(); try { this$1._sourceBuffers[type].appendBuffer(segment.data); this$1._isBufferFull = false; if (type === 'video' && segment.hasOwnProperty('info')) { this$1._idrList.appendArray(segment.info.syncPoints); } } catch (error) { this$1._pendingSegments[type].unshift(segment); if (error.code === 22) { // QuotaExceededError /* Notice that FireFox may not throw QuotaExceededError if SourceBuffer is full * Currently we can only do lazy-load to avoid SourceBuffer become scattered. * SourceBuffer eviction policy may be changed in future version of FireFox. * * Related issues: * https://bugzilla.mozilla.org/show_bug.cgi?id=1279885 * https://bugzilla.mozilla.org/show_bug.cgi?id=1280023 */ // report buffer full, abort network IO if (!this$1._isBufferFull) { this$1._emitter.emit(_mseEvents2.default.BUFFER_FULL); } this$1._isBufferFull = true; } else { _logger2.default.e(this$1.TAG, error.message); this$1._emitter.emit(_mseEvents2.default.ERROR, { code: error.code, msg: error.message }); } } } } } }, { key: '_onSourceOpen', value: function _onSourceOpen() { var this$1 = this; _logger2.default.v(this.TAG, 'MediaSource onSourceOpen'); this._mediaSource.removeEventListener('sourceopen', this.e.onSourceOpen); // deferred sourcebuffer creation / initialization if (this._pendingSourceBufferInit.length > 0) { var pendings = this._pendingSourceBufferInit; while (pendings.length) { var segment = pendings.shift(); this$1.appendInitSegment(segment, true); } } // there may be some pending media segments, append them if (this._hasPendingSegments()) { this._doAppendSegments(); } this._emitter.emit(_mseEvents2.default.SOURCE_OPEN); } }, { key: '_onSourceEnded', value: function _onSourceEnded() { // fired on endOfStream _logger2.default.v(this.TAG, 'MediaSource onSourceEnded'); } }, { key: '_onSourceClose', value: function _onSourceClose() { // fired on detaching from media element _logger2.default.v(this.TAG, 'MediaSource onSourceClose'); if (this._mediaSource && this.e != null) { this._mediaSource.removeEventListener('sourceopen', this.e.onSourceOpen); this._mediaSource.removeEventListener('sourceended', this.e.onSourceEnded); this._mediaSource.removeEventListener('sourceclose', this.e.onSourceClose); } } }, { key: '_hasPendingSegments', value: function _hasPendingSegments() { var ps = this._pendingSegments; return ps.video.length > 0 || ps.audio.length > 0; } }, { key: '_hasPendingRemoveRanges', value: function _hasPendingRemoveRanges() { var prr = this._pendingRemoveRanges; return prr.video.length > 0 || prr.audio.length > 0; } }, { key: '_onSourceBufferUpdateEnd', value: function _onSourceBufferUpdateEnd() { if (this._hasPendingRemoveRanges()) { this._doRemoveRanges(); } else if (this._hasPendingSegments()) { this._doAppendSegments(); } else if (this._hasPendingEos) { this.endOfStream(); } this._emitter.emit(_mseEvents2.default.UPDATE_END); } }, { key: '_onSourceBufferError', value: function _onSourceBufferError(e) { _logger2.default.e(this.TAG, 'SourceBuffer Error: ' + e); // this error might not always be fatal, just ignore it } }]); return MSEController; }(); exports.default = MSEController; },{"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./media-segment-info.js":8,"./mse-events.js":10,"events":2}],10:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var MSEEvents = { ERROR: 'error', SOURCE_OPEN: 'source_open', UPDATE_END: 'update_end', BUFFER_FULL: 'buffer_full' }; exports.default = MSEEvents; },{}],11:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _loggingControl = _dereq_('../utils/logging-control.js'); var _loggingControl2 = _interopRequireDefault(_loggingControl); var _transmuxingController = _dereq_('./transmuxing-controller.js'); var _transmuxingController2 = _interopRequireDefault(_transmuxingController); var _transmuxingEvents = _dereq_('./transmuxing-events.js'); var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents); var _transmuxingWorker = _dereq_('./transmuxing-worker.js'); var _transmuxingWorker2 = _interopRequireDefault(_transmuxingWorker); var _mediaInfo = _dereq_('./media-info.js'); var _mediaInfo2 = _interopRequireDefault(_mediaInfo); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var Transmuxer = function () { function Transmuxer(mediaDataSource, config) { _classCallCheck(this, Transmuxer); this.TAG = 'Transmuxer'; this._emitter = new _events2.default(); if (config.enableWorker && typeof Worker !== 'undefined') { try { var work = _dereq_('webworkify'); this._worker = work(_transmuxingWorker2.default); this._workerDestroying = false; this._worker.addEventListener('message', this._onWorkerMessage.bind(this)); this._worker.postMessage({ cmd: 'init', param: [mediaDataSource, config] }); this.e = { onLoggingConfigChanged: this._onLoggingConfigChanged.bind(this) }; _loggingControl2.default.registerListener(this.e.onLoggingConfigChanged); this._worker.postMessage({ cmd: 'logging_config', param: _loggingControl2.default.getConfig() }); } catch (error) { _logger2.default.e(this.TAG, 'Error while initialize transmuxing worker, fallback to inline transmuxing'); this._worker = null; this._controller = new _transmuxingController2.default(mediaDataSource, config); } } else { this._controller = new _transmuxingController2.default(mediaDataSource, config); } if (this._controller) { var ctl = this._controller; ctl.on(_transmuxingEvents2.default.IO_ERROR, this._onIOError.bind(this)); ctl.on(_transmuxingEvents2.default.DEMUX_ERROR, this._onDemuxError.bind(this)); ctl.on(_transmuxingEvents2.default.INIT_SEGMENT, this._onInitSegment.bind(this)); ctl.on(_transmuxingEvents2.default.MEDIA_SEGMENT, this._onMediaSegment.bind(this)); ctl.on(_transmuxingEvents2.default.LOADING_COMPLETE, this._onLoadingComplete.bind(this)); ctl.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, this._onRecoveredEarlyEof.bind(this)); ctl.on(_transmuxingEvents2.default.MEDIA_INFO, this._onMediaInfo.bind(this)); ctl.on(_transmuxingEvents2.default.STATISTICS_INFO, this._onStatisticsInfo.bind(this)); ctl.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, this._onRecommendSeekpoint.bind(this)); } } _createClass(Transmuxer, [{ key: 'destroy', value: function destroy() { if (this._worker) { if (!this._workerDestroying) { this._workerDestroying = true; this._worker.postMessage({ cmd: 'destroy' }); _loggingControl2.default.removeListener(this.e.onLoggingConfigChanged); this.e = null; } } else { this._controller.destroy(); this._controller = null; } this._emitter.removeAllListeners(); this._emitter = null; } }, { key: 'on', value: function on(event, listener) { this._emitter.addListener(event, listener); } }, { key: 'off', value: function off(event, listener) { this._emitter.removeListener(event, listener); } }, { key: 'hasWorker', value: function hasWorker() { return this._worker != null; } }, { key: 'open', value: function open() { if (this._worker) { this._worker.postMessage({ cmd: 'start' }); } else { this._controller.start(); } } }, { key: 'close', value: function close() { if (this._worker) { this._worker.postMessage({ cmd: 'stop' }); } else { this._controller.stop(); } } }, { key: 'seek', value: function seek(milliseconds) { if (this._worker) { this._worker.postMessage({ cmd: 'seek', param: milliseconds }); } else { this._controller.seek(milliseconds); } } }, { key: 'pause', value: function pause() { if (this._worker) { this._worker.postMessage({ cmd: 'pause' }); } else { this._controller.pause(); } } }, { key: 'resume', value: function resume() { if (this._worker) { this._worker.postMessage({ cmd: 'resume' }); } else { this._controller.resume(); } } }, { key: '_onInitSegment', value: function _onInitSegment(type, initSegment) { var _this = this; // do async invoke Promise.resolve().then(function () { _this._emitter.emit(_transmuxingEvents2.default.INIT_SEGMENT, type, initSegment); }); } }, { key: '_onMediaSegment', value: function _onMediaSegment(type, mediaSegment) { var _this2 = this; Promise.resolve().then(function () { _this2._emitter.emit(_transmuxingEvents2.default.MEDIA_SEGMENT, type, mediaSegment); }); } }, { key: '_onLoadingComplete', value: function _onLoadingComplete() { var _this3 = this; Promise.resolve().then(function () { _this3._emitter.emit(_transmuxingEvents2.default.LOADING_COMPLETE); }); } }, { key: '_onRecoveredEarlyEof', value: function _onRecoveredEarlyEof() { var _this4 = this; Promise.resolve().then(function () { _this4._emitter.emit(_transmuxingEvents2.default.RECOVERED_EARLY_EOF); }); } }, { key: '_onMediaInfo', value: function _onMediaInfo(mediaInfo) { var _this5 = this; Promise.resolve().then(function () { _this5._emitter.emit(_transmuxingEvents2.default.MEDIA_INFO, mediaInfo); }); } }, { key: '_onStatisticsInfo', value: function _onStatisticsInfo(statisticsInfo) { var _this6 = this; Promise.resolve().then(function () { _this6._emitter.emit(_transmuxingEvents2.default.STATISTICS_INFO, statisticsInfo); }); } }, { key: '_onIOError', value: function _onIOError(type, info) { var _this7 = this; Promise.resolve().then(function () { _this7._emitter.emit(_transmuxingEvents2.default.IO_ERROR, type, info); }); } }, { key: '_onDemuxError', value: function _onDemuxError(type, info) { var _this8 = this; Promise.resolve().then(function () { _this8._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, type, info); }); } }, { key: '_onRecommendSeekpoint', value: function _onRecommendSeekpoint(milliseconds) { var _this9 = this; Promise.resolve().then(function () { _this9._emitter.emit(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, milliseconds); }); } }, { key: '_onLoggingConfigChanged', value: function _onLoggingConfigChanged(config) { if (this._worker) { this._worker.postMessage({ cmd: 'logging_config', param: config }); } } }, { key: '_onWorkerMessage', value: function _onWorkerMessage(e) { var message = e.data; var data = message.data; if (message.msg === 'destroyed' || this._workerDestroying) { this._workerDestroying = false; this._worker.terminate(); this._worker = null; return; } switch (message.msg) { case _transmuxingEvents2.default.INIT_SEGMENT: case _transmuxingEvents2.default.MEDIA_SEGMENT: this._emitter.emit(message.msg, data.type, data.data); break; case _transmuxingEvents2.default.LOADING_COMPLETE: case _transmuxingEvents2.default.RECOVERED_EARLY_EOF: this._emitter.emit(message.msg); break; case _transmuxingEvents2.default.MEDIA_INFO: Object.setPrototypeOf(data, _mediaInfo2.default.prototype); this._emitter.emit(message.msg, data); break; case _transmuxingEvents2.default.STATISTICS_INFO: this._emitter.emit(message.msg, data); break; case _transmuxingEvents2.default.IO_ERROR: case _transmuxingEvents2.default.DEMUX_ERROR: this._emitter.emit(message.msg, data.type, data.info); break; case _transmuxingEvents2.default.RECOMMEND_SEEKPOINT: this._emitter.emit(message.msg, data); break; default: break; } } }]); return Transmuxer; }(); exports.default = Transmuxer; },{"../utils/logger.js":41,"../utils/logging-control.js":42,"./media-info.js":7,"./transmuxing-controller.js":12,"./transmuxing-events.js":13,"./transmuxing-worker.js":14,"events":2,"webworkify":4}],12:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _browser = _dereq_('../utils/browser.js'); var _browser2 = _interopRequireDefault(_browser); var _mediaInfo = _dereq_('./media-info.js'); var _mediaInfo2 = _interopRequireDefault(_mediaInfo); var _flvDemuxer = _dereq_('../demux/flv-demuxer.js'); var _flvDemuxer2 = _interopRequireDefault(_flvDemuxer); var _mp4Remuxer = _dereq_('../remux/mp4-remuxer.js'); var _mp4Remuxer2 = _interopRequireDefault(_mp4Remuxer); var _demuxErrors = _dereq_('../demux/demux-errors.js'); var _demuxErrors2 = _interopRequireDefault(_demuxErrors); var _ioController = _dereq_('../io/io-controller.js'); var _ioController2 = _interopRequireDefault(_ioController); var _transmuxingEvents = _dereq_('./transmuxing-events.js'); var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents); var _loader = _dereq_('../io/loader.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // Transmuxing (IO, Demuxing, Remuxing) controller, with multipart support var TransmuxingController = function () { function TransmuxingController(mediaDataSource, config) { _classCallCheck(this, TransmuxingController); this.TAG = 'TransmuxingController'; this._emitter = new _events2.default(); this._config = config; // treat single part media as multipart media, which has only one segment if (!mediaDataSource.segments) { mediaDataSource.segments = [{ duration: mediaDataSource.duration, filesize: mediaDataSource.filesize, url: mediaDataSource.url }]; } // fill in default IO params if not exists if (typeof mediaDataSource.cors !== 'boolean') { mediaDataSource.cors = true; } if (typeof mediaDataSource.withCredentials !== 'boolean') { mediaDataSource.withCredentials = false; } this._mediaDataSource = mediaDataSource; this._currentSegmentIndex = 0; var totalDuration = 0; this._mediaDataSource.segments.forEach(function (segment) { // timestampBase for each segment, and calculate total duration segment.timestampBase = totalDuration; totalDuration += segment.duration; // params needed by IOController segment.cors = mediaDataSource.cors; segment.withCredentials = mediaDataSource.withCredentials; }); if (!isNaN(totalDuration) && this._mediaDataSource.duration !== totalDuration) { this._mediaDataSource.duration = totalDuration; } this._mediaInfo = null; this._demuxer = null; this._remuxer = null; this._ioctl = null; this._pendingSeekTime = null; this._pendingResolveSeekPoint = null; this._statisticsReporter = null; } _createClass(TransmuxingController, [{ key: 'destroy', value: function destroy() { this._mediaInfo = null; this._mediaDataSource = null; if (this._statisticsReporter) { this._disableStatisticsReporter(); } if (this._ioctl) { this._ioctl.destroy(); this._ioctl = null; } if (this._demuxer) { this._demuxer.destroy(); this._demuxer = null; } if (this._remuxer) { this._remuxer.destroy(); this._remuxer = null; } this._emitter.removeAllListeners(); this._emitter = null; } }, { key: 'on', value: function on(event, listener) { this._emitter.addListener(event, listener); } }, { key: 'off', value: function off(event, listener) { this._emitter.removeListener(event, listener); } }, { key: 'start', value: function start() { this._loadSegment(0); this._enableStatisticsReporter(); } }, { key: '_loadSegment', value: function _loadSegment(segmentIndex, optionalFrom) { this._currentSegmentIndex = segmentIndex; var dataSource = this._mediaDataSource.segments[segmentIndex]; var ioctl = this._ioctl = new _ioController2.default(dataSource, this._config, segmentIndex); ioctl.onError = this._onIOException.bind(this); ioctl.onSeeked = this._onIOSeeked.bind(this); ioctl.onComplete = this._onIOComplete.bind(this); ioctl.onRecoveredEarlyEof = this._onIORecoveredEarlyEof.bind(this); if (optionalFrom) { this._demuxer.bindDataSource(this._ioctl); } else { ioctl.onDataArrival = this._onInitChunkArrival.bind(this); } ioctl.open(optionalFrom); } }, { key: 'stop', value: function stop() { this._internalAbort(); this._disableStatisticsReporter(); } }, { key: '_internalAbort', value: function _internalAbort() { if (this._ioctl) { this._ioctl.destroy(); this._ioctl = null; } } }, { key: 'pause', value: function pause() { // take a rest if (this._ioctl && this._ioctl.isWorking()) { this._ioctl.pause(); this._disableStatisticsReporter(); } } }, { key: 'resume', value: function resume() { if (this._ioctl && this._ioctl.isPaused()) { this._ioctl.resume(); this._enableStatisticsReporter(); } } }, { key: 'seek', value: function seek(milliseconds) { if (this._mediaInfo == null || !this._mediaInfo.isSeekable()) { return; } var targetSegmentIndex = this._searchSegmentIndexContains(milliseconds); if (targetSegmentIndex === this._currentSegmentIndex) { // intra-segment seeking var segmentInfo = this._mediaInfo.segments[targetSegmentIndex]; if (segmentInfo == undefined) { // current segment loading started, but mediainfo hasn't received yet // wait for the metadata loaded, then seek to expected position this._pendingSeekTime = milliseconds; } else { var keyframe = segmentInfo.getNearestKeyframe(milliseconds); this._remuxer.seek(keyframe.milliseconds); this._ioctl.seek(keyframe.fileposition); // Will be resolved in _onRemuxerMediaSegmentArrival() this._pendingResolveSeekPoint = keyframe.milliseconds; } } else { // cross-segment seeking var targetSegmentInfo = this._mediaInfo.segments[targetSegmentIndex]; if (targetSegmentInfo == undefined) { // target segment hasn't been loaded. We need metadata then seek to expected time this._pendingSeekTime = milliseconds; this._internalAbort(); this._remuxer.seek(); this._remuxer.insertDiscontinuity(); this._loadSegment(targetSegmentIndex); // Here we wait for the metadata loaded, then seek to expected position } else { // We have target segment's metadata, direct seek to target position var _keyframe = targetSegmentInfo.getNearestKeyframe(milliseconds); this._internalAbort(); this._remuxer.seek(milliseconds); this._remuxer.insertDiscontinuity(); this._demuxer.resetMediaInfo(); this._demuxer.timestampBase = this._mediaDataSource.segments[targetSegmentIndex].timestampBase; this._loadSegment(targetSegmentIndex, _keyframe.fileposition); this._pendingResolveSeekPoint = _keyframe.milliseconds; this._reportSegmentMediaInfo(targetSegmentIndex); } } this._enableStatisticsReporter(); } }, { key: '_searchSegmentIndexContains', value: function _searchSegmentIndexContains(milliseconds) { var segments = this._mediaDataSource.segments; var idx = segments.length - 1; for (var i = 0; i < segments.length; i++) { if (milliseconds < segments[i].timestampBase) { idx = i - 1; break; } } return idx; } }, { key: '_onInitChunkArrival', value: function _onInitChunkArrival(data, byteStart) { var _this = this; var probeData = null; var consumed = 0; if (byteStart > 0) { // IOController seeked immediately after opened, byteStart > 0 callback may received this._demuxer.bindDataSource(this._ioctl); this._demuxer.timestampBase = this._mediaDataSource.segments[this._currentSegmentIndex].timestampBase; consumed = this._demuxer.parseChunks(data, byteStart); } else if ((probeData = _flvDemuxer2.default.probe(data)).match) { // Always create new FLVDemuxer this._demuxer = new _flvDemuxer2.default(probeData, this._config); if (!this._remuxer) { this._remuxer = new _mp4Remuxer2.default(this._config); } var mds = this._mediaDataSource; if (mds.duration != undefined && !isNaN(mds.duration)) { this._demuxer.overridedDuration = mds.duration; } this._demuxer.timestampBase = mds.segments[this._currentSegmentIndex].timestampBase; this._demuxer.onError = this._onDemuxException.bind(this); this._demuxer.onMediaInfo = this._onMediaInfo.bind(this); this._remuxer.bindDataSource(this._demuxer.bindDataSource(this._ioctl)); this._remuxer.onInitSegment = this._onRemuxerInitSegmentArrival.bind(this); this._remuxer.onMediaSegment = this._onRemuxerMediaSegmentArrival.bind(this); consumed = this._demuxer.parseChunks(data, byteStart); } else { probeData = null; _logger2.default.e(this.TAG, 'Non-FLV, Unsupported media type!'); Promise.resolve().then(function () { _this._internalAbort(); }); this._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, _demuxErrors2.default.FORMAT_UNSUPPORTED, 'Non-FLV, Unsupported media type'); consumed = 0; } return consumed; } }, { key: '_onMediaInfo', value: function _onMediaInfo(mediaInfo) { var _this2 = this; if (this._mediaInfo == null) { // Store first segment's mediainfo as global mediaInfo this._mediaInfo = Object.assign({}, mediaInfo); this._mediaInfo.keyframesIndex = null; this._mediaInfo.segments = []; this._mediaInfo.segmentCount = this._mediaDataSource.segments.length; Object.setPrototypeOf(this._mediaInfo, _mediaInfo2.default.prototype); } var segmentInfo = Object.assign({}, mediaInfo); Object.setPrototypeOf(segmentInfo, _mediaInfo2.default.prototype); this._mediaInfo.segments[this._currentSegmentIndex] = segmentInfo; // notify mediaInfo update this._reportSegmentMediaInfo(this._currentSegmentIndex); if (this._pendingSeekTime != null) { Promise.resolve().then(function () { var target = _this2._pendingSeekTime; _this2._pendingSeekTime = null; _this2.seek(target); }); } } }, { key: '_onIOSeeked', value: function _onIOSeeked() { this._remuxer.insertDiscontinuity(); } }, { key: '_onIOComplete', value: function _onIOComplete(extraData) { var segmentIndex = extraData; var nextSegmentIndex = segmentIndex + 1; if (nextSegmentIndex < this._mediaDataSource.segments.length) { this._internalAbort(); this._loadSegment(nextSegmentIndex); } else { this._emitter.emit(_transmuxingEvents2.default.LOADING_COMPLETE); this._disableStatisticsReporter(); } } }, { key: '_onIORecoveredEarlyEof', value: function _onIORecoveredEarlyEof() { this._emitter.emit(_transmuxingEvents2.default.RECOVERED_EARLY_EOF); } }, { key: '_onIOException', value: function _onIOException(type, info) { _logger2.default.e(this.TAG, 'IOException: type = ' + type + ', code = ' + info.code + ', msg = ' + info.msg); this._emitter.emit(_transmuxingEvents2.default.IO_ERROR, type, info); this._disableStatisticsReporter(); } }, { key: '_onDemuxException', value: function _onDemuxException(type, info) { _logger2.default.e(this.TAG, 'DemuxException: type = ' + type + ', info = ' + info); this._emitter.emit(_transmuxingEvents2.default.DEMUX_ERROR, type, info); } }, { key: '_onRemuxerInitSegmentArrival', value: function _onRemuxerInitSegmentArrival(type, initSegment) { this._emitter.emit(_transmuxingEvents2.default.INIT_SEGMENT, type, initSegment); } }, { key: '_onRemuxerMediaSegmentArrival', value: function _onRemuxerMediaSegmentArrival(type, mediaSegment) { if (this._pendingSeekTime != null) { // Media segments after new-segment cross-seeking should be dropped. return; } this._emitter.emit(_transmuxingEvents2.default.MEDIA_SEGMENT, type, mediaSegment); // Resolve pending seekPoint if (this._pendingResolveSeekPoint != null && type === 'video') { var syncPoints = mediaSegment.info.syncPoints; var seekpoint = this._pendingResolveSeekPoint; this._pendingResolveSeekPoint = null; // Safari: Pass PTS for recommend_seekpoint if (_browser2.default.safari && syncPoints.length > 0 && syncPoints[0].originalDts === seekpoint) { seekpoint = syncPoints[0].pts; } // else: use original DTS (keyframe.milliseconds) this._emitter.emit(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, seekpoint); } } }, { key: '_enableStatisticsReporter', value: function _enableStatisticsReporter() { if (this._statisticsReporter == null) { this._statisticsReporter = self.setInterval(this._reportStatisticsInfo.bind(this), this._config.statisticsInfoReportInterval); } } }, { key: '_disableStatisticsReporter', value: function _disableStatisticsReporter() { if (this._statisticsReporter) { self.clearInterval(this._statisticsReporter); this._statisticsReporter = null; } } }, { key: '_reportSegmentMediaInfo', value: function _reportSegmentMediaInfo(segmentIndex) { var segmentInfo = this._mediaInfo.segments[segmentIndex]; var exportInfo = Object.assign({}, segmentInfo); exportInfo.duration = this._mediaInfo.duration; exportInfo.segmentCount = this._mediaInfo.segmentCount; delete exportInfo.segments; delete exportInfo.keyframesIndex; this._emitter.emit(_transmuxingEvents2.default.MEDIA_INFO, exportInfo); } }, { key: '_reportStatisticsInfo', value: function _reportStatisticsInfo() { var info = {}; info.url = this._ioctl.currentUrl; info.speed = this._ioctl.currentSpeed; info.loaderType = this._ioctl.loaderType; info.currentSegmentIndex = this._currentSegmentIndex; info.totalSegmentCount = this._mediaDataSource.segments.length; this._emitter.emit(_transmuxingEvents2.default.STATISTICS_INFO, info); } }]); return TransmuxingController; }(); exports.default = TransmuxingController; },{"../demux/demux-errors.js":16,"../demux/flv-demuxer.js":18,"../io/io-controller.js":23,"../io/loader.js":24,"../remux/mp4-remuxer.js":38,"../utils/browser.js":39,"../utils/logger.js":41,"./media-info.js":7,"./transmuxing-events.js":13,"events":2}],13:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var TransmuxingEvents = { IO_ERROR: 'io_error', DEMUX_ERROR: 'demux_error', INIT_SEGMENT: 'init_segment', MEDIA_SEGMENT: 'media_segment', LOADING_COMPLETE: 'loading_complete', RECOVERED_EARLY_EOF: 'recovered_early_eof', MEDIA_INFO: 'media_info', STATISTICS_INFO: 'statistics_info', RECOMMEND_SEEKPOINT: 'recommend_seekpoint' }; exports.default = TransmuxingEvents; },{}],14:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _loggingControl = _dereq_('../utils/logging-control.js'); var _loggingControl2 = _interopRequireDefault(_loggingControl); var _polyfill = _dereq_('../utils/polyfill.js'); var _polyfill2 = _interopRequireDefault(_polyfill); var _transmuxingController = _dereq_('./transmuxing-controller.js'); var _transmuxingController2 = _interopRequireDefault(_transmuxingController); var _transmuxingEvents = _dereq_('./transmuxing-events.js'); var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } /* post message to worker: data: { cmd: string param: any } receive message from worker: data: { msg: string, data: any } */ var TransmuxingWorker = function TransmuxingWorker(self) { var TAG = 'TransmuxingWorker'; var controller = null; _polyfill2.default.install(); self.addEventListener('message', function (e) { switch (e.data.cmd) { case 'init': controller = new _transmuxingController2.default(e.data.param[0], e.data.param[1]); controller.on(_transmuxingEvents2.default.IO_ERROR, onIOError.bind(this)); controller.on(_transmuxingEvents2.default.DEMUX_ERROR, onDemuxError.bind(this)); controller.on(_transmuxingEvents2.default.INIT_SEGMENT, onInitSegment.bind(this)); controller.on(_transmuxingEvents2.default.MEDIA_SEGMENT, onMediaSegment.bind(this)); controller.on(_transmuxingEvents2.default.LOADING_COMPLETE, onLoadingComplete.bind(this)); controller.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, onRecoveredEarlyEof.bind(this)); controller.on(_transmuxingEvents2.default.MEDIA_INFO, onMediaInfo.bind(this)); controller.on(_transmuxingEvents2.default.STATISTICS_INFO, onStatisticsInfo.bind(this)); controller.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, onRecommendSeekpoint.bind(this)); break; case 'destroy': if (controller) { controller.destroy(); controller = null; } self.postMessage({ msg: 'destroyed' }); break; case 'start': controller.start(); break; case 'stop': controller.stop(); break; case 'seek': controller.seek(e.data.param); break; case 'pause': controller.pause(); break; case 'resume': controller.resume(); break; case 'logging_config': _loggingControl2.default.applyConfig(e.data.param); break; } }); function onInitSegment(type, initSegment) { var obj = { msg: _transmuxingEvents2.default.INIT_SEGMENT, data: { type: type, data: initSegment } }; self.postMessage(obj, [initSegment.data]); // data: ArrayBuffer } function onMediaSegment(type, mediaSegment) { var obj = { msg: _transmuxingEvents2.default.MEDIA_SEGMENT, data: { type: type, data: mediaSegment } }; self.postMessage(obj, [mediaSegment.data]); // data: ArrayBuffer } function onLoadingComplete() { var obj = { msg: _transmuxingEvents2.default.LOADING_COMPLETE }; self.postMessage(obj); } function onRecoveredEarlyEof() { var obj = { msg: _transmuxingEvents2.default.RECOVERED_EARLY_EOF }; self.postMessage(obj); } function onMediaInfo(mediaInfo) { var obj = { msg: _transmuxingEvents2.default.MEDIA_INFO, data: mediaInfo }; self.postMessage(obj); } function onStatisticsInfo(statInfo) { var obj = { msg: _transmuxingEvents2.default.STATISTICS_INFO, data: statInfo }; self.postMessage(obj); } function onIOError(type, info) { self.postMessage({ msg: _transmuxingEvents2.default.IO_ERROR, data: { type: type, info: info } }); } function onDemuxError(type, info) { self.postMessage({ msg: _transmuxingEvents2.default.DEMUX_ERROR, data: { type: type, info: info } }); } function onRecommendSeekpoint(milliseconds) { self.postMessage({ msg: _transmuxingEvents2.default.RECOMMEND_SEEKPOINT, data: milliseconds }); } }; /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ exports.default = TransmuxingWorker; },{"../utils/logger.js":41,"../utils/logging-control.js":42,"../utils/polyfill.js":43,"./transmuxing-controller.js":12,"./transmuxing-events.js":13}],15:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _utf8Conv = _dereq_('../utils/utf8-conv.js'); var _utf8Conv2 = _interopRequireDefault(_utf8Conv); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var le = function () { var buf = new ArrayBuffer(2); new DataView(buf).setInt16(0, 256, true); // little-endian write return new Int16Array(buf)[0] === 256; // platform-spec read, if equal then LE }(); var AMF = function () { function AMF() { _classCallCheck(this, AMF); } _createClass(AMF, null, [{ key: 'parseScriptData', value: function parseScriptData(arrayBuffer, dataOffset, dataSize) { var data = {}; try { var name = AMF.parseValue(arrayBuffer, dataOffset, dataSize); var value = AMF.parseValue(arrayBuffer, dataOffset + name.size, dataSize - name.size); data[name.data] = value.data; } catch (e) { _logger2.default.e('AMF', e.toString()); } return data; } }, { key: 'parseObject', value: function parseObject(arrayBuffer, dataOffset, dataSize) { if (dataSize < 3) { throw new _exception.IllegalStateException('Data not enough when parse ScriptDataObject'); } var name = AMF.parseString(arrayBuffer, dataOffset, dataSize); var value = AMF.parseValue(arrayBuffer, dataOffset + name.size, dataSize - name.size); var isObjectEnd = value.objectEnd; return { data: { name: name.data, value: value.data }, size: name.size + value.size, objectEnd: isObjectEnd }; } }, { key: 'parseVariable', value: function parseVariable(arrayBuffer, dataOffset, dataSize) { return AMF.parseObject(arrayBuffer, dataOffset, dataSize); } }, { key: 'parseString', value: function parseString(arrayBuffer, dataOffset, dataSize) { if (dataSize < 2) { throw new _exception.IllegalStateException('Data not enough when parse String'); } var v = new DataView(arrayBuffer, dataOffset, dataSize); var length = v.getUint16(0, !le); var str = void 0; if (length > 0) { str = (0, _utf8Conv2.default)(new Uint8Array(arrayBuffer, dataOffset + 2, length)); } else { str = ''; } return { data: str, size: 2 + length }; } }, { key: 'parseLongString', value: function parseLongString(arrayBuffer, dataOffset, dataSize) { if (dataSize < 4) { throw new _exception.IllegalStateException('Data not enough when parse LongString'); } var v = new DataView(arrayBuffer, dataOffset, dataSize); var length = v.getUint32(0, !le); var str = void 0; if (length > 0) { str = (0, _utf8Conv2.default)(new Uint8Array(arrayBuffer, dataOffset + 4, length)); } else { str = ''; } return { data: str, size: 4 + length }; } }, { key: 'parseDate', value: function parseDate(arrayBuffer, dataOffset, dataSize) { if (dataSize < 10) { throw new _exception.IllegalStateException('Data size invalid when parse Date'); } var v = new DataView(arrayBuffer, dataOffset, dataSize); var timestamp = v.getFloat64(0, !le); var localTimeOffset = v.getInt16(8, !le); timestamp += localTimeOffset * 60 * 1000; // get UTC time return { data: new Date(timestamp), size: 8 + 2 }; } }, { key: 'parseValue', value: function parseValue(arrayBuffer, dataOffset, dataSize) { if (dataSize < 1) { throw new _exception.IllegalStateException('Data not enough when parse Value'); } var v = new DataView(arrayBuffer, dataOffset, dataSize); var offset = 1; var type = v.getUint8(0); var value = void 0; var objectEnd = false; try { switch (type) { case 0: // Number(Double) type value = v.getFloat64(1, !le); offset += 8; break; case 1: { // Boolean type var b = v.getUint8(1); value = b ? true : false; offset += 1; break; } case 2: { // String type var amfstr = AMF.parseString(arrayBuffer, dataOffset + 1, dataSize - 1); value = amfstr.data; offset += amfstr.size; break; } case 3: { // Object(s) type value = {}; var terminal = 0; // workaround for malformed Objects which has missing ScriptDataObjectEnd if ((v.getUint32(dataSize - 4, !le) & 0x00FFFFFF) === 9) { terminal = 3; } while (offset < dataSize - 4) { // 4 === type(UI8) + ScriptDataObjectEnd(UI24) var amfobj = AMF.parseObject(arrayBuffer, dataOffset + offset, dataSize - offset - terminal); if (amfobj.objectEnd) { break; } value[amfobj.data.name] = amfobj.data.value; offset += amfobj.size; } if (offset <= dataSize - 3) { var marker = v.getUint32(offset - 1, !le) & 0x00FFFFFF; if (marker === 9) { offset += 3; } } break; } case 8: { // ECMA array type (Mixed array) value = {}; offset += 4; // ECMAArrayLength(UI32) var _terminal = 0; // workaround for malformed MixedArrays which has missing ScriptDataObjectEnd if ((v.getUint32(dataSize - 4, !le) & 0x00FFFFFF) === 9) { _terminal = 3; } while (offset < dataSize - 8) { // 8 === type(UI8) + ECMAArrayLength(UI32) + ScriptDataVariableEnd(UI24) var amfvar = AMF.parseVariable(arrayBuffer, dataOffset + offset, dataSize - offset - _terminal); if (amfvar.objectEnd) { break; } value[amfvar.data.name] = amfvar.data.value; offset += amfvar.size; } if (offset <= dataSize - 3) { var _marker = v.getUint32(offset - 1, !le) & 0x00FFFFFF; if (_marker === 9) { offset += 3; } } break; } case 9: // ScriptDataObjectEnd value = undefined; offset = 1; objectEnd = true; break; case 10: { // Strict array type // ScriptDataValue[n]. NOTE: according to video_file_format_spec_v10_1.pdf value = []; var strictArrayLength = v.getUint32(1, !le); offset += 4; for (var i = 0; i < strictArrayLength; i++) { var val = AMF.parseValue(arrayBuffer, dataOffset + offset, dataSize - offset); value.push(val.data); offset += val.size; } break; } case 11: { // Date type var date = AMF.parseDate(arrayBuffer, dataOffset + 1, dataSize - 1); value = date.data; offset += date.size; break; } case 12: { // Long string type var amfLongStr = AMF.parseString(arrayBuffer, dataOffset + 1, dataSize - 1); value = amfLongStr.data; offset += amfLongStr.size; break; } default: // ignore and skip offset = dataSize; _logger2.default.w('AMF', 'Unsupported AMF value type ' + type); } } catch (e) { _logger2.default.e('AMF', e.toString()); } return { data: value, size: offset, objectEnd: objectEnd }; } }]); return AMF; }(); exports.default = AMF; },{"../utils/exception.js":40,"../utils/logger.js":41,"../utils/utf8-conv.js":44}],16:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var DemuxErrors = { OK: 'OK', FORMAT_ERROR: 'FormatError', FORMAT_UNSUPPORTED: 'FormatUnsupported', CODEC_UNSUPPORTED: 'CodecUnsupported' }; exports.default = DemuxErrors; },{}],17:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _exception = _dereq_('../utils/exception.js'); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // Exponential-Golomb buffer decoder var ExpGolomb = function () { function ExpGolomb(uint8array) { _classCallCheck(this, ExpGolomb); this.TAG = 'ExpGolomb'; this._buffer = uint8array; this._buffer_index = 0; this._total_bytes = uint8array.byteLength; this._total_bits = uint8array.byteLength * 8; this._current_word = 0; this._current_word_bits_left = 0; } _createClass(ExpGolomb, [{ key: 'destroy', value: function destroy() { this._buffer = null; } }, { key: '_fillCurrentWord', value: function _fillCurrentWord() { var buffer_bytes_left = this._total_bytes - this._buffer_index; if (buffer_bytes_left <= 0) { throw new _exception.IllegalStateException('ExpGolomb: _fillCurrentWord() but no bytes available'); } var bytes_read = Math.min(4, buffer_bytes_left); var word = new Uint8Array(4); word.set(this._buffer.subarray(this._buffer_index, this._buffer_index + bytes_read)); this._current_word = new DataView(word.buffer).getUint32(0, false); this._buffer_index += bytes_read; this._current_word_bits_left = bytes_read * 8; } }, { key: 'readBits', value: function readBits(bits) { if (bits > 32) { throw new _exception.InvalidArgumentException('ExpGolomb: readBits() bits exceeded max 32bits!'); } if (bits <= this._current_word_bits_left) { var _result = this._current_word >>> 32 - bits; this._current_word <<= bits; this._current_word_bits_left -= bits; return _result; } var result = this._current_word_bits_left ? this._current_word : 0; result = result >>> 32 - this._current_word_bits_left; var bits_need_left = bits - this._current_word_bits_left; this._fillCurrentWord(); var bits_read_next = Math.min(bits_need_left, this._current_word_bits_left); var result2 = this._current_word >>> 32 - bits_read_next; this._current_word <<= bits_read_next; this._current_word_bits_left -= bits_read_next; result = result << bits_read_next | result2; return result; } }, { key: 'readBool', value: function readBool() { return this.readBits(1) === 1; } }, { key: 'readByte', value: function readByte() { return this.readBits(8); } }, { key: '_skipLeadingZero', value: function _skipLeadingZero() { var this$1 = this; var zero_count = void 0; for (zero_count = 0; zero_count < this._current_word_bits_left; zero_count++) { if (0 !== (this$1._current_word & 0x80000000 >>> zero_count)) { this$1._current_word <<= zero_count; this$1._current_word_bits_left -= zero_count; return zero_count; } } this._fillCurrentWord(); return zero_count + this._skipLeadingZero(); } }, { key: 'readUEG', value: function readUEG() { // unsigned exponential golomb var leading_zeros = this._skipLeadingZero(); return this.readBits(leading_zeros + 1) - 1; } }, { key: 'readSEG', value: function readSEG() { // signed exponential golomb var value = this.readUEG(); if (value & 0x01) { return value + 1 >>> 1; } else { return -1 * (value >>> 1); } } }]); return ExpGolomb; }(); exports.default = ExpGolomb; },{"../utils/exception.js":40}],18:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _amfParser = _dereq_('./amf-parser.js'); var _amfParser2 = _interopRequireDefault(_amfParser); var _spsParser = _dereq_('./sps-parser.js'); var _spsParser2 = _interopRequireDefault(_spsParser); var _demuxErrors = _dereq_('./demux-errors.js'); var _demuxErrors2 = _interopRequireDefault(_demuxErrors); var _mediaInfo = _dereq_('../core/media-info.js'); var _mediaInfo2 = _interopRequireDefault(_mediaInfo); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function Swap16(src) { return src >>> 8 & 0xFF | (src & 0xFF) << 8; } function Swap32(src) { return (src & 0xFF000000) >>> 24 | (src & 0x00FF0000) >>> 8 | (src & 0x0000FF00) << 8 | (src & 0x000000FF) << 24; } function ReadBig32(array, index) { return array[index] << 24 | array[index + 1] << 16 | array[index + 2] << 8 | array[index + 3]; } var FLVDemuxer = function () { function FLVDemuxer(probeData, config) { _classCallCheck(this, FLVDemuxer); this.TAG = 'FLVDemuxer'; this._config = config; this._onError = null; this._onMediaInfo = null; this._onTrackMetadata = null; this._onDataAvailable = null; this._dataOffset = probeData.dataOffset; this._firstParse = true; this._dispatch = false; this._hasAudio = probeData.hasAudioTrack; this._hasVideo = probeData.hasVideoTrack; this._audioInitialMetadataDispatched = false; this._videoInitialMetadataDispatched = false; this._mediaInfo = new _mediaInfo2.default(); this._mediaInfo.hasAudio = this._hasAudio; this._mediaInfo.hasVideo = this._hasVideo; this._metadata = null; this._audioMetadata = null; this._videoMetadata = null; this._naluLengthSize = 4; this._timestampBase = 0; // int32, in milliseconds this._timescale = 1000; this._duration = 0; // int32, in milliseconds this._durationOverrided = false; this._referenceFrameRate = { fixed: true, fps: 23.976, fps_num: 23976, fps_den: 1000 }; this._videoTrack = { type: 'video', id: 1, sequenceNumber: 0, samples: [], length: 0 }; this._audioTrack = { type: 'audio', id: 2, sequenceNumber: 0, samples: [], length: 0 }; this._littleEndian = function () { var buf = new ArrayBuffer(2); new DataView(buf).setInt16(0, 256, true); // little-endian write return new Int16Array(buf)[0] === 256; // platform-spec read, if equal then LE }(); } _createClass(FLVDemuxer, [{ key: 'destroy', value: function destroy() { this._mediaInfo = null; this._metadata = null; this._audioMetadata = null; this._videoMetadata = null; this._videoTrack = null; this._audioTrack = null; this._onError = null; this._onMediaInfo = null; this._onTrackMetadata = null; this._onDataAvailable = null; } }, { key: 'bindDataSource', value: function bindDataSource(loader) { loader.onDataArrival = this.parseChunks.bind(this); return this; } // prototype: function(type: string, metadata: any): void }, { key: 'resetMediaInfo', value: function resetMediaInfo() { this._mediaInfo = new _mediaInfo2.default(); } }, { key: '_isInitialMetadataDispatched', value: function _isInitialMetadataDispatched() { if (this._hasAudio && this._hasVideo) { // both audio & video return this._audioInitialMetadataDispatched && this._videoInitialMetadataDispatched; } if (this._hasAudio && !this._hasVideo) { // audio only return this._audioInitialMetadataDispatched; } if (!this._hasAudio && this._hasVideo) { // video only return this._videoInitialMetadataDispatched; } } // function parseChunks(chunk: ArrayBuffer, byteStart: number): number; }, { key: 'parseChunks', value: function parseChunks(chunk, byteStart) { var this$1 = this; if (!this._onError || !this._onMediaInfo || !this._onTrackMetadata || !this._onDataAvailable) { throw new _exception.IllegalStateException('Flv: onError & onMediaInfo & onTrackMetadata & onDataAvailable callback must be specified'); } var offset = 0; var le = this._littleEndian; if (byteStart === 0) { // buffer with FLV header if (chunk.byteLength > 13) { var probeData = FLVDemuxer.probe(chunk); offset = probeData.dataOffset; byteStart = probeData.dataOffset; } else { return 0; } } if (this._firstParse) { // handle PreviousTagSize0 before Tag1 this._firstParse = false; if (byteStart !== this._dataOffset) { _logger2.default.w(this.TAG, 'First time parsing but chunk byteStart invalid!'); } var v = new DataView(chunk, offset); var prevTagSize0 = v.getUint32(0, !le); if (prevTagSize0 !== 0) { _logger2.default.w(this.TAG, 'PrevTagSize0 !== 0 !!!'); } offset += 4; } while (offset < chunk.byteLength) { this$1._dispatch = true; var _v = new DataView(chunk, offset); if (offset + 11 + 4 > chunk.byteLength) { // data not enough for parsing an flv tag break; } var tagType = _v.getUint8(0); var dataSize = _v.getUint32(0, !le) & 0x00FFFFFF; if (offset + 11 + dataSize + 4 > chunk.byteLength) { // data not enough for parsing actual data body break; } if (tagType !== 8 && tagType !== 9 && tagType !== 18) { _logger2.default.w(this$1.TAG, 'Unsupported tag type ' + tagType + ', skipped'); // consume the whole tag (skip it) offset += 11 + dataSize + 4; continue; } var ts2 = _v.getUint8(4); var ts1 = _v.getUint8(5); var ts0 = _v.getUint8(6); var ts3 = _v.getUint8(7); var timestamp = ts0 | ts1 << 8 | ts2 << 16 | ts3 << 24; var streamId = _v.getUint32(7, !le) & 0x00FFFFFF; if (streamId !== 0) { _logger2.default.w(this$1.TAG, 'Meet tag which has StreamID != 0!'); } var dataOffset = offset + 11; switch (tagType) { case 8: // Audio this$1._parseAudioData(chunk, dataOffset, dataSize, timestamp); break; case 9: // Video this$1._parseVideoData(chunk, dataOffset, dataSize, timestamp, byteStart + offset); break; case 18: // ScriptDataObject this$1._parseScriptData(chunk, dataOffset, dataSize); break; } var prevTagSize = _v.getUint32(11 + dataSize, !le); if (prevTagSize !== 11 + dataSize) { _logger2.default.w(this$1.TAG, 'Invalid PrevTagSize ' + prevTagSize); } offset += 11 + dataSize + 4; // tagBody + dataSize + prevTagSize } // dispatch parsed frames to consumer (typically, the remuxer) if (this._isInitialMetadataDispatched()) { if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) { this._onDataAvailable(this._audioTrack, this._videoTrack); } } return offset; // consumed bytes, just equals latest offset index } }, { key: '_parseScriptData', value: function _parseScriptData(arrayBuffer, dataOffset, dataSize) { var scriptData = _amfParser2.default.parseScriptData(arrayBuffer, dataOffset, dataSize); if (scriptData.hasOwnProperty('onMetaData')) { if (this._metadata) { _logger2.default.w(this.TAG, 'Found another onMetaData tag!'); } this._metadata = scriptData; var onMetaData = this._metadata.onMetaData; if (typeof onMetaData.hasAudio === 'boolean') { // hasAudio this._hasAudio = onMetaData.hasAudio; this._mediaInfo.hasAudio = this._hasAudio; } if (typeof onMetaData.hasVideo === 'boolean') { // hasVideo this._hasVideo = onMetaData.hasVideo; this._mediaInfo.hasVideo = this._hasVideo; } if (typeof onMetaData.audiodatarate === 'number') { // audiodatarate this._mediaInfo.audioDataRate = onMetaData.audiodatarate; } if (typeof onMetaData.videodatarate === 'number') { // videodatarate this._mediaInfo.videoDataRate = onMetaData.videodatarate; } if (typeof onMetaData.width === 'number') { // width this._mediaInfo.width = onMetaData.width; } if (typeof onMetaData.height === 'number') { // height this._mediaInfo.height = onMetaData.height; } if (typeof onMetaData.duration === 'number') { // duration if (!this._durationOverrided) { var duration = Math.floor(onMetaData.duration * this._timescale); this._duration = duration; this._mediaInfo.duration = duration; } } else { this._mediaInfo.duration = 0; } if (typeof onMetaData.framerate === 'number') { // framerate var fps_num = Math.floor(onMetaData.framerate * 1000); if (fps_num > 0) { var fps = fps_num / 1000; this._referenceFrameRate.fixed = true; this._referenceFrameRate.fps = fps; this._referenceFrameRate.fps_num = fps_num; this._referenceFrameRate.fps_den = 1000; this._mediaInfo.fps = fps; } } if (_typeof(onMetaData.keyframes) === 'object') { // keyframes this._mediaInfo.hasKeyframesIndex = true; var keyframes = onMetaData.keyframes; this._mediaInfo.keyframesIndex = this._parseKeyframesIndex(keyframes); onMetaData.keyframes = null; // keyframes has been extracted, remove it } else { this._mediaInfo.hasKeyframesIndex = false; } this._dispatch = false; this._mediaInfo.metadata = onMetaData; _logger2.default.v(this.TAG, 'Parsed onMetaData'); if (this._mediaInfo.isComplete()) { this._onMediaInfo(this._mediaInfo); } } } }, { key: '_parseKeyframesIndex', value: function _parseKeyframesIndex(keyframes) { var this$1 = this; var times = []; var filepositions = []; // ignore first keyframe which is actually AVC Sequence Header (AVCDecoderConfigurationRecord) for (var i = 1; i < keyframes.times.length; i++) { var time = this$1._timestampBase + Math.floor(keyframes.times[i] * 1000); times.push(time); filepositions.push(keyframes.filepositions[i]); } return { times: times, filepositions: filepositions }; } }, { key: '_parseAudioData', value: function _parseAudioData(arrayBuffer, dataOffset, dataSize, tagTimestamp) { if (dataSize <= 1) { _logger2.default.w(this.TAG, 'Flv: Invalid audio packet, missing SoundData payload!'); return; } var meta = this._audioMetadata; var track = this._audioTrack; if (!meta || !meta.codec) { // initial metadata meta = this._audioMetadata = {}; meta.type = 'audio'; meta.id = track.id; meta.timescale = this._timescale; meta.duration = this._duration; var le = this._littleEndian; var v = new DataView(arrayBuffer, dataOffset, dataSize); var soundSpec = v.getUint8(0); var soundFormat = soundSpec >>> 4; if (soundFormat !== 10) { // AAC // TODO: support MP3 audio codec this._onError(_demuxErrors2.default.CODEC_UNSUPPORTED, 'Flv: Unsupported audio codec idx: ' + soundFormat); return; } var soundRate = 0; var soundRateIndex = (soundSpec & 12) >>> 2; var soundRateTable = [5500, 11025, 22050, 44100, 48000]; if (soundRateIndex < soundRateTable.length) { soundRate = soundRateTable[soundRateIndex]; } else { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid audio sample rate idx: ' + soundRateIndex); return; } var soundSize = (soundSpec & 2) >>> 1; // unused var soundType = soundSpec & 1; meta.audioSampleRate = soundRate; meta.channelCount = soundType === 0 ? 1 : 2; meta.refSampleDuration = Math.floor(1024 / meta.audioSampleRate * meta.timescale); meta.codec = 'mp4a.40.5'; } var aacData = this._parseAACAudioData(arrayBuffer, dataOffset + 1, dataSize - 1); if (aacData == undefined) { return; } if (aacData.packetType === 0) { // AAC sequence header (AudioSpecificConfig) if (meta.config) { _logger2.default.w(this.TAG, 'Found another AudioSpecificConfig!'); } var misc = aacData.data; meta.audioSampleRate = misc.samplingRate; meta.channelCount = misc.channelCount; meta.codec = misc.codec; meta.config = misc.config; // The decode result of an aac sample is 1024 PCM samples meta.refSampleDuration = Math.floor(1024 / meta.audioSampleRate * meta.timescale); _logger2.default.v(this.TAG, 'Parsed AudioSpecificConfig'); if (this._isInitialMetadataDispatched()) { // Non-initial metadata, force dispatch (or flush) parsed frames to remuxer if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) { this._onDataAvailable(this._audioTrack, this._videoTrack); } } else { this._audioInitialMetadataDispatched = true; } // then notify new metadata this._dispatch = false; this._onTrackMetadata('audio', meta); var mi = this._mediaInfo; mi.audioCodec = 'mp4a.40.' + misc.originalAudioObjectType; mi.audioSampleRate = meta.audioSampleRate; mi.audioChannelCount = meta.channelCount; if (mi.hasVideo) { if (mi.videoCodec != null) { mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + ',' + mi.audioCodec + '"'; } } else { mi.mimeType = 'video/x-flv; codecs="' + mi.audioCodec + '"'; } if (mi.isComplete()) { this._onMediaInfo(mi); } return; } else if (aacData.packetType === 1) { // AAC raw frame data var dts = this._timestampBase + tagTimestamp; var aacSample = { unit: aacData.data, dts: dts, pts: dts }; track.samples.push(aacSample); track.length += aacData.data.length; } else { _logger2.default.e(this.TAG, 'Flv: Unsupported AAC data type ' + aacData.packetType); } } }, { key: '_parseAACAudioData', value: function _parseAACAudioData(arrayBuffer, dataOffset, dataSize) { if (dataSize <= 1) { _logger2.default.w(this.TAG, 'Flv: Invalid AAC packet, missing AACPacketType or/and Data!'); return; } var result = {}; var array = new Uint8Array(arrayBuffer, dataOffset, dataSize); result.packetType = array[0]; if (array[0] === 0) { result.data = this._parseAACAudioSpecificConfig(arrayBuffer, dataOffset + 1, dataSize - 1); } else { result.data = array.subarray(1); } return result; } }, { key: '_parseAACAudioSpecificConfig', value: function _parseAACAudioSpecificConfig(arrayBuffer, dataOffset, dataSize) { var array = new Uint8Array(arrayBuffer, dataOffset, dataSize); var config = null; var mpegSamplingRates = [96000, 88200, 64000, 48000, 44100, 32000, 24000, 22050, 16000, 12000, 11025, 8000, 7350]; /* Audio Object Type: 0: Null 1: AAC Main 2: AAC LC 3: AAC SSR (Scalable Sample Rate) 4: AAC LTP (Long Term Prediction) 5: HE-AAC / SBR (Spectral Band Replication) 6: AAC Scalable */ var audioObjectType = 0; var originalAudioObjectType = 0; var audioExtensionObjectType = null; var samplingIndex = 0; var extensionSamplingIndex = null; // 5 bits audioObjectType = originalAudioObjectType = array[0] >>> 3; // 4 bits samplingIndex = (array[0] & 0x07) << 1 | array[1] >>> 7; if (samplingIndex < 0 || samplingIndex >= mpegSamplingRates.length) { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: AAC invalid sampling frequency index!'); return; } var samplingFrequence = mpegSamplingRates[samplingIndex]; // 4 bits var channelConfig = (array[1] & 0x78) >>> 3; if (channelConfig < 0 || channelConfig >= 8) { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: AAC invalid channel configuration'); return; } if (audioObjectType === 5) { // HE-AAC? // 4 bits extensionSamplingIndex = (array[1] & 0x07) << 1 | array[2] >>> 7; // 5 bits audioExtensionObjectType = (array[2] & 0x7C) >>> 2; } // workarounds for various browsers var userAgent = self.navigator.userAgent.toLowerCase(); if (userAgent.indexOf('firefox') !== -1) { // firefox: use SBR (HE-AAC) if freq less than 24kHz if (samplingIndex >= 6) { audioObjectType = 5; config = new Array(4); extensionSamplingIndex = samplingIndex - 3; } else { // use LC-AAC audioObjectType = 2; config = new Array(2); extensionSamplingIndex = samplingIndex; } } else if (userAgent.indexOf('android') !== -1) { // android: always use LC-AAC audioObjectType = 2; config = new Array(2); extensionSamplingIndex = samplingIndex; } else { // for other browsers, e.g. chrome... // Always use HE-AAC to make it easier to switch aac codec profile audioObjectType = 5; extensionSamplingIndex = samplingIndex; config = new Array(4); if (samplingIndex >= 6) { extensionSamplingIndex = samplingIndex - 3; } else if (channelConfig === 1) { // Mono channel audioObjectType = 2; config = new Array(2); extensionSamplingIndex = samplingIndex; } } config[0] = audioObjectType << 3; config[0] |= (samplingIndex & 0x0F) >>> 1; config[1] = (samplingIndex & 0x0F) << 7; config[1] |= (channelConfig & 0x0F) << 3; if (audioObjectType === 5) { config[1] |= (extensionSamplingIndex & 0x0F) >>> 1; config[2] = (extensionSamplingIndex & 0x01) << 7; // extended audio object type: force to 2 (LC-AAC) config[2] |= 2 << 2; config[3] = 0; } return { config: config, samplingRate: samplingFrequence, channelCount: channelConfig, codec: 'mp4a.40.' + audioObjectType, originalAudioObjectType: originalAudioObjectType }; } }, { key: '_parseVideoData', value: function _parseVideoData(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition) { if (dataSize <= 1) { _logger2.default.w(this.TAG, 'Flv: Invalid video packet, missing VideoData payload!'); return; } var spec = new Uint8Array(arrayBuffer, dataOffset, dataSize)[0]; var frameType = (spec & 240) >>> 4; var codecId = spec & 15; if (codecId !== 7) { this._onError(_demuxErrors2.default.CODEC_UNSUPPORTED, 'Flv: Unsupported codec in video frame: ' + codecId); return; } this._parseAVCVideoPacket(arrayBuffer, dataOffset + 1, dataSize - 1, tagTimestamp, tagPosition, frameType); } }, { key: '_parseAVCVideoPacket', value: function _parseAVCVideoPacket(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition, frameType) { if (dataSize < 4) { _logger2.default.w(this.TAG, 'Flv: Invalid AVC packet, missing AVCPacketType or/and CompositionTime'); return; } var le = this._littleEndian; var v = new DataView(arrayBuffer, dataOffset, dataSize); var packetType = v.getUint8(0); var cts = v.getUint32(0, !le) & 0x00FFFFFF; if (packetType === 0) { // AVCDecoderConfigurationRecord this._parseAVCDecoderConfigurationRecord(arrayBuffer, dataOffset + 4, dataSize - 4); } else if (packetType === 1) { // One or more Nalus this._parseAVCVideoData(arrayBuffer, dataOffset + 4, dataSize - 4, tagTimestamp, tagPosition, frameType, cts); } else if (packetType === 2) { // empty, AVC end of sequence } else { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid video packet type ' + packetType); return; } } }, { key: '_parseAVCDecoderConfigurationRecord', value: function _parseAVCDecoderConfigurationRecord(arrayBuffer, dataOffset, dataSize) { var this$1 = this; if (dataSize < 7) { _logger2.default.w(this.TAG, 'Flv: Invalid AVCDecoderConfigurationRecord, lack of data!'); return; } var meta = this._videoMetadata; var track = this._videoTrack; var le = this._littleEndian; var v = new DataView(arrayBuffer, dataOffset, dataSize); if (!meta) { meta = this._videoMetadata = {}; meta.type = 'video'; meta.id = track.id; meta.timescale = this._timescale; meta.duration = this._duration; } else { if (typeof meta.avcc !== 'undefined') { _logger2.default.w(this.TAG, 'Found another AVCDecoderConfigurationRecord!'); } } var version = v.getUint8(0); // configurationVersion var avcProfile = v.getUint8(1); // avcProfileIndication var profileCompatibility = v.getUint8(2); // profile_compatibility var avcLevel = v.getUint8(3); // AVCLevelIndication if (version !== 1 || avcProfile === 0) { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid AVCDecoderConfigurationRecord'); return; } this._naluLengthSize = (v.getUint8(4) & 3) + 1; // lengthSizeMinusOne if (this._naluLengthSize !== 3 && this._naluLengthSize !== 4) { // holy shit!!! this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Strange NaluLengthSizeMinusOne: ' + (this._naluLengthSize - 1)); return; } var spsCount = v.getUint8(5) & 31; // numOfSequenceParameterSets if (spsCount === 0 || spsCount > 1) { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid H264 SPS count: ' + spsCount); return; } var offset = 6; for (var i = 0; i < spsCount; i++) { var len = v.getUint16(offset, !le); // sequenceParameterSetLength offset += 2; if (len === 0) { continue; } // Notice: Nalu without startcode header (00 00 00 01) var sps = new Uint8Array(arrayBuffer, dataOffset + offset, len); offset += len; var config = _spsParser2.default.parseSPS(sps); meta.codecWidth = config.codec_size.width; meta.codecHeight = config.codec_size.height; meta.presentWidth = config.present_size.width; meta.presentHeight = config.present_size.height; meta.profile = config.profile_string; meta.level = config.level_string; meta.bitDepth = config.bit_depth; meta.chromaFormat = config.chroma_format; meta.sarRatio = config.sar_ratio; meta.frameRate = config.frame_rate; if (config.frame_rate.fixed === false || config.frame_rate.fps_num === 0 || config.frame_rate.fps_den === 0) { meta.frameRate = this$1._referenceFrameRate; } var fps_den = meta.frameRate.fps_den; var fps_num = meta.frameRate.fps_num; meta.refSampleDuration = Math.floor(meta.timescale * (fps_den / fps_num)); var codecArray = sps.subarray(1, 4); var codecString = 'avc1.'; for (var j = 0; j < 3; j++) { var h = codecArray[j].toString(16); if (h.length < 2) { h = '0' + h; } codecString += h; } meta.codec = codecString; var mi = this$1._mediaInfo; mi.width = meta.codecWidth; mi.height = meta.codecHeight; mi.fps = meta.frameRate.fps; mi.profile = meta.profile; mi.level = meta.level; mi.chromaFormat = config.chroma_format_string; mi.sarNum = meta.sarRatio.width; mi.sarDen = meta.sarRatio.height; mi.videoCodec = codecString; if (mi.hasAudio) { if (mi.audioCodec != null) { mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + ',' + mi.audioCodec + '"'; } } else { mi.mimeType = 'video/x-flv; codecs="' + mi.videoCodec + '"'; } if (mi.isComplete()) { this$1._onMediaInfo(mi); } } var ppsCount = v.getUint8(offset); // numOfPictureParameterSets if (ppsCount === 0 || ppsCount > 1) { this._onError(_demuxErrors2.default.FORMAT_ERROR, 'Flv: Invalid H264 PPS count: ' + ppsCount); return; } offset++; for (var _i = 0; _i < ppsCount; _i++) { var _len = v.getUint16(offset, !le); // pictureParameterSetLength offset += 2; if (_len === 0) { continue; } // pps is useless for extracting video information offset += _len; } meta.avcc = new Uint8Array(dataSize); meta.avcc.set(new Uint8Array(arrayBuffer, dataOffset, dataSize), 0); _logger2.default.v(this.TAG, 'Parsed AVCDecoderConfigurationRecord'); if (this._isInitialMetadataDispatched()) { // flush parsed frames if (this._dispatch && (this._audioTrack.length || this._videoTrack.length)) { this._onDataAvailable(this._audioTrack, this._videoTrack); } } else { this._videoInitialMetadataDispatched = true; } // notify new metadata this._dispatch = false; this._onTrackMetadata('video', meta); } }, { key: '_parseAVCVideoData', value: function _parseAVCVideoData(arrayBuffer, dataOffset, dataSize, tagTimestamp, tagPosition, frameType, cts) { var this$1 = this; var le = this._littleEndian; var v = new DataView(arrayBuffer, dataOffset, dataSize); var units = [], length = 0; var offset = 0; var lengthSize = this._naluLengthSize; var dts = this._timestampBase + tagTimestamp; var keyframe = frameType === 1; // from FLV Frame Type constants while (offset < dataSize) { if (offset + 4 >= dataSize) { _logger2.default.w(this$1.TAG, 'Malformed Nalu near timestamp ' + dts + ', offset = ' + offset + ', dataSize = ' + dataSize); break; // data not enough for next Nalu } // Nalu with length-header (AVC1) var naluSize = v.getUint32(offset, !le); // Big-Endian read if (lengthSize === 3) { naluSize >>>= 8; } if (naluSize > dataSize - lengthSize) { _logger2.default.w(this$1.TAG, 'Malformed Nalus near timestamp ' + dts + ', NaluSize > DataSize!'); return; } var unitType = v.getUint8(offset + lengthSize) & 0x1F; if (unitType === 5) { // IDR keyframe = true; } var data = new Uint8Array(arrayBuffer, dataOffset + offset, lengthSize + naluSize); var unit = { type: unitType, data: data }; units.push(unit); length += data.byteLength; offset += lengthSize + naluSize; } if (units.length) { var track = this._videoTrack; var avcSample = { units: units, length: length, isKeyframe: keyframe, dts: dts, cts: cts, pts: dts + cts }; if (keyframe) { avcSample.fileposition = tagPosition; } track.samples.push(avcSample); track.length += length; } } }, { key: 'onTrackMetadata', get: function get() { return this._onTrackMetadata; }, set: function set(callback) { this._onTrackMetadata = callback; } // prototype: function(mediaInfo: MediaInfo): void }, { key: 'onMediaInfo', get: function get() { return this._onMediaInfo; }, set: function set(callback) { this._onMediaInfo = callback; } // prototype: function(type: number, info: string): void }, { key: 'onError', get: function get() { return this._onError; }, set: function set(callback) { this._onError = callback; } // prototype: function(videoTrack: any, audioTrack: any): void }, { key: 'onDataAvailable', get: function get() { return this._onDataAvailable; }, set: function set(callback) { this._onDataAvailable = callback; } // timestamp base for output samples, must be in milliseconds }, { key: 'timestampBase', get: function get() { return this._timestampBase; }, set: function set(base) { this._timestampBase = base; } }, { key: 'overridedDuration', get: function get() { return this._duration; } // Force-override media duration. Must be in milliseconds, int32 , set: function set(duration) { this._durationOverrided = true; this._duration = duration; this._mediaInfo.duration = duration; } }], [{ key: 'probe', value: function probe(buffer) { var data = new Uint8Array(buffer); var mismatch = { match: false }; if (data[0] !== 0x46 || data[1] !== 0x4C || data[2] !== 0x56 || data[3] !== 0x01) { return mismatch; } var hasAudio = (data[4] & 4) >>> 2 !== 0; var hasVideo = (data[4] & 1) !== 0; if (!hasAudio && !hasVideo) { return mismatch; } var offset = ReadBig32(data, 5); if (offset < 9) { return mismatch; } return { match: true, consumed: offset, dataOffset: offset, hasAudioTrack: hasAudio, hasVideoTrack: hasVideo }; } }]); return FLVDemuxer; }(); exports.default = FLVDemuxer; },{"../core/media-info.js":7,"../utils/exception.js":40,"../utils/logger.js":41,"./amf-parser.js":15,"./demux-errors.js":16,"./sps-parser.js":19}],19:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _expGolomb = _dereq_('./exp-golomb.js'); var _expGolomb2 = _interopRequireDefault(_expGolomb); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var SPSParser = function () { function SPSParser() { _classCallCheck(this, SPSParser); } _createClass(SPSParser, null, [{ key: '_ebsp2rbsp', value: function _ebsp2rbsp(uint8array) { var src = uint8array; var src_length = src.byteLength; var dst = new Uint8Array(src_length); var dst_idx = 0; for (var i = 0; i < src_length; i++) { if (i >= 2) { // Unescape: Skip 0x03 after 00 00 if (src[i] === 0x03 && src[i - 1] === 0x00 && src[i - 2] === 0x00) { continue; } } dst[dst_idx] = src[i]; dst_idx++; } return new Uint8Array(dst.buffer, 0, dst_idx); } }, { key: 'parseSPS', value: function parseSPS(uint8array) { var rbsp = SPSParser._ebsp2rbsp(uint8array); var gb = new _expGolomb2.default(rbsp); gb.readByte(); var profile_idc = gb.readByte(); // profile_idc gb.readByte(); // constraint_set_flags[5] + reserved_zero[3] var level_idc = gb.readByte(); // level_idc gb.readUEG(); // seq_parameter_set_id var profile_string = SPSParser.getProfileString(profile_idc); var level_string = SPSParser.getLevelString(level_idc); var chroma_format_idc = 1; var chroma_format = 420; var chroma_format_table = [0, 420, 422, 444]; var bit_depth = 8; if (profile_idc === 100 || profile_idc === 110 || profile_idc === 122 || profile_idc === 244 || profile_idc === 44 || profile_idc === 83 || profile_idc === 86 || profile_idc === 118 || profile_idc === 128 || profile_idc === 138 || profile_idc === 144) { chroma_format_idc = gb.readUEG(); if (chroma_format_idc === 3) { gb.readBits(1); // separate_colour_plane_flag } if (chroma_format_idc <= 3) { chroma_format = chroma_format_table[chroma_format_idc]; } bit_depth = gb.readUEG() + 8; // bit_depth_luma_minus8 gb.readUEG(); // bit_depth_chroma_minus8 gb.readBits(1); // qpprime_y_zero_transform_bypass_flag if (gb.readBool()) { // seq_scaling_matrix_present_flag var scaling_list_count = chroma_format_idc !== 3 ? 8 : 12; for (var i = 0; i < scaling_list_count; i++) { if (gb.readBool()) { // seq_scaling_list_present_flag if (i < 6) { SPSParser._skipScalingList(gb, 16); } else { SPSParser._skipScalingList(gb, 64); } } } } } gb.readUEG(); // log2_max_frame_num_minus4 var pic_order_cnt_type = gb.readUEG(); if (pic_order_cnt_type === 0) { gb.readUEG(); // log2_max_pic_order_cnt_lsb_minus_4 } else if (pic_order_cnt_type === 1) { gb.readBits(1); // delta_pic_order_always_zero_flag gb.readSEG(); // offset_for_non_ref_pic gb.readSEG(); // offset_for_top_to_bottom_field var num_ref_frames_in_pic_order_cnt_cycle = gb.readUEG(); for (var _i = 0; _i < num_ref_frames_in_pic_order_cnt_cycle; _i++) { gb.readSEG(); // offset_for_ref_frame } } gb.readUEG(); // max_num_ref_frames gb.readBits(1); // gaps_in_frame_num_value_allowed_flag var pic_width_in_mbs_minus1 = gb.readUEG(); var pic_height_in_map_units_minus1 = gb.readUEG(); var frame_mbs_only_flag = gb.readBits(1); if (frame_mbs_only_flag === 0) { gb.readBits(1); // mb_adaptive_frame_field_flag } gb.readBits(1); // direct_8x8_inference_flag var frame_crop_left_offset = 0; var frame_crop_right_offset = 0; var frame_crop_top_offset = 0; var frame_crop_bottom_offset = 0; var frame_cropping_flag = gb.readBool(); if (frame_cropping_flag) { frame_crop_left_offset = gb.readUEG(); frame_crop_right_offset = gb.readUEG(); frame_crop_top_offset = gb.readUEG(); frame_crop_bottom_offset = gb.readUEG(); } var sar_width = 1, sar_height = 1; var fps = 0, fps_fixed = true, fps_num = 0, fps_den = 0; var vui_parameters_present_flag = gb.readBool(); if (vui_parameters_present_flag) { if (gb.readBool()) { // aspect_ratio_info_present_flag var aspect_ratio_idc = gb.readByte(); var sar_w_table = [1, 12, 10, 16, 40, 24, 20, 32, 80, 18, 15, 64, 160, 4, 3, 2]; var sar_h_table = [1, 11, 11, 11, 33, 11, 11, 11, 33, 11, 11, 33, 99, 3, 2, 1]; if (aspect_ratio_idc > 0 && aspect_ratio_idc < 16) { sar_width = sar_w_table[aspect_ratio_idc - 1]; sar_height = sar_h_table[aspect_ratio_idc - 1]; } else if (aspect_ratio_idc === 255) { sar_width = gb.readByte() << 8 | gb.readByte(); sar_height = gb.readByte() << 8 | gb.readByte(); } } if (gb.readBool()) { // overscan_info_present_flag gb.readBool(); // overscan_appropriate_flag } if (gb.readBool()) { // video_signal_type_present_flag gb.readBits(4); // video_format & video_full_range_flag if (gb.readBool()) { // colour_description_present_flag gb.readBits(24); // colour_primaries & transfer_characteristics & matrix_coefficients } } if (gb.readBool()) { // chroma_loc_info_present_flag gb.readUEG(); // chroma_sample_loc_type_top_field gb.readUEG(); // chroma_sample_loc_type_bottom_field } if (gb.readBool()) { // timing_info_present_flag var num_units_in_tick = gb.readBits(32); var time_scale = gb.readBits(32); fps_fixed = gb.readBool(); // fixed_frame_rate_flag fps_num = time_scale; fps_den = num_units_in_tick * 2; fps = fps_num / fps_den; } } var sarScale = 1; if (sar_width !== 1 || sar_height !== 1) { sarScale = sar_width / sar_height; } var crop_unit_x = 0, crop_unit_y = 0; if (chroma_format_idc === 0) { crop_unit_x = 1; crop_unit_y = 2 - frame_mbs_only_flag; } else { var sub_wc = chroma_format_idc === 3 ? 1 : 2; var sub_hc = chroma_format_idc === 1 ? 2 : 1; crop_unit_x = sub_wc; crop_unit_y = sub_hc * (2 - frame_mbs_only_flag); } var codec_width = (pic_width_in_mbs_minus1 + 1) * 16; var codec_height = (2 - frame_mbs_only_flag) * ((pic_height_in_map_units_minus1 + 1) * 16); codec_width -= (frame_crop_left_offset + frame_crop_right_offset) * crop_unit_x; codec_height -= (frame_crop_top_offset + frame_crop_bottom_offset) * crop_unit_y; var present_width = Math.ceil(codec_width * sarScale); gb.destroy(); gb = null; return { profile_string: profile_string, // baseline, high, high10, ... level_string: level_string, // 3, 3.1, 4, 4.1, 5, 5.1, ... bit_depth: bit_depth, // 8bit, 10bit, ... chroma_format: chroma_format, // 4:2:0, 4:2:2, ... chroma_format_string: SPSParser.getChromaFormatString(chroma_format), frame_rate: { fixed: fps_fixed, fps: fps, fps_den: fps_den, fps_num: fps_num }, sar_ratio: { width: sar_width, height: sar_height }, codec_size: { width: codec_width, height: codec_height }, present_size: { width: present_width, height: codec_height } }; } }, { key: '_skipScalingList', value: function _skipScalingList(gb, count) { var last_scale = 8, next_scale = 8; var delta_scale = 0; for (var i = 0; i < count; i++) { if (next_scale !== 0) { delta_scale = gb.readSEG(); next_scale = (last_scale + delta_scale + 256) % 256; } last_scale = next_scale === 0 ? last_scale : next_scale; } } }, { key: 'getProfileString', value: function getProfileString(profile_idc) { switch (profile_idc) { case 66: return 'Baseline'; case 77: return 'Main'; case 88: return 'Extended'; case 100: return 'High'; case 110: return 'High10'; case 122: return 'High422'; case 244: return 'High444'; default: return 'Unknown'; } } }, { key: 'getLevelString', value: function getLevelString(level_idc) { return (level_idc / 10).toFixed(1); } }, { key: 'getChromaFormatString', value: function getChromaFormatString(chroma) { switch (chroma) { case 420: return '4:2:0'; case 422: return '4:2:2'; case 444: return '4:4:4'; default: return 'Unknown'; } } }]); return SPSParser; }(); exports.default = SPSParser; },{"./exp-golomb.js":17}],20:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _polyfill = _dereq_('./utils/polyfill.js'); var _polyfill2 = _interopRequireDefault(_polyfill); var _features = _dereq_('./core/features.js'); var _features2 = _interopRequireDefault(_features); var _flvPlayer = _dereq_('./player/flv-player.js'); var _flvPlayer2 = _interopRequireDefault(_flvPlayer); var _nativePlayer = _dereq_('./player/native-player.js'); var _nativePlayer2 = _interopRequireDefault(_nativePlayer); var _playerEvents = _dereq_('./player/player-events.js'); var _playerEvents2 = _interopRequireDefault(_playerEvents); var _playerErrors = _dereq_('./player/player-errors.js'); var _loggingControl = _dereq_('./utils/logging-control.js'); var _loggingControl2 = _interopRequireDefault(_loggingControl); var _exception = _dereq_('./utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } // here are all the interfaces // install polyfills _polyfill2.default.install(); // factory method function createPlayer(mediaDataSource, optionalConfig) { var mds = mediaDataSource; if (mds == null || (typeof mds === 'undefined' ? 'undefined' : _typeof(mds)) !== 'object') { throw new _exception.InvalidArgumentException('MediaDataSource must be an javascript object!'); } if (!mds.hasOwnProperty('type')) { throw new _exception.InvalidArgumentException('MediaDataSource must has type field to indicate video file type!'); } switch (mds.type) { case 'flv': return new _flvPlayer2.default(mds, optionalConfig); default: return new _nativePlayer2.default(mds, optionalConfig); } } // feature detection function isSupported() { return _features2.default.supportMSEH264Playback(); } function getFeatureList() { return _features2.default.getFeatureList(); } // interfaces var flvjs = {}; flvjs.createPlayer = createPlayer; flvjs.isSupported = isSupported; flvjs.getFeatureList = getFeatureList; flvjs.Events = _playerEvents2.default; flvjs.ErrorTypes = _playerErrors.ErrorTypes; flvjs.ErrorDetails = _playerErrors.ErrorDetails; flvjs.FlvPlayer = _flvPlayer2.default; flvjs.NativePlayer = _nativePlayer2.default; flvjs.LoggingControl = _loggingControl2.default; Object.defineProperty(flvjs, 'version', { enumerable: true, get: function get() { // replaced by browserify-versionify transform return '1.1.0'; } }); exports.default = flvjs; },{"./core/features.js":6,"./player/flv-player.js":32,"./player/native-player.js":33,"./player/player-errors.js":34,"./player/player-events.js":35,"./utils/exception.js":40,"./utils/logging-control.js":42,"./utils/polyfill.js":43}],21:[function(_dereq_,module,exports){ 'use strict'; // entry/index file // make it compatible with browserify's umd wrapper module.exports = _dereq_('./flv.js').default; },{"./flv.js":20}],22:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _get = function get(object, property, receiver) { if (object === null) { object = Function.prototype; } var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _browser = _dereq_('../utils/browser.js'); var _browser2 = _interopRequireDefault(_browser); var _loader = _dereq_('./loader.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* fetch + stream IO loader. Currently working on chrome 43+. * fetch provides a better alternative http API to XMLHttpRequest * * fetch spec https://fetch.spec.whatwg.org/ * stream spec https://streams.spec.whatwg.org/ */ var FetchStreamLoader = function (_BaseLoader) { _inherits(FetchStreamLoader, _BaseLoader); _createClass(FetchStreamLoader, null, [{ key: 'isSupported', value: function isSupported() { try { // fetch + stream is broken on Microsoft Edge. Disable for now. // see https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/8196907/ return self.fetch && self.ReadableStream && !_browser2.default.msedge; } catch (e) { return false; } } }]); function FetchStreamLoader(seekHandler) { _classCallCheck(this, FetchStreamLoader); var _this = _possibleConstructorReturn(this, (FetchStreamLoader.__proto__ || Object.getPrototypeOf(FetchStreamLoader)).call(this, 'fetch-stream-loader')); _this.TAG = 'FetchStreamLoader'; _this._seekHandler = seekHandler; _this._needStash = true; _this._requestAbort = false; _this._contentLength = null; _this._receivedLength = 0; return _this; } _createClass(FetchStreamLoader, [{ key: 'destroy', value: function destroy() { if (this.isWorking()) { this.abort(); } _get(FetchStreamLoader.prototype.__proto__ || Object.getPrototypeOf(FetchStreamLoader.prototype), 'destroy', this).call(this); } }, { key: 'open', value: function open(dataSource, range) { var _this2 = this; this._dataSource = dataSource; this._range = range; var seekConfig = this._seekHandler.getConfig(dataSource.url, range); var headers = new self.Headers(); if (_typeof(seekConfig.headers) === 'object') { var configHeaders = seekConfig.headers; for (var key in configHeaders) { if (configHeaders.hasOwnProperty(key)) { headers.append(key, configHeaders[key]); } } } var url = seekConfig.url; var params = { method: 'GET', headers: headers, mode: 'cors', cache: 'default' }; // cors is enabled by default if (dataSource.cors === false) { // no-cors means 'disregard cors policy', which can only be used in ServiceWorker params.mode = 'same-origin'; } // withCredentials is disabled by default if (dataSource.withCredentials) { params.credentials = 'include'; } this._status = _loader.LoaderStatus.kConnecting; self.fetch(url, params).then(function (res) { if (_this2._requestAbort) { _this2._requestAbort = false; _this2._status = _loader.LoaderStatus.kIdle; return; } if (res.ok && res.status >= 200 && res.status <= 299) { var lengthHeader = res.headers.get('Content-Length'); if (lengthHeader != null) { _this2._contentLength = parseInt(lengthHeader); if (_this2._contentLength !== 0) { if (_this2._onContentLengthKnown) { _this2._onContentLengthKnown(_this2._contentLength); } } } return _this2._pump.call(_this2, res.body.getReader()); } else { _this2._status = _loader.LoaderStatus.kError; if (_this2._onError) { _this2._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: res.status, msg: res.statusText }); } else { throw new _exception.RuntimeException('FetchStreamLoader: Http code invalid, ' + res.status + ' ' + res.statusText); } } }).catch(function (e) { _this2._status = _loader.LoaderStatus.kError; if (_this2._onError) { _this2._onError(_loader.LoaderErrors.EXCEPTION, { code: -1, msg: e.message }); } else { throw e; } }); } }, { key: 'abort', value: function abort() { this._requestAbort = true; } }, { key: '_pump', value: function _pump(reader) { var _this3 = this; // ReadableStreamReader return reader.read().then(function (result) { if (result.done) { _this3._status = _loader.LoaderStatus.kComplete; if (_this3._onComplete) { _this3._onComplete(_this3._range.from, _this3._range.from + _this3._receivedLength - 1); } } else { if (_this3._requestAbort === true) { _this3._requestAbort = false; _this3._status = _loader.LoaderStatus.kComplete; return reader.cancel(); } _this3._status = _loader.LoaderStatus.kBuffering; var chunk = result.value.buffer; var byteStart = _this3._range.from + _this3._receivedLength; _this3._receivedLength += chunk.byteLength; if (_this3._onDataArrival) { _this3._onDataArrival(chunk, byteStart, _this3._receivedLength); } return _this3._pump(reader); } }).catch(function (e) { _this3._status = _loader.LoaderStatus.kError; var type = 0; var info = null; if (e.code === 19 && ( // NETWORK_ERR _this3._contentLength === null || _this3._contentLength !== null && _this3._receivedLength < _this3._contentLength)) { type = _loader.LoaderErrors.EARLY_EOF; info = { code: e.code, msg: 'Fetch stream meet Early-EOF' }; } else { type = _loader.LoaderErrors.EXCEPTION; info = { code: e.code, msg: e.message }; } if (_this3._onError) { _this3._onError(type, info); } else { throw new _exception.RuntimeException(info.msg); } }); } }]); return FetchStreamLoader; }(_loader.BaseLoader); exports.default = FetchStreamLoader; },{"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],23:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _speedSampler = _dereq_('./speed-sampler.js'); var _speedSampler2 = _interopRequireDefault(_speedSampler); var _loader = _dereq_('./loader.js'); var _fetchStreamLoader = _dereq_('./fetch-stream-loader.js'); var _fetchStreamLoader2 = _interopRequireDefault(_fetchStreamLoader); var _xhrMozChunkedLoader = _dereq_('./xhr-moz-chunked-loader.js'); var _xhrMozChunkedLoader2 = _interopRequireDefault(_xhrMozChunkedLoader); var _xhrMsstreamLoader = _dereq_('./xhr-msstream-loader.js'); var _xhrMsstreamLoader2 = _interopRequireDefault(_xhrMsstreamLoader); var _xhrRangeLoader = _dereq_('./xhr-range-loader.js'); var _xhrRangeLoader2 = _interopRequireDefault(_xhrRangeLoader); var _websocketLoader = _dereq_('./websocket-loader.js'); var _websocketLoader2 = _interopRequireDefault(_websocketLoader); var _rangeSeekHandler = _dereq_('./range-seek-handler.js'); var _rangeSeekHandler2 = _interopRequireDefault(_rangeSeekHandler); var _paramSeekHandler = _dereq_('./param-seek-handler.js'); var _paramSeekHandler2 = _interopRequireDefault(_paramSeekHandler); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /** * DataSource: { * url: string, * filesize: number, * cors: boolean, * withCredentials: boolean * } * */ // Manage IO Loaders var IOController = function () { function IOController(dataSource, config, extraData) { _classCallCheck(this, IOController); this.TAG = 'IOController'; this._config = config; this._extraData = extraData; this._stashInitialSize = 1024 * 384; // default initial size: 384KB if (config.isLive === true) { this._stashInitialSize = 1024 * 512; // default live initial size: 512KB } if (config.stashInitialSize != undefined && config.stashInitialSize > 0) { // apply from config this._stashInitialSize = config.stashInitialSize; } this._stashUsed = 0; this._stashSize = this._stashInitialSize; this._bufferSize = 1024 * 1024 * 3; // initial size: 3MB this._stashBuffer = new ArrayBuffer(this._bufferSize); this._stashByteStart = 0; this._enableStash = true; if (config.enableStashBuffer === false) { this._enableStash = false; } this._loader = null; this._loaderClass = null; this._seekHandler = null; this._dataSource = dataSource; this._isWebSocketURL = /wss?:\/\/(.+?)\//.test(dataSource.url); this._refTotalLength = dataSource.filesize ? dataSource.filesize : null; this._totalLength = this._refTotalLength; this._fullRequestFlag = false; this._currentRange = null; this._speedNormalized = 0; this._speedSampler = new _speedSampler2.default(); this._speedNormalizeList = [64, 128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096]; this._isEarlyEofReconnecting = false; this._paused = false; this._resumeFrom = 0; this._onDataArrival = null; this._onSeeked = null; this._onError = null; this._onComplete = null; this._onRecoveredEarlyEof = null; this._selectSeekHandler(); this._selectLoader(); this._createLoader(); } _createClass(IOController, [{ key: 'destroy', value: function destroy() { if (this._loader.isWorking()) { this._loader.abort(); } this._loader.destroy(); this._loader = null; this._loaderClass = null; this._dataSource = null; this._stashBuffer = null; this._stashUsed = this._stashSize = this._bufferSize = this._stashByteStart = 0; this._currentRange = null; this._speedSampler = null; this._isEarlyEofReconnecting = false; this._onDataArrival = null; this._onSeeked = null; this._onError = null; this._onComplete = null; this._onRecoveredEarlyEof = null; this._extraData = null; } }, { key: 'isWorking', value: function isWorking() { return this._loader && this._loader.isWorking() && !this._paused; } }, { key: 'isPaused', value: function isPaused() { return this._paused; } }, { key: '_selectSeekHandler', value: function _selectSeekHandler() { var config = this._config; if (config.seekType === 'range') { this._seekHandler = new _rangeSeekHandler2.default(this._config.rangeLoadZeroStart); } else if (config.seekType === 'param') { var paramStart = config.seekParamStart || 'bstart'; var paramEnd = config.seekParamEnd || 'bend'; this._seekHandler = new _paramSeekHandler2.default(paramStart, paramEnd); } else if (config.seekType === 'custom') { if (typeof config.customSeekHandler !== 'function') { throw new _exception.InvalidArgumentException('Custom seekType specified in config but invalid customSeekHandler!'); } this._seekHandler = new config.customSeekHandler(); } else { throw new _exception.InvalidArgumentException('Invalid seekType in config: ' + config.seekType); } } }, { key: '_selectLoader', value: function _selectLoader() { if (this._isWebSocketURL) { this._loaderClass = _websocketLoader2.default; } else if (_fetchStreamLoader2.default.isSupported()) { this._loaderClass = _fetchStreamLoader2.default; } else if (_xhrMozChunkedLoader2.default.isSupported()) { this._loaderClass = _xhrMozChunkedLoader2.default; } else if (_xhrRangeLoader2.default.isSupported()) { this._loaderClass = _xhrRangeLoader2.default; } else { throw new _exception.RuntimeException('Your browser doesn\'t support xhr with arraybuffer responseType!'); } } }, { key: '_createLoader', value: function _createLoader() { this._loader = new this._loaderClass(this._seekHandler); if (this._loader.needStashBuffer === false) { this._enableStash = false; } this._loader.onContentLengthKnown = this._onContentLengthKnown.bind(this); this._loader.onDataArrival = this._onLoaderChunkArrival.bind(this); this._loader.onComplete = this._onLoaderComplete.bind(this); this._loader.onError = this._onLoaderError.bind(this); } }, { key: 'open', value: function open(optionalFrom) { this._currentRange = { from: 0, to: -1 }; if (optionalFrom) { this._currentRange.from = optionalFrom; } this._speedSampler.reset(); if (!optionalFrom) { this._fullRequestFlag = true; } this._loader.open(this._dataSource, Object.assign({}, this._currentRange)); } }, { key: 'abort', value: function abort() { this._loader.abort(); if (this._paused) { this._paused = false; this._resumeFrom = 0; } } }, { key: 'pause', value: function pause() { if (this.isWorking()) { this._loader.abort(); if (this._stashUsed !== 0) { this._resumeFrom = this._stashByteStart; this._currentRange.to = this._stashByteStart - 1; } else { this._resumeFrom = this._currentRange.to + 1; } this._stashUsed = 0; this._stashByteStart = 0; this._paused = true; } } }, { key: 'resume', value: function resume() { if (this._paused) { this._paused = false; var bytes = this._resumeFrom; this._resumeFrom = 0; this._internalSeek(bytes, true); } } }, { key: 'seek', value: function seek(bytes) { this._paused = false; this._stashUsed = 0; this._stashByteStart = 0; this._internalSeek(bytes, true); } /** * When seeking request is from media seeking, unconsumed stash data should be dropped * However, stash data shouldn't be dropped if seeking requested from http reconnection * * @dropUnconsumed: Ignore and discard all unconsumed data in stash buffer */ }, { key: '_internalSeek', value: function _internalSeek(bytes, dropUnconsumed) { if (this._loader.isWorking()) { this._loader.abort(); } // dispatch & flush stash buffer before seek this._flushStashBuffer(dropUnconsumed); this._loader.destroy(); this._loader = null; var requestRange = { from: bytes, to: -1 }; this._currentRange = { from: requestRange.from, to: -1 }; this._speedSampler.reset(); this._stashSize = this._stashInitialSize; this._createLoader(); this._loader.open(this._dataSource, requestRange); if (this._onSeeked) { this._onSeeked(); } } }, { key: 'updateUrl', value: function updateUrl(url) { if (!url || typeof url !== 'string' || url.length === 0) { throw new _exception.InvalidArgumentException('Url must be a non-empty string!'); } this._dataSource.url = url; // TODO: replace with new url } }, { key: '_expandBuffer', value: function _expandBuffer(expectedBytes) { var bufferNewSize = this._stashSize; while (bufferNewSize + 1024 * 1024 * 1 < expectedBytes) { bufferNewSize *= 2; } bufferNewSize += 1024 * 1024 * 1; // bufferSize = stashSize + 1MB if (bufferNewSize === this._bufferSize) { return; } var newBuffer = new ArrayBuffer(bufferNewSize); if (this._stashUsed > 0) { // copy existing data into new buffer var stashOldArray = new Uint8Array(this._stashBuffer, 0, this._stashUsed); var stashNewArray = new Uint8Array(newBuffer, 0, bufferNewSize); stashNewArray.set(stashOldArray, 0); } this._stashBuffer = newBuffer; this._bufferSize = bufferNewSize; } }, { key: '_normalizeSpeed', value: function _normalizeSpeed(input) { var list = this._speedNormalizeList; var last = list.length - 1; var mid = 0; var lbound = 0; var ubound = last; if (input < list[0]) { return list[0]; } // binary search while (lbound <= ubound) { mid = lbound + Math.floor((ubound - lbound) / 2); if (mid === last || input >= list[mid] && input < list[mid + 1]) { return list[mid]; } else if (list[mid] < input) { lbound = mid + 1; } else { ubound = mid - 1; } } } }, { key: '_adjustStashSize', value: function _adjustStashSize(normalized) { var stashSizeKB = 0; if (this._config.isLive) { // live stream: always use single normalized speed for size of stashSizeKB stashSizeKB = normalized; } else { if (normalized < 512) { stashSizeKB = normalized; } else if (normalized >= 512 && normalized <= 1024) { stashSizeKB = Math.floor(normalized * 1.5); } else { stashSizeKB = normalized * 2; } } if (stashSizeKB > 8192) { stashSizeKB = 8192; } var bufferSize = stashSizeKB * 1024 + 1024 * 1024 * 1; // stashSize + 1MB if (this._bufferSize < bufferSize) { this._expandBuffer(bufferSize); } this._stashSize = stashSizeKB * 1024; } }, { key: '_dispatchChunks', value: function _dispatchChunks(chunks, byteStart) { this._currentRange.to = byteStart + chunks.byteLength - 1; return this._onDataArrival(chunks, byteStart); } }, { key: '_onContentLengthKnown', value: function _onContentLengthKnown(contentLength) { if (contentLength && this._fullRequestFlag) { this._totalLength = contentLength; this._fullRequestFlag = false; } } }, { key: '_onLoaderChunkArrival', value: function _onLoaderChunkArrival(chunk, byteStart, receivedLength) { if (!this._onDataArrival) { throw new _exception.IllegalStateException('IOController: No existing consumer (onDataArrival) callback!'); } if (this._paused) { return; } if (this._isEarlyEofReconnecting) { // Auto-reconnect for EarlyEof succeed, notify to upper-layer by callback this._isEarlyEofReconnecting = false; if (this._onRecoveredEarlyEof) { this._onRecoveredEarlyEof(); } } this._speedSampler.addBytes(chunk.byteLength); // adjust stash buffer size according to network speed dynamically var KBps = this._speedSampler.lastSecondKBps; if (KBps !== 0) { var normalized = this._normalizeSpeed(KBps); if (this._speedNormalized !== normalized) { this._speedNormalized = normalized; this._adjustStashSize(normalized); } } if (!this._enableStash) { // disable stash if (this._stashUsed === 0) { // dispatch chunk directly to consumer; // check ret value (consumed bytes) and stash unconsumed to stashBuffer var consumed = this._dispatchChunks(chunk, byteStart); if (consumed < chunk.byteLength) { // unconsumed data remain. var remain = chunk.byteLength - consumed; if (remain > this._bufferSize) { this._expandBuffer(remain); } var stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize); stashArray.set(new Uint8Array(chunk, consumed), 0); this._stashUsed += remain; this._stashByteStart = byteStart + consumed; } } else { // else: Merge chunk into stashBuffer, and dispatch stashBuffer to consumer. if (this._stashUsed + chunk.byteLength > this._bufferSize) { this._expandBuffer(this._stashUsed + chunk.byteLength); } var _stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize); _stashArray.set(new Uint8Array(chunk), this._stashUsed); this._stashUsed += chunk.byteLength; var _consumed = this._dispatchChunks(this._stashBuffer.slice(0, this._stashUsed), this._stashByteStart); if (_consumed < this._stashUsed && _consumed > 0) { // unconsumed data remain var remainArray = new Uint8Array(this._stashBuffer, _consumed); _stashArray.set(remainArray, 0); } this._stashUsed -= _consumed; this._stashByteStart += _consumed; } } else { // enable stash if (this._stashUsed === 0 && this._stashByteStart === 0) { // seeked? or init chunk? // This is the first chunk after seek action this._stashByteStart = byteStart; } if (this._stashUsed + chunk.byteLength <= this._stashSize) { // just stash var _stashArray2 = new Uint8Array(this._stashBuffer, 0, this._stashSize); _stashArray2.set(new Uint8Array(chunk), this._stashUsed); this._stashUsed += chunk.byteLength; } else { // stashUsed + chunkSize > stashSize, size limit exceeded var _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize); if (this._stashUsed > 0) { // There're stash datas in buffer // dispatch the whole stashBuffer, and stash remain data // then append chunk to stashBuffer (stash) var buffer = this._stashBuffer.slice(0, this._stashUsed); var _consumed2 = this._dispatchChunks(buffer, this._stashByteStart); if (_consumed2 < buffer.byteLength) { if (_consumed2 > 0) { var _remainArray = new Uint8Array(buffer, _consumed2); _stashArray3.set(_remainArray, 0); this._stashUsed = _remainArray.byteLength; this._stashByteStart += _consumed2; } } else { this._stashUsed = 0; this._stashByteStart += _consumed2; } if (this._stashUsed + chunk.byteLength > this._bufferSize) { this._expandBuffer(this._stashUsed + chunk.byteLength); _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize); } _stashArray3.set(new Uint8Array(chunk), this._stashUsed); this._stashUsed += chunk.byteLength; } else { // stash buffer empty, but chunkSize > stashSize (oh, holy shit) // dispatch chunk directly and stash remain data var _consumed3 = this._dispatchChunks(chunk, byteStart); if (_consumed3 < chunk.byteLength) { var _remain = chunk.byteLength - _consumed3; if (_remain > this._bufferSize) { this._expandBuffer(_remain); _stashArray3 = new Uint8Array(this._stashBuffer, 0, this._bufferSize); } _stashArray3.set(new Uint8Array(chunk, _consumed3), 0); this._stashUsed += _remain; this._stashByteStart = byteStart + _consumed3; } } } } } }, { key: '_flushStashBuffer', value: function _flushStashBuffer(dropUnconsumed) { if (this._stashUsed > 0) { var buffer = this._stashBuffer.slice(0, this._stashUsed); var consumed = this._dispatchChunks(buffer, this._stashByteStart); var remain = buffer.byteLength - consumed; if (consumed < buffer.byteLength) { if (dropUnconsumed) { _logger2.default.w(this.TAG, remain + ' bytes unconsumed data remain when flush buffer, dropped'); } else { if (consumed > 0) { var stashArray = new Uint8Array(this._stashBuffer, 0, this._bufferSize); var remainArray = new Uint8Array(buffer, consumed); stashArray.set(remainArray, 0); this._stashUsed = remainArray.byteLength; this._stashByteStart += consumed; } return 0; } } this._stashUsed = 0; this._stashByteStart = 0; return remain; } return 0; } }, { key: '_onLoaderComplete', value: function _onLoaderComplete(from, to) { // Force-flush stash buffer, and drop unconsumed data this._flushStashBuffer(true); if (this._onComplete) { this._onComplete(this._extraData); } } }, { key: '_onLoaderError', value: function _onLoaderError(type, data) { _logger2.default.e(this.TAG, 'Loader error, code = ' + data.code + ', msg = ' + data.msg); this._flushStashBuffer(false); if (this._isEarlyEofReconnecting) { // Auto-reconnect for EarlyEof failed, throw UnrecoverableEarlyEof error to upper-layer this._isEarlyEofReconnecting = false; type = _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF; } switch (type) { case _loader.LoaderErrors.EARLY_EOF: { if (!this._config.isLive) { // Do internal http reconnect if not live stream if (this._totalLength) { var nextFrom = this._currentRange.to + 1; if (nextFrom < this._totalLength) { _logger2.default.w(this.TAG, 'Connection lost, trying reconnect...'); this._isEarlyEofReconnecting = true; this._internalSeek(nextFrom, false); } return; } // else: We don't know totalLength, throw UnrecoverableEarlyEof } // live stream: throw UnrecoverableEarlyEof error to upper-layer type = _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF; break; } case _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF: case _loader.LoaderErrors.CONNECTING_TIMEOUT: case _loader.LoaderErrors.HTTP_STATUS_CODE_INVALID: case _loader.LoaderErrors.EXCEPTION: break; } if (this._onError) { this._onError(type, data); } else { throw new _exception.RuntimeException('IOException: ' + data.msg); } } }, { key: 'status', get: function get() { return this._loader.status; } }, { key: 'extraData', get: function get() { return this._extraData; }, set: function set(data) { this._extraData = data; } // prototype: function onDataArrival(chunks: ArrayBuffer, byteStart: number): number }, { key: 'onDataArrival', get: function get() { return this._onDataArrival; }, set: function set(callback) { this._onDataArrival = callback; } }, { key: 'onSeeked', get: function get() { return this._onSeeked; }, set: function set(callback) { this._onSeeked = callback; } // prototype: function onError(type: number, info: {code: number, msg: string}): void }, { key: 'onError', get: function get() { return this._onError; }, set: function set(callback) { this._onError = callback; } }, { key: 'onComplete', get: function get() { return this._onComplete; }, set: function set(callback) { this._onComplete = callback; } }, { key: 'onRecoveredEarlyEof', get: function get() { return this._onRecoveredEarlyEof; }, set: function set(callback) { this._onRecoveredEarlyEof = callback; } }, { key: 'currentUrl', get: function get() { return this._dataSource.url; } // in KB/s }, { key: 'currentSpeed', get: function get() { if (this._loaderClass === _xhrRangeLoader2.default) { // SpeedSampler is inaccuracy if loader is RangeLoader return this._loader.currentSpeed; } return this._speedSampler.lastSecondKBps; } }, { key: 'loaderType', get: function get() { return this._loader.type; } }]); return IOController; }(); exports.default = IOController; },{"../utils/exception.js":40,"../utils/logger.js":41,"./fetch-stream-loader.js":22,"./loader.js":24,"./param-seek-handler.js":25,"./range-seek-handler.js":26,"./speed-sampler.js":27,"./websocket-loader.js":28,"./xhr-moz-chunked-loader.js":29,"./xhr-msstream-loader.js":30,"./xhr-range-loader.js":31}],24:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.BaseLoader = exports.LoaderErrors = exports.LoaderStatus = undefined; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _exception = _dereq_('../utils/exception.js'); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var LoaderStatus = exports.LoaderStatus = { kIdle: 0, kConnecting: 1, kBuffering: 2, kError: 3, kComplete: 4 }; var LoaderErrors = exports.LoaderErrors = { OK: 'OK', EXCEPTION: 'Exception', HTTP_STATUS_CODE_INVALID: 'HttpStatusCodeInvalid', CONNECTING_TIMEOUT: 'ConnectingTimeout', EARLY_EOF: 'EarlyEof', UNRECOVERABLE_EARLY_EOF: 'UnrecoverableEarlyEof' }; /* Loader has callbacks which have following prototypes: * function onContentLengthKnown(contentLength: number): void * function onDataArrival(chunk: ArrayBuffer, byteStart: number, receivedLength: number): void * function onError(errorType: number, errorInfo: {code: number, msg: string}): void * function onComplete(rangeFrom: number, rangeTo: number): void */ var BaseLoader = exports.BaseLoader = function () { function BaseLoader(typeName) { _classCallCheck(this, BaseLoader); this._type = typeName || 'undefined'; this._status = LoaderStatus.kIdle; this._needStash = false; // callbacks this._onContentLengthKnown = null; this._onDataArrival = null; this._onError = null; this._onComplete = null; } _createClass(BaseLoader, [{ key: 'destroy', value: function destroy() { this._status = LoaderStatus.kIdle; this._onContentLengthKnown = null; this._onDataArrival = null; this._onError = null; this._onComplete = null; } }, { key: 'isWorking', value: function isWorking() { return this._status === LoaderStatus.kConnecting || this._status === LoaderStatus.kBuffering; } }, { key: 'open', // pure virtual value: function open(dataSource, range) { throw new _exception.NotImplementedException('Unimplemented abstract function!'); } }, { key: 'abort', value: function abort() { throw new _exception.NotImplementedException('Unimplemented abstract function!'); } }, { key: 'type', get: function get() { return this._type; } }, { key: 'status', get: function get() { return this._status; } }, { key: 'needStashBuffer', get: function get() { return this._needStash; } }, { key: 'onContentLengthKnown', get: function get() { return this._onContentLengthKnown; }, set: function set(callback) { this._onContentLengthKnown = callback; } }, { key: 'onDataArrival', get: function get() { return this._onDataArrival; }, set: function set(callback) { this._onDataArrival = callback; } }, { key: 'onError', get: function get() { return this._onError; }, set: function set(callback) { this._onError = callback; } }, { key: 'onComplete', get: function get() { return this._onComplete; }, set: function set(callback) { this._onComplete = callback; } }]); return BaseLoader; }(); },{"../utils/exception.js":40}],25:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var ParamSeekHandler = function () { function ParamSeekHandler(paramStart, paramEnd) { _classCallCheck(this, ParamSeekHandler); this._startName = paramStart; this._endName = paramEnd; } _createClass(ParamSeekHandler, [{ key: 'getConfig', value: function getConfig(baseUrl, range) { var url = baseUrl; if (range.from !== 0 || range.to !== -1) { var needAnd = true; if (url.indexOf('?') === -1) { url += '?'; needAnd = false; } if (needAnd) { url += '&'; } url += this._startName + '=' + range.from.toString(); if (range.to !== -1) { url += '&' + this._endName + '=' + range.to.toString(); } } return { url: url, headers: {} }; } }]); return ParamSeekHandler; }(); exports.default = ParamSeekHandler; },{}],26:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var RangeSeekHandler = function () { function RangeSeekHandler(zeroStart) { _classCallCheck(this, RangeSeekHandler); this._zeroStart = zeroStart || false; } _createClass(RangeSeekHandler, [{ key: 'getConfig', value: function getConfig(url, range) { var headers = {}; if (range.from !== 0 || range.to !== -1) { var param = void 0; if (range.to !== -1) { param = 'bytes=' + range.from.toString() + '-' + range.to.toString(); } else { param = 'bytes=' + range.from.toString() + '-'; } headers['Range'] = param; } else if (this._zeroStart) { headers['Range'] = 'bytes=0-'; } return { url: url, headers: headers }; } }]); return RangeSeekHandler; }(); exports.default = RangeSeekHandler; },{}],27:[function(_dereq_,module,exports){ "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // Utility class to calculate realtime network I/O speed var SpeedSampler = function () { function SpeedSampler() { _classCallCheck(this, SpeedSampler); // milliseconds this._firstCheckpoint = 0; this._lastCheckpoint = 0; this._intervalBytes = 0; this._totalBytes = 0; this._lastSecondBytes = 0; // compatibility detection if (self.performance && self.performance.now) { this._now = self.performance.now.bind(self.performance); } else { this._now = Date.now; } } _createClass(SpeedSampler, [{ key: "reset", value: function reset() { this._firstCheckpoint = this._lastCheckpoint = 0; this._totalBytes = this._intervalBytes = 0; this._lastSecondBytes = 0; } }, { key: "addBytes", value: function addBytes(bytes) { if (this._firstCheckpoint === 0) { this._firstCheckpoint = this._now(); this._lastCheckpoint = this._firstCheckpoint; this._intervalBytes += bytes; this._totalBytes += bytes; } else if (this._now() - this._lastCheckpoint < 1000) { this._intervalBytes += bytes; this._totalBytes += bytes; } else { // duration >= 1000 this._lastSecondBytes = this._intervalBytes; this._intervalBytes = bytes; this._totalBytes += bytes; this._lastCheckpoint = this._now(); } } }, { key: "currentKBps", get: function get() { this.addBytes(0); var durationSeconds = (this._now() - this._lastCheckpoint) / 1000; if (durationSeconds == 0) { durationSeconds = 1; } return this._intervalBytes / durationSeconds / 1024; } }, { key: "lastSecondKBps", get: function get() { this.addBytes(0); if (this._lastSecondBytes !== 0) { return this._lastSecondBytes / 1024; } else { // lastSecondBytes === 0 if (this._now() - this._lastCheckpoint >= 500) { // if time interval since last checkpoint has exceeded 500ms // the speed is nearly accurate return this.currentKBps; } else { // We don't know return 0; } } } }, { key: "averageKBps", get: function get() { var durationSeconds = (this._now() - this._firstCheckpoint) / 1000; return this._totalBytes / durationSeconds / 1024; } }]); return SpeedSampler; }(); exports.default = SpeedSampler; },{}],28:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _get = function get(object, property, receiver) { if (object === null) { object = Function.prototype; } var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _loader = _dereq_('./loader.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // For FLV over WebSocket live stream var WebSocketLoader = function (_BaseLoader) { _inherits(WebSocketLoader, _BaseLoader); _createClass(WebSocketLoader, null, [{ key: 'isSupported', value: function isSupported() { try { return typeof self.WebSocket !== 'undefined'; } catch (e) { return false; } } }]); function WebSocketLoader() { _classCallCheck(this, WebSocketLoader); var _this = _possibleConstructorReturn(this, (WebSocketLoader.__proto__ || Object.getPrototypeOf(WebSocketLoader)).call(this, 'websocket-loader')); _this.TAG = 'WebSocketLoader'; _this._needStash = true; _this._ws = null; _this._requestAbort = false; _this._receivedLength = 0; return _this; } _createClass(WebSocketLoader, [{ key: 'destroy', value: function destroy() { if (this._ws) { this.abort(); } _get(WebSocketLoader.prototype.__proto__ || Object.getPrototypeOf(WebSocketLoader.prototype), 'destroy', this).call(this); } }, { key: 'open', value: function open(dataSource) { try { var ws = this._ws = new self.WebSocket(dataSource.url); ws.binaryType = 'arraybuffer'; ws.onopen = this._onWebSocketOpen.bind(this); ws.onclose = this._onWebSocketClose.bind(this); ws.onmessage = this._onWebSocketMessage.bind(this); ws.onerror = this._onWebSocketError.bind(this); this._status = _loader.LoaderStatus.kConnecting; } catch (e) { this._status = _loader.LoaderStatus.kError; var info = { code: e.code, msg: e.message }; if (this._onError) { this._onError(_loader.LoaderErrors.EXCEPTION, info); } else { throw new _exception.RuntimeException(info.msg); } } } }, { key: 'abort', value: function abort() { var ws = this._ws; if (ws && (ws.readyState === 0 || ws.readyState === 1)) { // CONNECTING || OPEN this._requestAbort = true; ws.close(); } this._ws = null; this._status = _loader.LoaderStatus.kComplete; } }, { key: '_onWebSocketOpen', value: function _onWebSocketOpen(e) { this._status = _loader.LoaderStatus.kBuffering; } }, { key: '_onWebSocketClose', value: function _onWebSocketClose(e) { if (this._requestAbort === true) { this._requestAbort = false; return; } this._status = _loader.LoaderStatus.kComplete; if (this._onComplete) { this._onComplete(0, this._receivedLength - 1); } } }, { key: '_onWebSocketMessage', value: function _onWebSocketMessage(e) { var _this2 = this; if (e.data instanceof ArrayBuffer) { this._dispatchArrayBuffer(e.data); } else if (e.data instanceof Blob) { (function () { var reader = new FileReader(); reader.onload = function () { _this2._dispatchArrayBuffer(reader.result); }; reader.readAsArrayBuffer(e.data); })(); } else { this._status = _loader.LoaderStatus.kError; var info = { code: -1, msg: 'Unsupported WebSocket message type: ' + e.data.constructor.name }; if (this._onError) { this._onError(_loader.LoaderErrors.EXCEPTION, info); } else { throw new _exception.RuntimeException(info.msg); } } } }, { key: '_dispatchArrayBuffer', value: function _dispatchArrayBuffer(arraybuffer) { var chunk = arraybuffer; var byteStart = this._receivedLength; this._receivedLength += chunk.byteLength; if (this._onDataArrival) { this._onDataArrival(chunk, byteStart, this._receivedLength); } } }, { key: '_onWebSocketError', value: function _onWebSocketError(e) { this._status = _loader.LoaderStatus.kError; var info = { code: e.code, msg: e.message }; if (this._onError) { this._onError(_loader.LoaderErrors.EXCEPTION, info); } else { throw new _exception.RuntimeException(info.msg); } } }]); return WebSocketLoader; }(_loader.BaseLoader); exports.default = WebSocketLoader; },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],29:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _get = function get(object, property, receiver) { if (object === null) { object = Function.prototype; } var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _loader = _dereq_('./loader.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // For FireFox browser which supports `xhr.responseType = 'moz-chunked-arraybuffer'` var MozChunkedLoader = function (_BaseLoader) { _inherits(MozChunkedLoader, _BaseLoader); _createClass(MozChunkedLoader, null, [{ key: 'isSupported', value: function isSupported() { try { var xhr = new XMLHttpRequest(); // Firefox 37- requires .open() to be called before setting responseType xhr.open('GET', 'https://example.com', true); xhr.responseType = 'moz-chunked-arraybuffer'; return xhr.responseType === 'moz-chunked-arraybuffer'; } catch (e) { _logger2.default.w('MozChunkedLoader', e.message); return false; } } }]); function MozChunkedLoader(seekHandler) { _classCallCheck(this, MozChunkedLoader); var _this = _possibleConstructorReturn(this, (MozChunkedLoader.__proto__ || Object.getPrototypeOf(MozChunkedLoader)).call(this, 'xhr-moz-chunked-loader')); _this.TAG = 'MozChunkedLoader'; _this._seekHandler = seekHandler; _this._needStash = true; _this._xhr = null; _this._requestAbort = false; _this._contentLength = null; _this._receivedLength = 0; return _this; } _createClass(MozChunkedLoader, [{ key: 'destroy', value: function destroy() { if (this.isWorking()) { this.abort(); } if (this._xhr) { this._xhr.onreadystatechange = null; this._xhr.onprogress = null; this._xhr.onloadend = null; this._xhr.onerror = null; this._xhr = null; } _get(MozChunkedLoader.prototype.__proto__ || Object.getPrototypeOf(MozChunkedLoader.prototype), 'destroy', this).call(this); } }, { key: 'open', value: function open(dataSource, range) { this._dataSource = dataSource; this._range = range; var seekConfig = this._seekHandler.getConfig(dataSource.url, range); var xhr = this._xhr = new XMLHttpRequest(); xhr.open('GET', seekConfig.url, true); xhr.responseType = 'moz-chunked-arraybuffer'; xhr.onreadystatechange = this._onReadyStateChange.bind(this); xhr.onprogress = this._onProgress.bind(this); xhr.onloadend = this._onLoadEnd.bind(this); xhr.onerror = this._onXhrError.bind(this); // cors is auto detected and enabled by xhr // withCredentials is disabled by default if (dataSource.withCredentials && xhr['withCredentials']) { xhr.withCredentials = true; } if (_typeof(seekConfig.headers) === 'object') { var headers = seekConfig.headers; for (var key in headers) { if (headers.hasOwnProperty(key)) { xhr.setRequestHeader(key, headers[key]); } } } this._status = _loader.LoaderStatus.kConnecting; xhr.send(); } }, { key: 'abort', value: function abort() { this._requestAbort = true; if (this._xhr) { this._xhr.abort(); } this._status = _loader.LoaderStatus.kComplete; } }, { key: '_onReadyStateChange', value: function _onReadyStateChange(e) { var xhr = e.target; if (xhr.readyState === 2) { // HEADERS_RECEIVED if (xhr.status !== 0 && (xhr.status < 200 || xhr.status > 299)) { this._status = _loader.LoaderStatus.kError; if (this._onError) { this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText }); } else { throw new _exception.RuntimeException('MozChunkedLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText); } } else { this._status = _loader.LoaderStatus.kBuffering; } } } }, { key: '_onProgress', value: function _onProgress(e) { if (this._contentLength === null) { if (e.total !== null && e.total !== 0) { this._contentLength = e.total; if (this._onContentLengthKnown) { this._onContentLengthKnown(this._contentLength); } } } var chunk = e.target.response; var byteStart = this._range.from + this._receivedLength; this._receivedLength += chunk.byteLength; if (this._onDataArrival) { this._onDataArrival(chunk, byteStart, this._receivedLength); } } }, { key: '_onLoadEnd', value: function _onLoadEnd(e) { if (this._requestAbort === true) { this._requestAbort = false; return; } else if (this._status === _loader.LoaderStatus.kError) { return; } this._status = _loader.LoaderStatus.kComplete; if (this._onComplete) { this._onComplete(this._range.from, this._range.from + this._receivedLength - 1); } } }, { key: '_onXhrError', value: function _onXhrError(e) { this._status = _loader.LoaderStatus.kError; var type = 0; var info = null; if (this._contentLength && e.loaded < this._contentLength) { type = _loader.LoaderErrors.EARLY_EOF; info = { code: -1, msg: 'Moz-Chunked stream meet Early-Eof' }; } else { type = _loader.LoaderErrors.EXCEPTION; info = { code: -1, msg: e.constructor.name + ' ' + e.type }; } if (this._onError) { this._onError(type, info); } else { throw new _exception.RuntimeException(info.msg); } } }]); return MozChunkedLoader; }(_loader.BaseLoader); exports.default = MozChunkedLoader; },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],30:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _get = function get(object, property, receiver) { if (object === null) { object = Function.prototype; } var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _loader = _dereq_('./loader.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* Notice: ms-stream may cause IE/Edge browser crash if seek too frequently!!! * The browser may crash in wininet.dll. Disable for now. * * For IE11/Edge browser by microsoft which supports `xhr.responseType = 'ms-stream'` * Notice that ms-stream API sucks. The buffer is always expanding along with downloading. * * We need to abort the xhr if buffer size exceeded limit size (e.g. 16 MiB), then do reconnect. * in order to release previous ArrayBuffer to avoid memory leak * * Otherwise, the ArrayBuffer will increase to a terrible size that equals final file size. */ var MSStreamLoader = function (_BaseLoader) { _inherits(MSStreamLoader, _BaseLoader); _createClass(MSStreamLoader, null, [{ key: 'isSupported', value: function isSupported() { try { if (typeof self.MSStream === 'undefined' || typeof self.MSStreamReader === 'undefined') { return false; } var xhr = new XMLHttpRequest(); xhr.open('GET', 'https://example.com', true); xhr.responseType = 'ms-stream'; return xhr.responseType === 'ms-stream'; } catch (e) { _logger2.default.w('MSStreamLoader', e.message); return false; } } }]); function MSStreamLoader(seekHandler) { _classCallCheck(this, MSStreamLoader); var _this = _possibleConstructorReturn(this, (MSStreamLoader.__proto__ || Object.getPrototypeOf(MSStreamLoader)).call(this, 'xhr-msstream-loader')); _this.TAG = 'MSStreamLoader'; _this._seekHandler = seekHandler; _this._needStash = true; _this._xhr = null; _this._reader = null; // MSStreamReader _this._totalRange = null; _this._currentRange = null; _this._contentLength = null; _this._receivedLength = 0; _this._bufferLimit = 16 * 1024 * 1024; // 16MB _this._lastTimeBufferSize = 0; _this._isReconnecting = false; return _this; } _createClass(MSStreamLoader, [{ key: 'destroy', value: function destroy() { if (this.isWorking()) { this.abort(); } if (this._reader) { this._reader.onprogress = null; this._reader.onload = null; this._reader.onerror = null; this._reader = null; } if (this._xhr) { this._xhr.onreadystatechange = null; this._xhr = null; } _get(MSStreamLoader.prototype.__proto__ || Object.getPrototypeOf(MSStreamLoader.prototype), 'destroy', this).call(this); } }, { key: 'open', value: function open(dataSource, range) { this._internalOpen(dataSource, range, false); } }, { key: '_internalOpen', value: function _internalOpen(dataSource, range, isSubrange) { this._dataSource = dataSource; if (!isSubrange) { this._totalRange = range; } else { this._currentRange = range; } var seekConfig = this._seekHandler.getConfig(dataSource.url, range); var reader = this._reader = new self.MSStreamReader(); reader.onprogress = this._msrOnProgress.bind(this); reader.onload = this._msrOnLoad.bind(this); reader.onerror = this._msrOnError.bind(this); var xhr = this._xhr = new XMLHttpRequest(); xhr.open('GET', seekConfig.url, true); xhr.responseType = 'ms-stream'; xhr.onreadystatechange = this._xhrOnReadyStateChange.bind(this); xhr.onerror = this._xhrOnError.bind(this); if (dataSource.withCredentials) { xhr.withCredentials = true; } if (_typeof(seekConfig.headers) === 'object') { var headers = seekConfig.headers; for (var key in headers) { if (headers.hasOwnProperty(key)) { xhr.setRequestHeader(key, headers[key]); } } } if (this._isReconnecting) { this._isReconnecting = false; } else { this._status = _loader.LoaderStatus.kConnecting; } xhr.send(); } }, { key: 'abort', value: function abort() { this._internalAbort(); this._status = _loader.LoaderStatus.kComplete; } }, { key: '_internalAbort', value: function _internalAbort() { if (this._reader) { if (this._reader.readyState === 1) { // LOADING this._reader.abort(); } this._reader.onprogress = null; this._reader.onload = null; this._reader.onerror = null; this._reader = null; } if (this._xhr) { this._xhr.abort(); this._xhr.onreadystatechange = null; this._xhr = null; } } }, { key: '_xhrOnReadyStateChange', value: function _xhrOnReadyStateChange(e) { var xhr = e.target; if (xhr.readyState === 3) { if (xhr.status >= 200 && xhr.status <= 299) { this._status = _loader.LoaderStatus.kBuffering; var lengthHeader = xhr.getResponseHeader('Content-Length'); if (lengthHeader != null && this._contentLength == null) { var length = parseInt(lengthHeader); if (length > 0) { this._contentLength = length; if (this._onContentLengthKnown) { this._onContentLengthKnown(this._contentLength); } } } var msstream = xhr.response; this._reader.readAsArrayBuffer(msstream); } else { this._status = _loader.LoaderStatus.kError; if (this._onError) { this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText }); } else { throw new _exception.RuntimeException('MSStreamLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText); } } } } }, { key: '_xhrOnError', value: function _xhrOnError(e) { this._status = _loader.LoaderStatus.kError; var type = _loader.LoaderErrors.EXCEPTION; var info = { code: -1, msg: e.constructor.name + ' ' + e.type }; if (this._onError) { this._onError(type, info); } else { throw new _exception.RuntimeException(info.msg); } } }, { key: '_msrOnProgress', value: function _msrOnProgress(e) { var reader = e.target; var bigbuffer = reader.result; if (bigbuffer == null) { // result may be null, workaround for buggy M$ this._doReconnectIfNeeded(); return; } var slice = bigbuffer.slice(this._lastTimeBufferSize); this._lastTimeBufferSize = bigbuffer.byteLength; var byteStart = this._totalRange.from + this._receivedLength; this._receivedLength += slice.byteLength; if (this._onDataArrival) { this._onDataArrival(slice, byteStart, this._receivedLength); } if (bigbuffer.byteLength >= this._bufferLimit) { _logger2.default.v(this.TAG, 'MSStream buffer exceeded max size near ' + (byteStart + slice.byteLength) + ', reconnecting...'); this._doReconnectIfNeeded(); } } }, { key: '_doReconnectIfNeeded', value: function _doReconnectIfNeeded() { if (this._contentLength == null || this._receivedLength < this._contentLength) { this._isReconnecting = true; this._lastTimeBufferSize = 0; this._internalAbort(); var range = { from: this._totalRange.from + this._receivedLength, to: -1 }; this._internalOpen(this._dataSource, range, true); } } }, { key: '_msrOnLoad', value: function _msrOnLoad(e) { // actually it is onComplete event this._status = _loader.LoaderStatus.kComplete; if (this._onComplete) { this._onComplete(this._totalRange.from, this._totalRange.from + this._receivedLength - 1); } } }, { key: '_msrOnError', value: function _msrOnError(e) { this._status = _loader.LoaderStatus.kError; var type = 0; var info = null; if (this._contentLength && this._receivedLength < this._contentLength) { type = _loader.LoaderErrors.EARLY_EOF; info = { code: -1, msg: 'MSStream meet Early-Eof' }; } else { type = _loader.LoaderErrors.EARLY_EOF; info = { code: -1, msg: e.constructor.name + ' ' + e.type }; } if (this._onError) { this._onError(type, info); } else { throw new _exception.RuntimeException(info.msg); } } }]); return MSStreamLoader; }(_loader.BaseLoader); exports.default = MSStreamLoader; },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24}],31:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _get = function get(object, property, receiver) { if (object === null) { object = Function.prototype; } var desc = Object.getOwnPropertyDescriptor(object, property); if (desc === undefined) { var parent = Object.getPrototypeOf(object); if (parent === null) { return undefined; } else { return get(parent, property, receiver); } } else if ("value" in desc) { return desc.value; } else { var getter = desc.get; if (getter === undefined) { return undefined; } return getter.call(receiver); } }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _speedSampler = _dereq_('./speed-sampler.js'); var _speedSampler2 = _interopRequireDefault(_speedSampler); var _loader = _dereq_('./loader.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // Universal IO Loader, implemented by adding Range header in xhr's request header var RangeLoader = function (_BaseLoader) { _inherits(RangeLoader, _BaseLoader); _createClass(RangeLoader, null, [{ key: 'isSupported', value: function isSupported() { try { var xhr = new XMLHttpRequest(); xhr.open('GET', 'https://example.com', true); xhr.responseType = 'arraybuffer'; return xhr.responseType === 'arraybuffer'; } catch (e) { _logger2.default.w('RangeLoader', e.message); return false; } } }]); function RangeLoader(seekHandler) { _classCallCheck(this, RangeLoader); var _this = _possibleConstructorReturn(this, (RangeLoader.__proto__ || Object.getPrototypeOf(RangeLoader)).call(this, 'xhr-range-loader')); _this.TAG = 'RangeLoader'; _this._seekHandler = seekHandler; _this._needStash = false; _this._chunkSizeKBList = [128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096, 5120, 6144, 7168, 8192]; _this._currentChunkSizeKB = 384; _this._currentSpeedNormalized = 0; _this._zeroSpeedChunkCount = 0; _this._xhr = null; _this._speedSampler = new _speedSampler2.default(); _this._requestAbort = false; _this._waitForTotalLength = false; _this._totalLengthReceived = false; _this._currentRequestRange = null; _this._totalLength = null; // size of the entire file _this._contentLength = null; // Content-Length of entire request range _this._receivedLength = 0; // total received bytes _this._lastTimeLoaded = 0; // received bytes of current request sub-range return _this; } _createClass(RangeLoader, [{ key: 'destroy', value: function destroy() { if (this.isWorking()) { this.abort(); } if (this._xhr) { this._xhr.onreadystatechange = null; this._xhr.onprogress = null; this._xhr.onload = null; this._xhr.onerror = null; this._xhr = null; } _get(RangeLoader.prototype.__proto__ || Object.getPrototypeOf(RangeLoader.prototype), 'destroy', this).call(this); } }, { key: 'open', value: function open(dataSource, range) { this._dataSource = dataSource; this._range = range; this._status = _loader.LoaderStatus.kConnecting; if (!this._totalLengthReceived) { // We need total filesize this._waitForTotalLength = true; this._internalOpen(this._dataSource, { from: 0, to: -1 }); } else { // We have filesize, start loading this._openSubRange(); } } }, { key: '_openSubRange', value: function _openSubRange() { var chunkSize = this._currentChunkSizeKB * 1024; var from = this._range.from + this._receivedLength; var to = from + chunkSize; if (this._contentLength != null) { if (to - this._range.from >= this._contentLength) { to = this._range.from + this._contentLength - 1; } } this._currentRequestRange = { from: from, to: to }; this._internalOpen(this._dataSource, this._currentRequestRange); } }, { key: '_internalOpen', value: function _internalOpen(dataSource, range) { this._lastTimeLoaded = 0; var seekConfig = this._seekHandler.getConfig(dataSource.url, range); var xhr = this._xhr = new XMLHttpRequest(); xhr.open('GET', seekConfig.url, true); xhr.responseType = 'arraybuffer'; xhr.onreadystatechange = this._onReadyStateChange.bind(this); xhr.onprogress = this._onProgress.bind(this); xhr.onload = this._onLoad.bind(this); xhr.onerror = this._onXhrError.bind(this); if (dataSource.withCredentials && xhr['withCredentials']) { xhr.withCredentials = true; } if (_typeof(seekConfig.headers) === 'object') { var headers = seekConfig.headers; for (var key in headers) { if (headers.hasOwnProperty(key)) { xhr.setRequestHeader(key, headers[key]); } } } xhr.send(); } }, { key: 'abort', value: function abort() { this._requestAbort = true; this._internalAbort(); this._status = _loader.LoaderStatus.kComplete; } }, { key: '_internalAbort', value: function _internalAbort() { if (this._xhr) { this._xhr.onreadystatechange = null; this._xhr.onprogress = null; this._xhr.onload = null; this._xhr.onerror = null; this._xhr.abort(); this._xhr = null; } } }, { key: '_onReadyStateChange', value: function _onReadyStateChange(e) { var xhr = e.target; if (xhr.readyState === 2) { // HEADERS_RECEIVED if (xhr.status >= 200 && xhr.status < 300) { if (this._waitForTotalLength) { return; } this._status = _loader.LoaderStatus.kBuffering; } else { this._status = _loader.LoaderStatus.kError; if (this._onError) { this._onError(_loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, { code: xhr.status, msg: xhr.statusText }); } else { throw new _exception.RuntimeException('RangeLoader: Http code invalid, ' + xhr.status + ' ' + xhr.statusText); } } } } }, { key: '_onProgress', value: function _onProgress(e) { if (this._contentLength === null) { var openNextRange = false; if (this._waitForTotalLength) { this._waitForTotalLength = false; this._totalLengthReceived = true; openNextRange = true; var total = e.total; this._internalAbort(); if (total != null & total !== 0) { this._totalLength = total; } } // calculate currrent request range's contentLength if (this._range.to === -1) { this._contentLength = this._totalLength - this._range.from; } else { // to !== -1 this._contentLength = this._range.to - this._range.from + 1; } if (openNextRange) { this._openSubRange(); return; } if (this._onContentLengthKnown) { this._onContentLengthKnown(this._contentLength); } } var delta = e.loaded - this._lastTimeLoaded; this._lastTimeLoaded = e.loaded; this._speedSampler.addBytes(delta); } }, { key: '_normalizeSpeed', value: function _normalizeSpeed(input) { var list = this._chunkSizeKBList; var last = list.length - 1; var mid = 0; var lbound = 0; var ubound = last; if (input < list[0]) { return list[0]; } while (lbound <= ubound) { mid = lbound + Math.floor((ubound - lbound) / 2); if (mid === last || input >= list[mid] && input < list[mid + 1]) { return list[mid]; } else if (list[mid] < input) { lbound = mid + 1; } else { ubound = mid - 1; } } } }, { key: '_onLoad', value: function _onLoad(e) { if (this._waitForTotalLength) { this._waitForTotalLength = false; return; } this._lastTimeLoaded = 0; var KBps = this._speedSampler.lastSecondKBps; if (KBps === 0) { this._zeroSpeedChunkCount++; if (this._zeroSpeedChunkCount >= 3) { // Try get currentKBps after 3 chunks KBps = this._speedSampler.currentKBps; } } if (KBps !== 0) { var normalized = this._normalizeSpeed(KBps); if (this._currentSpeedNormalized !== normalized) { this._currentSpeedNormalized = normalized; this._currentChunkSizeKB = normalized; } } var chunk = e.target.response; var byteStart = this._range.from + this._receivedLength; this._receivedLength += chunk.byteLength; var reportComplete = false; if (this._contentLength != null && this._receivedLength < this._contentLength) { // continue load next chunk this._openSubRange(); } else { reportComplete = true; } // dispatch received chunk if (this._onDataArrival) { this._onDataArrival(chunk, byteStart, this._receivedLength); } if (reportComplete) { this._status = _loader.LoaderStatus.kComplete; if (this._onComplete) { this._onComplete(this._range.from, this._range.from + this._receivedLength - 1); } } } }, { key: '_onXhrError', value: function _onXhrError(e) { this._status = _loader.LoaderStatus.kError; var type = 0; var info = null; if (this._contentLength && this._receivedLength > 0 && this._receivedLength < this._contentLength) { type = _loader.LoaderErrors.EARLY_EOF; info = { code: -1, msg: 'RangeLoader meet Early-Eof' }; } else { type = _loader.LoaderErrors.EXCEPTION; info = { code: -1, msg: e.constructor.name + ' ' + e.type }; } if (this._onError) { this._onError(type, info); } else { throw new _exception.RuntimeException(info.msg); } } }, { key: 'currentSpeed', get: function get() { return this._speedSampler.lastSecondKBps; } }]); return RangeLoader; }(_loader.BaseLoader); exports.default = RangeLoader; },{"../utils/exception.js":40,"../utils/logger.js":41,"./loader.js":24,"./speed-sampler.js":27}],32:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _browser = _dereq_('../utils/browser.js'); var _browser2 = _interopRequireDefault(_browser); var _playerEvents = _dereq_('./player-events.js'); var _playerEvents2 = _interopRequireDefault(_playerEvents); var _transmuxer = _dereq_('../core/transmuxer.js'); var _transmuxer2 = _interopRequireDefault(_transmuxer); var _transmuxingEvents = _dereq_('../core/transmuxing-events.js'); var _transmuxingEvents2 = _interopRequireDefault(_transmuxingEvents); var _mseController = _dereq_('../core/mse-controller.js'); var _mseController2 = _interopRequireDefault(_mseController); var _mseEvents = _dereq_('../core/mse-events.js'); var _mseEvents2 = _interopRequireDefault(_mseEvents); var _playerErrors = _dereq_('./player-errors.js'); var _config = _dereq_('../config.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var FlvPlayer = function () { function FlvPlayer(mediaDataSource, config) { _classCallCheck(this, FlvPlayer); this.TAG = 'FlvPlayer'; this._type = 'FlvPlayer'; this._emitter = new _events2.default(); this._config = (0, _config.createDefaultConfig)(); if ((typeof config === 'undefined' ? 'undefined' : _typeof(config)) === 'object') { Object.assign(this._config, config); } if (mediaDataSource.type.toLowerCase() !== 'flv') { throw new _exception.InvalidArgumentException('FlvPlayer requires an flv MediaDataSource input!'); } if (mediaDataSource.isLive === true) { this._config.isLive = true; } this.e = { onvLoadedMetadata: this._onvLoadedMetadata.bind(this), onvSeeking: this._onvSeeking.bind(this), onvCanPlay: this._onvCanPlay.bind(this), onvStalled: this._onvStalled.bind(this), onvProgress: this._onvProgress.bind(this) }; if (self.performance && self.performance.now) { this._now = self.performance.now.bind(self.performance); } else { this._now = Date.now; } this._pendingSeekTime = null; // in seconds this._requestSetTime = false; this._seekpointRecord = null; this._progressChecker = null; this._mediaDataSource = mediaDataSource; this._mediaElement = null; this._msectl = null; this._transmuxer = null; this._mseSourceOpened = false; this._hasPendingLoad = false; this._receivedCanPlay = false; this._mediaInfo = null; this._statisticsInfo = null; var chromeNeedIDRFix = _browser2.default.chrome && (_browser2.default.version.major < 50 || _browser2.default.version.major === 50 && _browser2.default.version.build < 2661); this._alwaysSeekKeyframe = chromeNeedIDRFix || _browser2.default.msedge || _browser2.default.msie ? true : false; if (this._alwaysSeekKeyframe) { this._config.accurateSeek = false; } } _createClass(FlvPlayer, [{ key: 'destroy', value: function destroy() { if (this._progressChecker != null) { window.clearInterval(this._progressChecker); this._progressChecker = null; } if (this._transmuxer) { this.unload(); } if (this._mediaElement) { this.detachMediaElement(); } this.e = null; this._mediaDataSource = null; this._emitter.removeAllListeners(); this._emitter = null; } }, { key: 'on', value: function on(event, listener) { var _this = this; if (event === _playerEvents2.default.MEDIA_INFO) { if (this._mediaInfo != null) { Promise.resolve().then(function () { _this._emitter.emit(_playerEvents2.default.MEDIA_INFO, _this.mediaInfo); }); } } else if (event === _playerEvents2.default.STATISTICS_INFO) { if (this._statisticsInfo != null) { Promise.resolve().then(function () { _this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, _this.statisticsInfo); }); } } this._emitter.addListener(event, listener); } }, { key: 'off', value: function off(event, listener) { this._emitter.removeListener(event, listener); } }, { key: 'attachMediaElement', value: function attachMediaElement(mediaElement) { var _this2 = this; this._mediaElement = mediaElement; mediaElement.addEventListener('loadedmetadata', this.e.onvLoadedMetadata); mediaElement.addEventListener('seeking', this.e.onvSeeking); mediaElement.addEventListener('canplay', this.e.onvCanPlay); mediaElement.addEventListener('stalled', this.e.onvStalled); mediaElement.addEventListener('progress', this.e.onvProgress); this._msectl = new _mseController2.default(); this._msectl.on(_mseEvents2.default.UPDATE_END, this._onmseUpdateEnd.bind(this)); this._msectl.on(_mseEvents2.default.BUFFER_FULL, this._onmseBufferFull.bind(this)); this._msectl.on(_mseEvents2.default.SOURCE_OPEN, function () { _this2._mseSourceOpened = true; if (_this2._hasPendingLoad) { _this2._hasPendingLoad = false; _this2.load(); } }); this._msectl.on(_mseEvents2.default.ERROR, function (info) { _this2._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.MEDIA_ERROR, _playerErrors.ErrorDetails.MEDIA_MSE_ERROR, info); }); this._msectl.attachMediaElement(mediaElement); if (this._pendingSeekTime != null) { try { mediaElement.currentTime = this._pendingSeekTime; this._pendingSeekTime = null; } catch (e) { // IE11 may throw InvalidStateError if readyState === 0 // We can defer set currentTime operation after loadedmetadata } } } }, { key: 'detachMediaElement', value: function detachMediaElement() { if (this._mediaElement) { this._msectl.detachMediaElement(); this._mediaElement.removeEventListener('loadedmetadata', this.e.onvLoadedMetadata); this._mediaElement.removeEventListener('seeking', this.e.onvSeeking); this._mediaElement.removeEventListener('canplay', this.e.onvCanPlay); this._mediaElement.removeEventListener('stalled', this.e.onvStalled); this._mediaElement.removeEventListener('progress', this.e.onvProgress); this._mediaElement = null; } if (this._msectl) { this._msectl.destroy(); this._msectl = null; } } }, { key: 'load', value: function load() { var _this3 = this; if (!this._mediaElement) { throw new _exception.IllegalStateException('HTMLMediaElement must be attached before load()!'); } if (this._transmuxer) { throw new _exception.IllegalStateException('FlvPlayer.load() has been called, please call unload() first!'); } if (this._hasPendingLoad) { return; } if (this._config.deferLoadAfterSourceOpen && this._mseSourceOpened === false) { this._hasPendingLoad = true; return; } if (this._mediaElement.readyState > 0) { this._requestSetTime = true; // IE11 may throw InvalidStateError if readyState === 0 this._mediaElement.currentTime = 0; } this._transmuxer = new _transmuxer2.default(this._mediaDataSource, this._config); this._transmuxer.on(_transmuxingEvents2.default.INIT_SEGMENT, function (type, is) { _this3._msectl.appendInitSegment(is); }); this._transmuxer.on(_transmuxingEvents2.default.MEDIA_SEGMENT, function (type, ms) { _this3._msectl.appendMediaSegment(ms); // lazyLoad check if (_this3._config.lazyLoad && !_this3._config.isLive) { var currentTime = _this3._mediaElement.currentTime; if (ms.info.endDts >= (currentTime + _this3._config.lazyLoadMaxDuration) * 1000) { if (_this3._progressChecker == null) { _logger2.default.v(_this3.TAG, 'Maximum buffering duration exceeded, suspend transmuxing task'); _this3._suspendTransmuxer(); } } } }); this._transmuxer.on(_transmuxingEvents2.default.LOADING_COMPLETE, function () { _this3._msectl.endOfStream(); _this3._emitter.emit(_playerEvents2.default.LOADING_COMPLETE); }); this._transmuxer.on(_transmuxingEvents2.default.RECOVERED_EARLY_EOF, function () { _this3._emitter.emit(_playerEvents2.default.RECOVERED_EARLY_EOF); }); this._transmuxer.on(_transmuxingEvents2.default.IO_ERROR, function (detail, info) { _this3._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.NETWORK_ERROR, detail, info); }); this._transmuxer.on(_transmuxingEvents2.default.DEMUX_ERROR, function (detail, info) { _this3._emitter.emit(_playerEvents2.default.ERROR, _playerErrors.ErrorTypes.MEDIA_ERROR, detail, { code: -1, msg: info }); }); this._transmuxer.on(_transmuxingEvents2.default.MEDIA_INFO, function (mediaInfo) { _this3._mediaInfo = mediaInfo; _this3._emitter.emit(_playerEvents2.default.MEDIA_INFO, Object.assign({}, mediaInfo)); }); this._transmuxer.on(_transmuxingEvents2.default.STATISTICS_INFO, function (statInfo) { _this3._statisticsInfo = _this3._fillStatisticsInfo(statInfo); _this3._emitter.emit(_playerEvents2.default.STATISTICS_INFO, Object.assign({}, _this3._statisticsInfo)); }); this._transmuxer.on(_transmuxingEvents2.default.RECOMMEND_SEEKPOINT, function (milliseconds) { if (_this3._mediaElement && !_this3._config.accurateSeek) { _this3._requestSetTime = true; _this3._mediaElement.currentTime = milliseconds / 1000; } }); this._transmuxer.open(); } }, { key: 'unload', value: function unload() { if (this._mediaElement) { this._mediaElement.pause(); } if (this._msectl) { this._msectl.seek(0); } if (this._transmuxer) { this._transmuxer.close(); this._transmuxer.destroy(); this._transmuxer = null; } } }, { key: 'play', value: function play() { this._mediaElement.play(); } }, { key: 'pause', value: function pause() { this._mediaElement.pause(); } }, { key: '_fillStatisticsInfo', value: function _fillStatisticsInfo(statInfo) { statInfo.playerType = this._type; if (!(this._mediaElement instanceof HTMLVideoElement)) { return statInfo; } var hasQualityInfo = true; var decoded = 0; var dropped = 0; if (this._mediaElement.getVideoPlaybackQuality) { var quality = this._mediaElement.getVideoPlaybackQuality(); decoded = quality.totalVideoFrames; dropped = quality.droppedVideoFrames; } else if (this._mediaElement.webkitDecodedFrameCount != undefined) { decoded = this._mediaElement.webkitDecodedFrameCount; dropped = this._mediaElement.webkitDroppedFrameCount; } else { hasQualityInfo = false; } if (hasQualityInfo) { statInfo.decodedFrames = decoded; statInfo.droppedFrames = dropped; } return statInfo; } }, { key: '_onmseUpdateEnd', value: function _onmseUpdateEnd() { if (!this._config.lazyLoad || this._config.isLive) { return; } var buffered = this._mediaElement.buffered; var currentTime = this._mediaElement.currentTime; var currentRangeStart = 0; var currentRangeEnd = 0; for (var i = 0; i < buffered.length; i++) { var start = buffered.start(i); var end = buffered.end(i); if (start <= currentTime && currentTime < end) { currentRangeStart = start; currentRangeEnd = end; break; } } if (currentRangeEnd >= currentTime + this._config.lazyLoadMaxDuration && this._progressChecker == null) { _logger2.default.v(this.TAG, 'Maximum buffering duration exceeded, suspend transmuxing task'); this._suspendTransmuxer(); } } }, { key: '_onmseBufferFull', value: function _onmseBufferFull() { _logger2.default.v(this.TAG, 'MSE SourceBuffer is full, suspend transmuxing task'); if (this._progressChecker == null) { this._suspendTransmuxer(); } } }, { key: '_suspendTransmuxer', value: function _suspendTransmuxer() { if (this._transmuxer) { this._transmuxer.pause(); if (this._progressChecker == null) { this._progressChecker = window.setInterval(this._checkProgressAndResume.bind(this), 1000); } } } }, { key: '_checkProgressAndResume', value: function _checkProgressAndResume() { var currentTime = this._mediaElement.currentTime; var buffered = this._mediaElement.buffered; var needResume = false; for (var i = 0; i < buffered.length; i++) { var from = buffered.start(i); var to = buffered.end(i); if (currentTime >= from && currentTime < to) { if (currentTime >= to - 30) { needResume = true; } break; } } if (needResume) { window.clearInterval(this._progressChecker); this._progressChecker = null; if (needResume) { _logger2.default.v(this.TAG, 'Continue loading from paused position'); this._transmuxer.resume(); } } } }, { key: '_isTimepointBuffered', value: function _isTimepointBuffered(seconds) { var buffered = this._mediaElement.buffered; for (var i = 0; i < buffered.length; i++) { var from = buffered.start(i); var to = buffered.end(i); if (seconds >= from && seconds < to) { return true; } } return false; } }, { key: '_internalSeek', value: function _internalSeek(seconds) { var directSeek = this._isTimepointBuffered(seconds); var directSeekBegin = false; var directSeekBeginTime = 0; if (seconds < 1.0 && this._mediaElement.buffered.length > 0) { var videoBeginTime = this._mediaElement.buffered.start(0); if (videoBeginTime < 1.0 && seconds < videoBeginTime || _browser2.default.safari) { directSeekBegin = true; // also workaround for Safari: Seek to 0 may cause video stuck, use 0.1 to avoid directSeekBeginTime = _browser2.default.safari ? 0.1 : videoBeginTime; } } if (directSeekBegin) { // seek to video begin, set currentTime directly if beginPTS buffered this._requestSetTime = true; this._mediaElement.currentTime = directSeekBeginTime; } else if (directSeek) { // buffered position if (!this._alwaysSeekKeyframe) { this._requestSetTime = true; this._mediaElement.currentTime = seconds; } else { var idr = this._msectl.getNearestKeyframe(Math.floor(seconds * 1000)); this._requestSetTime = true; if (idr != null) { this._mediaElement.currentTime = idr.dts / 1000; } else { this._mediaElement.currentTime = seconds; } } if (this._progressChecker != null) { this._checkProgressAndResume(); } } else { if (this._progressChecker != null) { window.clearInterval(this._progressChecker); this._progressChecker = null; } this._msectl.seek(seconds); this._transmuxer.seek(Math.floor(seconds * 1000)); // in milliseconds // no need to set mediaElement.currentTime if non-accurateSeek, // just wait for the recommend_seekpoint callback if (this._config.accurateSeek) { this._requestSetTime = true; this._mediaElement.currentTime = seconds; } } } }, { key: '_checkAndApplyUnbufferedSeekpoint', value: function _checkAndApplyUnbufferedSeekpoint() { if (this._seekpointRecord) { if (this._seekpointRecord.recordTime <= this._now() - 100) { var target = this._mediaElement.currentTime; this._seekpointRecord = null; if (!this._isTimepointBuffered(target)) { if (this._progressChecker != null) { window.clearTimeout(this._progressChecker); this._progressChecker = null; } // .currentTime is consists with .buffered timestamp // Chrome/Edge use DTS, while FireFox/Safari use PTS this._msectl.seek(target); this._transmuxer.seek(Math.floor(target * 1000)); // set currentTime if accurateSeek, or wait for recommend_seekpoint callback if (this._config.accurateSeek) { this._requestSetTime = true; this._mediaElement.currentTime = target; } } } else { window.setTimeout(this._checkAndApplyUnbufferedSeekpoint.bind(this), 50); } } } }, { key: '_checkAndResumeStuckPlayback', value: function _checkAndResumeStuckPlayback(stalled) { var media = this._mediaElement; if (stalled || !this._receivedCanPlay || media.readyState < 2) { // HAVE_CURRENT_DATA var buffered = media.buffered; if (buffered.length > 0 && media.currentTime < buffered.start(0)) { _logger2.default.w(this.TAG, 'Playback seems stuck at ' + media.currentTime + ', seek to ' + buffered.start(0)); this._requestSetTime = true; this._mediaElement.currentTime = buffered.start(0); this._mediaElement.removeEventListener('progress', this.e.onvProgress); } } else { // Playback didn't stuck, remove progress event listener this._mediaElement.removeEventListener('progress', this.e.onvProgress); } } }, { key: '_onvLoadedMetadata', value: function _onvLoadedMetadata(e) { if (this._pendingSeekTime != null) { this._mediaElement.currentTime = this._pendingSeekTime; this._pendingSeekTime = null; } } }, { key: '_onvSeeking', value: function _onvSeeking(e) { // handle seeking request from browser's progress bar var target = this._mediaElement.currentTime; var buffered = this._mediaElement.buffered; if (this._requestSetTime) { this._requestSetTime = false; return; } if (target < 1.0 && buffered.length > 0) { // seek to video begin, set currentTime directly if beginPTS buffered var videoBeginTime = buffered.start(0); if (videoBeginTime < 1.0 && target < videoBeginTime || _browser2.default.safari) { this._requestSetTime = true; // also workaround for Safari: Seek to 0 may cause video stuck, use 0.1 to avoid this._mediaElement.currentTime = _browser2.default.safari ? 0.1 : videoBeginTime; return; } } if (this._isTimepointBuffered(target)) { if (this._alwaysSeekKeyframe) { var idr = this._msectl.getNearestKeyframe(Math.floor(target * 1000)); if (idr != null) { this._requestSetTime = true; this._mediaElement.currentTime = idr.dts / 1000; } } if (this._progressChecker != null) { this._checkProgressAndResume(); } return; } this._seekpointRecord = { seekPoint: target, recordTime: this._now() }; window.setTimeout(this._checkAndApplyUnbufferedSeekpoint.bind(this), 50); } }, { key: '_onvCanPlay', value: function _onvCanPlay(e) { this._receivedCanPlay = true; this._mediaElement.removeEventListener('canplay', this.e.onvCanPlay); } }, { key: '_onvStalled', value: function _onvStalled(e) { this._checkAndResumeStuckPlayback(true); } }, { key: '_onvProgress', value: function _onvProgress(e) { this._checkAndResumeStuckPlayback(); } }, { key: 'type', get: function get() { return this._type; } }, { key: 'buffered', get: function get() { return this._mediaElement.buffered; } }, { key: 'duration', get: function get() { return this._mediaElement.duration; } }, { key: 'volume', get: function get() { return this._mediaElement.volume; }, set: function set(value) { this._mediaElement.volume = value; } }, { key: 'muted', get: function get() { return this._mediaElement.muted; }, set: function set(muted) { this._mediaElement.muted = muted; } }, { key: 'currentTime', get: function get() { if (this._mediaElement) { return this._mediaElement.currentTime; } return 0; }, set: function set(seconds) { if (this._mediaElement) { this._internalSeek(seconds); } else { this._pendingSeekTime = seconds; } } }, { key: 'mediaInfo', get: function get() { return Object.assign({}, this._mediaInfo); } }, { key: 'statisticsInfo', get: function get() { if (this._statisticsInfo == null) { this._statisticsInfo = {}; } this._statisticsInfo = this._fillStatisticsInfo(this._statisticsInfo); return Object.assign({}, this._statisticsInfo); } }]); return FlvPlayer; }(); exports.default = FlvPlayer; },{"../config.js":5,"../core/mse-controller.js":9,"../core/mse-events.js":10,"../core/transmuxer.js":11,"../core/transmuxing-events.js":13,"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./player-errors.js":34,"./player-events.js":35,"events":2}],33:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _playerEvents = _dereq_('./player-events.js'); var _playerEvents2 = _interopRequireDefault(_playerEvents); var _config = _dereq_('../config.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // Player wrapper for browser's native player (HTMLVideoElement) without MediaSource src. var NativePlayer = function () { function NativePlayer(mediaDataSource, config) { _classCallCheck(this, NativePlayer); this.TAG = 'NativePlayer'; this._type = 'NativePlayer'; this._emitter = new _events2.default(); this._config = (0, _config.createDefaultConfig)(); if ((typeof config === 'undefined' ? 'undefined' : _typeof(config)) === 'object') { Object.assign(this._config, config); } if (mediaDataSource.type.toLowerCase() === 'flv') { throw new _exception.InvalidArgumentException('NativePlayer does\'t support flv MediaDataSource input!'); } if (mediaDataSource.hasOwnProperty('segments')) { throw new _exception.InvalidArgumentException('NativePlayer(' + mediaDataSource.type + ') doesn\'t support multipart playback!'); } this.e = { onvLoadedMetadata: this._onvLoadedMetadata.bind(this) }; this._pendingSeekTime = null; this._statisticsReporter = null; this._mediaDataSource = mediaDataSource; this._mediaElement = null; } _createClass(NativePlayer, [{ key: 'destroy', value: function destroy() { if (this._mediaElement) { this.unload(); this.detachMediaElement(); } this.e = null; this._mediaDataSource = null; this._emitter.removeAllListeners(); this._emitter = null; } }, { key: 'on', value: function on(event, listener) { var _this = this; if (event === _playerEvents2.default.MEDIA_INFO) { if (this._mediaElement != null && this._mediaElement.readyState !== 0) { // HAVE_NOTHING Promise.resolve().then(function () { _this._emitter.emit(_playerEvents2.default.MEDIA_INFO, _this.mediaInfo); }); } } else if (event === _playerEvents2.default.STATISTICS_INFO) { if (this._mediaElement != null && this._mediaElement.readyState !== 0) { Promise.resolve().then(function () { _this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, _this.statisticsInfo); }); } } this._emitter.addListener(event, listener); } }, { key: 'off', value: function off(event, listener) { this._emitter.removeListener(event, listener); } }, { key: 'attachMediaElement', value: function attachMediaElement(mediaElement) { this._mediaElement = mediaElement; mediaElement.addEventListener('loadedmetadata', this.e.onvLoadedMetadata); if (this._pendingSeekTime != null) { try { mediaElement.currentTime = this._pendingSeekTime; this._pendingSeekTime = null; } catch (e) { // IE11 may throw InvalidStateError if readyState === 0 // Defer set currentTime operation after loadedmetadata } } } }, { key: 'detachMediaElement', value: function detachMediaElement() { if (this._mediaElement) { this._mediaElement.src = ''; this._mediaElement.removeAttribute('src'); this._mediaElement.removeEventListener('loadedmetadata', this.e.onvLoadedMetadata); this._mediaElement = null; } if (this._statisticsReporter != null) { window.clearInterval(this._statisticsReporter); this._statisticsReporter = null; } } }, { key: 'load', value: function load() { if (!this._mediaElement) { throw new _exception.IllegalStateException('HTMLMediaElement must be attached before load()!'); } this._mediaElement.src = this._mediaDataSource.url; if (this._mediaElement.readyState > 0) { this._mediaElement.currentTime = 0; } this._mediaElement.preload = 'auto'; this._mediaElement.load(); this._statisticsReporter = window.setInterval(this._reportStatisticsInfo.bind(this), this._config.statisticsInfoReportInterval); } }, { key: 'unload', value: function unload() { if (this._mediaElement) { this._mediaElement.src = ''; this._mediaElement.removeAttribute('src'); } if (this._statisticsReporter != null) { window.clearInterval(this._statisticsReporter); this._statisticsReporter = null; } } }, { key: 'play', value: function play() { this._mediaElement.play(); } }, { key: 'pause', value: function pause() { this._mediaElement.pause(); } }, { key: '_onvLoadedMetadata', value: function _onvLoadedMetadata(e) { if (this._pendingSeekTime != null) { this._mediaElement.currentTime = this._pendingSeekTime; this._pendingSeekTime = null; } this._emitter.emit(_playerEvents2.default.MEDIA_INFO, this.mediaInfo); } }, { key: '_reportStatisticsInfo', value: function _reportStatisticsInfo() { this._emitter.emit(_playerEvents2.default.STATISTICS_INFO, this.statisticsInfo); } }, { key: 'type', get: function get() { return this._type; } }, { key: 'buffered', get: function get() { return this._mediaElement.buffered; } }, { key: 'duration', get: function get() { return this._mediaElement.duration; } }, { key: 'volume', get: function get() { return this._mediaElement.volume; }, set: function set(value) { this._mediaElement.volume = value; } }, { key: 'muted', get: function get() { return this._mediaElement.muted; }, set: function set(muted) { this._mediaElement.muted = muted; } }, { key: 'currentTime', get: function get() { if (this._mediaElement) { return this._mediaElement.currentTime; } return 0; }, set: function set(seconds) { if (this._mediaElement) { this._mediaElement.currentTime = seconds; } else { this._pendingSeekTime = seconds; } } }, { key: 'mediaInfo', get: function get() { var mediaPrefix = this._mediaElement instanceof HTMLAudioElement ? 'audio/' : 'video/'; var info = { mimeType: mediaPrefix + this._mediaDataSource.type }; if (this._mediaElement) { info.duration = Math.floor(this._mediaElement.duration * 1000); if (this._mediaElement instanceof HTMLVideoElement) { info.width = this._mediaElement.videoWidth; info.height = this._mediaElement.videoHeight; } } return info; } }, { key: 'statisticsInfo', get: function get() { var info = { playerType: this._type, url: this._mediaDataSource.url }; if (!(this._mediaElement instanceof HTMLVideoElement)) { return info; } var hasQualityInfo = true; var decoded = 0; var dropped = 0; if (this._mediaElement.getVideoPlaybackQuality) { var quality = this._mediaElement.getVideoPlaybackQuality(); decoded = quality.totalVideoFrames; dropped = quality.droppedVideoFrames; } else if (this._mediaElement.webkitDecodedFrameCount != undefined) { decoded = this._mediaElement.webkitDecodedFrameCount; dropped = this._mediaElement.webkitDroppedFrameCount; } else { hasQualityInfo = false; } if (hasQualityInfo) { info.decodedFrames = decoded; info.droppedFrames = dropped; } return info; } }]); return NativePlayer; }(); exports.default = NativePlayer; },{"../config.js":5,"../utils/exception.js":40,"./player-events.js":35,"events":2}],34:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.ErrorDetails = exports.ErrorTypes = undefined; var _loader = _dereq_('../io/loader.js'); var _demuxErrors = _dereq_('../demux/demux-errors.js'); var _demuxErrors2 = _interopRequireDefault(_demuxErrors); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var ErrorTypes = exports.ErrorTypes = { NETWORK_ERROR: 'NetworkError', MEDIA_ERROR: 'MediaError', OTHER_ERROR: 'OtherError' }; var ErrorDetails = exports.ErrorDetails = { NETWORK_EXCEPTION: _loader.LoaderErrors.EXCEPTION, NETWORK_STATUS_CODE_INVALID: _loader.LoaderErrors.HTTP_STATUS_CODE_INVALID, NETWORK_TIMEOUT: _loader.LoaderErrors.CONNECTING_TIMEOUT, NETWORK_UNRECOVERABLE_EARLY_EOF: _loader.LoaderErrors.UNRECOVERABLE_EARLY_EOF, MEDIA_MSE_ERROR: 'MediaMSEError', MEDIA_FORMAT_ERROR: _demuxErrors2.default.FORMAT_ERROR, MEDIA_FORMAT_UNSUPPORTED: _demuxErrors2.default.FORMAT_UNSUPPORTED, MEDIA_CODEC_UNSUPPORTED: _demuxErrors2.default.CODEC_UNSUPPORTED }; },{"../demux/demux-errors.js":16,"../io/loader.js":24}],35:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var PlayerEvents = { ERROR: 'error', LOADING_COMPLETE: 'loading_complete', RECOVERED_EARLY_EOF: 'recovered_early_eof', MEDIA_INFO: 'media_info', STATISTICS_INFO: 'statistics_info' }; exports.default = PlayerEvents; },{}],36:[function(_dereq_,module,exports){ "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * This file is modified from dailymotion's hls.js library (hls.js/src/helper/aac.js) * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var AAC = function () { function AAC() { _classCallCheck(this, AAC); } _createClass(AAC, null, [{ key: "getSilentFrame", value: function getSilentFrame(channelCount) { if (channelCount === 1) { return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x23, 0x80]); } else if (channelCount === 2) { return new Uint8Array([0x21, 0x00, 0x49, 0x90, 0x02, 0x19, 0x00, 0x23, 0x80]); } else if (channelCount === 3) { return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x8e]); } else if (channelCount === 4) { return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x80, 0x2c, 0x80, 0x08, 0x02, 0x38]); } else if (channelCount === 5) { return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x82, 0x30, 0x04, 0x99, 0x00, 0x21, 0x90, 0x02, 0x38]); } else if (channelCount === 6) { return new Uint8Array([0x00, 0xc8, 0x00, 0x80, 0x20, 0x84, 0x01, 0x26, 0x40, 0x08, 0x64, 0x00, 0x82, 0x30, 0x04, 0x99, 0x00, 0x21, 0x90, 0x02, 0x00, 0xb2, 0x00, 0x20, 0x08, 0xe0]); } return null; } }]); return AAC; }(); exports.default = AAC; },{}],37:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * This file is derived from dailymotion's hls.js library (hls.js/src/remux/mp4-generator.js) * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // MP4 boxes generator for ISO BMFF (ISO Base Media File Format, defined in ISO/IEC 14496-12) var MP4 = function () { function MP4() { _classCallCheck(this, MP4); } _createClass(MP4, null, [{ key: 'init', value: function init() { MP4.types = { avc1: [], avcC: [], btrt: [], dinf: [], dref: [], esds: [], ftyp: [], hdlr: [], mdat: [], mdhd: [], mdia: [], mfhd: [], minf: [], moof: [], moov: [], mp4a: [], mvex: [], mvhd: [], sdtp: [], stbl: [], stco: [], stsc: [], stsd: [], stsz: [], stts: [], tfdt: [], tfhd: [], traf: [], trak: [], trun: [], trex: [], tkhd: [], vmhd: [], smhd: [] }; for (var name in MP4.types) { if (MP4.types.hasOwnProperty(name)) { MP4.types[name] = [name.charCodeAt(0), name.charCodeAt(1), name.charCodeAt(2), name.charCodeAt(3)]; } } var constants = MP4.constants = {}; constants.FTYP = new Uint8Array([0x69, 0x73, 0x6F, 0x6D, // major_brand: isom 0x0, 0x0, 0x0, 0x1, // minor_version: 0x01 0x69, 0x73, 0x6F, 0x6D, // isom 0x61, 0x76, 0x63, 0x31 // avc1 ]); constants.STSD_PREFIX = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x01 // entry_count ]); constants.STTS = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00 // entry_count ]); constants.STSC = constants.STCO = constants.STTS; constants.STSZ = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // sample_size 0x00, 0x00, 0x00, 0x00 // sample_count ]); constants.HDLR_VIDEO = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // pre_defined 0x76, 0x69, 0x64, 0x65, // handler_type: 'vide' 0x00, 0x00, 0x00, 0x00, // reserved: 3 * 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x56, 0x69, 0x64, 0x65, 0x6F, 0x48, 0x61, 0x6E, 0x64, 0x6C, 0x65, 0x72, 0x00 // name: VideoHandler ]); constants.HDLR_AUDIO = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // pre_defined 0x73, 0x6F, 0x75, 0x6E, // handler_type: 'soun' 0x00, 0x00, 0x00, 0x00, // reserved: 3 * 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x53, 0x6F, 0x75, 0x6E, 0x64, 0x48, 0x61, 0x6E, 0x64, 0x6C, 0x65, 0x72, 0x00 // name: SoundHandler ]); constants.DREF = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x01, // entry_count 0x00, 0x00, 0x00, 0x0C, // entry_size 0x75, 0x72, 0x6C, 0x20, // type 'url ' 0x00, 0x00, 0x00, 0x01 // version(0) + flags ]); // Sound media header constants.SMHD = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00 // balance(2) + reserved(2) ]); // video media header constants.VMHD = new Uint8Array([0x00, 0x00, 0x00, 0x01, // version(0) + flags 0x00, 0x00, // graphicsmode: 2 bytes 0x00, 0x00, 0x00, 0x00, // opcolor: 3 * 2 bytes 0x00, 0x00]); } // Generate a box }, { key: 'box', value: function box(type) { var size = 8; var result = null; var datas = Array.prototype.slice.call(arguments, 1); var arrayCount = datas.length; for (var i = 0; i < arrayCount; i++) { size += datas[i].byteLength; } result = new Uint8Array(size); result[0] = size >>> 24 & 0xFF; // size result[1] = size >>> 16 & 0xFF; result[2] = size >>> 8 & 0xFF; result[3] = size & 0xFF; result.set(type, 4); // type var offset = 8; for (var _i = 0; _i < arrayCount; _i++) { // data body result.set(datas[_i], offset); offset += datas[_i].byteLength; } return result; } // emit ftyp & moov }, { key: 'generateInitSegment', value: function generateInitSegment(meta) { var ftyp = MP4.box(MP4.types.ftyp, MP4.constants.FTYP); var moov = MP4.moov(meta); var result = new Uint8Array(ftyp.byteLength + moov.byteLength); result.set(ftyp, 0); result.set(moov, ftyp.byteLength); return result; } // Movie metadata box }, { key: 'moov', value: function moov(meta) { var mvhd = MP4.mvhd(meta.timescale, meta.duration); var trak = MP4.trak(meta); var mvex = MP4.mvex(meta); return MP4.box(MP4.types.moov, mvhd, trak, mvex); } // Movie header box }, { key: 'mvhd', value: function mvhd(timescale, duration) { return MP4.box(MP4.types.mvhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // creation_time 0x00, 0x00, 0x00, 0x00, // modification_time timescale >>> 24 & 0xFF, // timescale: 4 bytes timescale >>> 16 & 0xFF, timescale >>> 8 & 0xFF, timescale & 0xFF, duration >>> 24 & 0xFF, // duration: 4 bytes duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x00, 0x01, 0x00, 0x00, // Preferred rate: 1.0 0x01, 0x00, 0x00, 0x00, // PreferredVolume(1.0, 2bytes) + reserved(2bytes) 0x00, 0x00, 0x00, 0x00, // reserved: 4 + 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, // ----begin composition matrix---- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, // ----end composition matrix---- 0x00, 0x00, 0x00, 0x00, // ----begin pre_defined 6 * 4 bytes---- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // ----end pre_defined 6 * 4 bytes---- 0xFF, 0xFF, 0xFF, 0xFF // next_track_ID ])); } // Track box }, { key: 'trak', value: function trak(meta) { return MP4.box(MP4.types.trak, MP4.tkhd(meta), MP4.mdia(meta)); } // Track header box }, { key: 'tkhd', value: function tkhd(meta) { var trackId = meta.id, duration = meta.duration; var width = meta.presentWidth, height = meta.presentHeight; return MP4.box(MP4.types.tkhd, new Uint8Array([0x00, 0x00, 0x00, 0x07, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // creation_time 0x00, 0x00, 0x00, 0x00, // modification_time trackId >>> 24 & 0xFF, // track_ID: 4 bytes trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF, 0x00, 0x00, 0x00, 0x00, // reserved: 4 bytes duration >>> 24 & 0xFF, // duration: 4 bytes duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x00, 0x00, 0x00, 0x00, // reserved: 2 * 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // layer(2bytes) + alternate_group(2bytes) 0x00, 0x00, 0x00, 0x00, // volume(2bytes) + reserved(2bytes) 0x00, 0x01, 0x00, 0x00, // ----begin composition matrix---- 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, // ----end composition matrix---- width >>> 8 & 0xFF, // width and height width & 0xFF, 0x00, 0x00, height >>> 8 & 0xFF, height & 0xFF, 0x00, 0x00])); } // Media Box }, { key: 'mdia', value: function mdia(meta) { return MP4.box(MP4.types.mdia, MP4.mdhd(meta), MP4.hdlr(meta), MP4.minf(meta)); } // Media header box }, { key: 'mdhd', value: function mdhd(meta) { var timescale = meta.timescale; var duration = meta.duration; return MP4.box(MP4.types.mdhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags 0x00, 0x00, 0x00, 0x00, // creation_time 0x00, 0x00, 0x00, 0x00, // modification_time timescale >>> 24 & 0xFF, // timescale: 4 bytes timescale >>> 16 & 0xFF, timescale >>> 8 & 0xFF, timescale & 0xFF, duration >>> 24 & 0xFF, // duration: 4 bytes duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, 0x55, 0xC4, // language: und (undetermined) 0x00, 0x00 // pre_defined = 0 ])); } // Media handler reference box }, { key: 'hdlr', value: function hdlr(meta) { var data = null; if (meta.type === 'audio') { data = MP4.constants.HDLR_AUDIO; } else { data = MP4.constants.HDLR_VIDEO; } return MP4.box(MP4.types.hdlr, data); } // Media infomation box }, { key: 'minf', value: function minf(meta) { var xmhd = null; if (meta.type === 'audio') { xmhd = MP4.box(MP4.types.smhd, MP4.constants.SMHD); } else { xmhd = MP4.box(MP4.types.vmhd, MP4.constants.VMHD); } return MP4.box(MP4.types.minf, xmhd, MP4.dinf(), MP4.stbl(meta)); } // Data infomation box }, { key: 'dinf', value: function dinf() { var result = MP4.box(MP4.types.dinf, MP4.box(MP4.types.dref, MP4.constants.DREF)); return result; } // Sample table box }, { key: 'stbl', value: function stbl(meta) { var result = MP4.box(MP4.types.stbl, // type: stbl MP4.stsd(meta), // Sample Description Table MP4.box(MP4.types.stts, MP4.constants.STTS), // Time-To-Sample MP4.box(MP4.types.stsc, MP4.constants.STSC), // Sample-To-Chunk MP4.box(MP4.types.stsz, MP4.constants.STSZ), // Sample size MP4.box(MP4.types.stco, MP4.constants.STCO) // Chunk offset ); return result; } // Sample description box }, { key: 'stsd', value: function stsd(meta) { if (meta.type === 'audio') { return MP4.box(MP4.types.stsd, MP4.constants.STSD_PREFIX, MP4.mp4a(meta)); } else { return MP4.box(MP4.types.stsd, MP4.constants.STSD_PREFIX, MP4.avc1(meta)); } } }, { key: 'mp4a', value: function mp4a(meta) { var channelCount = meta.channelCount; var sampleRate = meta.audioSampleRate; var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // reserved(4) 0x00, 0x00, 0x00, 0x01, // reserved(2) + data_reference_index(2) 0x00, 0x00, 0x00, 0x00, // reserved: 2 * 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, channelCount, // channelCount(2) 0x00, 0x10, // sampleSize(2) 0x00, 0x00, 0x00, 0x00, // reserved(4) sampleRate >>> 8 & 0xFF, // Audio sample rate sampleRate & 0xFF, 0x00, 0x00]); return MP4.box(MP4.types.mp4a, data, MP4.esds(meta)); } }, { key: 'esds', value: function esds(meta) { var config = meta.config; var configSize = config.length; var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version 0 + flags 0x03, // descriptor_type 0x17 + configSize, // length3 0x00, 0x01, // es_id 0x00, // stream_priority 0x04, // descriptor_type 0x0F + configSize, // length 0x40, // codec: mpeg4_audio 0x15, // stream_type: Audio 0x00, 0x00, 0x00, // buffer_size 0x00, 0x00, 0x00, 0x00, // maxBitrate 0x00, 0x00, 0x00, 0x00, // avgBitrate 0x05 // descriptor_type ].concat([configSize]).concat(config).concat([0x06, 0x01, 0x02 // GASpecificConfig ])); return MP4.box(MP4.types.esds, data); } }, { key: 'avc1', value: function avc1(meta) { var avcc = meta.avcc; var width = meta.codecWidth, height = meta.codecHeight; var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // reserved(4) 0x00, 0x00, 0x00, 0x01, // reserved(2) + data_reference_index(2) 0x00, 0x00, 0x00, 0x00, // pre_defined(2) + reserved(2) 0x00, 0x00, 0x00, 0x00, // pre_defined: 3 * 4 bytes 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, width >>> 8 & 0xFF, // width: 2 bytes width & 0xFF, height >>> 8 & 0xFF, // height: 2 bytes height & 0xFF, 0x00, 0x48, 0x00, 0x00, // horizresolution: 4 bytes 0x00, 0x48, 0x00, 0x00, // vertresolution: 4 bytes 0x00, 0x00, 0x00, 0x00, // reserved: 4 bytes 0x00, 0x01, // frame_count 0x0A, // strlen 0x78, 0x71, 0x71, 0x2F, // compressorname: 32 bytes 0x66, 0x6C, 0x76, 0x2E, 0x6A, 0x73, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, // depth 0xFF, 0xFF // pre_defined = -1 ]); return MP4.box(MP4.types.avc1, data, MP4.box(MP4.types.avcC, avcc)); } // Movie Extends box }, { key: 'mvex', value: function mvex(meta) { return MP4.box(MP4.types.mvex, MP4.trex(meta)); } // Track Extends box }, { key: 'trex', value: function trex(meta) { var trackId = meta.id; var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) + flags trackId >>> 24 & 0xFF, // track_ID trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF, 0x00, 0x00, 0x00, 0x01, // default_sample_description_index 0x00, 0x00, 0x00, 0x00, // default_sample_duration 0x00, 0x00, 0x00, 0x00, // default_sample_size 0x00, 0x01, 0x00, 0x01 // default_sample_flags ]); return MP4.box(MP4.types.trex, data); } // Movie fragment box }, { key: 'moof', value: function moof(track, baseMediaDecodeTime) { return MP4.box(MP4.types.moof, MP4.mfhd(track.sequenceNumber), MP4.traf(track, baseMediaDecodeTime)); } }, { key: 'mfhd', value: function mfhd(sequenceNumber) { var data = new Uint8Array([0x00, 0x00, 0x00, 0x00, sequenceNumber >>> 24 & 0xFF, // sequence_number: int32 sequenceNumber >>> 16 & 0xFF, sequenceNumber >>> 8 & 0xFF, sequenceNumber & 0xFF]); return MP4.box(MP4.types.mfhd, data); } // Track fragment box }, { key: 'traf', value: function traf(track, baseMediaDecodeTime) { var trackId = track.id; // Track fragment header box var tfhd = MP4.box(MP4.types.tfhd, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) & flags trackId >>> 24 & 0xFF, // track_ID trackId >>> 16 & 0xFF, trackId >>> 8 & 0xFF, trackId & 0xFF])); // Track Fragment Decode Time var tfdt = MP4.box(MP4.types.tfdt, new Uint8Array([0x00, 0x00, 0x00, 0x00, // version(0) & flags baseMediaDecodeTime >>> 24 & 0xFF, // baseMediaDecodeTime: int32 baseMediaDecodeTime >>> 16 & 0xFF, baseMediaDecodeTime >>> 8 & 0xFF, baseMediaDecodeTime & 0xFF])); var sdtp = MP4.sdtp(track); var trun = MP4.trun(track, sdtp.byteLength + 16 + 16 + 8 + 16 + 8 + 8); return MP4.box(MP4.types.traf, tfhd, tfdt, trun, sdtp); } // Sample Dependency Type box }, { key: 'sdtp', value: function sdtp(track) { var samples = track.samples || []; var sampleCount = samples.length; var data = new Uint8Array(4 + sampleCount); // 0~4 bytes: version(0) & flags for (var i = 0; i < sampleCount; i++) { var flags = samples[i].flags; data[i + 4] = flags.isLeading << 6 | // is_leading: 2 (bit) flags.dependsOn << 4 // sample_depends_on | flags.isDependedOn << 2 // sample_is_depended_on | flags.hasRedundancy; // sample_has_redundancy } return MP4.box(MP4.types.sdtp, data); } // Track fragment run box }, { key: 'trun', value: function trun(track, offset) { var samples = track.samples || []; var sampleCount = samples.length; var dataSize = 12 + 16 * sampleCount; var data = new Uint8Array(dataSize); offset += 8 + dataSize; data.set([0x00, 0x00, 0x0F, 0x01, // version(0) & flags sampleCount >>> 24 & 0xFF, // sample_count sampleCount >>> 16 & 0xFF, sampleCount >>> 8 & 0xFF, sampleCount & 0xFF, offset >>> 24 & 0xFF, // data_offset offset >>> 16 & 0xFF, offset >>> 8 & 0xFF, offset & 0xFF], 0); for (var i = 0; i < sampleCount; i++) { var duration = samples[i].duration; var size = samples[i].size; var flags = samples[i].flags; var cts = samples[i].cts; data.set([duration >>> 24 & 0xFF, // sample_duration duration >>> 16 & 0xFF, duration >>> 8 & 0xFF, duration & 0xFF, size >>> 24 & 0xFF, // sample_size size >>> 16 & 0xFF, size >>> 8 & 0xFF, size & 0xFF, flags.isLeading << 2 | flags.dependsOn, // sample_flags flags.isDependedOn << 6 | flags.hasRedundancy << 4 | flags.isNonSync, 0x00, 0x00, // sample_degradation_priority cts >>> 24 & 0xFF, // sample_composition_time_offset cts >>> 16 & 0xFF, cts >>> 8 & 0xFF, cts & 0xFF], 12 + 16 * i); } return MP4.box(MP4.types.trun, data); } }, { key: 'mdat', value: function mdat(data) { return MP4.box(MP4.types.mdat, data); } }]); return MP4; }(); MP4.init(); exports.default = MP4; },{}],38:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _logger = _dereq_('../utils/logger.js'); var _logger2 = _interopRequireDefault(_logger); var _mp4Generator = _dereq_('./mp4-generator.js'); var _mp4Generator2 = _interopRequireDefault(_mp4Generator); var _aacSilent = _dereq_('./aac-silent.js'); var _aacSilent2 = _interopRequireDefault(_aacSilent); var _browser = _dereq_('../utils/browser.js'); var _browser2 = _interopRequireDefault(_browser); var _mediaSegmentInfo = _dereq_('../core/media-segment-info.js'); var _exception = _dereq_('../utils/exception.js'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // Fragmented mp4 remuxer var MP4Remuxer = function () { function MP4Remuxer(config) { _classCallCheck(this, MP4Remuxer); this.TAG = 'MP4Remuxer'; this._config = config; this._isLive = config.isLive === true ? true : false; this._dtsBase = -1; this._dtsBaseInited = false; this._audioDtsBase = Infinity; this._videoDtsBase = Infinity; this._audioNextDts = undefined; this._videoNextDts = undefined; this._audioMeta = null; this._videoMeta = null; this._audioSegmentInfoList = new _mediaSegmentInfo.MediaSegmentInfoList('audio'); this._videoSegmentInfoList = new _mediaSegmentInfo.MediaSegmentInfoList('video'); this._onInitSegment = null; this._onMediaSegment = null; // Workaround for chrome < 50: Always force first sample as a Random Access Point in media segment // see https://bugs.chromium.org/p/chromium/issues/detail?id=229412 this._forceFirstIDR = _browser2.default.chrome && (_browser2.default.version.major < 50 || _browser2.default.version.major === 50 && _browser2.default.version.build < 2661) ? true : false; // Workaround for IE11/Edge: Fill silent aac frame after keyframe-seeking // Make audio beginDts equals with video beginDts, in order to fix seek freeze this._fillSilentAfterSeek = _browser2.default.msedge || _browser2.default.msie; } _createClass(MP4Remuxer, [{ key: 'destroy', value: function destroy() { this._dtsBase = -1; this._dtsBaseInited = false; this._audioMeta = null; this._videoMeta = null; this._audioSegmentInfoList.clear(); this._audioSegmentInfoList = null; this._videoSegmentInfoList.clear(); this._videoSegmentInfoList = null; this._onInitSegment = null; this._onMediaSegment = null; } }, { key: 'bindDataSource', value: function bindDataSource(producer) { producer.onDataAvailable = this.remux.bind(this); producer.onTrackMetadata = this._onTrackMetadataReceived.bind(this); return this; } /* prototype: function onInitSegment(type: string, initSegment: ArrayBuffer): void InitSegment: { type: string, data: ArrayBuffer, codec: string, container: string } */ }, { key: 'insertDiscontinuity', value: function insertDiscontinuity() { this._audioNextDts = this._videoNextDts = undefined; } }, { key: 'seek', value: function seek(originalDts) { this._videoSegmentInfoList.clear(); this._audioSegmentInfoList.clear(); } }, { key: 'remux', value: function remux(audioTrack, videoTrack) { if (!this._onMediaSegment) { throw new _exception.IllegalStateException('MP4Remuxer: onMediaSegment callback must be specificed!'); } if (!this._dtsBaseInited) { this._calculateDtsBase(audioTrack, videoTrack); } this._remuxVideo(videoTrack); this._remuxAudio(audioTrack); } }, { key: '_onTrackMetadataReceived', value: function _onTrackMetadataReceived(type, metadata) { var metabox = null; if (type === 'audio') { this._audioMeta = metadata; metabox = _mp4Generator2.default.generateInitSegment(metadata); } else if (type === 'video') { this._videoMeta = metadata; metabox = _mp4Generator2.default.generateInitSegment(metadata); } else { return; } // dispatch metabox (Initialization Segment) if (!this._onInitSegment) { throw new _exception.IllegalStateException('MP4Remuxer: onInitSegment callback must be specified!'); } this._onInitSegment(type, { type: type, data: metabox.buffer, codec: metadata.codec, container: type + '/mp4' }); } }, { key: '_calculateDtsBase', value: function _calculateDtsBase(audioTrack, videoTrack) { if (this._dtsBaseInited) { return; } if (audioTrack.samples && audioTrack.samples.length) { this._audioDtsBase = audioTrack.samples[0].dts; } if (videoTrack.samples && videoTrack.samples.length) { this._videoDtsBase = videoTrack.samples[0].dts; } this._dtsBase = Math.min(this._audioDtsBase, this._videoDtsBase); this._dtsBaseInited = true; } }, { key: '_remuxAudio', value: function _remuxAudio(audioTrack) { var this$1 = this; var track = audioTrack; var samples = track.samples; var dtsCorrection = undefined; var firstDts = -1, lastDts = -1, lastPts = -1; var remuxSilentFrame = false; var silentFrameDuration = -1; if (!samples || samples.length === 0) { return; } var bytes = 8 + track.length; var mdatbox = new Uint8Array(bytes); mdatbox[0] = bytes >>> 24 & 0xFF; mdatbox[1] = bytes >>> 16 & 0xFF; mdatbox[2] = bytes >>> 8 & 0xFF; mdatbox[3] = bytes & 0xFF; mdatbox.set(_mp4Generator2.default.types.mdat, 4); var offset = 8; // size + type var mp4Samples = []; while (samples.length) { var aacSample = samples.shift(); var unit = aacSample.unit; var originalDts = aacSample.dts - this$1._dtsBase; if (dtsCorrection == undefined) { if (this$1._audioNextDts == undefined) { if (this$1._audioSegmentInfoList.isEmpty()) { dtsCorrection = 0; if (this$1._fillSilentAfterSeek && !this$1._videoSegmentInfoList.isEmpty()) { remuxSilentFrame = true; } } else { var lastSample = this$1._audioSegmentInfoList.getLastSampleBefore(originalDts); if (lastSample != null) { var distance = originalDts - (lastSample.originalDts + lastSample.duration); if (distance <= 3) { distance = 0; } var expectedDts = lastSample.dts + lastSample.duration + distance; dtsCorrection = originalDts - expectedDts; } else { // lastSample == null dtsCorrection = 0; } } } else { dtsCorrection = originalDts - this$1._audioNextDts; } } var dts = originalDts - dtsCorrection; if (remuxSilentFrame) { // align audio segment beginDts to match with current video segment's beginDts var videoSegment = this$1._videoSegmentInfoList.getLastSegmentBefore(originalDts); if (videoSegment != null && videoSegment.beginDts < dts) { silentFrameDuration = dts - videoSegment.beginDts; dts = videoSegment.beginDts; } else { remuxSilentFrame = false; } } if (firstDts === -1) { firstDts = dts; } if (remuxSilentFrame) { remuxSilentFrame = false; samples.unshift(aacSample); var frame = this$1._generateSilentAudio(dts, silentFrameDuration); if (frame == null) { continue; } var _mp4Sample = frame.mp4Sample; var _unit = frame.unit; mp4Samples.push(_mp4Sample); // re-allocate mdatbox buffer with new size, to fit with this silent frame bytes += _unit.byteLength; mdatbox = new Uint8Array(bytes); mdatbox[0] = bytes >>> 24 & 0xFF; mdatbox[1] = bytes >>> 16 & 0xFF; mdatbox[2] = bytes >>> 8 & 0xFF; mdatbox[3] = bytes & 0xFF; mdatbox.set(_mp4Generator2.default.types.mdat, 4); // fill data now mdatbox.set(_unit, offset); offset += _unit.byteLength; continue; } var sampleDuration = 0; if (samples.length >= 1) { var nextDts = samples[0].dts - this$1._dtsBase - dtsCorrection; sampleDuration = nextDts - dts; } else { if (mp4Samples.length >= 1) { // use second last sample duration sampleDuration = mp4Samples[mp4Samples.length - 1].duration; } else { // the only one sample, use reference sample duration sampleDuration = this$1._audioMeta.refSampleDuration; } } var mp4Sample = { dts: dts, pts: dts, cts: 0, size: unit.byteLength, duration: sampleDuration, originalDts: originalDts, flags: { isLeading: 0, dependsOn: 1, isDependedOn: 0, hasRedundancy: 0 } }; mp4Samples.push(mp4Sample); mdatbox.set(unit, offset); offset += unit.byteLength; } var latest = mp4Samples[mp4Samples.length - 1]; lastDts = latest.dts + latest.duration; this._audioNextDts = lastDts; // fill media segment info & add to info list var info = new _mediaSegmentInfo.MediaSegmentInfo(); info.beginDts = firstDts; info.endDts = lastDts; info.beginPts = firstDts; info.endPts = lastDts; info.originalBeginDts = mp4Samples[0].originalDts; info.originalEndDts = latest.originalDts + latest.duration; info.firstSample = new _mediaSegmentInfo.SampleInfo(mp4Samples[0].dts, mp4Samples[0].pts, mp4Samples[0].duration, mp4Samples[0].originalDts, false); info.lastSample = new _mediaSegmentInfo.SampleInfo(latest.dts, latest.pts, latest.duration, latest.originalDts, false); if (!this._isLive) { this._audioSegmentInfoList.append(info); } track.samples = mp4Samples; track.sequenceNumber++; var moofbox = _mp4Generator2.default.moof(track, firstDts); track.samples = []; track.length = 0; this._onMediaSegment('audio', { type: 'audio', data: this._mergeBoxes(moofbox, mdatbox).buffer, sampleCount: mp4Samples.length, info: info }); } }, { key: '_generateSilentAudio', value: function _generateSilentAudio(dts, frameDuration) { _logger2.default.v(this.TAG, 'GenerateSilentAudio: dts = ' + dts + ', duration = ' + frameDuration); var unit = _aacSilent2.default.getSilentFrame(this._audioMeta.channelCount); if (unit == null) { _logger2.default.w(this.TAG, 'Cannot generate silent aac frame for channelCount = ' + this._audioMeta.channelCount); return null; } var mp4Sample = { dts: dts, pts: dts, cts: 0, size: unit.byteLength, duration: frameDuration, originalDts: dts, flags: { isLeading: 0, dependsOn: 1, isDependedOn: 0, hasRedundancy: 0 } }; return { unit: unit, mp4Sample: mp4Sample }; } }, { key: '_remuxVideo', value: function _remuxVideo(videoTrack) { var this$1 = this; var track = videoTrack; var samples = track.samples; var dtsCorrection = undefined; var firstDts = -1, lastDts = -1; var firstPts = -1, lastPts = -1; if (!samples || samples.length === 0) { return; } var bytes = 8 + videoTrack.length; var mdatbox = new Uint8Array(bytes); mdatbox[0] = bytes >>> 24 & 0xFF; mdatbox[1] = bytes >>> 16 & 0xFF; mdatbox[2] = bytes >>> 8 & 0xFF; mdatbox[3] = bytes & 0xFF; mdatbox.set(_mp4Generator2.default.types.mdat, 4); var offset = 8; var mp4Samples = []; var info = new _mediaSegmentInfo.MediaSegmentInfo(); while (samples.length) { var avcSample = samples.shift(); var keyframe = avcSample.isKeyframe; var originalDts = avcSample.dts - this$1._dtsBase; if (dtsCorrection == undefined) { if (this$1._videoNextDts == undefined) { if (this$1._videoSegmentInfoList.isEmpty()) { dtsCorrection = 0; } else { var lastSample = this$1._videoSegmentInfoList.getLastSampleBefore(originalDts); if (lastSample != null) { var distance = originalDts - (lastSample.originalDts + lastSample.duration); if (distance <= 3) { distance = 0; } var expectedDts = lastSample.dts + lastSample.duration + distance; dtsCorrection = originalDts - expectedDts; } else { // lastSample == null dtsCorrection = 0; } } } else { dtsCorrection = originalDts - this$1._videoNextDts; } } var dts = originalDts - dtsCorrection; var cts = avcSample.cts; var pts = dts + cts; if (firstDts === -1) { firstDts = dts; firstPts = pts; } // fill mdat box var sampleSize = 0; while (avcSample.units.length) { var unit = avcSample.units.shift(); var data = unit.data; mdatbox.set(data, offset); offset += data.byteLength; sampleSize += data.byteLength; } var sampleDuration = 0; if (samples.length >= 1) { var nextDts = samples[0].dts - this$1._dtsBase - dtsCorrection; sampleDuration = nextDts - dts; } else { if (mp4Samples.length >= 1) { // lastest sample, use second last duration sampleDuration = mp4Samples[mp4Samples.length - 1].duration; } else { // the only one sample, use reference duration sampleDuration = this$1._videoMeta.refSampleDuration; } } if (keyframe) { var syncPoint = new _mediaSegmentInfo.SampleInfo(dts, pts, sampleDuration, avcSample.dts, true); syncPoint.fileposition = avcSample.fileposition; info.appendSyncPoint(syncPoint); } var mp4Sample = { dts: dts, pts: pts, cts: cts, size: sampleSize, isKeyframe: keyframe, duration: sampleDuration, originalDts: originalDts, flags: { isLeading: 0, dependsOn: keyframe ? 2 : 1, isDependedOn: keyframe ? 1 : 0, hasRedundancy: 0, isNonSync: keyframe ? 0 : 1 } }; mp4Samples.push(mp4Sample); } var latest = mp4Samples[mp4Samples.length - 1]; lastDts = latest.dts + latest.duration; lastPts = latest.pts + latest.duration; this._videoNextDts = lastDts; // fill media segment info & add to info list info.beginDts = firstDts; info.endDts = lastDts; info.beginPts = firstPts; info.endPts = lastPts; info.originalBeginDts = mp4Samples[0].originalDts; info.originalEndDts = latest.originalDts + latest.duration; info.firstSample = new _mediaSegmentInfo.SampleInfo(mp4Samples[0].dts, mp4Samples[0].pts, mp4Samples[0].duration, mp4Samples[0].originalDts, mp4Samples[0].isKeyframe); info.lastSample = new _mediaSegmentInfo.SampleInfo(latest.dts, latest.pts, latest.duration, latest.originalDts, latest.isKeyframe); if (!this._isLive) { this._videoSegmentInfoList.append(info); } track.samples = mp4Samples; track.sequenceNumber++; // workaround for chrome < 50: force first sample as a random access point // see https://bugs.chromium.org/p/chromium/issues/detail?id=229412 if (this._forceFirstIDR) { var flags = mp4Samples[0].flags; flags.dependsOn = 2; flags.isNonSync = 0; } var moofbox = _mp4Generator2.default.moof(track, firstDts); track.samples = []; track.length = 0; this._onMediaSegment('video', { type: 'video', data: this._mergeBoxes(moofbox, mdatbox).buffer, sampleCount: mp4Samples.length, info: info }); } }, { key: '_mergeBoxes', value: function _mergeBoxes(moof, mdat) { var result = new Uint8Array(moof.byteLength + mdat.byteLength); result.set(moof, 0); result.set(mdat, moof.byteLength); return result; } }, { key: 'onInitSegment', get: function get() { return this._onInitSegment; }, set: function set(callback) { this._onInitSegment = callback; } /* prototype: function onMediaSegment(type: string, mediaSegment: MediaSegment): void MediaSegment: { type: string, data: ArrayBuffer, sampleCount: int32 info: MediaSegmentInfo } */ }, { key: 'onMediaSegment', get: function get() { return this._onMediaSegment; }, set: function set(callback) { this._onMediaSegment = callback; } }]); return MP4Remuxer; }(); exports.default = MP4Remuxer; },{"../core/media-segment-info.js":8,"../utils/browser.js":39,"../utils/exception.js":40,"../utils/logger.js":41,"./aac-silent.js":36,"./mp4-generator.js":37}],39:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var Browser = {}; function detect() { // modified from jquery-browser-plugin var ua = self.navigator.userAgent.toLowerCase(); var match = /(edge)\/([\w.]+)/.exec(ua) || /(opr)[\/]([\w.]+)/.exec(ua) || /(chrome)[ \/]([\w.]+)/.exec(ua) || /(iemobile)[\/]([\w.]+)/.exec(ua) || /(version)(applewebkit)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec(ua) || /(webkit)[ \/]([\w.]+).*(version)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec(ua) || /(webkit)[ \/]([\w.]+)/.exec(ua) || /(opera)(?:.*version|)[ \/]([\w.]+)/.exec(ua) || /(msie) ([\w.]+)/.exec(ua) || ua.indexOf('trident') >= 0 && /(rv)(?::| )([\w.]+)/.exec(ua) || ua.indexOf('compatible') < 0 && /(mozilla)(?:.*? rv:([\w.]+)|)/.exec(ua) || []; var platform_match = /(ipad)/.exec(ua) || /(ipod)/.exec(ua) || /(windows phone)/.exec(ua) || /(iphone)/.exec(ua) || /(kindle)/.exec(ua) || /(android)/.exec(ua) || /(windows)/.exec(ua) || /(mac)/.exec(ua) || /(linux)/.exec(ua) || /(cros)/.exec(ua) || []; var matched = { browser: match[5] || match[3] || match[1] || '', version: match[2] || match[4] || '0', majorVersion: match[4] || match[2] || '0', platform: platform_match[0] || '' }; var browser = {}; if (matched.browser) { browser[matched.browser] = true; var versionArray = matched.majorVersion.split('.'); browser.version = { major: parseInt(matched.majorVersion, 10), string: matched.version }; if (versionArray.length > 1) { browser.version.minor = parseInt(versionArray[1], 10); } if (versionArray.length > 2) { browser.version.build = parseInt(versionArray[2], 10); } } if (matched.platform) { browser[matched.platform] = true; } if (browser.chrome || browser.opr || browser.safari) { browser.webkit = true; } // MSIE. IE11 has 'rv' identifer if (browser.rv || browser.iemobile) { if (browser.rv) { delete browser.rv; } var msie = 'msie'; matched.browser = msie; browser[msie] = true; } // Microsoft Edge if (browser.edge) { delete browser.edge; var msedge = 'msedge'; matched.browser = msedge; browser[msedge] = true; } // Opera 15+ if (browser.opr) { var opera = 'opera'; matched.browser = opera; browser[opera] = true; } // Stock android browsers are marked as Safari if (browser.safari && browser.android) { var android = 'android'; matched.browser = android; browser[android] = true; } browser.name = matched.browser; browser.platform = matched.platform; for (var key in Browser) { if (Browser.hasOwnProperty(key)) { delete Browser[key]; } } Object.assign(Browser, browser); } detect(); exports.default = Browser; },{}],40:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) { Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var RuntimeException = exports.RuntimeException = function () { function RuntimeException(message) { _classCallCheck(this, RuntimeException); this._message = message; } _createClass(RuntimeException, [{ key: 'toString', value: function toString() { return this.name + ': ' + this.message; } }, { key: 'name', get: function get() { return 'RuntimeException'; } }, { key: 'message', get: function get() { return this._message; } }]); return RuntimeException; }(); var IllegalStateException = exports.IllegalStateException = function (_RuntimeException) { _inherits(IllegalStateException, _RuntimeException); function IllegalStateException(message) { _classCallCheck(this, IllegalStateException); return _possibleConstructorReturn(this, (IllegalStateException.__proto__ || Object.getPrototypeOf(IllegalStateException)).call(this, message)); } _createClass(IllegalStateException, [{ key: 'name', get: function get() { return 'IllegalStateException'; } }]); return IllegalStateException; }(RuntimeException); var InvalidArgumentException = exports.InvalidArgumentException = function (_RuntimeException2) { _inherits(InvalidArgumentException, _RuntimeException2); function InvalidArgumentException(message) { _classCallCheck(this, InvalidArgumentException); return _possibleConstructorReturn(this, (InvalidArgumentException.__proto__ || Object.getPrototypeOf(InvalidArgumentException)).call(this, message)); } _createClass(InvalidArgumentException, [{ key: 'name', get: function get() { return 'InvalidArgumentException'; } }]); return InvalidArgumentException; }(RuntimeException); var NotImplementedException = exports.NotImplementedException = function (_RuntimeException3) { _inherits(NotImplementedException, _RuntimeException3); function NotImplementedException(message) { _classCallCheck(this, NotImplementedException); return _possibleConstructorReturn(this, (NotImplementedException.__proto__ || Object.getPrototypeOf(NotImplementedException)).call(this, message)); } _createClass(NotImplementedException, [{ key: 'name', get: function get() { return 'NotImplementedException'; } }]); return NotImplementedException; }(RuntimeException); },{}],41:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var Log = function () { function Log() { _classCallCheck(this, Log); } _createClass(Log, null, [{ key: 'e', value: function e(tag, msg) { if (!Log.ENABLE_ERROR) { return; } if (!tag || Log.FORCE_GLOBAL_TAG) { tag = Log.GLOBAL_TAG; } var str = '[' + tag + '] > ' + msg; if (console.error) { console.error(str); } else if (console.warn) { console.warn(str); } else { console.log(str); } } }, { key: 'i', value: function i(tag, msg) { if (!Log.ENABLE_INFO) { return; } if (!tag || Log.FORCE_GLOBAL_TAG) { tag = Log.GLOBAL_TAG; } var str = '[' + tag + '] > ' + msg; if (console.info) { console.info(str); } else { console.log(str); } } }, { key: 'w', value: function w(tag, msg) { if (!Log.ENABLE_WARN) { return; } if (!tag || Log.FORCE_GLOBAL_TAG) { tag = Log.GLOBAL_TAG; } var str = '[' + tag + '] > ' + msg; if (console.warn) { console.warn(str); } else { console.log(str); } } }, { key: 'd', value: function d(tag, msg) { if (!Log.ENABLE_DEBUG) { return; } if (!tag || Log.FORCE_GLOBAL_TAG) { tag = Log.GLOBAL_TAG; } var str = '[' + tag + '] > ' + msg; if (console.debug) { console.debug(str); } else { console.log(str); } } }, { key: 'v', value: function v(tag, msg) { if (!Log.ENABLE_VERBOSE) { return; } if (!tag || Log.FORCE_GLOBAL_TAG) { tag = Log.GLOBAL_TAG; } console.log('[' + tag + '] > ' + msg); } }]); return Log; }(); Log.GLOBAL_TAG = 'flv.js'; Log.FORCE_GLOBAL_TAG = false; Log.ENABLE_ERROR = true; Log.ENABLE_INFO = true; Log.ENABLE_WARN = true; Log.ENABLE_DEBUG = true; Log.ENABLE_VERBOSE = true; exports.default = Log; },{}],42:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var _events = _dereq_('events'); var _events2 = _interopRequireDefault(_events); var _logger = _dereq_('./logger.js'); var _logger2 = _interopRequireDefault(_logger); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } var LoggingControl = function () { function LoggingControl() { _classCallCheck(this, LoggingControl); } _createClass(LoggingControl, null, [{ key: 'getConfig', value: function getConfig() { return { globalTag: _logger2.default.GLOBAL_TAG, forceGlobalTag: _logger2.default.FORCE_GLOBAL_TAG, enableVerbose: _logger2.default.ENABLE_VERBOSE, enableDebug: _logger2.default.ENABLE_DEBUG, enableInfo: _logger2.default.ENABLE_INFO, enableWarn: _logger2.default.ENABLE_WARN, enableError: _logger2.default.ENABLE_ERROR }; } }, { key: 'applyConfig', value: function applyConfig(config) { _logger2.default.GLOBAL_TAG = config.globalTag; _logger2.default.FORCE_GLOBAL_TAG = config.forceGlobalTag; _logger2.default.ENABLE_VERBOSE = config.enableVerbose; _logger2.default.ENABLE_DEBUG = config.enableDebug; _logger2.default.ENABLE_INFO = config.enableInfo; _logger2.default.ENABLE_WARN = config.enableWarn; _logger2.default.ENABLE_ERROR = config.enableError; } }, { key: '_notifyChange', value: function _notifyChange() { var emitter = LoggingControl.emitter; if (emitter.listenerCount('change') > 0) { var config = LoggingControl.getConfig(); emitter.emit('change', config); } } }, { key: 'registerListener', value: function registerListener(listener) { LoggingControl.emitter.addListener('change', listener); } }, { key: 'removeListener', value: function removeListener(listener) { LoggingControl.emitter.removeListener('change', listener); } }, { key: 'forceGlobalTag', get: function get() { return _logger2.default.FORCE_GLOBAL_TAG; }, set: function set(enable) { _logger2.default.FORCE_GLOBAL_TAG = enable; LoggingControl._notifyChange(); } }, { key: 'globalTag', get: function get() { return _logger2.default.GLOBAL_TAG; }, set: function set(tag) { _logger2.default.GLOBAL_TAG = tag; LoggingControl._notifyChange(); } }, { key: 'enableAll', get: function get() { return _logger2.default.ENABLE_VERBOSE && _logger2.default.ENABLE_DEBUG && _logger2.default.ENABLE_INFO && _logger2.default.ENABLE_WARN && _logger2.default.ENABLE_ERROR; }, set: function set(enable) { _logger2.default.ENABLE_VERBOSE = enable; _logger2.default.ENABLE_DEBUG = enable; _logger2.default.ENABLE_INFO = enable; _logger2.default.ENABLE_WARN = enable; _logger2.default.ENABLE_ERROR = enable; LoggingControl._notifyChange(); } }, { key: 'enableDebug', get: function get() { return _logger2.default.ENABLE_DEBUG; }, set: function set(enable) { _logger2.default.ENABLE_DEBUG = enable; LoggingControl._notifyChange(); } }, { key: 'enableVerbose', get: function get() { return _logger2.default.ENABLE_VERBOSE; }, set: function set(enable) { _logger2.default.ENABLE_VERBOSE = enable; LoggingControl._notifyChange(); } }, { key: 'enableInfo', get: function get() { return _logger2.default.ENABLE_INFO; }, set: function set(enable) { _logger2.default.ENABLE_INFO = enable; LoggingControl._notifyChange(); } }, { key: 'enableWarn', get: function get() { return _logger2.default.ENABLE_WARN; }, set: function set(enable) { _logger2.default.ENABLE_WARN = enable; LoggingControl._notifyChange(); } }, { key: 'enableError', get: function get() { return _logger2.default.ENABLE_ERROR; }, set: function set(enable) { _logger2.default.ENABLE_ERROR = enable; LoggingControl._notifyChange(); } }]); return LoggingControl; }(); LoggingControl.emitter = new _events2.default(); exports.default = LoggingControl; },{"./logger.js":41,"events":2}],43:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) { descriptor.writable = true; } Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) { defineProperties(Constructor.prototype, protoProps); } if (staticProps) { defineProperties(Constructor, staticProps); } return Constructor; }; }(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var Polyfill = function () { function Polyfill() { _classCallCheck(this, Polyfill); } _createClass(Polyfill, null, [{ key: 'install', value: function install() { // ES6 Object.setPrototypeOf Object.setPrototypeOf = Object.setPrototypeOf || function (obj, proto) { obj.__proto__ = proto; return obj; }; // ES6 Object.assign Object.assign = Object.assign || function (target) { var arguments$1 = arguments; if (target === undefined || target === null) { throw new TypeError('Cannot convert undefined or null to object'); } var output = Object(target); for (var i = 1; i < arguments.length; i++) { var source = arguments$1[i]; if (source !== undefined && source !== null) { for (var key in source) { if (source.hasOwnProperty(key)) { output[key] = source[key]; } } } } return output; }; // ES6 Promise (missing support in IE11) if (typeof self.Promise !== 'function') { _dereq_('es6-promise').polyfill(); } } }]); return Polyfill; }(); Polyfill.install(); exports.default = Polyfill; },{"es6-promise":1}],44:[function(_dereq_,module,exports){ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); /* * Copyright (C) 2016 Bilibili. All Rights Reserved. * * This file is derived from C++ project libWinTF8 (https://github.com/m13253/libWinTF8) * @author zheng qian * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ function checkContinuation(uint8array, start, checkLength) { var array = uint8array; if (start + checkLength < array.length) { while (checkLength--) { if ((array[++start] & 0xC0) !== 0x80) { return false; } } return true; } else { return false; } } function decodeUTF8(uint8array) { var out = []; var input = uint8array; var i = 0; var length = uint8array.length; while (i < length) { if (input[i] < 0x80) { out.push(String.fromCharCode(input[i])); ++i; continue; } else if (input[i] < 0xC0) { // fallthrough } else if (input[i] < 0xE0) { if (checkContinuation(input, i, 1)) { var ucs4 = (input[i] & 0x1F) << 6 | input[i + 1] & 0x3F; if (ucs4 >= 0x80) { out.push(String.fromCharCode(ucs4 & 0xFFFF)); i += 2; continue; } } } else if (input[i] < 0xF0) { if (checkContinuation(input, i, 2)) { var _ucs = (input[i] & 0xF) << 12 | (input[i + 1] & 0x3F) << 6 | input[i + 2] & 0x3F; if (_ucs >= 0x800 && (_ucs & 0xF800) !== 0xD800) { out.push(String.fromCharCode(_ucs & 0xFFFF)); i += 3; continue; } } } else if (input[i] < 0xF8) { if (checkContinuation(input, i, 3)) { var _ucs2 = (input[i] & 0x7) << 18 | (input[i + 1] & 0x3F) << 12 | (input[i + 2] & 0x3F) << 6 | input[i + 3] & 0x3F; if (_ucs2 > 0x10000 && _ucs2 < 0x110000) { _ucs2 -= 0x10000; out.push(String.fromCharCode(_ucs2 >>> 10 | 0xD800)); out.push(String.fromCharCode(_ucs2 & 0x3FF | 0xDC00)); i += 4; continue; } } } out.push(String.fromCharCode(0xFFFD)); ++i; } return out.join(''); } exports.default = decodeUTF8; },{}]},{},[21])(21) }); }); var flvjs = unwrapExports(flv); var emptyFunc = function () { return null; }; var requestFullScreen = function () { var de = document.documentElement; if (de.requestFullscreen) { de.requestFullscreen(); } else if (de.mozRequestFullScreen) { de.mozRequestFullScreen(); } else if (de.webkitRequestFullScreen) { de.webkitRequestFullScreen(); } }; function exitFullscreen() { var de = document; if (de.exitFullscreen) { de.exitFullscreen(); } else if (de.mozCancelFullScreen) { de.mozCancelFullScreen(); } else if (de.webkitCancelFullScreen) { de.webkitCancelFullScreen(); } } function findInParent (node, toFind) { while ((node !== toFind) && (node !== null)) { node = node.parentElement; } return node !== null } function DanmuPlayerControls (listener) { var this$1 = this; this.onReload = emptyFunc; this.onSendDanmu = emptyFunc; Object.keys(listener).forEach(function (key) { return this$1[key] = listener[key]; }); } var DanmuPlayer = function DanmuPlayer (controls, isSelf) { var this$1 = this; var poolSize = 100; var playerContainer = document.createElement('div'); var playerWrap = document.createElement('div'); var playerCtrl = document.createElement('div'); var danmuLayout = document.createElement('div'); var videoBox = document.createElement('div'); var msgBox = document.createElement('div'); var msgInput = document.createElement('input'); var videoEl = document.createElement('video'); videoEl.style.width = videoEl.style.height = '100%'; msgInput.type = 'text'; msgInput.placeholder = '发送弹幕...'; msgBox.className = 'danmu-input'; videoBox.className = 'danmu-video'; playerCtrl.className = 'danmu-ctrl'; danmuLayout.className = 'danmu-layout'; playerWrap.className = 'danmu-wrap'; playerContainer.className = 'danmu-container'; videoBox.appendChild(videoEl); msgBox.appendChild(msgInput); playerWrap.appendChild(videoBox); playerWrap.appendChild(playerCtrl); playerWrap.appendChild(danmuLayout); playerWrap.appendChild(msgBox); playerContainer.appendChild(playerWrap); playerWrap.addEventListener('mousemove', function (event) { // const hoverCtl = event.path.indexOf(playerCtrl) !== -1 var hoverCtl = findInParent(event.target, playerCtrl); if (event.offsetY - playerWrap.lastY == 0) { return } playerWrap.lastY = event.offsetY; var height = playerWrap.getBoundingClientRect().height; if (event.offsetY > 0) { playerWrap.setAttribute('hover', ''); if (this$1._moveId) { clearTimeout(this$1._moveId); } if (!hoverCtl) { this$1._moveId = setTimeout(function () { return playerWrap.removeAttribute('hover'); }, 1000); } } else { playerWrap.removeAttribute('hover'); } }); this.inputing = false; playerWrap.addEventListener('click', function (event) { // if (event.path.indexOf(msgBox) !== -1) return if (findInParent(event.target, msgBox)) { return } playerWrap.removeAttribute('inputing'); this$1.inputing = false; }); document.addEventListener('keydown', function (event) { if (event.keyCode == 13) { // enter if (playerWrap.getAttribute('fullpage') === null) { return } this$1.inputing = !this$1.inputing; if (this$1.inputing) { msgInput.value = ''; playerWrap.setAttribute('inputing', ''); msgInput.focus(); } else { if (msgInput.value.length > 0) { controls.onSendDanmu(msgInput.value); } playerWrap.removeAttribute('inputing'); } } else if (event.keyCode == 27) { // esc if (this$1.wrap.getAttribute('fullpage') === '') { this$1.onFullpage(); } } }); document.addEventListener('webkitfullscreenchange', function (event) { this$1.fullscreen = !this$1.fullscreen; if (!this$1.fullscreen) { this$1.onFullpage(); } }); window.addEventListener('unload', function (event) { this$1.stop(); }); this.parsePic = function (i) { return i; }; this.isSelf = isSelf || emptyFunc; this.controls = controls; this.video = videoEl; this.el = playerContainer; this.wrap = playerWrap; this.dmLayout = danmuLayout; this.playerCtrl = playerCtrl; this.curDanmu = []; this.pool = []; this.rows = []; this.baseTop = 10; this.maxRow = 10; this.deferTime = 0; // 暂停时间 this.deferQueue = []; this.hideDanmu = false; this.playing = true; this.muted = false; this.fullscreen = false; this._src = ''; this.player = null; for (var i = 0; i < poolSize; i++) { var dm = document.createElement('div'); dm.using = false; danmuLayout.appendChild(dm); this$1.pool.push(dm); } this.transparent = this.transparent; }; var prototypeAccessors = { transparent: {},src: {} }; prototypeAccessors.transparent.get = function () { return parseInt(window.localStorage.getItem('h5plr-transparent') || '0') }; prototypeAccessors.transparent.set = function (val) { window.localStorage.setItem('h5plr-transparent', val); this.dmLayout.style.opacity = 1 - val / 100; }; DanmuPlayer.prototype.onStat = function onStat (e) { this.setTip(parseInt(e.speed*10)/10 + 'KB/s'); }; DanmuPlayer.prototype.createFlvjs = function createFlvjs () { var this$1 = this; var sourceConfig = { isLive: true, type: 'flv', url: this.src }; var playerConfig = { enableWorker: false, deferLoadAfterSourceOpen: true, stashInitialSize: 512*1024, enableStashBuffer: true }; var player = flvjs.createPlayer(sourceConfig, playerConfig); player.on(flvjs.Events.ERROR, function(e, t) { console.error('播放器发生错误:' + e + ' - ' + t); player.unload(); }); player.on(flvjs.Events.STATISTICS_INFO, function (e) { return this$1.onStat(e); }); player.attachMediaElement(this.video); player.load(); player.play(); return player }; DanmuPlayer.prototype.stop = function stop () { if (this.player) { this.player.unload(); this.player.detachMediaElement(); this.player = null; } }; prototypeAccessors.src.set = function (val) { this._src = val; this.stop(); var player = this.createFlvjs(); this.player = player; if (!this._ctrl) { this.initControls(); this._ctrl = true; } }; prototypeAccessors.src.get = function () { return this._src }; DanmuPlayer.prototype.createVolume = function createVolume (cb) { var volume = document.createElement('div'); var progress = document.createElement('div'); var input = document.createElement('input'); volume.className = 'danmu-volume'; progress.className = 'progress'; input.type = 'range'; volume.appendChild(input); volume.appendChild(progress); input.value = localStorage.getItem('volume') || '100'; cb(input.value / 100); input.addEventListener('input', function (event) { progress.style.width = (input.value) + "%"; cb(input.value / 100); localStorage.setItem('volume', input.value); }); progress.style.width = (input.value) + "%"; return volume }; DanmuPlayer.prototype.initControls = function initControls () { var this$1 = this; var bar = this.playerCtrl; var now = function () { return new Date().getTime(); }; var addBtn = function (cls, cb) { var btn = document.createElement('div'); btn.className = ['danmu-btn', 'danmu-'+cls].join(' '); btn.addEventListener('click', cb); bar.appendChild(btn); return btn }; this.video.addEventListener('dblclick', function (event) { this$1.onFullpage(); event.preventDefault(); event.stopPropagation(); }); var beginTime = 0; var playPause = addBtn('playpause', function () { this$1.playing = !this$1.playing; if (this$1.playing) { this$1.deferTime += now() - beginTime; this$1.player.play(); playPause.setAttribute('pause', ''); } else { beginTime = now(); this$1.player.pause(); playPause.removeAttribute('pause'); } }); playPause.setAttribute('pause', ''); var reload = addBtn('reload', function () { this$1.deferTime = 0; this$1.controls.onReload(); }); var fullscreen = addBtn('fullscreen', function () { this$1.onFullpage(true); }); var fullpage = addBtn('fullpage', function () { this$1.onFullpage(); }); var volume = this.createVolume(function (percent) { // volume this$1.player.volume = percent; }); bar.appendChild(volume); var lastVolume; var mute = addBtn('mute', function () { this$1.muted = !this$1.muted; if (this$1.muted) { lastVolume = this$1.player.volume; this$1.player.volume = 0; mute.setAttribute('muted', ''); } else { this$1.player.volume = lastVolume; mute.removeAttribute('muted'); } }); var danmuSwitch = addBtn('switch', function () { this$1.hideDanmu = !this$1.hideDanmu; danmuSwitch.innerText = this$1.hideDanmu ? '开启弹幕' : '关闭弹幕'; this$1.dmLayout.style.display = this$1.hideDanmu ? 'none' : 'block'; }); danmuSwitch.innerText = this.hideDanmu ? '开启弹幕' : '关闭弹幕'; var tip = document.createElement('div'); tip.className = 'danmu-tip'; bar.appendChild(tip); this.tipEl = tip; }; DanmuPlayer.prototype.setTip = function setTip (tip) { this.tipEl.innerText = tip; }; DanmuPlayer.prototype.tryPlay = function tryPlay () { if (this.playing) { try { this.video.play(); } catch (e) {} } }; DanmuPlayer.prototype.onFullpage = function onFullpage (fullScreen) { if (this.wrap.getAttribute('fullpage') === null) { if (fullScreen) { requestFullScreen(); } this.wrap.setAttribute('fullpage', ''); document.body.appendChild(this.wrap); document.body.style.overflow = 'hidden'; } else { exitFullscreen(); this.wrap.removeAttribute('fullpage'); this.el.appendChild(this.wrap); document.body.style.overflow = 'auto'; } this.tryPlay(); }; DanmuPlayer.prototype.onDanmu = function onDanmu (pkg) { var example = { "type": "chatmsg", "rid": "510541", "ct": "1", // 酬勤 "uid": "59839409", "nn": "登辛", "txt": "3ds没有鼓棒先生吗", "cid": "ce554df5bf2841e41459070000000000", "ic": "avatar/face/201607/27/12d23d30a9a7790e955d7affc54335ad", "level": "17", "gt": "2", // "rg": "4", // "el": "eid@A=1500000005@Setp@A=1@Ssc@A=1@Sef@A=0@S/" }; var getColor = function (c) { return ["#ff0000", "#1e87f0", "#7ac84b", "#ff7f00", "#9b39f4", "#ff69b4"][c-1]; }; if (pkg.txt.length > 0) { var cls = []; var color = getColor(pkg.col) || '#ffffff'; if (this.isSelf(pkg)) { cls.push('danmu-self'); } this.fireDanmu(pkg.txt, color, cls); } }; DanmuPlayer.prototype.calcRect = function calcRect () { return this.dmLayout.getBoundingClientRect() }; DanmuPlayer.prototype.reset = function reset () { this.playing = true; this.deferTime = 0; }; DanmuPlayer.prototype.doDefer = function doDefer () { var this$1 = this; var top = this.deferQueue[0]; var now = new Date().getTime(); if (this.playing && ((top.oriTime + this.deferTime) <= now)) { // console.log(top.oriTime, this.deferTime, now) top.run(); this.deferQueue.shift(); } if (this.deferQueue.length !== 0) { // const next = this.deferQueue[0] setTimeout(function () { return this$1.doDefer(); }, 100); } }; DanmuPlayer.prototype.fireDanmu = function fireDanmu (text, color, cls) { var this$1 = this; var fire = function () { var rect = this$1.calcRect(); var duration = rect.width * 7; var dm = this$1.pool.shift(); setTimeout(function () { dm.removeAttribute('style'); this$1.pool.push(dm); }, duration); dm.innerText = text; dm.innerHTML = this$1.parsePic(dm.innerHTML); if (Array.isArray(cls)) { cls = cls.join(' '); } dm.className = cls || ''; dm.style.left = (rect.width) + "px"; dm.style.display = 'inline-block'; dm.style.color = color; setTimeout(function () { var dmRect = dm.getBoundingClientRect(); // console.log(dmRect) var row = this$1.calcRow(dmRect.width, duration); // console.log('row', text, row) dm.style.top = (this$1.baseTop + row * dmRect.height) + "px"; dm.style.transition = "transform " + (duration/1000) + "s linear"; dm.style.transform = "translateX(-" + (rect.width+dmRect.width) + "px)"; }, 0); }; var now = new Date().getTime(); if (!this.playing || this.deferTime > 0) { if (this.deferQueue.length === 0) { setTimeout(function () { return this$1.doDefer(); }, 100); } this.deferQueue.push({ oriTime: now, run: function () { return fire(); } }); return } if (this.hideDanmu) { return } if (this.pool.length == 0) { return } fire(); }; DanmuPlayer.prototype.calcRow = function calcRow (width, duration) { var this$1 = this; var rect = this.calcRect(); var now = new Date().getTime(); var check = function (row) { row = this$1.rows[row]; if (!row) { return true } if (row.endTime <= now) { this$1.rows[row] = null; return true } else { var distance = rect.width + row.width; var percent = (now - row.beginTime) / row.duration; var left = rect.width - distance * percent; if (left + row.width >= rect.width) { return false } var remainTime = row.endTime - now; var myDistance = rect.width + width; var leftX = rect.width - (myDistance) * (remainTime / duration); if (leftX < 0) { return false } } return true }; var add = function (row) { this$1.rows[row] = { duration: duration, beginTime: now, endTime: now + duration, width: width }; }; var i = 0; while(true) { if (check(i)) { add(i); return i % this$1.maxRow } i++; } }; Object.defineProperties( DanmuPlayer.prototype, prototypeAccessors ); var createMenu = function (x, y) { var wrap = document.createElement('div'); var menu = document.createElement('div'); wrap.className = 'player-menu'; menu.className = 'menu'; wrap.appendChild(menu); menu.style.left = x + "px"; menu.style.top = y + "px"; menu.close = function () { return document.body.removeChild(wrap); }; wrap.addEventListener('mousedown', function (event) { if (event.target === wrap) { document.body.removeChild(wrap); } }); wrap.addEventListener('contextmenu', function (event) { return event.preventDefault(); }); document.body.appendChild(wrap); return menu }; var addTextMenu = function (menu, text, cb) { var item = document.createElement('div'); item.className = 'menu-item'; item.innerText = text; menu.appendChild(item); item.addEventListener('click', function () { menu.close(); cb(); }); }; var addEleMenu = function (menu, ele) { var item = document.createElement('div'); item.className = 'menu-ele'; item.appendChild(ele); menu.appendChild(item); }; var addLabelMenu = function (menu, label) { var item = document.createElement('div'); item.className = 'menu-item'; item.innerText = label; menu.appendChild(item); }; var addDash = function (menu) { var item = document.createElement('div'); item.className = 'menu-dash'; menu.appendChild(item); }; function bindMenu (el, menuItems) { el.addEventListener('contextmenu', function (event) { var menu = createMenu(event.clientX, event.clientY); var items = menuItems; if (typeof items === 'function') { items = items(); } for (var i = 0, list = items; i < list.length; i += 1) { var item = list[i]; if (item.text) { addTextMenu(menu, item.text, item.cb); } else if (item.el) { addEleMenu(menu, item.el, item.cb); } else if (item.label) { addLabelMenu(menu, item.label); } else { addDash(menu); } } var rect = menu.getBoundingClientRect(); if (menu.offsetLeft + menu.offsetWidth > document.documentElement.clientWidth) { menu.style.left = (rect.left - rect.width) + "px"; } if (menu.offsetTop + menu.offsetHeight > document.documentElement.clientHeight) { menu.style.top = (rect.top - rect.height) + "px"; } event.preventDefault(); }); } var crypt = createCommonjsModule(function (module) { (function() { var base64map = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/', crypt = { // Bit-wise rotation left rotl: function(n, b) { return (n << b) | (n >>> (32 - b)); }, // Bit-wise rotation right rotr: function(n, b) { return (n << (32 - b)) | (n >>> b); }, // Swap big-endian to little-endian and vice versa endian: function(n) { // If number given, swap endian if (n.constructor == Number) { return crypt.rotl(n, 8) & 0x00FF00FF | crypt.rotl(n, 24) & 0xFF00FF00; } // Else, assume array and swap all items for (var i = 0; i < n.length; i++) { n[i] = crypt.endian(n[i]); } return n; }, // Generate an array of any length of random bytes randomBytes: function(n) { for (var bytes = []; n > 0; n--) { bytes.push(Math.floor(Math.random() * 256)); } return bytes; }, // Convert a byte array to big-endian 32-bit words bytesToWords: function(bytes) { for (var words = [], i = 0, b = 0; i < bytes.length; i++, b += 8) { words[b >>> 5] |= bytes[i] << (24 - b % 32); } return words; }, // Convert big-endian 32-bit words to a byte array wordsToBytes: function(words) { for (var bytes = [], b = 0; b < words.length * 32; b += 8) { bytes.push((words[b >>> 5] >>> (24 - b % 32)) & 0xFF); } return bytes; }, // Convert a byte array to a hex string bytesToHex: function(bytes) { for (var hex = [], i = 0; i < bytes.length; i++) { hex.push((bytes[i] >>> 4).toString(16)); hex.push((bytes[i] & 0xF).toString(16)); } return hex.join(''); }, // Convert a hex string to a byte array hexToBytes: function(hex) { for (var bytes = [], c = 0; c < hex.length; c += 2) { bytes.push(parseInt(hex.substr(c, 2), 16)); } return bytes; }, // Convert a byte array to a base-64 string bytesToBase64: function(bytes) { for (var base64 = [], i = 0; i < bytes.length; i += 3) { var triplet = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2]; for (var j = 0; j < 4; j++) { if (i * 8 + j * 6 <= bytes.length * 8) { base64.push(base64map.charAt((triplet >>> 6 * (3 - j)) & 0x3F)); } else { base64.push('='); } } } return base64.join(''); }, // Convert a base-64 string to a byte array base64ToBytes: function(base64) { // Remove non-base-64 characters base64 = base64.replace(/[^A-Z0-9+\/]/ig, ''); for (var bytes = [], i = 0, imod4 = 0; i < base64.length; imod4 = ++i % 4) { if (imod4 == 0) { continue; } bytes.push(((base64map.indexOf(base64.charAt(i - 1)) & (Math.pow(2, -2 * imod4 + 8) - 1)) << (imod4 * 2)) | (base64map.indexOf(base64.charAt(i)) >>> (6 - imod4 * 2))); } return bytes; } }; module.exports = crypt; })(); }); /*! * Determine if an object is a Buffer * * @author Feross Aboukhadijeh * @license MIT */ // The _isBuffer check is for Safari 5-7 support, because it's missing // Object.prototype.constructor. Remove this eventually var index = function (obj) { return obj != null && (isBuffer(obj) || isSlowBuffer(obj) || !!obj._isBuffer) }; function isBuffer (obj) { return !!obj.constructor && typeof obj.constructor.isBuffer === 'function' && obj.constructor.isBuffer(obj) } // For Node v0.10 support. Remove this eventually. function isSlowBuffer (obj) { return typeof obj.readFloatLE === 'function' && typeof obj.slice === 'function' && isBuffer(obj.slice(0, 0)) } var charenc = { // UTF-8 encoding utf8: { // Convert a string to a byte array stringToBytes: function(str) { return charenc.bin.stringToBytes(unescape(encodeURIComponent(str))); }, // Convert a byte array to a string bytesToString: function(bytes) { return decodeURIComponent(escape(charenc.bin.bytesToString(bytes))); } }, // Binary encoding bin: { // Convert a string to a byte array stringToBytes: function(str) { for (var bytes = [], i = 0; i < str.length; i++) { bytes.push(str.charCodeAt(i) & 0xFF); } return bytes; }, // Convert a byte array to a string bytesToString: function(bytes) { for (var str = [], i = 0; i < bytes.length; i++) { str.push(String.fromCharCode(bytes[i])); } return str.join(''); } } }; var charenc_1 = charenc; var md5 = createCommonjsModule(function (module) { (function(){ var crypt$$1 = crypt, utf8 = charenc_1.utf8, isBuffer = index, bin = charenc_1.bin, // The core md5 = function (message, options) { // Convert to byte array if (message.constructor == String) { if (options && options.encoding === 'binary') { message = bin.stringToBytes(message); } else { message = utf8.stringToBytes(message); } } else if (isBuffer(message)) { message = Array.prototype.slice.call(message, 0); } else if (!Array.isArray(message)) { message = message.toString(); } // else, assume byte array already var m = crypt$$1.bytesToWords(message), l = message.length * 8, a = 1732584193, b = -271733879, c = -1732584194, d = 271733878; // Swap endian for (var i = 0; i < m.length; i++) { m[i] = ((m[i] << 8) | (m[i] >>> 24)) & 0x00FF00FF | ((m[i] << 24) | (m[i] >>> 8)) & 0xFF00FF00; } // Padding m[l >>> 5] |= 0x80 << (l % 32); m[(((l + 64) >>> 9) << 4) + 14] = l; // Method shortcuts var FF = md5._ff, GG = md5._gg, HH = md5._hh, II = md5._ii; for (var i = 0; i < m.length; i += 16) { var aa = a, bb = b, cc = c, dd = d; a = FF(a, b, c, d, m[i+ 0], 7, -680876936); d = FF(d, a, b, c, m[i+ 1], 12, -389564586); c = FF(c, d, a, b, m[i+ 2], 17, 606105819); b = FF(b, c, d, a, m[i+ 3], 22, -1044525330); a = FF(a, b, c, d, m[i+ 4], 7, -176418897); d = FF(d, a, b, c, m[i+ 5], 12, 1200080426); c = FF(c, d, a, b, m[i+ 6], 17, -1473231341); b = FF(b, c, d, a, m[i+ 7], 22, -45705983); a = FF(a, b, c, d, m[i+ 8], 7, 1770035416); d = FF(d, a, b, c, m[i+ 9], 12, -1958414417); c = FF(c, d, a, b, m[i+10], 17, -42063); b = FF(b, c, d, a, m[i+11], 22, -1990404162); a = FF(a, b, c, d, m[i+12], 7, 1804603682); d = FF(d, a, b, c, m[i+13], 12, -40341101); c = FF(c, d, a, b, m[i+14], 17, -1502002290); b = FF(b, c, d, a, m[i+15], 22, 1236535329); a = GG(a, b, c, d, m[i+ 1], 5, -165796510); d = GG(d, a, b, c, m[i+ 6], 9, -1069501632); c = GG(c, d, a, b, m[i+11], 14, 643717713); b = GG(b, c, d, a, m[i+ 0], 20, -373897302); a = GG(a, b, c, d, m[i+ 5], 5, -701558691); d = GG(d, a, b, c, m[i+10], 9, 38016083); c = GG(c, d, a, b, m[i+15], 14, -660478335); b = GG(b, c, d, a, m[i+ 4], 20, -405537848); a = GG(a, b, c, d, m[i+ 9], 5, 568446438); d = GG(d, a, b, c, m[i+14], 9, -1019803690); c = GG(c, d, a, b, m[i+ 3], 14, -187363961); b = GG(b, c, d, a, m[i+ 8], 20, 1163531501); a = GG(a, b, c, d, m[i+13], 5, -1444681467); d = GG(d, a, b, c, m[i+ 2], 9, -51403784); c = GG(c, d, a, b, m[i+ 7], 14, 1735328473); b = GG(b, c, d, a, m[i+12], 20, -1926607734); a = HH(a, b, c, d, m[i+ 5], 4, -378558); d = HH(d, a, b, c, m[i+ 8], 11, -2022574463); c = HH(c, d, a, b, m[i+11], 16, 1839030562); b = HH(b, c, d, a, m[i+14], 23, -35309556); a = HH(a, b, c, d, m[i+ 1], 4, -1530992060); d = HH(d, a, b, c, m[i+ 4], 11, 1272893353); c = HH(c, d, a, b, m[i+ 7], 16, -155497632); b = HH(b, c, d, a, m[i+10], 23, -1094730640); a = HH(a, b, c, d, m[i+13], 4, 681279174); d = HH(d, a, b, c, m[i+ 0], 11, -358537222); c = HH(c, d, a, b, m[i+ 3], 16, -722521979); b = HH(b, c, d, a, m[i+ 6], 23, 76029189); a = HH(a, b, c, d, m[i+ 9], 4, -640364487); d = HH(d, a, b, c, m[i+12], 11, -421815835); c = HH(c, d, a, b, m[i+15], 16, 530742520); b = HH(b, c, d, a, m[i+ 2], 23, -995338651); a = II(a, b, c, d, m[i+ 0], 6, -198630844); d = II(d, a, b, c, m[i+ 7], 10, 1126891415); c = II(c, d, a, b, m[i+14], 15, -1416354905); b = II(b, c, d, a, m[i+ 5], 21, -57434055); a = II(a, b, c, d, m[i+12], 6, 1700485571); d = II(d, a, b, c, m[i+ 3], 10, -1894986606); c = II(c, d, a, b, m[i+10], 15, -1051523); b = II(b, c, d, a, m[i+ 1], 21, -2054922799); a = II(a, b, c, d, m[i+ 8], 6, 1873313359); d = II(d, a, b, c, m[i+15], 10, -30611744); c = II(c, d, a, b, m[i+ 6], 15, -1560198380); b = II(b, c, d, a, m[i+13], 21, 1309151649); a = II(a, b, c, d, m[i+ 4], 6, -145523070); d = II(d, a, b, c, m[i+11], 10, -1120210379); c = II(c, d, a, b, m[i+ 2], 15, 718787259); b = II(b, c, d, a, m[i+ 9], 21, -343485551); a = (a + aa) >>> 0; b = (b + bb) >>> 0; c = (c + cc) >>> 0; d = (d + dd) >>> 0; } return crypt$$1.endian([a, b, c, d]); }; // Auxiliary functions md5._ff = function (a, b, c, d, x, s, t) { var n = a + (b & c | ~b & d) + (x >>> 0) + t; return ((n << s) | (n >>> (32 - s))) + b; }; md5._gg = function (a, b, c, d, x, s, t) { var n = a + (b & d | c & ~d) + (x >>> 0) + t; return ((n << s) | (n >>> (32 - s))) + b; }; md5._hh = function (a, b, c, d, x, s, t) { var n = a + (b ^ c ^ d) + (x >>> 0) + t; return ((n << s) | (n >>> (32 - s))) + b; }; md5._ii = function (a, b, c, d, x, s, t) { var n = a + (c ^ (b | ~d)) + (x >>> 0) + t; return ((n << s) | (n >>> (32 - s))) + b; }; // Package private blocksize md5._blocksize = 16; md5._digestsize = 16; module.exports = function (message, options) { if (message === undefined || message === null) { throw new Error('Illegal argument ' + message); } var digestbytes = crypt$$1.wordsToBytes(md5(message, options)); return options && options.asBytes ? digestbytes : options && options.asString ? bin.bytesToString(digestbytes) : crypt$$1.bytesToHex(digestbytes); }; })(); }); function getSourceURL (rid, cdn, rate) { var API_KEY = 'A12Svb&%1UUmf@hC'; var tt = Math.round(new Date().getTime() / 60 / 1000); var did = md5(Math.random().toString()).toUpperCase(); var signContent = [rid, did, API_KEY, tt].join(''); var sign = md5(signContent); var body = { 'cdn': cdn, 'rate': rate, 'ver': '2016102501', 'tt': tt, 'did': did, 'sign': sign }; body = Object.keys(body).map(function (key) { return (key + "=" + (encodeURIComponent(body[key]))); }).join('&'); return fetch(("https://www.douyu.com/lapi/live/getPlay/" + rid), { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, body: body }) .then(function (res) { return res.json(); }) .then(function (videoInfo) { var baseUrl = videoInfo.data.rtmp_url; var livePath = videoInfo.data.rtmp_live; if (baseUrl && livePath) { var videoUrl = baseUrl + "/" + livePath; console.log('RoomId', rid, 'SourceURL:', videoUrl); return videoUrl } else { throw new Error('未开播或获取失败') } }) } var getSwfApi = function (rid) { var API_KEY = 'bLFlashflowlad92'; var tt = Math.round(new Date().getTime() / 60 / 1000); var signContent = [rid, API_KEY, tt].join(''); var sign = md5(signContent); return fetch(("http://www.douyutv.com/swf_api/room/" + rid + "?cdn=&nofan=yes&_t=" + tt + "&sign=" + sign)) .then(function (res) { return res.json(); }) .then(function (r) { return r.data; }) }; var DouyuSource = function DouyuSource (roomId) { this._cdn = 'ws'; this._rate = '0'; this.url = ''; this.roomId = roomId; this.swfApi = null; this.onChange = function () { return null; }; }; var prototypeAccessors$1 = { cdn: {},rate: {},cdnsWithName: {} }; prototypeAccessors$1.cdn.set = function (val) { this._cdn = val; this.getUrl(); }; prototypeAccessors$1.cdn.get = function () { return this._cdn }; prototypeAccessors$1.rate.set = function (val) { this._rate = val; this.getUrl(); }; prototypeAccessors$1.rate.get = function () { return this._rate }; prototypeAccessors$1.cdnsWithName.get = function () { if (this.swfApi) { return this.swfApi.cdnsWithName } else { return [{ name: '主要线路', cdn: 'ws' }] } }; DouyuSource.prototype.getUrl = function getUrl () { var this$1 = this; var chain = Promise.resolve(); if (!this.swfApi) { chain = chain.then(function () { return getSwfApi(this$1.roomId); }).then(function (swfApi) { this$1.swfApi = swfApi; this$1._cdn = swfApi.cdns[0]; }); } chain = chain.then(function () { return getSourceURL(this$1.roomId, this$1.cdn, this$1.rate); }) .then(function (url) { this$1.url = url; this$1.onChange(url); return url }); return chain }; Object.defineProperties( DouyuSource.prototype, prototypeAccessors$1 ); function utf8ToUtf16(utf8_bytes) { var unicode_codes = []; var unicode_code = 0; var num_followed = 0; for (var i = 0; i < utf8_bytes.length; ++i) { var utf8_byte = utf8_bytes[i]; if (utf8_byte >= 0x100) { // Malformed utf8 byte ignored. } else if ((utf8_byte & 0xC0) == 0x80) { if (num_followed > 0) { unicode_code = (unicode_code << 6) | (utf8_byte & 0x3f); num_followed -= 1; } else { // Malformed UTF-8 sequence ignored. } } else { if (num_followed == 0) { unicode_codes.push(unicode_code); } else { // Malformed UTF-8 sequence ignored. } if (utf8_byte < 0x80){ // 1-byte unicode_code = utf8_byte; num_followed = 0; } else if ((utf8_byte & 0xE0) == 0xC0) { // 2-byte unicode_code = utf8_byte & 0x1f; num_followed = 1; } else if ((utf8_byte & 0xF0) == 0xE0) { // 3-byte unicode_code = utf8_byte & 0x0f; num_followed = 2; } else if ((utf8_byte & 0xF8) == 0xF0) { // 4-byte unicode_code = utf8_byte & 0x07; num_followed = 3; } else { // Malformed UTF-8 sequence ignored. } } } if (num_followed == 0) { unicode_codes.push(unicode_code); } else { // Malformed UTF-8 sequence ignored. } unicode_codes.shift(); // Trim the first element. var utf16_codes = []; for (var i = 0; i < unicode_codes.length; ++i) { var unicode_code = unicode_codes[i]; if (unicode_code < (1 << 16)) { utf16_codes.push(unicode_code); } else { var first = ((unicode_code - (1 << 16)) / (1 << 10)) + 0xD800; var second = (unicode_code % (1 << 10)) + 0xDC00; utf16_codes.push(first); utf16_codes.push(second); } } return utf16_codes; } function utf8_to_ascii( str ) { // return unescape(encodeURIComponent(str)) var char2bytes = function (unicode_code) { var utf8_bytes = []; if (unicode_code < 0x80) { // 1-byte utf8_bytes.push(unicode_code); } else if (unicode_code < (1 << 11)) { // 2-byte utf8_bytes.push((unicode_code >>> 6) | 0xC0); utf8_bytes.push((unicode_code & 0x3F) | 0x80); } else if (unicode_code < (1 << 16)) { // 3-byte utf8_bytes.push((unicode_code >>> 12) | 0xE0); utf8_bytes.push(((unicode_code >> 6) & 0x3f) | 0x80); utf8_bytes.push((unicode_code & 0x3F) | 0x80); } else if (unicode_code < (1 << 21)) { // 4-byte utf8_bytes.push((unicode_code >>> 18) | 0xF0); utf8_bytes.push(((unicode_code >> 12) & 0x3F) | 0x80); utf8_bytes.push(((unicode_code >> 6) & 0x3F) | 0x80); utf8_bytes.push((unicode_code & 0x3F) | 0x80); } return utf8_bytes; }; var o = []; for (var i = 0; i < str.length; i++) { o = o.concat(char2bytes(str.charCodeAt(i))); } return o.map(function (i) { return String.fromCharCode(i); }).join('') } function ascii_to_utf8( str ) { // return decodeURIComponent(escape(str)) var bytes = str.split('').map(function (i) { return i.charCodeAt(0); }); return utf8ToUtf16(bytes).map(function (i) { return String.fromCharCode(i); }).join('') } /* * Jsocket - Socket on Javascript * Author: Masahiro Chiba * Depends: * - jQuery: http://jquery.com/ * - jQuery TOOLS - Flashembed: http://flowplayer.org/tools/flashembed.html * SYNOPSIS: * JSocket.init('/static/JSocket.swf', function () { * socket = new JSocket({ * connectHandler: connectHandler, * dataHandler: dataHandler, * closeHandler: closeHandler, * errorHandler: errorHandler * }); * socket.connect(location.hostname, location.port || 80); * }); * function connectHandler() { * socket.writeFlush("GET / HTTP/1.0\x0D\x0A"); * socket.write("Host: " + location.hostname + "\x0D\x0A\x0D\x0A"); * socket.flush(); * } * function dataHandler(data) { * alert(data); * socket.close(); * } * function closeHandler() { * alert('lost connection') * } * function errorHandler(errorstr) { * alert(errorstr); * } * * */ function JSocket() { this.initialize.apply(this, arguments); } JSocket.VERSION = '0.05'; JSocket.init = function(src, swfloadedcb) { var flash = ['', '', '', '', '', '', '', '', '', ""].join(""); var div = document.createElement('div'); div.style.width = '1px'; div.style.height = '1px'; document.body.appendChild(div); div.innerHTML = flash; var api = document.querySelector('#jsocket'); console.log(div, api); JSocket.flashapi = api; if ( JSocket.flashapi.newsocket ) { // for IE(because already construct) swfloadedcb(); } else { JSocket.swfloadedcb = swfloadedcb; } }; JSocket.swfloaded = function() { if ( JSocket.swfloadedcb ) { setTimeout(JSocket.swfloadedcb, 0); } }; JSocket.handlers = new Array(); JSocket.defaultHandlers = { connectHandler: function () {}, dataHandler: function () {}, closeHandler: function () {}, errorHandler: function () {} }; JSocket.connectHandler = function(socid) { JSocket.handlers[socid].connectHandler(); }; JSocket.dataHandler = function(socid, data) { // setTimeout(() => JSocket.handlers[socid].dataHandler(atob(data)), 0) try { JSocket.handlers[socid].dataHandler(atob(data)); } catch (e) { console.error(e); } }; JSocket.closeHandler = function(socid) { JSocket.handlers[socid].closeHandler(); }; JSocket.errorHandler = function(socid, str) { JSocket.handlers[socid].errorHandler(str); }; JSocket.prototype = { initialize: function(handlers, newsocketopt) { this.socid = JSocket.flashapi.newsocket(newsocketopt); JSocket.handlers[this.socid] = handlers; }, connect: function(host, port) { JSocket.flashapi.connect(this.socid, host, port); }, write: function(data) { JSocket.flashapi.write(this.socid, btoa(data)); }, writeFlush: function(data) { JSocket.flashapi.writeFlush(this.socid, btoa(data)); }, close: function() { JSocket.flashapi.close(this.socid); }, flush: function() { JSocket.flashapi.flush(this.socid); } }; window.JSocket = JSocket; var p32 = function (i) { return [i, i / 256, i / 65536, i / 16777216].map(function (i) { return String.fromCharCode(Math.floor(i) % 256); }).join(''); }; var u32 = function (s) { return s.split('').map(function (i) { return i.charCodeAt(0); }).reduce(function (a, b) { return b * 256 + a; }); }; function ACJ (id, data) { if (typeof data == 'object') { data = douyuClient.douyuEncode(data); } try { _ACJ_([id, data]); } catch (e) { console.error(id, data, e); } } var getACF = function (key) { try { return new RegExp(("acf_" + key + "=(.*?);")).exec(document.cookie)[1] } catch (e) { return '' } }; var douyuClient = function douyuClient () { var this$1 = this; this.map = {}; this.buffer = ''; this.socket = new JSocket({ connectHandler: function () { return this$1.connectHandler(); }, dataHandler: function (data) { return this$1.dataHandler(data); }, closeHandler: function () { return this$1.closeHandler(); }, errorHandler: function () { return this$1.errorHandler(); } }); // let oldSend = this.send // this.send = data => oldSend(data) }; douyuClient.prototype.connectHandler = function connectHandler () {}; douyuClient.prototype.dataHandler = function dataHandler (data) { var this$1 = this; this.buffer += data; var buffer = this.buffer; var map = this.map; while (buffer.length >= 4) { var size = u32(buffer.substr(0, 4)); if (buffer.length >= size) { var pkg = ''; try { pkg = ascii_to_utf8(buffer.substr(12, size-8)); } catch (e) { console.log('deocde fail', escape(buffer.substr(12, size-8))); } this$1.buffer = buffer = buffer.substr(size+4); if (pkg.length === 0) { continue } try { var rawString = pkg; var caller = function (func) { return func(pkg, function (data) { return this$1.send(data); }, { ACJ: ACJ, rawString: rawString, decode: douyuClient.douyuDecode, encode: douyuClient.douyuEncode }); }; pkg = douyuClient.douyuDecode(pkg); if (map) { var cb = map[pkg.type]; if (cb) { if (typeof cb == 'string') { ACJ(cb, pkg); } else { caller(map[pkg.type]); } } else { map.default && caller(map.default); } } } catch (e) { console.error('call map', e); } } else { break } } }; douyuClient.prototype.connect = function connect (ip, port) { this.socket.connect(ip, port); }; douyuClient.filterEnc = function filterEnc (s) { s = s.toString(); s = s.replace(/@/g, '@A'); return s.replace(/\//g, '@S') }; douyuClient.filterDec = function filterDec (s) { s = s.toString(); s = s.replace(/@S/g, '/'); return s.replace(/@A/g, '@') }; douyuClient.douyuEncode = function douyuEncode (data) { return Object.keys(data).map(function (key) { return (key + "@=" + (douyuClient.filterEnc(data[key]))); }).join('/') + '/' }; douyuClient.douyuDecode = function douyuDecode (data) { var out = {}; data.split('/').filter(function (i) { return i.length > 2; }).some(function (i) { var e = i.split('@='); out[e[0]] = douyuClient.filterDec(e[1]); }); return out }; douyuClient.encode = function encode (data) { return douyuClient.douyuEncode(data) }; douyuClient.decode = function decode (data) { return douyuClient.douyuDecode(data) }; douyuClient.decodeList = function decodeList (list) { return list = list.split('/').filter(function (i) { return i.length > 2; }).map(douyuClient.filterDec).map(douyuClient.douyuDecode) }; douyuClient.prototype.closeHandler = function closeHandler () { console.error('lost connection'); }; douyuClient.prototype.errorHandler = function errorHandler (errorstr) { console.error(errorstr); }; douyuClient.prototype.send = function send (data) { var msg = douyuClient.douyuEncode(data) + '\0'; msg = utf8_to_ascii(msg); msg = p32(msg.length+8) + p32(msg.length+8) + p32(689) + msg; this.socket.writeFlush(msg); }; //rlcn //import './start' var onload = function () { function getURL (src) { if (src.substr(0, 5) !== 'blob:') { src = chrome.runtime.getURL(src); } return src } function addScript (src) { var script = document.createElement('script'); // blob: script.src = getURL(src); document.head.appendChild(script); } function addCss (src, rel, type) { var link = document.createElement('link'); link.rel = rel || 'stylesheet'; link.type = type || 'text/css'; link.href = getURL(src); document.head.appendChild(link); } function createBlobURL (content, type) { var blob = new Blob([content], { type: type }); return URL.createObjectURL(blob) } // addCss('src/danmu.less', 'stylesheet/less', 'text/css') if (window.__space_inject) { var ref = window.__space_inject; var script = ref.script; var css = ref.css; addCss(createBlobURL(css, 'text/css')); addScript(createBlobURL(script, 'text/javascript')); } else { addCss('dist/danmu.css'); addScript('dist/douyuInject.js'); } // addScript('libs/less.min.js') var uid = getACF('uid'); flvjs.LoggingControl.forceGlobalTag = true; flvjs.LoggingControl.enableAll = true; var makeMenu = function (player, source) { var cdnMenu = function () { return source.cdnsWithName.map(function (i) { var suffix = ''; if (i.cdn == source.cdn) { suffix = ' √'; } return { text: i.name + suffix, cb: function cb () { source.cdn = i.cdn; } } }); }; var rateMenu = function () { var rates = [{ text: '超清', rate: '0' }, { text: '高清', rate: '2' }, { text: '普清', rate: '1' }]; return rates.map(function (i) { var suffix = ''; if (i.rate == source.rate) { suffix = ' √'; } return { text: i.text + suffix, cb: function cb () { source.rate = i.rate; } } }) }; var transparentMenu = function () { var opts = [{ text: '0%', transparent: 0 }, { text: '25%', transparent: 25 }, { text: '50%', transparent: 50 }]; return [{ label: '弹幕透明度:' }].concat(opts.map(function (i) { var suffix = ''; if (i.transparent == player.transparent) { suffix = ' √'; } return { text: i.text + suffix, cb: function cb () { player.transparent = i.transparent; } } })) }; var dash = {}; bindMenu(player.video, function () { return [].concat(cdnMenu(), dash, rateMenu(), dash, transparentMenu()); }); }; var loadVideo = function (roomId, replace) { var source = new DouyuSource(roomId); var danmuPlayer = new DanmuPlayer(new DanmuPlayerControls({ onReload: function () { return source.getUrl(); }, onSendDanmu: function onSendDanmu (txt) { window.postMessage({ type: "SENDANMU", data: txt }, "*"); } }), function (pkg) { return pkg.uid == uid; }); source.onChange = function (videoUrl) { danmuPlayer.src = videoUrl; }; danmuPlayer.parsePic = function (s) { return s.replace( /\[emot:dy(.*?)\]/g, function (_, i) { return (""); }// `
` ); }; // let roomVideo = document.querySelector('#js-room-video') // if (!roomVideo) { // roomVideo = document.querySelector('.live_site_player_container') // } // roomVideo.removeChild(roomVideo.children[0]) // roomVideo.insertBefore(danmuPlayer.el, roomVideo.children[0]) replace(danmuPlayer.el); makeMenu(danmuPlayer, source); window.danmu = danmuPlayer; return source.getUrl().then(function () { return danmuPlayer; }) }; var danmuPlayer = null; window.addEventListener('message', function (event) { if (event.source != window) { return } if (event.data.type) { var data = event.data.data; switch (event.data.type) { case 'DANMU': danmuPlayer && danmuPlayer.onDanmu(data); break case 'VIDEOID': // getRoomId() console.log('onVideoId', data); var ctr = document.querySelector(("#" + (data.id))); loadVideo(data.roomId, function (el) { ctr.parentNode.replaceChild(el, ctr); }).then(function (dp) { danmuPlayer = dp; }); break } } }, false); }; //document.addEventListener('DOMContentLoaded', onload) onload(); })));