// ==UserScript== // @description 基于 flv.js 的斗鱼HTML5播放器. // @icon https://ojiju7xvu.qnssl.com/d5hp/icon.png // @name 斗鱼HTML5播放器 // @require https://cdn.bootcss.com/flv.js/1.3.3/flv.min.js // @namespace http://imspace.cn/gms // @run-at document_end // @version 0.7.4 // @grant GM_xmlhttpRequest // @match *://*.douyu.com/* // @downloadURL https://update.greasyfork.icu/scripts/26901/%E6%96%97%E9%B1%BCHTML5%E6%92%AD%E6%94%BE%E5%99%A8.user.js // @updateURL https://update.greasyfork.icu/scripts/26901/%E6%96%97%E9%B1%BCHTML5%E6%92%AD%E6%94%BE%E5%99%A8.meta.js // ==/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 conf.responseType = 'arraybuffer' return new Promise((resolve, reject) => { conf.onload = (response) => { if (response.status === 200) { resolve({ json () { const enc = new TextDecoder('utf-8') return Promise.resolve(JSON.parse(enc.decode(new Uint8Array(response.response)))) }, arrayBuffer () { return Promise.resolve(response.response) } }) } 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"+ " Copyright (c) Microsoft Corporation. All rights reserved.\r\n"+ " Licensed under the Apache License, Version 2.0 (the \"License\"); you may not use\r\n"+ " this file except in compliance with the License. You may obtain a copy of the\r\n"+ " License at http://www.apache.org/licenses/LICENSE-2.0\r\n"+ "\r\n"+ " THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY\r\n"+ " KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED\r\n"+ " WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE,\r\n"+ " MERCHANTABLITY OR NON-INFRINGEMENT.\r\n"+ "\r\n"+ " See the Apache Version 2.0 License for specific language governing permissions\r\n"+ " and limitations under the License.\r\n"+ " ***************************************************************************** */\r\n"+ " /* global Reflect, Promise */\r\n"+ "\r\n"+ " var extendStatics = Object.setPrototypeOf ||\r\n"+ " ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n"+ " function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " function __decorate(decorators, target, key, desc) {\r\n"+ " var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n"+ " if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n"+ " else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n"+ " return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " function __awaiter(thisArg, _arguments, P, generator) {\r\n"+ " return new (P || (P = Promise))(function (resolve, reject) {\r\n"+ " function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n"+ " function rejected(value) { try { step(generator.throw(value)); } catch (e) { reject(e); } }\r\n"+ " function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\r\n"+ " step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n"+ " });\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " function __read(o, n) {\r\n"+ " var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n"+ " if (!m) return o;\r\n"+ " var i = m.call(o), r, ar = [], e;\r\n"+ " try {\r\n"+ " while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n"+ " }\r\n"+ " catch (error) { e = { error: error }; }\r\n"+ " finally {\r\n"+ " try {\r\n"+ " if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n"+ " }\r\n"+ " finally { if (e) throw e.error; }\r\n"+ " }\r\n"+ " return ar;\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " function __await(v) {\r\n"+ " return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n"+ " }\n"+ "\n"+ " class JSocket {\r\n"+ " static init() {\r\n"+ " return __awaiter(this, void 0, void 0, function* () {\r\n"+ " const src = 'https://imspace.nos-eastchina1.126.net/JSocket2.swf';\r\n"+ " const flash = ['', '', '', '', '', '', '', '', '', \"\"].join(\"\");\r\n"+ " let div = document.createElement('div');\r\n"+ " div.className = 'jsocket-cls';\r\n"+ " document.body.appendChild(div);\r\n"+ " JSocket.el = 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"+ " if (JSocket.flashapi.newsocket) {\r\n"+ " return;\r\n"+ " }\r\n"+ " else {\r\n"+ " return new Promise((res, rej) => {\r\n"+ " const id = setTimeout(rej, 10 * 1000);\r\n"+ " JSocket.swfloadedcb = () => {\r\n"+ " clearTimeout(id);\r\n"+ " res();\r\n"+ " };\r\n"+ " });\r\n"+ " }\r\n"+ " });\r\n"+ " }\r\n"+ " static swfloaded() {\r\n"+ " JSocket.swfloadedcb();\r\n"+ " }\r\n"+ " static connectHandler(socid) {\r\n"+ " JSocket.handlers[socid].connectHandler();\r\n"+ " }\r\n"+ " static dataHandler(socid, data) {\r\n"+ " try {\r\n"+ " JSocket.handlers[socid].dataHandler(atob(data));\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " console.error(e);\r\n"+ " }\r\n"+ " }\r\n"+ " static closeHandler(socid) {\r\n"+ " JSocket.handlers[socid].closeHandler();\r\n"+ " }\r\n"+ " static errorHandler(socid, str) {\r\n"+ " JSocket.handlers[socid].errorHandler(str);\r\n"+ " }\r\n"+ " init(handlers, newsocketopt) {\r\n"+ " this.socid = JSocket.flashapi.newsocket(newsocketopt);\r\n"+ " JSocket.handlers[this.socid] = handlers;\r\n"+ " }\r\n"+ " connect(host, port) {\r\n"+ " JSocket.flashapi.connect(this.socid, host, port);\r\n"+ " }\r\n"+ " write(data) {\r\n"+ " JSocket.flashapi.write(this.socid, btoa(data));\r\n"+ " }\r\n"+ " writeFlush(data) {\r\n"+ " JSocket.flashapi.writeFlush(this.socid, btoa(data));\r\n"+ " }\r\n"+ " close() {\r\n"+ " JSocket.flashapi.close(this.socid);\r\n"+ " }\r\n"+ " flush() {\r\n"+ " JSocket.flashapi.flush(this.socid);\r\n"+ " }\r\n"+ " }\r\n"+ " JSocket.VERSION = '0.1';\r\n"+ " JSocket.handlers = [];\r\n"+ " window.JSocket = JSocket;\n"+ "\n"+ " function utf8ToUtf16(utf8_bytes) {\r\n"+ " let unicode_codes = [];\r\n"+ " let unicode_code = 0;\r\n"+ " let num_followed = 0;\r\n"+ " for (let i = 0; i < utf8_bytes.length; ++i) {\r\n"+ " let utf8_byte = utf8_bytes[i];\r\n"+ " if (utf8_byte >= 0x100) {\r\n"+ " }\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"+ " }\r\n"+ " else {\r\n"+ " }\r\n"+ " }\r\n"+ " else {\r\n"+ " if (num_followed == 0) {\r\n"+ " unicode_codes.push(unicode_code);\r\n"+ " }\r\n"+ " else {\r\n"+ " }\r\n"+ " if (utf8_byte < 0x80) {\r\n"+ " unicode_code = utf8_byte;\r\n"+ " num_followed = 0;\r\n"+ " }\r\n"+ " else if ((utf8_byte & 0xE0) == 0xC0) {\r\n"+ " unicode_code = utf8_byte & 0x1f;\r\n"+ " num_followed = 1;\r\n"+ " }\r\n"+ " else if ((utf8_byte & 0xF0) == 0xE0) {\r\n"+ " unicode_code = utf8_byte & 0x0f;\r\n"+ " num_followed = 2;\r\n"+ " }\r\n"+ " else if ((utf8_byte & 0xF8) == 0xF0) {\r\n"+ " unicode_code = utf8_byte & 0x07;\r\n"+ " num_followed = 3;\r\n"+ " }\r\n"+ " else {\r\n"+ " }\r\n"+ " }\r\n"+ " }\r\n"+ " if (num_followed == 0) {\r\n"+ " unicode_codes.push(unicode_code);\r\n"+ " }\r\n"+ " else {\r\n"+ " }\r\n"+ " unicode_codes.shift();\r\n"+ " let utf16_codes = [];\r\n"+ " for (var i = 0; i < unicode_codes.length; ++i) {\r\n"+ " unicode_code = unicode_codes[i];\r\n"+ " if (unicode_code < (1 << 16)) {\r\n"+ " utf16_codes.push(unicode_code);\r\n"+ " }\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"+ " function utf8_to_ascii(str) {\r\n"+ " const char2bytes = (unicode_code) => {\r\n"+ " let utf8_bytes = [];\r\n"+ " if (unicode_code < 0x80) {\r\n"+ " utf8_bytes.push(unicode_code);\r\n"+ " }\r\n"+ " else if (unicode_code < (1 << 11)) {\r\n"+ " utf8_bytes.push((unicode_code >>> 6) | 0xC0);\r\n"+ " utf8_bytes.push((unicode_code & 0x3F) | 0x80);\r\n"+ " }\r\n"+ " else if (unicode_code < (1 << 16)) {\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"+ " }\r\n"+ " else if (unicode_code < (1 << 21)) {\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"+ " let o = [];\r\n"+ " for (let i = 0; i < str.length; i++) {\r\n"+ " o = o.concat(char2bytes(str.charCodeAt(i)));\r\n"+ " }\r\n"+ " return o.map(i => String.fromCharCode(i)).join('');\r\n"+ " }\r\n"+ " function ascii_to_utf8(str) {\r\n"+ " let bytes = str.split('').map(i => i.charCodeAt(0));\r\n"+ " return utf8ToUtf16(bytes).map(i => String.fromCharCode(i)).join('');\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ " function getURL(src) {\r\n"+ " if (src.substr(0, 5) !== 'blob:') {\r\n"+ " src = chrome.runtime.getURL(src);\r\n"+ " }\r\n"+ " return src;\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " const p32 = (i) => [i, i / 256, i / 65536, i / 16777216].map(i => String.fromCharCode(Math.floor(i) % 256)).join('');\r\n"+ " const u32 = (s) => s.split('').map(i => i.charCodeAt(0)).reduce((a, b) => b * 256 + a);\r\n"+ " let messageMap = {};\r\n"+ " function onMessage(type, cb) {\r\n"+ " messageMap[type] = cb;\r\n"+ " }\r\n"+ " function postMessage(type, data) {\r\n"+ " window.postMessage({\r\n"+ " type: type,\r\n"+ " data: data\r\n"+ " }, \"*\");\r\n"+ " }\r\n"+ " let msgCallbacks = [];\r\n"+ " let lastCbId = 0;\r\n"+ "\r\n"+ " window.addEventListener('message', (event) => __awaiter(window, void 0, void 0, function* () {\r\n"+ " const data = event.data;\r\n"+ " if (data.cb) {\r\n"+ " let cb = msgCallbacks[data.cbId];\r\n"+ " if (cb && (typeof cb === 'function')) {\r\n"+ " cb(data.cbResult);\r\n"+ " }\r\n"+ " }\r\n"+ " else if (data.type) {\r\n"+ " let result = undefined;\r\n"+ " if (typeof messageMap[data.type] === 'function') {\r\n"+ " result = messageMap[data.type](data.data);\r\n"+ " if (result instanceof Promise) {\r\n"+ " result = yield result;\r\n"+ " }\r\n"+ " if (data.cbId) {\r\n"+ " window.postMessage({\r\n"+ " cb: true,\r\n"+ " cbId: data.cbId,\r\n"+ " cbResult: result\r\n"+ " }, '*');\r\n"+ " }\r\n"+ " }\r\n"+ " }\r\n"+ " }), false);\r\n"+ " function retry(promise, times) {\r\n"+ " return __awaiter(this, void 0, void 0, function* () {\r\n"+ " let err = [];\r\n"+ " for (let i = 0; i < times; i++) {\r\n"+ " try {\r\n"+ " return yield promise();\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " err.push(e);\r\n"+ " }\r\n"+ " }\r\n"+ " throw err;\r\n"+ " });\r\n"+ " }\r\n"+ " function getSync() {\r\n"+ " return new Promise((res, rej) => {\r\n"+ " if (chrome && chrome.storage && chrome.storage.sync) {\r\n"+ " chrome.storage.sync.get(items => {\r\n"+ " res(items);\r\n"+ " });\r\n"+ " }\r\n"+ " else {\r\n"+ " rej(new Error('不支持的存储方式'));\r\n"+ " }\r\n"+ " });\r\n"+ " }\r\n"+ "\r\n"+ "\r\n"+ "\r\n"+ " const defaultBgListener = (request) => __awaiter(window, void 0, void 0, function* () { return null; });\r\n"+ " let bgListener = defaultBgListener;\n"+ "\n"+ " function md5cycle(x, k) {\r\n"+ " var a = x[0], b = x[1], c = x[2], d = x[3];\r\n"+ " a = ff(a, b, c, d, k[0], 7, -680876936);\r\n"+ " d = ff(d, a, b, c, k[1], 12, -389564586);\r\n"+ " c = ff(c, d, a, b, k[2], 17, 606105819);\r\n"+ " b = ff(b, c, d, a, k[3], 22, -1044525330);\r\n"+ " a = ff(a, b, c, d, k[4], 7, -176418897);\r\n"+ " d = ff(d, a, b, c, k[5], 12, 1200080426);\r\n"+ " c = ff(c, d, a, b, k[6], 17, -1473231341);\r\n"+ " b = ff(b, c, d, a, k[7], 22, -45705983);\r\n"+ " a = ff(a, b, c, d, k[8], 7, 1770035416);\r\n"+ " d = ff(d, a, b, c, k[9], 12, -1958414417);\r\n"+ " c = ff(c, d, a, b, k[10], 17, -42063);\r\n"+ " b = ff(b, c, d, a, k[11], 22, -1990404162);\r\n"+ " a = ff(a, b, c, d, k[12], 7, 1804603682);\r\n"+ " d = ff(d, a, b, c, k[13], 12, -40341101);\r\n"+ " c = ff(c, d, a, b, k[14], 17, -1502002290);\r\n"+ " b = ff(b, c, d, a, k[15], 22, 1236535329);\r\n"+ " a = gg(a, b, c, d, k[1], 5, -165796510);\r\n"+ " d = gg(d, a, b, c, k[6], 9, -1069501632);\r\n"+ " c = gg(c, d, a, b, k[11], 14, 643717713);\r\n"+ " b = gg(b, c, d, a, k[0], 20, -373897302);\r\n"+ " a = gg(a, b, c, d, k[5], 5, -701558691);\r\n"+ " d = gg(d, a, b, c, k[10], 9, 38016083);\r\n"+ " c = gg(c, d, a, b, k[15], 14, -660478335);\r\n"+ " b = gg(b, c, d, a, k[4], 20, -405537848);\r\n"+ " a = gg(a, b, c, d, k[9], 5, 568446438);\r\n"+ " d = gg(d, a, b, c, k[14], 9, -1019803690);\r\n"+ " c = gg(c, d, a, b, k[3], 14, -187363961);\r\n"+ " b = gg(b, c, d, a, k[8], 20, 1163531501);\r\n"+ " a = gg(a, b, c, d, k[13], 5, -1444681467);\r\n"+ " d = gg(d, a, b, c, k[2], 9, -51403784);\r\n"+ " c = gg(c, d, a, b, k[7], 14, 1735328473);\r\n"+ " b = gg(b, c, d, a, k[12], 20, -1926607734);\r\n"+ " a = hh(a, b, c, d, k[5], 4, -378558);\r\n"+ " d = hh(d, a, b, c, k[8], 11, -2022574463);\r\n"+ " c = hh(c, d, a, b, k[11], 16, 1839030562);\r\n"+ " b = hh(b, c, d, a, k[14], 23, -35309556);\r\n"+ " a = hh(a, b, c, d, k[1], 4, -1530992060);\r\n"+ " d = hh(d, a, b, c, k[4], 11, 1272893353);\r\n"+ " c = hh(c, d, a, b, k[7], 16, -155497632);\r\n"+ " b = hh(b, c, d, a, k[10], 23, -1094730640);\r\n"+ " a = hh(a, b, c, d, k[13], 4, 681279174);\r\n"+ " d = hh(d, a, b, c, k[0], 11, -358537222);\r\n"+ " c = hh(c, d, a, b, k[3], 16, -722521979);\r\n"+ " b = hh(b, c, d, a, k[6], 23, 76029189);\r\n"+ " a = hh(a, b, c, d, k[9], 4, -640364487);\r\n"+ " d = hh(d, a, b, c, k[12], 11, -421815835);\r\n"+ " c = hh(c, d, a, b, k[15], 16, 530742520);\r\n"+ " b = hh(b, c, d, a, k[2], 23, -995338651);\r\n"+ " a = ii(a, b, c, d, k[0], 6, -198630844);\r\n"+ " d = ii(d, a, b, c, k[7], 10, 1126891415);\r\n"+ " c = ii(c, d, a, b, k[14], 15, -1416354905);\r\n"+ " b = ii(b, c, d, a, k[5], 21, -57434055);\r\n"+ " a = ii(a, b, c, d, k[12], 6, 1700485571);\r\n"+ " d = ii(d, a, b, c, k[3], 10, -1894986606);\r\n"+ " c = ii(c, d, a, b, k[10], 15, -1051523);\r\n"+ " b = ii(b, c, d, a, k[1], 21, -2054922799);\r\n"+ " a = ii(a, b, c, d, k[8], 6, 1873313359);\r\n"+ " d = ii(d, a, b, c, k[15], 10, -30611744);\r\n"+ " c = ii(c, d, a, b, k[6], 15, -1560198380);\r\n"+ " b = ii(b, c, d, a, k[13], 21, 1309151649);\r\n"+ " a = ii(a, b, c, d, k[4], 6, -145523070);\r\n"+ " d = ii(d, a, b, c, k[11], 10, -1120210379);\r\n"+ " c = ii(c, d, a, b, k[2], 15, 718787259);\r\n"+ " b = ii(b, c, d, a, k[9], 21, -343485551);\r\n"+ " x[0] = add32(a, x[0]);\r\n"+ " x[1] = add32(b, x[1]);\r\n"+ " x[2] = add32(c, x[2]);\r\n"+ " x[3] = add32(d, x[3]);\r\n"+ " }\r\n"+ " function cmn(q, a, b, x, s, t) {\r\n"+ " a = add32(add32(a, q), add32(x, t));\r\n"+ " return add32((a << s) | (a >>> (32 - s)), b);\r\n"+ " }\r\n"+ " function ff(a, b, c, d, x, s, t) {\r\n"+ " return cmn((b & c) | ((~b) & d), a, b, x, s, t);\r\n"+ " }\r\n"+ " function gg(a, b, c, d, x, s, t) {\r\n"+ " return cmn((b & d) | (c & (~d)), a, b, x, s, t);\r\n"+ " }\r\n"+ " function hh(a, b, c, d, x, s, t) {\r\n"+ " return cmn(b ^ c ^ d, a, b, x, s, t);\r\n"+ " }\r\n"+ " function ii(a, b, c, d, x, s, t) {\r\n"+ " return cmn(c ^ (b | (~d)), a, b, x, s, t);\r\n"+ " }\r\n"+ " function md51(s) {\r\n"+ " var txt = '';\r\n"+ " var n = s.length, state = [1732584193, -271733879, -1732584194, 271733878], i;\r\n"+ " for (i = 64; i <= s.length; i += 64) {\r\n"+ " md5cycle(state, md5blk(s.substring(i - 64, i)));\r\n"+ " }\r\n"+ " s = s.substring(i - 64);\r\n"+ " var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];\r\n"+ " for (i = 0; i < s.length; i++)\r\n"+ " tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3);\r\n"+ " tail[i >> 2] |= 0x80 << ((i % 4) << 3);\r\n"+ " if (i > 55) {\r\n"+ " md5cycle(state, tail);\r\n"+ " for (i = 0; i < 16; i++)\r\n"+ " tail[i] = 0;\r\n"+ " }\r\n"+ " tail[14] = n * 8;\r\n"+ " md5cycle(state, tail);\r\n"+ " return state;\r\n"+ " }\r\n"+ " function md5blk(s) {\r\n"+ " var md5blks = [], i;\r\n"+ " for (i = 0; i < 64; i += 4) {\r\n"+ " md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) << 24);\r\n"+ " }\r\n"+ " return md5blks;\r\n"+ " }\r\n"+ " var hex_chr = '0123456789abcdef'.split('');\r\n"+ " function rhex(n) {\r\n"+ " var s = '', j = 0;\r\n"+ " for (; j < 4; j++)\r\n"+ " s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F];\r\n"+ " return s;\r\n"+ " }\r\n"+ " function hex(x) {\r\n"+ " return x.map(rhex).join('');\r\n"+ " }\r\n"+ " function md5(s) {\r\n"+ " return hex(md51(s));\r\n"+ " }\r\n"+ " var add32 = function (a, b) {\r\n"+ " return (a + b) & 0xFFFFFFFF;\r\n"+ " };\r\n"+ " if (md5('hello') != '5d41402abc4b2a76b9719d911017c592') {\r\n"+ " add32 = function (x, y) {\r\n"+ " var lsw = (x & 0xFFFF) + (y & 0xFFFF), msw = (x >> 16) + (y >> 16) + (lsw >> 16);\r\n"+ " return (msw << 16) | (lsw & 0xFFFF);\r\n"+ " };\r\n"+ " }\n"+ "\n"+ " const getACF = (key) => {\r\n"+ " try {\r\n"+ " return new RegExp(`acf_${key}=(.*?);`).exec(document.cookie)[1];\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " return '';\r\n"+ " }\r\n"+ " };\r\n"+ " 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"+ " 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"+ " function douyuEncode(data) {\r\n"+ " return Object.keys(data).map(key => `${key}@=${filterEnc(data[key])}`).join('/') + '/';\r\n"+ " }\r\n"+ " function douyuDecode(data) {\r\n"+ " let out = {\r\n"+ " type: '!!missing!!'\r\n"+ " };\r\n"+ " try {\r\n"+ " data.split('/').filter(i => i.length > 2).forEach(i => {\r\n"+ " let e = i.split('@=');\r\n"+ " out[e[0]] = filterDec(e[1]);\r\n"+ " });\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " console.error(e);\r\n"+ " console.log(data);\r\n"+ " }\r\n"+ " return out;\r\n"+ " }\r\n"+ " function ACJ(id, data) {\r\n"+ " if (typeof data == 'object') {\r\n"+ " data = douyuEncode(data);\r\n"+ " }\r\n"+ " try {\r\n"+ " window._ACJ_([id, data]);\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " console.error(id, data, e);\r\n"+ " }\r\n"+ " }\r\n"+ " class DouyuProtocol extends JSocket {\r\n"+ " constructor(listener) {\r\n"+ " super();\r\n"+ " this.listener = listener;\r\n"+ " this.connectHandler = () => null;\r\n"+ " this.init(this, {});\r\n"+ " this.buffer = '';\r\n"+ " }\r\n"+ " connectAsync(host, port) {\r\n"+ " super.connect(host, port);\r\n"+ " return new Promise((res, rej) => {\r\n"+ " const prevConnHandler = this.connectHandler;\r\n"+ " const prevErrHandler = this.errorHandler;\r\n"+ " const recover = () => {\r\n"+ " this.connectHandler = prevConnHandler;\r\n"+ " this.errorHandler = prevErrHandler;\r\n"+ " };\r\n"+ " this.connectHandler = () => {\r\n"+ " recover();\r\n"+ " res();\r\n"+ " };\r\n"+ " this.errorHandler = () => {\r\n"+ " recover();\r\n"+ " rej();\r\n"+ " };\r\n"+ " });\r\n"+ " }\r\n"+ " dataHandler(data) {\r\n"+ " this.buffer += data;\r\n"+ " let buffer = this.buffer;\r\n"+ " while (buffer.length >= 4) {\r\n"+ " let size = u32(buffer.substr(0, 4));\r\n"+ " if (buffer.length >= size) {\r\n"+ " let pkgStr = '';\r\n"+ " try {\r\n"+ " pkgStr = ascii_to_utf8(buffer.substr(12, size - 8));\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " console.log('deocde fail', escape(buffer.substr(12, size - 8)));\r\n"+ " }\r\n"+ " this.buffer = buffer = buffer.substr(size + 4);\r\n"+ " if (pkgStr.length === 0)\r\n"+ " continue;\r\n"+ " try {\r\n"+ " let pkg = douyuDecode(pkgStr);\r\n"+ " this.listener && this.listener.onPackage(pkg, pkgStr);\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " console.error('call map', e);\r\n"+ " }\r\n"+ " }\r\n"+ " else {\r\n"+ " break;\r\n"+ " }\r\n"+ " }\r\n"+ " }\r\n"+ " closeHandler() {\r\n"+ " console.error('lost connection');\r\n"+ " this.listener && this.listener.onClose();\r\n"+ " }\r\n"+ " errorHandler(err) {\r\n"+ " console.error(err);\r\n"+ " this.listener && this.listener.onError(err);\r\n"+ " }\r\n"+ " send(data) {\r\n"+ " let msg = 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.writeFlush(msg);\r\n"+ " }\r\n"+ " }\r\n"+ " function Type(type) {\r\n"+ " return (target, propertyKey, descriptor) => {\r\n"+ " if (!target.map) {\r\n"+ " target.map = {};\r\n"+ " }\r\n"+ " target.map[type] = target[propertyKey];\r\n"+ " };\r\n"+ " }\r\n"+ " class DouyuBaseClient {\r\n"+ " constructor(roomId) {\r\n"+ " this.roomId = roomId;\r\n"+ " this.lastIP = null;\r\n"+ " this.lastPort = null;\r\n"+ " this.keepaliveId = null;\r\n"+ " this.redirect = {};\r\n"+ " this.prot = new DouyuProtocol(this);\r\n"+ " }\r\n"+ " static getRoomArgs() {\r\n"+ " if (window._room_args)\r\n"+ " return window._room_args;\r\n"+ " if (window.room_args) {\r\n"+ " return window.room_args;\r\n"+ " }\r\n"+ " else {\r\n"+ " return window.$ROOM.args;\r\n"+ " }\r\n"+ " }\r\n"+ " reconnect() {\r\n"+ " return __awaiter(this, void 0, void 0, function* () {\r\n"+ " console.log('reconnect');\r\n"+ " this.prot.listener = null;\r\n"+ " this.prot = new DouyuProtocol(this);\r\n"+ " try {\r\n"+ " yield this.connectAsync(this.lastIP, this.lastPort);\r\n"+ " }\r\n"+ " catch (e) {\r\n"+ " this.onError();\r\n"+ " }\r\n"+ " });\r\n"+ " }\r\n"+ " onClose() {\r\n"+ " setTimeout(() => this.reconnect(), 1000);\r\n"+ " }\r\n"+ " onError() {\r\n"+ " this.onClose();\r\n"+ " }\r\n"+ " onPackage(pkg, pkgStr) {\r\n"+ " const type = pkg.type;\r\n"+ " if (this.redirect[type]) {\r\n"+ " ACJ(this.redirect[type], pkg);\r\n"+ " return;\r\n"+ " }\r\n"+ " if (this.map[type]) {\r\n"+ " this.map[type].call(this, pkg, pkgStr);\r\n"+ " return;\r\n"+ " }\r\n"+ " this.onDefault(pkg);\r\n"+ " }\r\n"+ " onDefault(pkg) {\r\n"+ " }\r\n"+ " send(pkg) {\r\n"+ " this.prot.send(pkg);\r\n"+ " }\r\n"+ " connectAsync(ip, port) {\r\n"+ " return __awaiter(this, void 0, void 0, function* () {\r\n"+ " this.lastIP = ip;\r\n"+ " this.lastPort = port;\r\n"+ " yield this.prot.connectAsync(ip, port);\r\n"+ " this.send(this.loginreq());\r\n"+ " });\r\n"+ " }\r\n"+ " keepalivePkg() {\r\n"+ " return {\r\n"+ " type: 'keeplive',\r\n"+ " tick: Math.round(new Date().getTime() / 1000).toString()\r\n"+ " };\r\n"+ " }\r\n"+ " loginreq() {\r\n"+ " const rt = Math.round(new Date().getTime() / 1000);\r\n"+ " const devid = getACF('devid');\r\n"+ " const 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: this.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"+ " startKeepalive() {\r\n"+ " this.send(this.keepalivePkg());\r\n"+ " if (this.keepaliveId) {\r\n"+ " clearInterval(this.keepaliveId);\r\n"+ " }\r\n"+ " this.keepaliveId = setInterval(() => this.send(this.keepalivePkg()), 30 * 1000);\r\n"+ " }\r\n"+ " }\r\n"+ " let blacklist = [];\r\n"+ " function onChatMsg(data) {\r\n"+ " if (blacklist.indexOf(data.uid) !== -1) {\r\n"+ " console.log('black');\r\n"+ " return;\r\n"+ " }\r\n"+ " try {\r\n"+ " postMessage('DANMU', 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(douyuEncode(data));\r\n"+ " }\r\n"+ " }\r\n"+ " class DouyuClient extends DouyuBaseClient {\r\n"+ " constructor(roomId, danmuClient) {\r\n"+ " super(roomId);\r\n"+ " this.danmuClient = danmuClient;\r\n"+ " this.redirect = {\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"+ " frank: 'room_data_handler',\r\n"+ " online_noble_list: 'room_data_handler',\r\n"+ " };\r\n"+ " }\r\n"+ " reqOnlineGift(loginres) {\r\n"+ " return {\r\n"+ " type: 'reqog',\r\n"+ " uid: loginres.userid\r\n"+ " };\r\n"+ " }\r\n"+ " chatmsg(data) {\r\n"+ " onChatMsg(data);\r\n"+ " }\r\n"+ " resog(data) {\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(data) {\r\n"+ " console.log('loginres ms', data);\r\n"+ " this.uid = data.userid;\r\n"+ " this.send(this.reqOnlineGift(data));\r\n"+ " this.startKeepalive();\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(data, rawString) {\r\n"+ " ACJ('room_data_userc', data.uc);\r\n"+ " ACJ('room_data_tbredpacket', rawString);\r\n"+ " }\r\n"+ " setmsggroup(data) {\r\n"+ " console.log('joingroup', data);\r\n"+ " this.danmuClient.send({\r\n"+ " type: 'joingroup',\r\n"+ " rid: data.rid,\r\n"+ " gid: data.gid\r\n"+ " });\r\n"+ " }\r\n"+ " onDefault(data) {\r\n"+ " ACJ('room_data_handler', data);\r\n"+ " console.log('ms', data);\r\n"+ " }\r\n"+ " }\r\n"+ " __decorate([\r\n"+ " Type('chatmsg')\r\n"+ " ], DouyuClient.prototype, \"chatmsg\", null);\r\n"+ " __decorate([\r\n"+ " Type('resog')\r\n"+ " ], DouyuClient.prototype, \"resog\", null);\r\n"+ " __decorate([\r\n"+ " Type('loginres')\r\n"+ " ], DouyuClient.prototype, \"loginres\", null);\r\n"+ " __decorate([\r\n"+ " Type('keeplive')\r\n"+ " ], DouyuClient.prototype, \"keeplive\", null);\r\n"+ " __decorate([\r\n"+ " Type('setmsggroup')\r\n"+ " ], DouyuClient.prototype, \"setmsggroup\", null);\r\n"+ " class DouyuDanmuClient extends DouyuBaseClient {\r\n"+ " constructor(roomId) {\r\n"+ " super(roomId);\r\n"+ " this.redirect = {\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"+ " gpbc: 'room_data_handler',\r\n"+ " synexp: 'room_data_handler',\r\n"+ " frank: 'room_data_handler',\r\n"+ " ggbb: 'room_data_sabonusget',\r\n"+ " online_noble_list: 'room_data_handler',\r\n"+ " };\r\n"+ " }\r\n"+ " chatmsg(pkg) {\r\n"+ " onChatMsg(pkg);\r\n"+ " }\r\n"+ " loginres(data) {\r\n"+ " console.log('loginres dm', data);\r\n"+ " this.startKeepalive();\r\n"+ " }\r\n"+ " onDefault(data) {\r\n"+ " ACJ('room_data_handler', data);\r\n"+ " console.log('dm', data);\r\n"+ " }\r\n"+ " }\r\n"+ " __decorate([\r\n"+ " Type('chatmsg')\r\n"+ " ], DouyuDanmuClient.prototype, \"chatmsg\", null);\r\n"+ " __decorate([\r\n"+ " Type('loginres')\r\n"+ " ], DouyuDanmuClient.prototype, \"loginres\", null);\r\n"+ " function hookDouyu(roomId, miscClient) {\r\n"+ " let oldExe;\r\n"+ " const repeatPacket = (text) => douyuDecode(text);\r\n"+ " const jsMap = {\r\n"+ " js_getRankScore: repeatPacket,\r\n"+ " js_getnoble: repeatPacket,\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_sendmsg(msg) {\r\n"+ " let pkg = douyuDecode(msg);\r\n"+ " pkg.type = 'chatmessage';\r\n"+ " return pkg;\r\n"+ " },\r\n"+ " js_giveGift(gift) {\r\n"+ " let pkg = douyuDecode(gift);\r\n"+ " if (pkg.type === 'dn_s_gf') {\r\n"+ " pkg.type = 'sgq';\r\n"+ " pkg.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() { },\r\n"+ " js_myblacklist(list) {\r\n"+ " console.log('add blacklist', list);\r\n"+ " blacklist = list.split('|');\r\n"+ " },\r\n"+ " js_medal_opera(opt) {\r\n"+ " let pkg = douyuDecode(opt);\r\n"+ " return pkg;\r\n"+ " }\r\n"+ " };\r\n"+ " const api = window['require']('douyu/page/room/base/api');\r\n"+ " const hookd = function hookd(...args) {\r\n"+ " let 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 && miscClient.send(req);\r\n"+ " }\r\n"+ " else {\r\n"+ " console.log('exe', args);\r\n"+ " try {\r\n"+ " return oldExe.apply(api, args);\r\n"+ " }\r\n"+ " catch (e) { }\r\n"+ " }\r\n"+ " };\r\n"+ " if (api) {\r\n"+ " if (api.exe !== hookd) {\r\n"+ " oldExe = api.exe;\r\n"+ " api.exe = hookd;\r\n"+ " }\r\n"+ " }\r\n"+ " else if (window.thisMovie) {\r\n"+ " window.thisMovie = () => new Proxy({}, {\r\n"+ " get(target, key, receiver) {\r\n"+ " return (...args) => hookd.apply(null, [key].concat(args));\r\n"+ " }\r\n"+ " });\r\n"+ " }\r\n"+ " }\r\n"+ " function douyuApi(roomId) {\r\n"+ " return __awaiter(this, void 0, void 0, function* () {\r\n"+ " const res = yield fetch('/swf_api/get_room_args');\r\n"+ " const args = yield res.json();\r\n"+ " const servers = JSON.parse(decodeURIComponent(args.server_config));\r\n"+ " const mserver = servers[Math.floor(Math.random() * servers.length)];\r\n"+ " const ports = [8601, 8602, 12601, 12602];\r\n"+ " const danmuServer = {\r\n"+ " ip: 'danmu.douyu.com',\r\n"+ " port: ports[Math.floor(Math.random() * ports.length)]\r\n"+ " };\r\n"+ " let danmuClient = new DouyuDanmuClient(roomId);\r\n"+ " let miscClient = new DouyuClient(roomId, danmuClient);\r\n"+ " yield danmuClient.connectAsync(danmuServer.ip, danmuServer.port);\r\n"+ " yield miscClient.connectAsync(mserver.ip, mserver.port);\r\n"+ " return {\r\n"+ " sendDanmu(content) {\r\n"+ " miscClient.send({\r\n"+ " col: '0',\r\n"+ " content: content,\r\n"+ " dy: '',\r\n"+ " pid: '',\r\n"+ " sender: miscClient.uid,\r\n"+ " type: 'chatmessage'\r\n"+ " });\r\n"+ " },\r\n"+ " serverSend(pkg) {\r\n"+ " return miscClient.send(pkg);\r\n"+ " },\r\n"+ " hookExe() {\r\n"+ " hookDouyu(roomId, miscClient);\r\n"+ " }\r\n"+ " };\r\n"+ " });\r\n"+ " }\n"+ "\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"+ " const children = flash.children;\r\n"+ " for (let i = 0; i < children.length; i++) {\r\n"+ " const param = children[i];\r\n"+ " if (param.name == name) {\r\n"+ " return param.value;\r\n"+ " }\r\n"+ " }\r\n"+ " return '';\r\n"+ " }\r\n"+ " function getRoomIdFromFlash(s) {\r\n"+ " return s.split('&').filter(i => i.substr(0, 6) == 'RoomId')[0].split('=')[1];\r\n"+ " }\r\n"+ " hookFunc(document, 'createElement', (old, args) => {\r\n"+ " var ret = old.apply(null, args);\r\n"+ " if (args[0] == 'object') {\r\n"+ " hookFunc(ret, 'setAttribute', (old, args) => {\r\n"+ " if (args[0] == 'data') {\r\n"+ " if (/WebRoom/.test(args[1])) {\r\n"+ " setTimeout(() => {\r\n"+ " let roomId = getRoomIdFromFlash(getParam(ret, 'flashvars'));\r\n"+ " console.log('RoomId', roomId);\r\n"+ " postMessage('VIDEOID', {\r\n"+ " roomId: roomId,\r\n"+ " id: ret.id\r\n"+ " });\r\n"+ " }, 1);\r\n"+ " }\r\n"+ " }\r\n"+ " return old.apply(null, args);\r\n"+ " });\r\n"+ " }\r\n"+ " return ret;\r\n"+ " });\r\n"+ " let api;\r\n"+ " onMessage('BEGINAPI', (data) => __awaiter(window, void 0, void 0, function* () {\r\n"+ " yield retry(() => JSocket.init(), 3);\r\n"+ " api = yield douyuApi(data.roomId);\r\n"+ " api.hookExe();\r\n"+ " window.api = api;\r\n"+ " }));\r\n"+ " onMessage('SENDANMU', data => {\r\n"+ " api.sendDanmu(data);\r\n"+ " });\r\n"+ " onMessage('ACJ', data => {\r\n"+ " ACJ(data.id, data.data);\r\n"+ " });\n"+ "\n"+ "})));\n"+ "\n", css: ".jsocket-cls,\n"+ ".big-flash-cls {\n"+ " width: 80vw;\n"+ " height: 80vh;\n"+ " position: absolute;\n"+ " top: -100vh;\n"+ " left: 0;\n"+ "}\n"+ ".donate-dialog {\n"+ " position: fixed;\n"+ " z-index: 100002;\n"+ " left: 0;\n"+ " top: 0;\n"+ " right: 0;\n"+ " bottom: 0;\n"+ " display: flex;\n"+ " align-items: center;\n"+ " justify-content: center;\n"+ "}\n"+ ".donate-dialog .donate-title {\n"+ " font-size: 20px;\n"+ "}\n"+ ".donate-dialog .donate-content {\n"+ " margin-bottom: 10px;\n"+ "}\n"+ ".donate-dialog .donate-wrap {\n"+ " width: 400px;\n"+ " padding: 10px;\n"+ " background: #fff;\n"+ " border-radius: 5px;\n"+ " border: 1px solid #aaa;\n"+ "}\n"+ ".donate-dialog .donate-qrcode-bar {\n"+ " display: flex;\n"+ " justify-content: space-between;\n"+ "}\n"+ ".donate-dialog .donate-qrcode-bar img {\n"+ " height: 168px;\n"+ "}\n"+ ".donate-dialog .donate-qrcode-desc {\n"+ " text-align: center;\n"+ "}\n"+ ".donate-dialog .donate-close-btn {\n"+ " margin-top: 10px;\n"+ "}\n"+ ".donate-dialog .donate-close-btn:before {\n"+ " content: '关闭';\n"+ " text-align: center;\n"+ " display: block;\n"+ "}\n"+ ".player-menu {\n"+ " position: fixed;\n"+ " top: 0;\n"+ " left: 0;\n"+ " right: 0;\n"+ " bottom: 0;\n"+ " z-index: 100001;\n"+ "}\n"+ ".player-menu .menu {\n"+ " position: absolute;\n"+ " background-color: #fff;\n"+ " min-width: 200px;\n"+ " border: 1px solid #aaa;\n"+ " box-shadow: 2px 2px 5px rgba(0, 0, 0, 0.5);\n"+ " -webkit-user-select: none;\n"+ " -moz-user-select: none;\n"+ "}\n"+ ".player-menu .menu .menu-dash {\n"+ " height: 1px;\n"+ " background-color: #aaa;\n"+ "}\n"+ ".player-menu .menu .menu-item {\n"+ " color: #000;\n"+ " padding: 5px;\n"+ " cursor: default;\n"+ "}\n"+ ".player-menu .menu .menu-item:last-child {\n"+ " border-bottom: 0;\n"+ "}\n"+ ".player-menu .menu .menu-item:hover {\n"+ " background-color: #ddd;\n"+ "}\n"+ ".danmu-container {\n"+ " width: 100%;\n"+ " height: 100%;\n"+ " border: 1px solid #e5e4e4;\n"+ " box-sizing: border-box;\n"+ "}\n"+ ".danmu-wrap {\n"+ " width: 100%;\n"+ " height: 100%;\n"+ " position: relative;\n"+ " background-color: #000;\n"+ "}\n"+ ".danmu-wrap .danmu-input {\n"+ " display: none;\n"+ "}\n"+ ".danmu-wrap .danmu-video {\n"+ " width: 100%;\n"+ " height: calc(100% - 42px);\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl {\n"+ " box-sizing: border-box;\n"+ " border: 1px solid #fafafa;\n"+ " border-left: 0;\n"+ " border-right: 0;\n"+ " width: 100%;\n"+ " height: 42px;\n"+ " padding: 5px;\n"+ " color: #5a5a5a;\n"+ " background: #fafafa;\n"+ "}\n"+ ".danmu-wrap .danmu-ctrl > a {\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(require('flv.js')) : typeof define === 'function' && define.amd ? define(['flv.js'], factory) : (factory(global.flvjs)); }(this, (function (flv_js) { 'use strict'; /*! ***************************************************************************** Copyright (c) Microsoft Corporation. All rights reserved. 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 THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE, MERCHANTABLITY OR NON-INFRINGEMENT. See the Apache Version 2.0 License for specific language governing permissions and limitations under the License. ***************************************************************************** */ /* global Reflect, Promise */ var extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; function __decorate(decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; } function __awaiter(thisArg, _arguments, P, generator) { return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator.throw(value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); } function __read(o, n) { var m = typeof Symbol === "function" && o[Symbol.iterator]; if (!m) return o; var i = m.call(o), r, ar = [], e; try { while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value); } catch (error) { e = { error: error }; } finally { try { if (r && !r.done && (m = i["return"])) m.call(i); } finally { if (e) throw e.error; } } return ar; } function __await(v) { return this instanceof __await ? (this.v = v, this) : new __await(v); } function hookFetchCode () { // let self = this const convertHeader = function convertHeader (headers) { let out = new Headers(); for (let key of Object.keys(headers)) { out.set(key, headers[key]); } return out }; const hideHookStack = stack => { return stack.replace(/^\s*at\s.*?hookfetch\.js:\d.*$\n/mg, '') }; const base64ToUint8 = (b64) => { const s = atob(b64); const length = s.length; let ret = new Uint8Array(length); for (let i = 0; i < length; i++) { ret[i] = s.charCodeAt(i); } return ret }; class WrapPort { constructor (port) { this.curMethod = ''; this.curResolve = null; this.curReject = null; this.stack = ''; this.port = port; this.lastDone = true; port.onMessage.addListener(msg => this.onMessage(msg)); } post (method, args) { if (!this.lastDone) { throw new Error('Last post is not done') } this.stack = new Error().stack; return new Promise((resolve, reject) => { this.lastDone = false; this.curMethod = method; this.curResolve = resolve; this.curReject = reject; this.port.postMessage({ method: method, args: args }); }) } onMessage (msg) { if (msg.method === this.curMethod) { if (msg.err) { let err = new Error(msg.err.message); err.oriName = msg.err.name; err.stack = hideHookStack(this.stack); // console.log('fetch err', err) this.curReject.call(null, err); } else { this.curResolve.apply(null, msg.args); } this.curResolve = null; this.curReject = null; this.lastDone = true; } else { console.error('wtf?'); } } } class PortReader { constructor (port) { this.port = port; this.hasReader = false; } _requireReader () { if (this.hasReader) { return Promise.resolve() } else { return this.port.post('body.getReader').then(() => this.hasReader = true) } } read () { return this._requireReader() .then(() => this.port.post('reader.read')) .then(r => { if (r.done == false) { r.value = base64ToUint8(r.value); } return r }) } cancel () { return this._requireReader().then(() => this.port.post('reader.cancel')) } } class PortBody { constructor (port) { this.port = port; } getReader () { return new PortReader(this.port) } } class PortFetch { constructor () { this.port = new WrapPort(chrome.runtime.connect({name: 'fetch'})); } fetch (...args) { return this.port.post('fetch', args).then(r => { r.json = () => this.port.post('json'); r.arrayBuffer = () => this.port.post('arrayBuffer').then(buf => { return new Uint8Array(buf).buffer }); r.headers = convertHeader(r.headers); r.body = new PortBody(this.port); return r }) } } const bgFetch = function bgFetch (...args) { const fetch = new PortFetch(); return fetch.fetch(...args) }; function hookFetch () { if (fetch !== bgFetch) { fetch = bgFetch; } } const oldBlob = Blob; const 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(); } function isFirefox () { return /Firefox/.test(navigator.userAgent) } if (!isFirefox()) { hookFetchCode(); } function utf8ToUtf16(utf8_bytes) { let unicode_codes = []; let unicode_code = 0; let num_followed = 0; for (let i = 0; i < utf8_bytes.length; ++i) { let utf8_byte = utf8_bytes[i]; if (utf8_byte >= 0x100) { } else if ((utf8_byte & 0xC0) == 0x80) { if (num_followed > 0) { unicode_code = (unicode_code << 6) | (utf8_byte & 0x3f); num_followed -= 1; } else { } } else { if (num_followed == 0) { unicode_codes.push(unicode_code); } else { } if (utf8_byte < 0x80) { unicode_code = utf8_byte; num_followed = 0; } else if ((utf8_byte & 0xE0) == 0xC0) { unicode_code = utf8_byte & 0x1f; num_followed = 1; } else if ((utf8_byte & 0xF0) == 0xE0) { unicode_code = utf8_byte & 0x0f; num_followed = 2; } else if ((utf8_byte & 0xF8) == 0xF0) { unicode_code = utf8_byte & 0x07; num_followed = 3; } else { } } } if (num_followed == 0) { unicode_codes.push(unicode_code); } else { } unicode_codes.shift(); let utf16_codes = []; for (var i = 0; i < unicode_codes.length; ++i) { 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) { const char2bytes = (unicode_code) => { let utf8_bytes = []; if (unicode_code < 0x80) { utf8_bytes.push(unicode_code); } else if (unicode_code < (1 << 11)) { utf8_bytes.push((unicode_code >>> 6) | 0xC0); utf8_bytes.push((unicode_code & 0x3F) | 0x80); } else if (unicode_code < (1 << 16)) { 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)) { 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; }; let o = []; for (let i = 0; i < str.length; i++) { o = o.concat(char2bytes(str.charCodeAt(i))); } return o.map(i => String.fromCharCode(i)).join(''); } function ascii_to_utf8(str) { let bytes = str.split('').map(i => i.charCodeAt(0)); return utf8ToUtf16(bytes).map(i => String.fromCharCode(i)).join(''); } function requestFullScreen() { let de = document.documentElement; if (de.requestFullscreen) { de.requestFullscreen(); } else if (de.mozRequestFullScreen) { de.mozRequestFullScreen(); } else if (de.webkitRequestFullScreen) { de.webkitRequestFullScreen(); } } function exitFullscreen() { let de = document; if (de.exitFullscreen) { de.exitFullscreen(); } else if (de.mozCancelFullScreen) { de.mozCancelFullScreen(); } else if (de.webkitCancelFullScreen) { de.webkitCancelFullScreen(); } } class LocalStorage { constructor(domain) { this.domain = domain; } getItem(key, def) { return window.localStorage.getItem(`${this.domain}-${key}`) || def; } setItem(key, data) { window.localStorage.setItem(`${this.domain}-${key}`, data); } } class Timer { constructor(delay) { this.delay = delay; } reset() { if (this.id) { clearTimeout(this.id); } this.id = window.setTimeout(this.onTimer, this.delay); } } function getURL(src) { if (src.substr(0, 5) !== 'blob:') { src = chrome.runtime.getURL(src); } return src; } function addScript(src) { var script = document.createElement('script'); script.src = getURL(src); document.head.appendChild(script); } function addCss(src, rel = 'stylesheet', type = 'text/css') { var link = document.createElement('link'); link.rel = rel; link.type = type; link.href = getURL(src); document.head.appendChild(link); } function createBlobURL(content, type) { var blob = new Blob([content], { type }); return URL.createObjectURL(blob); } const p32 = (i) => [i, i / 256, i / 65536, i / 16777216].map(i => String.fromCharCode(Math.floor(i) % 256)).join(''); const u32 = (s) => s.split('').map(i => i.charCodeAt(0)).reduce((a, b) => b * 256 + a); let messageMap = {}; function onMessage(type, cb) { messageMap[type] = cb; } function postMessage(type, data) { window.postMessage({ type: type, data: data }, "*"); } let msgCallbacks = []; let lastCbId = 0; window.addEventListener('message', (event) => __awaiter(window, void 0, void 0, function* () { const data = event.data; if (data.cb) { let cb = msgCallbacks[data.cbId]; if (cb && (typeof cb === 'function')) { cb(data.cbResult); } } else if (data.type) { let result = undefined; if (typeof messageMap[data.type] === 'function') { result = messageMap[data.type](data.data); if (result instanceof Promise) { result = yield result; } if (data.cbId) { window.postMessage({ cb: true, cbId: data.cbId, cbResult: result }, '*'); } } } }), false); function getSync() { return new Promise((res, rej) => { if (chrome && chrome.storage && chrome.storage.sync) { chrome.storage.sync.get(items => { res(items); }); } else { rej(new Error('不支持的存储方式')); } }); } function setSync(item) { return new Promise((res, rej) => { if (chrome && chrome.storage && chrome.storage.sync) { chrome.storage.sync.set(item, res); } else { rej(new Error('不支持的存储方式')); } }); } function getSetting() { return __awaiter(this, void 0, void 0, function* () { let setting; try { setting = yield getSync(); } catch (e) { } if (!setting) { setting = {}; } if (!setting.blacklist) { setting.blacklist = []; } return setting; }); } function setSetting(setting) { return __awaiter(this, void 0, void 0, function* () { yield setSync(setting); }); } const defaultBgListener = (request) => __awaiter(window, void 0, void 0, function* () { return null; }); let bgListener = defaultBgListener; function setBgListener(listener) { if (bgListener === defaultBgListener) { if ((typeof chrome !== 'undefined') && chrome.runtime && chrome.runtime.onMessage) { chrome.runtime.onMessage.addListener((request, sender, sendResponse) => __awaiter(this, void 0, void 0, function* () { sendResponse(yield bgListener(request)); })); } } else { console.warn('多次设置BgListener'); } bgListener = listener; } class DelayNotify { constructor(defaultValue) { this.defaultValue = defaultValue; this.notified = false; this.tmid = null; this.res = null; } notify(value) { if (this.notified) { return; } this.notified = true; this.value = value; if (this.res) { this.res(this.value); } } wait(timeout = 1000 * 10) { if (this.notified) { return Promise.resolve(this.value); } return new Promise((resolve, reject) => { if (timeout > 0) { window.setTimeout(() => { resolve(this.defaultValue); }, timeout); } this.res = (value) => { return resolve(value); }; }); } reset() { this.notified = false; } } /*! typestate - v1.0.4 - 2016-09-07 * https://github.com/eonarheim/TypeState * Copyright (c) 2016 Erik Onarheim; Licensed BSD-2-Clause*/ var typestate; (function (typestate) { /** * Transition grouping to faciliate fluent api */ var Transitions = (function () { function Transitions(fsm) { this.fsm = fsm; } /** * Specify the end state(s) of a transition function */ Transitions.prototype.to = function () { var states = []; for (var _i = 0; _i < arguments.length; _i++) { states[_i - 0] = arguments[_i]; } this.toStates = states; this.fsm.addTransitions(this); }; /** * Specify that any state in the state enum is value * Takes the state enum as an argument */ Transitions.prototype.toAny = function (states) { var toStates = []; for (var s in states) { if (states.hasOwnProperty(s)) { toStates.push(states[s]); } } this.toStates = toStates; this.fsm.addTransitions(this); }; return Transitions; }()); typestate.Transitions = Transitions; /** * Internal representation of a transition function */ var TransitionFunction = (function () { function TransitionFunction(fsm, from, to) { this.fsm = fsm; this.from = from; this.to = to; } return TransitionFunction; }()); typestate.TransitionFunction = TransitionFunction; /** * A simple finite state machine implemented in TypeScript, the templated argument is meant to be used * with an enumeration. */ var FiniteStateMachine = (function () { function FiniteStateMachine(startState) { this._transitionFunctions = []; this._onCallbacks = {}; this._exitCallbacks = {}; this._enterCallbacks = {}; this._invalidTransitionCallback = null; this.currentState = startState; this._startState = startState; } FiniteStateMachine.prototype.addTransitions = function (fcn) { var _this = this; fcn.fromStates.forEach(function (from) { fcn.toStates.forEach(function (to) { // self transitions are invalid and don't add duplicates if (from !== to && !_this._validTransition(from, to)) { _this._transitionFunctions.push(new TransitionFunction(_this, from, to)); } }); }); }; /** * Listen for the transition to this state and fire the associated callback */ FiniteStateMachine.prototype.on = function (state, callback) { var key = state.toString(); if (!this._onCallbacks[key]) { this._onCallbacks[key] = []; } this._onCallbacks[key].push(callback); return this; }; /** * Listen for the transition to this state and fire the associated callback, returning * false in the callback will block the transition to this state. */ FiniteStateMachine.prototype.onEnter = function (state, callback) { var key = state.toString(); if (!this._enterCallbacks[key]) { this._enterCallbacks[key] = []; } this._enterCallbacks[key].push(callback); return this; }; /** * Listen for the transition to this state and fire the associated callback, returning * false in the callback will block the transition from this state. */ FiniteStateMachine.prototype.onExit = function (state, callback) { var key = state.toString(); if (!this._exitCallbacks[key]) { this._exitCallbacks[key] = []; } this._exitCallbacks[key].push(callback); return this; }; /** * List for an invalid transition and handle the error, returning a falsy value will throw an * exception, a truthy one will swallow the exception */ FiniteStateMachine.prototype.onInvalidTransition = function (callback) { if (!this._invalidTransitionCallback) { this._invalidTransitionCallback = callback; } return this; }; /** * Declares the start state(s) of a transition function, must be followed with a '.to(...endStates)' */ FiniteStateMachine.prototype.from = function () { var states = []; for (var _i = 0; _i < arguments.length; _i++) { states[_i - 0] = arguments[_i]; } var _transition = new Transitions(this); _transition.fromStates = states; return _transition; }; FiniteStateMachine.prototype.fromAny = function (states) { var fromStates = []; for (var s in states) { if (states.hasOwnProperty(s)) { fromStates.push(states[s]); } } var _transition = new Transitions(this); _transition.fromStates = fromStates; return _transition; }; FiniteStateMachine.prototype._validTransition = function (from, to) { return this._transitionFunctions.some(function (tf) { return (tf.from === from && tf.to === to); }); }; /** * Check whether a transition to a new state is valid */ FiniteStateMachine.prototype.canGo = function (state) { return this.currentState === state || this._validTransition(this.currentState, state); }; /** * Transition to another valid state */ FiniteStateMachine.prototype.go = function (state) { if (!this.canGo(state)) { if (!this._invalidTransitionCallback || !this._invalidTransitionCallback(this.currentState, state)) { throw new Error('Error no transition function exists from state ' + this.currentState.toString() + ' to ' + state.toString()); } } else { this._transitionTo(state); } }; /** * This method is availble for overridding for the sake of extensibility. * It is called in the event of a successful transition. */ FiniteStateMachine.prototype.onTransition = function (from, to) { // pass, does nothing until overidden }; /** * Reset the finite state machine back to the start state, DO NOT USE THIS AS A SHORTCUT for a transition. * This is for starting the fsm from the beginning. */ FiniteStateMachine.prototype.reset = function () { this.currentState = this._startState; }; /** * Whether or not the current state equals the given state */ FiniteStateMachine.prototype.is = function (state) { return this.currentState === state; }; FiniteStateMachine.prototype._transitionTo = function (state) { var _this = this; if (!this._exitCallbacks[this.currentState.toString()]) { this._exitCallbacks[this.currentState.toString()] = []; } if (!this._enterCallbacks[state.toString()]) { this._enterCallbacks[state.toString()] = []; } if (!this._onCallbacks[state.toString()]) { this._onCallbacks[state.toString()] = []; } var canExit = this._exitCallbacks[this.currentState.toString()].reduce(function (accum, next) { return accum && next.call(_this, state); }, true); var canEnter = this._enterCallbacks[state.toString()].reduce(function (accum, next) { return accum && next.call(_this, _this.currentState); }, true); if (canExit && canEnter) { var old = this.currentState; this.currentState = state; this._onCallbacks[this.currentState.toString()].forEach(function (fcn) { fcn.call(_this, old); }); this.onTransition(old, state); } }; return FiniteStateMachine; }()); typestate.FiniteStateMachine = FiniteStateMachine; })(typestate || (typestate = {})); var TypeState_1 = typestate; const storage = new LocalStorage('h5plr'); function findInParent(node, toFind) { while ((node !== toFind) && (node !== null)) { node = node.parentElement; } return node !== null; } var PlayerState; (function (PlayerState) { PlayerState[PlayerState["Stopped"] = 0] = "Stopped"; PlayerState[PlayerState["Playing"] = 1] = "Playing"; PlayerState[PlayerState["Paused"] = 2] = "Paused"; PlayerState[PlayerState["Buffering"] = 3] = "Buffering"; })(PlayerState || (PlayerState = {})); var SizeState; (function (SizeState) { SizeState[SizeState["Normal"] = 0] = "Normal"; SizeState[SizeState["FullPage"] = 1] = "FullPage"; SizeState[SizeState["FullScreen"] = 2] = "FullScreen"; SizeState[SizeState["ExitFullScreen"] = 3] = "ExitFullScreen"; })(SizeState || (SizeState = {})); class SizeStateFSM extends TypeState_1.FiniteStateMachine { constructor() { super(SizeState.Normal); this.fromAny(SizeState).to(SizeState.Normal); this.fromAny(SizeState).to(SizeState.FullPage); this.fromAny(SizeState).to(SizeState.FullScreen); this.from(SizeState.FullScreen).to(SizeState.ExitFullScreen); } onTransition(from, to) { console.log('SizeFSM', from, to); } } class PlayerStateFSM extends TypeState_1.FiniteStateMachine { constructor() { super(PlayerState.Stopped); this.fromAny(PlayerState).to(PlayerState.Stopped); this.fromAny(PlayerState).to(PlayerState.Playing); this.from(PlayerState.Playing).to(PlayerState.Buffering); this.from(PlayerState.Playing).to(PlayerState.Paused); this.from(PlayerState.Buffering).to(PlayerState.Paused); } onTransition(from, to) { console.log('PlayerFSM', from, to); } } class PlayerUI { constructor(listener, state) { this.listener = listener; this.state = state; this.inputing = false; this.hideDanmu = false; this._muted = false; this._fullscreen = false; this._lastY = -1; const playerContainer = document.createElement('div'); const playerWrap = document.createElement('div'); const playerCtrl = document.createElement('div'); const danmuLayout = document.createElement('div'); const videoBox = document.createElement('div'); const msgBox = document.createElement('div'); const msgInput = document.createElement('input'); const videoEl = document.createElement('video'); this.sizeState = new SizeStateFSM(); let lastState; this.sizeState .on(SizeState.Normal, from => { switch (from) { case SizeState.FullPage: this._exitFullPage(); break; case SizeState.ExitFullScreen: this._exitFullScreen(); break; } }) .on(SizeState.FullPage, from => { switch (from) { case SizeState.Normal: this._enterFullPage(); break; case SizeState.ExitFullScreen: this._enterFullPage(); break; } }) .on(SizeState.FullScreen, from => { if (from == SizeState.FullScreen) return; lastState = from; switch (from) { case SizeState.Normal: this._enterFullScreen(); break; case SizeState.FullPage: this._enterFullScreen(); break; } }) .on(SizeState.ExitFullScreen, from => { this._exitFullScreen(); this.sizeState.go(lastState); }); 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); let timer = new Timer(1000); timer.onTimer = () => playerWrap.removeAttribute('hover'); playerWrap.addEventListener('mousemove', event => { const hoverCtl = findInParent(event.target, playerCtrl); if (event.offsetY - this._lastY == 0) return; this._lastY = event.offsetY; let height = playerWrap.getBoundingClientRect().height; if (event.offsetY > 0) { playerWrap.setAttribute('hover', ''); timer.reset(); } else { playerWrap.removeAttribute('hover'); } }); playerWrap.addEventListener('click', event => { if (findInParent(event.target, msgBox)) return; playerWrap.removeAttribute('inputing'); this.inputing = false; }); document.addEventListener('keydown', event => { if (event.keyCode == 13) { if (this.sizeState.is(SizeState.Normal)) return; if (event.target.nodeName.toUpperCase() === 'TEXTAREA') return; this.inputing = !this.inputing; if (this.inputing) { msgInput.value = ''; playerWrap.setAttribute('inputing', ''); msgInput.focus(); } else { if (msgInput.value.length > 0) { listener.onSendDanmu(msgInput.value); } playerWrap.removeAttribute('inputing'); } } else if (event.keyCode == 27) { if (this.sizeState.is(SizeState.FullPage)) { this.sizeState.go(SizeState.Normal); } if (this.sizeState.is(SizeState.FullScreen)) { this.sizeState.go(SizeState.ExitFullScreen); } } }); document.addEventListener('webkitfullscreenchange', event => { this._fullscreen = !this._fullscreen; if (!this._fullscreen) { if (this.sizeState.is(SizeState.FullScreen)) { this.sizeState.go(SizeState.ExitFullScreen); } } }); window.addEventListener('unload', event => { listener.onStop(); listener.onUnload(); }); this.video = videoEl; this.el = playerContainer; this.wrap = playerWrap; this.dmLayout = danmuLayout; this.playerCtrl = playerCtrl; this.transparent = this.transparent; } _exitFullScreen() { exitFullscreen(); this.wrap.removeAttribute('fullpage'); this.el.appendChild(this.wrap); document.body.style.overflow = document.body.parentElement.style.overflow = 'auto'; this.listener.onTryPlay(); } _enterFullScreen() { requestFullScreen(); this.wrap.setAttribute('fullpage', ''); document.body.appendChild(this.wrap); document.body.style.overflow = document.body.parentElement.style.overflow = 'hidden'; this.listener.onTryPlay(); } _exitFullPage() { this.wrap.removeAttribute('fullpage'); this.el.appendChild(this.wrap); document.body.style.overflow = document.body.parentElement.style.overflow = 'auto'; this.listener.onTryPlay(); } _enterFullPage() { this.wrap.setAttribute('fullpage', ''); document.body.appendChild(this.wrap); document.body.style.overflow = document.body.parentElement.style.overflow = 'hidden'; this.listener.onTryPlay(); } get transparent() { return parseInt(storage.getItem('transparent', '0')); } set transparent(val) { storage.setItem('transparent', val.toString()); this.dmLayout.style.opacity = (1 - val / 100).toString(); } get playing() { return this.state.is(PlayerState.Playing) || this.state.is(PlayerState.Buffering); } set playing(val) { if (val) { this.state.go(PlayerState.Playing); } else { this.state.go(PlayerState.Paused); } } get muted() { return this._muted; } set muted(v) { this.listener.onMute(v); if (v) { this.muteEl.setAttribute('muted', ''); } else { this.muteEl.removeAttribute('muted'); } this._muted = v; } notifyStateChange() { if (this.playing) { this.playPause.setAttribute('pause', ''); } else { this.playPause.removeAttribute('pause'); } } initControls() { if (this.tipEl) return; let bar = this.playerCtrl; const now = () => new Date().getTime(); const addBtn = (cls, cb) => { const btn = document.createElement('a'); btn.className = ['danmu-btn', 'danmu-' + cls].join(' '); btn.addEventListener('click', cb); bar.appendChild(btn); return btn; }; this.video.addEventListener('dblclick', event => { switch (this.sizeState.currentState) { case SizeState.Normal: this.sizeState.go(SizeState.FullPage); break; case SizeState.FullPage: this.sizeState.go(SizeState.Normal); break; case SizeState.FullScreen: this.sizeState.go(SizeState.ExitFullScreen); break; } event.preventDefault(); event.stopPropagation(); }); this.playPause = addBtn('playpause', () => { this.playing = !this.playing; this.notifyStateChange(); }); this.playPause.setAttribute('pause', ''); const reload = addBtn('reload', () => { this.listener.onReload(); }); const fullscreen = addBtn('fullscreen', () => { if (this.sizeState.is(SizeState.FullScreen)) { this.sizeState.go(SizeState.ExitFullScreen); } else { this.sizeState.go(SizeState.FullScreen); } }); const fullpage = addBtn('fullpage', () => { switch (this.sizeState.currentState) { case SizeState.Normal: this.sizeState.go(SizeState.FullPage); break; case SizeState.FullPage: this.sizeState.go(SizeState.Normal); break; case SizeState.FullScreen: this.sizeState.go(SizeState.ExitFullScreen); this.sizeState.go(SizeState.FullPage); break; } }); const volume = this.createVolume(percent => { this.listener.onVolumeChange(percent); }); bar.appendChild(volume); this.muteEl = addBtn('mute', () => { this.muted = !this.muted; }); const danmuSwitch = addBtn('switch', () => { this.hideDanmu = !this.hideDanmu; this.listener.onHideDanmu(this.hideDanmu); danmuSwitch.innerText = this.hideDanmu ? '开启弹幕' : '关闭弹幕'; this.dmLayout.style.display = this.hideDanmu ? 'none' : 'block'; }); danmuSwitch.innerText = this.hideDanmu ? '开启弹幕' : '关闭弹幕'; const tip = document.createElement('div'); tip.className = 'danmu-tip'; bar.appendChild(tip); this.tipEl = tip; } createVolume(cb) { const volume = document.createElement('div'); const progress = document.createElement('div'); const input = document.createElement('input'); volume.className = 'danmu-volume'; progress.className = 'progress'; input.type = 'range'; volume.appendChild(input); volume.appendChild(progress); input.value = storage.getItem('volume') || '100'; cb(parseInt(input.value) / 100); input.addEventListener('input', event => { progress.style.width = `${input.value}%`; cb(parseInt(input.value) / 100); storage.setItem('volume', input.value); }); progress.style.width = `${input.value}%`; return volume; } setTip(tip) { this.tipEl.innerText = tip; } } class PlayerBufferMonitor { constructor(dmPlayer) { this.dmPlayer = dmPlayer; this.intId = window.setInterval(() => { try { this.handler(); } catch (e) { console.error(e); } }, 200); this.reset(); } unload() { window.clearInterval(this.intId); } reset() { this.bufTime = 1; } get player() { return this.dmPlayer.player; } handler() { if (this.player) { const buffered = this.player.buffered; if (buffered.length === 0) return; const buf = buffered.end(buffered.length - 1) - this.player.currentTime; const state = this.dmPlayer.state; if (state.is(PlayerState.Playing)) { if (buf <= 1) { state.go(PlayerState.Buffering); this.dmPlayer.ui.notifyStateChange(); this.bufTime *= 2; if (this.bufTime > 8) { console.warn('网络不佳'); this.bufTime = 8; } } } else if (state.is(PlayerState.Buffering)) { if (buf > this.bufTime) { state.go(PlayerState.Playing); this.dmPlayer.player.currentTime -= 0.5; this.dmPlayer.ui.notifyStateChange(); } } } } } class DanmuPlayer { constructor(listener, ui) { this.inputing = false; this._src = ''; this.bufferMonitor = new PlayerBufferMonitor(this); this.state = new PlayerStateFSM(); const now = () => new Date().getTime(); let beginTime = 0; this.state .on(PlayerState.Stopped, () => { beginTime = 0; this.mgr.deferTime = 0; this.bufferMonitor.reset(); if (this.player) { this.player.unload(); this.player.detachMediaElement(); this.player = null; } }) .on(PlayerState.Paused, from => { beginTime = now(); this.player.pause(); }) .on(PlayerState.Playing, from => { if (beginTime !== 0) { this.mgr.deferTime += now() - beginTime; } this.player.play(); }) .on(PlayerState.Buffering, from => { beginTime = 0; this.player.pause(); }); this.initUI(); this.mgr = new DanmuManager(this.ui.dmLayout, this.state); this.listener = listener; } onVolumeChange(vol) { this.player.volume = vol; } onReload() { this.stop(); this.load(); } onSendDanmu(txt) { this.listener.onSendDanmu(txt); } onStop() { this.stop(); } onUnload() { this.bufferMonitor.unload(); } onTryPlay() { this.tryPlay(); } onMute(muted) { if (muted) { this.lastVolume = this.player.volume; this.player.volume = 0; } else { this.player.volume = this.lastVolume; } } onHideDanmu(hide) { this.mgr.hideDanmu = hide; } onStat(e) { this.ui.setTip(Math.round(e.speed * 10) / 10 + 'KB/s'); } load() { return __awaiter(this, void 0, void 0, function* () { this.src = yield this.listener.getSrc(); }); } createFlvjs() { const sourceConfig = { isLive: true, type: 'flv', url: this.src }; const playerConfig = { enableWorker: false, deferLoadAfterSourceOpen: true, stashInitialSize: 512 * 1024, enableStashBuffer: true, autoCleanupMinBackwardDuration: 20, autoCleanupMaxBackwardDuration: 40, autoCleanupSourceBuffer: true }; const player = flvjs.createPlayer(sourceConfig, playerConfig); player.on(flvjs.Events.ERROR, (e, t) => { console.error('播放器发生错误:' + e + ' - ' + t); player.unload(); }); player.on(flvjs.Events.STATISTICS_INFO, this.onStat.bind(this)); player.attachMediaElement(this.ui.video); player.load(); player.play(); return player; } stop() { this.state.go(PlayerState.Stopped); } set src(val) { this._src = val; this.stop(); let player = this.createFlvjs(); this.player = player; this.ui.initControls(); this.state.go(PlayerState.Playing); } get src() { return this._src; } initUI() { this.ui = new PlayerUI(this, this.state); } tryPlay() { if (this.state.is(PlayerState.Playing)) { try { this.ui.video.play(); } catch (e) { } } } fireDanmu(text, color, cls) { return this.mgr.fireDanmu(text, color, cls); } } class DanmuManager { constructor(danmuLayout, state) { this.danmuLayout = danmuLayout; this.state = state; this.pool = []; this.rows = []; this._deferTime = 0; this.maxRow = 10; this.baseTop = 10; this.deferId = null; this.deferQueue = []; this.hideDanmu = false; this.parsePic = (i) => i; const poolSize = 100; for (let i = 0; i < poolSize; i++) { let dm = document.createElement('div'); danmuLayout.appendChild(dm); this.pool.push({ el: dm, using: false }); } } get playing() { return this.state.is(PlayerState.Playing); } set deferTime(v) { this._deferTime = v; this.defering = v !== 0; } get deferTime() { return this._deferTime; } calcRect() { return this.danmuLayout.getBoundingClientRect(); } calcRow(width, duration) { let rect = this.calcRect(); const now = new Date().getTime(); const check = (idx) => { let row = this.rows[idx]; if (!row) return true; if (row.endTime <= now) { this.rows[idx] = null; return true; } else { const distance = rect.width + row.width; const percent = (now - row.beginTime) / row.duration; const left = rect.width - distance * percent; if (left + row.width >= rect.width) { return false; } const remainTime = row.endTime - now; const myDistance = rect.width + width; const leftX = rect.width - (myDistance) * (remainTime / duration); if (leftX < 0) { return false; } } return true; }; let i = 0; while (true) { if (check(i)) { this.rows[i] = { duration: duration, beginTime: now, endTime: now + duration, width: width }; return i % this.maxRow; } i++; } } doDefer() { if (this.deferQueue.length === 0) return; const top = this.deferQueue[0]; const now = new Date().getTime(); if (this.playing && ((top.oriTime + this.deferTime) <= now)) { top.run(); this.deferQueue.shift(); } } set defering(v) { if (this.deferId === null) { if (v) { this.deferId = window.setInterval(() => this.doDefer(), 100); } } else { if (v === false) { window.clearInterval(this.deferId); this.deferId = null; } } } fireDanmu(text, color, cls) { const fire = () => { let rect = this.calcRect(); const duration = rect.width * 7; let { el: dm } = this.pool.shift(); setTimeout(() => { dm.removeAttribute('style'); this.pool.push({ el: dm, using: false }); }, duration); dm.innerText = text; dm.innerHTML = this.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(() => { let dmRect = dm.getBoundingClientRect(); const row = this.calcRow(dmRect.width, duration); dm.style.top = `${this.baseTop + row * dmRect.height}px`; dm.style.transition = `transform ${duration / 1000}s linear`; dm.style.transform = `translateX(-${rect.width + dmRect.width}px)`; }, 0); }; const now = new Date().getTime(); if (!this.playing || this.deferTime > 0) { this.deferQueue.push({ oriTime: now, run: () => fire() }); return; } if (this.hideDanmu) return; if (this.pool.length == 0) return; fire(); } } const createMenu = (x, y) => { const wrap = document.createElement('div'); const 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 = () => document.body.removeChild(wrap); wrap.addEventListener('mousedown', event => { if (event.target === wrap) { document.body.removeChild(wrap); } }); wrap.addEventListener('contextmenu', event => event.preventDefault()); document.body.appendChild(wrap); return menu }; const addTextMenu = (menu, text, cb) => { const item = document.createElement('div'); item.className = 'menu-item'; item.innerText = text; menu.appendChild(item); item.addEventListener('click', () => { menu.close(); cb(); }); }; const addEleMenu = (menu, ele) => { const item = document.createElement('div'); item.className = 'menu-ele'; item.appendChild(ele); menu.appendChild(item); }; const addLabelMenu = (menu, label) => { const item = document.createElement('div'); item.className = 'menu-item'; item.innerText = label; menu.appendChild(item); }; const addDash = (menu) => { const item = document.createElement('div'); item.className = 'menu-dash'; menu.appendChild(item); }; function bindMenu (el, menuItems) { el.addEventListener('contextmenu', event => { const menu = createMenu(event.clientX, event.clientY); let items = menuItems; if (typeof items === 'function') items = items(); for (let item of items) { 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); } } const 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(); }); } function md5cycle(x, k) { var a = x[0], b = x[1], c = x[2], d = x[3]; a = ff(a, b, c, d, k[0], 7, -680876936); d = ff(d, a, b, c, k[1], 12, -389564586); c = ff(c, d, a, b, k[2], 17, 606105819); b = ff(b, c, d, a, k[3], 22, -1044525330); a = ff(a, b, c, d, k[4], 7, -176418897); d = ff(d, a, b, c, k[5], 12, 1200080426); c = ff(c, d, a, b, k[6], 17, -1473231341); b = ff(b, c, d, a, k[7], 22, -45705983); a = ff(a, b, c, d, k[8], 7, 1770035416); d = ff(d, a, b, c, k[9], 12, -1958414417); c = ff(c, d, a, b, k[10], 17, -42063); b = ff(b, c, d, a, k[11], 22, -1990404162); a = ff(a, b, c, d, k[12], 7, 1804603682); d = ff(d, a, b, c, k[13], 12, -40341101); c = ff(c, d, a, b, k[14], 17, -1502002290); b = ff(b, c, d, a, k[15], 22, 1236535329); a = gg(a, b, c, d, k[1], 5, -165796510); d = gg(d, a, b, c, k[6], 9, -1069501632); c = gg(c, d, a, b, k[11], 14, 643717713); b = gg(b, c, d, a, k[0], 20, -373897302); a = gg(a, b, c, d, k[5], 5, -701558691); d = gg(d, a, b, c, k[10], 9, 38016083); c = gg(c, d, a, b, k[15], 14, -660478335); b = gg(b, c, d, a, k[4], 20, -405537848); a = gg(a, b, c, d, k[9], 5, 568446438); d = gg(d, a, b, c, k[14], 9, -1019803690); c = gg(c, d, a, b, k[3], 14, -187363961); b = gg(b, c, d, a, k[8], 20, 1163531501); a = gg(a, b, c, d, k[13], 5, -1444681467); d = gg(d, a, b, c, k[2], 9, -51403784); c = gg(c, d, a, b, k[7], 14, 1735328473); b = gg(b, c, d, a, k[12], 20, -1926607734); a = hh(a, b, c, d, k[5], 4, -378558); d = hh(d, a, b, c, k[8], 11, -2022574463); c = hh(c, d, a, b, k[11], 16, 1839030562); b = hh(b, c, d, a, k[14], 23, -35309556); a = hh(a, b, c, d, k[1], 4, -1530992060); d = hh(d, a, b, c, k[4], 11, 1272893353); c = hh(c, d, a, b, k[7], 16, -155497632); b = hh(b, c, d, a, k[10], 23, -1094730640); a = hh(a, b, c, d, k[13], 4, 681279174); d = hh(d, a, b, c, k[0], 11, -358537222); c = hh(c, d, a, b, k[3], 16, -722521979); b = hh(b, c, d, a, k[6], 23, 76029189); a = hh(a, b, c, d, k[9], 4, -640364487); d = hh(d, a, b, c, k[12], 11, -421815835); c = hh(c, d, a, b, k[15], 16, 530742520); b = hh(b, c, d, a, k[2], 23, -995338651); a = ii(a, b, c, d, k[0], 6, -198630844); d = ii(d, a, b, c, k[7], 10, 1126891415); c = ii(c, d, a, b, k[14], 15, -1416354905); b = ii(b, c, d, a, k[5], 21, -57434055); a = ii(a, b, c, d, k[12], 6, 1700485571); d = ii(d, a, b, c, k[3], 10, -1894986606); c = ii(c, d, a, b, k[10], 15, -1051523); b = ii(b, c, d, a, k[1], 21, -2054922799); a = ii(a, b, c, d, k[8], 6, 1873313359); d = ii(d, a, b, c, k[15], 10, -30611744); c = ii(c, d, a, b, k[6], 15, -1560198380); b = ii(b, c, d, a, k[13], 21, 1309151649); a = ii(a, b, c, d, k[4], 6, -145523070); d = ii(d, a, b, c, k[11], 10, -1120210379); c = ii(c, d, a, b, k[2], 15, 718787259); b = ii(b, c, d, a, k[9], 21, -343485551); x[0] = add32(a, x[0]); x[1] = add32(b, x[1]); x[2] = add32(c, x[2]); x[3] = add32(d, x[3]); } function cmn(q, a, b, x, s, t) { a = add32(add32(a, q), add32(x, t)); return add32((a << s) | (a >>> (32 - s)), b); } function ff(a, b, c, d, x, s, t) { return cmn((b & c) | ((~b) & d), a, b, x, s, t); } function gg(a, b, c, d, x, s, t) { return cmn((b & d) | (c & (~d)), a, b, x, s, t); } function hh(a, b, c, d, x, s, t) { return cmn(b ^ c ^ d, a, b, x, s, t); } function ii(a, b, c, d, x, s, t) { return cmn(c ^ (b | (~d)), a, b, x, s, t); } function md51(s) { var txt = ''; var n = s.length, state = [1732584193, -271733879, -1732584194, 271733878], i; for (i = 64; i <= s.length; i += 64) { md5cycle(state, md5blk(s.substring(i - 64, i))); } s = s.substring(i - 64); var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; for (i = 0; i < s.length; i++) tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3); tail[i >> 2] |= 0x80 << ((i % 4) << 3); if (i > 55) { md5cycle(state, tail); for (i = 0; i < 16; i++) tail[i] = 0; } tail[14] = n * 8; md5cycle(state, tail); return state; } function md5blk(s) { var md5blks = [], i; for (i = 0; i < 64; i += 4) { md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) << 24); } return md5blks; } var hex_chr = '0123456789abcdef'.split(''); function rhex(n) { var s = '', j = 0; for (; j < 4; j++) s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F]; return s; } function hex(x) { return x.map(rhex).join(''); } function md5(s) { return hex(md51(s)); } var add32 = function (a, b) { return (a + b) & 0xFFFFFFFF; }; if (md5('hello') != '5d41402abc4b2a76b9719d911017c592') { add32 = function (x, y) { var lsw = (x & 0xFFFF) + (y & 0xFFFF), msw = (x >> 16) + (y >> 16) + (lsw >> 16); return (msw << 16) | (lsw & 0xFFFF); }; } class BaseSource { constructor() { this.onChange = () => null; } set url(v) { if (v === this._url) { this._url = v; return; } this.onChange(v); } get url() { return this._url; } } let m_signer = null; function getSourceURL(rid, cdn, rate) { return __awaiter(this, void 0, void 0, function* () { const tt = Math.round(new Date().getTime() / 60 / 1000); const did = md5(Math.random().toString()).toUpperCase(); if (m_signer === null) { throw new Error('Signer is not defined.'); } const sign = yield m_signer(rid, tt, did); let body = { 'cdn': cdn, 'rate': rate, 'ver': 'Douyu_h5_2017080201beta', 'tt': tt, 'did': did, 'sign': sign.sign, 'cptl': sign.cptl, 'ct': 'webh5' }; body = Object.keys(body).map(key => `${key}=${encodeURIComponent(body[key])}`).join('&'); const res = yield fetch(`https://www.douyu.com/lapi/live/getPlay/${rid}`, { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, body: body }); const videoInfo = yield res.json(); const baseUrl = videoInfo.data.rtmp_url; const livePath = videoInfo.data.rtmp_live; if (baseUrl && livePath) { const videoUrl = `${baseUrl}/${livePath}`; console.log('RoomId', rid, 'SourceURL:', videoUrl); return videoUrl; } else { throw new Error('未开播或获取失败'); } }); } function getSwfApi(rid) { return __awaiter(this, void 0, void 0, function* () { const API_KEY = 'bLFlashflowlad92'; const tt = Math.round(new Date().getTime() / 60 / 1000); const signContent = [rid, API_KEY, tt].join(''); const sign = md5(signContent); const res = yield fetch(`https://www.douyu.com/swf_api/room/${rid}?cdn=&nofan=yes&_t=${tt}&sign=${sign}`); const obj = yield res.json(); return yield obj.data; }); } class DouyuSource extends BaseSource { constructor(roomId, signer) { super(); m_signer = signer; this._cdn = 'ws'; this._rate = '0'; this.url = ''; this.roomId = roomId; this.swfApi = null; } set cdn(val) { this._cdn = val; this.getUrl(); } get cdn() { return this._cdn; } set rate(val) { this._rate = val; this.getUrl(); } get rate() { return this._rate; } get cdnsWithName() { if (this.swfApi) { return this.swfApi.cdnsWithName; } else { return [{ name: '主要线路', cdn: 'ws' }]; } } getUrl() { return __awaiter(this, void 0, void 0, function* () { if (!this.swfApi) { this.swfApi = yield getSwfApi(this.roomId); this._cdn = this.swfApi.cdns[0]; } let url = yield getSourceURL(this.roomId, this.cdn, this.rate); this.url = url; return url; }); } } class JSocket { static init() { return __awaiter(this, void 0, void 0, function* () { const src = 'https://imspace.nos-eastchina1.126.net/JSocket2.swf'; const flash = ['', '', '', '', '', '', '', '', '', ""].join(""); let div = document.createElement('div'); div.className = 'jsocket-cls'; document.body.appendChild(div); JSocket.el = div; div.innerHTML = flash; var api = document.querySelector('#jsocket'); console.log(div, api); JSocket.flashapi = api; if (JSocket.flashapi.newsocket) { return; } else { return new Promise((res, rej) => { const id = setTimeout(rej, 10 * 1000); JSocket.swfloadedcb = () => { clearTimeout(id); res(); }; }); } }); } static swfloaded() { JSocket.swfloadedcb(); } static connectHandler(socid) { JSocket.handlers[socid].connectHandler(); } static dataHandler(socid, data) { try { JSocket.handlers[socid].dataHandler(atob(data)); } catch (e) { console.error(e); } } static closeHandler(socid) { JSocket.handlers[socid].closeHandler(); } static errorHandler(socid, str) { JSocket.handlers[socid].errorHandler(str); } init(handlers, newsocketopt) { this.socid = JSocket.flashapi.newsocket(newsocketopt); JSocket.handlers[this.socid] = handlers; } connect(host, port) { JSocket.flashapi.connect(this.socid, host, port); } write(data) { JSocket.flashapi.write(this.socid, btoa(data)); } writeFlush(data) { JSocket.flashapi.writeFlush(this.socid, btoa(data)); } close() { JSocket.flashapi.close(this.socid); } flush() { JSocket.flashapi.flush(this.socid); } } JSocket.VERSION = '0.1'; JSocket.handlers = []; window.JSocket = JSocket; const getACF = (key) => { try { return new RegExp(`acf_${key}=(.*?);`).exec(document.cookie)[1]; } catch (e) { return ''; } }; function filterEnc(s) { s = s.toString(); s = s.replace(/@/g, '@A'); return s.replace(/\//g, '@S'); } function filterDec(s) { s = s.toString(); s = s.replace(/@S/g, '/'); return s.replace(/@A/g, '@'); } function douyuEncode(data) { return Object.keys(data).map(key => `${key}@=${filterEnc(data[key])}`).join('/') + '/'; } function douyuDecode(data) { let out = { type: '!!missing!!' }; try { data.split('/').filter(i => i.length > 2).forEach(i => { let e = i.split('@='); out[e[0]] = filterDec(e[1]); }); } catch (e) { console.error(e); console.log(data); } return out; } function ACJ(id, data) { if (typeof data == 'object') { data = douyuEncode(data); } try { window._ACJ_([id, data]); } catch (e) { console.error(id, data, e); } } class DouyuProtocol extends JSocket { constructor(listener) { super(); this.listener = listener; this.connectHandler = () => null; this.init(this, {}); this.buffer = ''; } connectAsync(host, port) { super.connect(host, port); return new Promise((res, rej) => { const prevConnHandler = this.connectHandler; const prevErrHandler = this.errorHandler; const recover = () => { this.connectHandler = prevConnHandler; this.errorHandler = prevErrHandler; }; this.connectHandler = () => { recover(); res(); }; this.errorHandler = () => { recover(); rej(); }; }); } dataHandler(data) { this.buffer += data; let buffer = this.buffer; while (buffer.length >= 4) { let size = u32(buffer.substr(0, 4)); if (buffer.length >= size) { let pkgStr = ''; try { pkgStr = ascii_to_utf8(buffer.substr(12, size - 8)); } catch (e) { console.log('deocde fail', escape(buffer.substr(12, size - 8))); } this.buffer = buffer = buffer.substr(size + 4); if (pkgStr.length === 0) continue; try { let pkg = douyuDecode(pkgStr); this.listener && this.listener.onPackage(pkg, pkgStr); } catch (e) { console.error('call map', e); } } else { break; } } } closeHandler() { console.error('lost connection'); this.listener && this.listener.onClose(); } errorHandler(err) { console.error(err); this.listener && this.listener.onError(err); } send(data) { let msg = douyuEncode(data) + '\0'; msg = utf8_to_ascii(msg); msg = p32(msg.length + 8) + p32(msg.length + 8) + p32(689) + msg; this.writeFlush(msg); } } function Type(type) { return (target, propertyKey, descriptor) => { if (!target.map) { target.map = {}; } target.map[type] = target[propertyKey]; }; } class DouyuBaseClient { constructor(roomId) { this.roomId = roomId; this.lastIP = null; this.lastPort = null; this.keepaliveId = null; this.redirect = {}; this.prot = new DouyuProtocol(this); } static getRoomArgs() { if (window._room_args) return window._room_args; if (window.room_args) { return window.room_args; } else { return window.$ROOM.args; } } reconnect() { return __awaiter(this, void 0, void 0, function* () { console.log('reconnect'); this.prot.listener = null; this.prot = new DouyuProtocol(this); try { yield this.connectAsync(this.lastIP, this.lastPort); } catch (e) { this.onError(); } }); } onClose() { setTimeout(() => this.reconnect(), 1000); } onError() { this.onClose(); } onPackage(pkg, pkgStr) { const type = pkg.type; if (this.redirect[type]) { ACJ(this.redirect[type], pkg); return; } if (this.map[type]) { this.map[type].call(this, pkg, pkgStr); return; } this.onDefault(pkg); } onDefault(pkg) { } send(pkg) { this.prot.send(pkg); } connectAsync(ip, port) { return __awaiter(this, void 0, void 0, function* () { this.lastIP = ip; this.lastPort = port; yield this.prot.connectAsync(ip, port); this.send(this.loginreq()); }); } keepalivePkg() { return { type: 'keeplive', tick: Math.round(new Date().getTime() / 1000).toString() }; } loginreq() { const rt = Math.round(new Date().getTime() / 1000); const devid = getACF('devid'); const username = getACF('username'); console.log('username', username, devid); return { type: 'loginreq', username: username, ct: 0, password: '', roomid: this.roomId, devid: devid, rt: rt, vk: md5(`${rt}r5*^5;}2#\${XF[h+;'./.Q'1;,-]f'p[${devid}`), ver: '20150929', aver: '2017012111', biz: getACF('biz'), stk: getACF('stk'), ltkid: getACF('ltkid') }; } startKeepalive() { this.send(this.keepalivePkg()); if (this.keepaliveId) { clearInterval(this.keepaliveId); } this.keepaliveId = setInterval(() => this.send(this.keepalivePkg()), 30 * 1000); } } let blacklist = []; function onChatMsg(data) { if (blacklist.indexOf(data.uid) !== -1) { console.log('black'); return; } try { postMessage('DANMU', data); } catch (e) { console.error('wtf', e); } ACJ('room_data_chat2', data); if (window.BarrageReturn) { window.BarrageReturn(douyuEncode(data)); } } class DouyuClient extends DouyuBaseClient { constructor(roomId, danmuClient) { super(roomId); this.danmuClient = danmuClient; this.redirect = { qtlr: 'room_data_tasklis', initcl: 'room_data_chatinit', memberinfores: 'room_data_info', ranklist: 'room_data_cqrank', rsm: 'room_data_brocast', qausrespond: 'data_rank_score', frank: 'room_data_handler', online_noble_list: 'room_data_handler', }; } reqOnlineGift(loginres) { return { type: 'reqog', uid: loginres.userid }; } chatmsg(data) { onChatMsg(data); } resog(data) { ACJ('room_data_chest', { lev: data.lv, lack_time: data.t, dl: data.dl }); } loginres(data) { console.log('loginres ms', data); this.uid = data.userid; this.send(this.reqOnlineGift(data)); this.startKeepalive(); ACJ('room_data_login', data); ACJ('room_data_getdid', { devid: getACF('devid') }); } keeplive(data, rawString) { ACJ('room_data_userc', data.uc); ACJ('room_data_tbredpacket', rawString); } setmsggroup(data) { console.log('joingroup', data); this.danmuClient.send({ type: 'joingroup', rid: data.rid, gid: data.gid }); } onDefault(data) { ACJ('room_data_handler', data); console.log('ms', data); } } __decorate([ Type('chatmsg') ], DouyuClient.prototype, "chatmsg", null); __decorate([ Type('resog') ], DouyuClient.prototype, "resog", null); __decorate([ Type('loginres') ], DouyuClient.prototype, "loginres", null); __decorate([ Type('keeplive') ], DouyuClient.prototype, "keeplive", null); __decorate([ Type('setmsggroup') ], DouyuClient.prototype, "setmsggroup", null); class DouyuDanmuClient extends DouyuBaseClient { constructor(roomId) { super(roomId); this.redirect = { chatres: 'room_data_chat2', initcl: 'room_data_chatinit', dgb: 'room_data_giftbat1', dgn: 'room_data_giftbat1', spbc: 'room_data_giftbat1', uenter: 'room_data_nstip2', upgrade: 'room_data_ulgrow', newblackres: 'room_data_sys', ranklist: 'room_data_cqrank', rankup: 'room_data_ulgrow', gift_title: 'room_data_schat', rss: 'room_data_state', srres: 'room_data_wbsharesuc', onlinegift: 'room_data_olyw', gpbc: 'room_data_handler', synexp: 'room_data_handler', frank: 'room_data_handler', ggbb: 'room_data_sabonusget', online_noble_list: 'room_data_handler', }; } chatmsg(pkg) { onChatMsg(pkg); } loginres(data) { console.log('loginres dm', data); this.startKeepalive(); } onDefault(data) { ACJ('room_data_handler', data); console.log('dm', data); } } __decorate([ Type('chatmsg') ], DouyuDanmuClient.prototype, "chatmsg", null); __decorate([ Type('loginres') ], DouyuDanmuClient.prototype, "loginres", null); const runtime = { sendMessage(message) { return chrome.runtime.sendMessage(message); }, connect(connectInfo) { return chrome.runtime.connect(connectInfo); } }; var SignerState; (function (SignerState) { SignerState[SignerState["None"] = 0] = "None"; SignerState[SignerState["Loaded"] = 1] = "Loaded"; SignerState[SignerState["Ready"] = 2] = "Ready"; SignerState[SignerState["Timeout"] = 3] = "Timeout"; })(SignerState || (SignerState = {})); function wrapPort(port) { let curMethod = ''; let curResolve = null; let curReject = null; let stack = new Error().stack; port.onMessage.addListener((msg) => { if (msg.method === curMethod) { curResolve(msg.args[0]); } else { curReject('wtf'); console.error('wtf?'); } }); return function (method, ...args) { return new Promise((resolve, reject) => { curMethod = method; curResolve = resolve; curReject = reject; port.postMessage({ method: method, args: args }); }); }; } let Signer; { const DB_NAME = 'shared-worker-signer'; const DB_VERSION = 1; const DB_STORE_NAME = 'cache'; class ManualCache { initDB() { return new Promise((resolve, reject) => { const that = this; const req = indexedDB.open(DB_NAME, DB_VERSION); req.onsuccess = function (evt) { that.db = this.result; resolve(that); }; req.onerror = function (evt) { that.errCode = evt.target.errorCode; reject(that.errCode); }; req.onupgradeneeded = function (evt) { const store = evt.currentTarget.result.createObjectStore(DB_STORE_NAME); }; }); } getArrayBuffer(key, url) { return this.getFile(key).then((file) => { if (file && file.url === url) { return file.data; } else { return this.fetchAndSave(key, url); } }); } putFile(key, url, data) { return new Promise((resolve, reject) => { const tx = this.db.transaction(DB_STORE_NAME, 'readwrite'); const store = tx.objectStore(DB_STORE_NAME); const req = store.put({ data: data, url: url }, key); req.onsuccess = function (evt) { resolve(); }; req.onerror = function () { reject(this.error); }; }); } getFile(key) { return new Promise((resolve, reject) => { const tx = this.db.transaction(DB_STORE_NAME, 'readonly'); const store = tx.objectStore(DB_STORE_NAME); const req = store.get(key); req.onsuccess = function ({ target }) { resolve(target.result); }; req.onerror = function () { reject(this.error); }; }); } fetchAndSave(key, url) { return fetch(url).then(res => res.arrayBuffer()).then(buffer => this.putFile(key, url, buffer).then(() => buffer)); } } const manualCache = new ManualCache(); const signerURL = `data:text/javascript,importScripts('https://imspace.nos-eastchina1.126.net/shared-worker-signer_v0.0.3.js')`; class SharedWorkerSigner { static _clean() { this._resolve = null; this._reject = null; } static onMessage({ data }) { console.log('onMessage', data); if (data.type === 'query') { if (data.data === true) { this._resolve(); this._clean(); this._stopQuery = true; } else { setTimeout(() => this.query(), 100); } } else if (data.type === 'sign') { this._resolve(data.data); this._clean(); } else if (data.type === 'error') { this._reject(data.data); this._clean(); } else if (data.type === 'getArrayBuffer') { manualCache.getArrayBuffer(data.key, data.url).then(buffer => { this._worker.port.postMessage({ type: 'getArrayBuffer', data: buffer }, [buffer]); }).catch((e) => { this._worker.port.postMessage({ type: 'getArrayBuffer', err: e }); }); } } static get state() { return this._state; } static sign(rid, tt, did) { return __awaiter(this, void 0, void 0, function* () { return new Promise((resolve, reject) => { this._worker.port.postMessage({ type: 'sign', args: [rid, tt, did] }); this._resolve = resolve; this._reject = reject; }); }); } static init() { if (this._inited) { return Promise.resolve(); } return manualCache.initDB().then(() => new Promise((resolve, reject) => { this._resolve = resolve; this._reject = reject; const worker = new SharedWorker(signerURL); this._worker = worker; worker.port.onmessage = e => this.onMessage(e); window.setTimeout(() => this.query(), 500); window.setTimeout(() => { if (this.state !== SignerState.Ready) { this._state = SignerState.Timeout; } if (this._stopQuery === false) { this._stopQuery = true; } }, 15 * 1000); })); } static query() { if (!this._stopQuery) { this._worker.port.postMessage({ type: 'query' }); } } } SharedWorkerSigner._inited = false; SharedWorkerSigner._state = SignerState.None; SharedWorkerSigner._stopQuery = false; Signer = SharedWorkerSigner; } let dialog = null; function getDialog(title, content, qrcodes) { if (dialog) { return dialog; } dialog = document.createElement('div'); dialog.className = 'donate-dialog'; const wrap = document.createElement('div'); wrap.className = 'donate-wrap'; const titleEl = document.createElement('h3'); titleEl.className = 'donate-title'; titleEl.innerText = title; const contentEl = document.createElement('div'); contentEl.className = 'donate-content'; contentEl.innerText = content; const qrcodeEl = document.createElement('div'); qrcodeEl.className = 'donate-qrcode-bar'; for (let i of qrcodes) { const qrcodeBox = document.createElement('div'); qrcodeBox.className = 'donate-qrcode-box'; const qrcode = document.createElement('img'); qrcode.className = 'donate-qrcode-img'; qrcode.src = i.src; const qrcodeDesc = document.createElement('div'); qrcodeDesc.className = 'donate-qrcode-desc'; qrcodeDesc.innerText = i.desc; qrcodeBox.appendChild(qrcode); qrcodeBox.appendChild(qrcodeDesc); qrcodeEl.appendChild(qrcodeBox); } const closeEl = document.createElement('div'); closeEl.className = 'donate-close-btn'; const close = () => { dialog.style.display = 'none'; }; closeEl.addEventListener('click', close); wrap.appendChild(titleEl); wrap.appendChild(contentEl); wrap.appendChild(qrcodeEl); wrap.appendChild(closeEl); dialog.appendChild(wrap); dialog.style.display = 'none'; return dialog; } const onload = () => { if (window.__space_inject) { const { script, css } = window.__space_inject; addCss(createBlobURL(css, 'text/css')); addScript(createBlobURL(script, 'text/javascript')); window.__space_inject = null; } else { addCss('dist/danmu.css'); addScript('dist/douyuInject.js'); } const uid = getACF('uid'); flvjs.LoggingControl.forceGlobalTag = true; flvjs.LoggingControl.enableAll = true; class DouyuPlayerUI extends PlayerUI { constructor(listener, state) { super(listener, state); this.douyuFullpage = false; this.wrap.style.position = 'inherit'; this.wrap.style.zIndex = 'inherit'; } _enterFullScreen() { this.wrap.style.position = ''; this.wrap.style.zIndex = ''; super._enterFullScreen(); } _exitFullScreen() { this.wrap.style.position = 'inherit'; this.wrap.style.zIndex = 'inherit'; super._exitFullScreen(); } _enterFullPage() { this.wrap.setAttribute('fullpage', ''); this.el.style.border = '0'; if (!this.douyuFullpage) { this.douyuFullpage = true; postMessage('ACJ', { id: 'room_bus_pagescr' }); } } _exitFullPage() { this.wrap.removeAttribute('fullpage'); this.el.style.border = ''; if (this.douyuFullpage) { this.douyuFullpage = false; postMessage('ACJ', { id: 'room_bus_pagescr' }); } } } class DouyuDanmuPlayer extends DanmuPlayer { constructor(roomId) { const source = new DouyuSource(roomId, (rid, tt, did) => __awaiter(this, void 0, void 0, function* () { let sign = yield Signer.sign(roomId, tt, did); return sign; })); source.onChange = videoUrl => { this.src = videoUrl; }; super({ getSrc: () => source.getUrl(), onSendDanmu(txt) { window.postMessage({ type: "SENDANMU", data: txt }, "*"); } }); this.source = source; } initUI() { this.ui = new DouyuPlayerUI(this, this.state); } onDanmuPkg(pkg) { const getColor = (c) => ['#ff0000', '#1e87f0', '#7ac84b', '#ff7f00', '#9b39f4', '#ff69b4'][c - 1]; if (pkg.txt.length > 0) { let cls = []; let color = getColor(pkg.col) || '#ffffff'; if (pkg.uid === uid) cls.push('danmu-self'); this.fireDanmu(pkg.txt, color, cls); } } } const makeMenu = (player, source) => { const cdnMenu = () => source.cdnsWithName.map((i) => { let suffix = ''; if (i.cdn == source.cdn) suffix = ' √'; return { text: i.name + suffix, cb() { source.cdn = i.cdn; } }; }); const rateMenu = () => { const rates = [{ text: '超清', rate: '0' }, { text: '高清', rate: '2' }, { text: '普清', rate: '1' }]; return rates.map(i => { let suffix = ''; if (i.rate == source.rate) suffix = ' √'; return { text: i.text + suffix, cb() { source.rate = i.rate; } }; }); }; const transparentMenu = () => { const opts = [{ text: '0%', transparent: 0 }, { text: '25%', transparent: 25 }, { text: '50%', transparent: 50 }]; return [{ label: '弹幕透明度:' }].concat(opts.map(i => { let suffix = ''; if (i.transparent == player.ui.transparent) suffix = ' √'; return { text: i.text + suffix, cb() { player.ui.transparent = i.transparent; }, label: null }; })); }; let mGetURL; { mGetURL = file => 'https://imspace.nos-eastchina1.126.net/img/' + file; } const dialog = getDialog('捐赠', '你的支持是我最大的动力.', [{ src: mGetURL('alipay.png'), desc: '支付宝' }, { src: mGetURL('wechat.png'), desc: '微信' }]); const donate = () => { return [{ text: '捐赠', cb() { document.body.appendChild(dialog); dialog.style.display = 'flex'; } }]; }; const dash = {}; bindMenu(player.ui.video, () => [].concat(cdnMenu(), dash, rateMenu(), dash, transparentMenu(), dash, donate())); }; const loadVideo = (roomId, replace) => { const danmuPlayer = new DouyuDanmuPlayer(roomId); danmuPlayer.mgr.parsePic = s => s.replace(/\[emot:dy(.*?)\]/g, (_, i) => ``); replace(danmuPlayer.ui.el); makeMenu(danmuPlayer, danmuPlayer.source); window.danmu = danmuPlayer; return danmuPlayer.source.getUrl().then(() => danmuPlayer); }; let danmuPlayer = null; let signerLoaded = new DelayNotify(false); Signer.init().then(() => true).catch(() => false).then((data) => { console.log('SIGNER_READY', data); signerLoaded.notify(data); }); onMessage('DANMU', data => { danmuPlayer && danmuPlayer.onDanmuPkg(data); }); onMessage('VIDEOID', (data) => __awaiter(window, void 0, void 0, function* () { console.log('onVideoId', data); const roomId = data.roomId; setBgListener((req) => __awaiter(this, void 0, void 0, function* () { switch (req.type) { case 'toggle': let setting = yield getSetting(); const id = setting.blacklist.indexOf(roomId); if (id === -1) { setting.blacklist.push(roomId); } else { setting.blacklist.splice(id, 1); } yield setSetting(setting); location.reload(); } })); console.log('wait signer'); if (!(yield signerLoaded.wait())) { console.warn('加载签名程序失败, 无法获取视频地址'); return; } console.log('start replace'); try { const setting = yield getSetting(); if (setting.blacklist.indexOf(roomId) !== -1) { if (runtime.sendMessage) { runtime.sendMessage({ type: 'disable' }); } return; } } catch (e) { console.warn(e); } let ctr = document.querySelector(`#${data.id}`); yield postMessage('BEGINAPI', { roomId }); danmuPlayer = yield loadVideo(roomId, el => { ctr.parentNode.replaceChild(el, ctr); }); })); }; onload(); })));