安卓uc斗鱼html5,斗鱼HTML5播放器

// ==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/*

// ==/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) => `dy%24%7Bi%7D.png?v=20161103`);

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

})));

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
PotPlayer是一款功能强大的媒体播放器,它支持多种格式的音视频文件播放。直播是一家知名的游戏直播平台,用户可以在这里观看游戏、电竞等各种类型的直播内容。通过PotPlayer,在电脑上观看直播是非常方便的。 首先,我们需要在PotPlayer中找到直播的URL地址或者是直播的直播间号码。我们可以在直播的官方网站或者是官方手机应用上找到这些信息。将直播的URL地址或直播间号复制后,在PotPlayer的播放界面中点击“打开”或者使用快捷键“CTRL+U”,将URL地址或直播间号码粘贴到弹出的窗口中。 接下来,PotPlayer会自动加载直播的视频流,并且开始播放。我们可以通过PotPlayer的播放控制按钮来进行播放、暂停、调节音量等操作。同时,PotPlayer还提供了全屏播放的功能,可以将直播的内容以更大的画面展示出来,提供更好的视觉体验。 此外,PotPlayer还支持弹幕功能。直播的弹幕是观众即时发送的文字信息,这些弹幕会在视频画面上滚动显示。通过PotPlayer,我们可以在直播的播放界面上打开弹幕功能,及时看到其他观众发送的弹幕内容,增加互动性和娱乐性。 总的来说,通过PotPlayer观看直播,我们可以在电脑上享受到高质量的直播内容,提供更好的观看体验。PotPlayer的强大功能和易用性,使得我们能够更方便地选择和播放直播的内容,并与其他观众进行互动。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值