Greasy Fork is available in English.

斗鱼HTML5播放器

基于 flv.js 的斗鱼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 = ['<object type=\"application/x-shockwave-flash\" ', 'id=\"jsocket\" ', 'name=\"jsocket\" ', 'align=\"middle\" ', 'allowscriptaccess=\"always\" ', 'allowfullscreen=\"true\" ', 'allowfullscreeninteractive=\"true\" ', 'wmode=\"transparent\" ', 'data=\"' + src + '\" ', 'width=\"100%\" ', 'height=\"100%\">', '<param name=\"src\" value=\"' + src + '\">', '<param name=\"quality\" value=\"high\">', '<param name=\"bgcolor\" value=\"#fff\">', '<param name=\"allowscriptaccess\" value=\"always\">', '<param name=\"allowfullscreen\" value=\"true\">', '<param name=\"wmode\" value=\"transparent\">', '<param name=\"allowFullScreenInteractive\" value=\"true\">', '<param name=\"flashvars\" value=\"\">', \"</object>\"].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(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI%2FPg0KPCFET0NUWVBFIHN2ZyBQVUJMSUMgIi0vL1czQy8vRFREIFNWRyAxLjEvL0VOIiANCiJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiPg0KDQo8c3ZnIHdpZHRoPSIyMCIgaGVpZ2h0PSIyMCIgdmVyc2lvbj0iMS4xIg0KeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBzdHlsZT0ic3Ryb2tlOiAjNWE1YTVhOyBmaWxsOnRyYW5zcGFyZW50OyBzdHJva2Utd2lkdGg6MiI%2BDQogIDxwYXRoIGQ9Ik05LDQgTDUsOCBMMiw4IEwyLDEyIEw1LDEyIEw5LDE2IFoiIHN0eWxlPSJmaWxsOiAjNWE1YTVhIiAvPg0KICA8cGF0aCBkPSJNMTIsMTIgQTMsMyAwIDAgMCAxMiw4IFoiIHN0eWxlPSJmaWxsOiAjNWE1YTVhIiAvPg0KICA8cGF0aCBkPSJNMTIsMiBBIDksOSAwIDAgMSAxMiwxOCIgc3R5bGU9InN0cm9rZS13aWR0aDoyIi8%2BDQo8L3N2Zz4%3D) no-repeat center;\n"+
"}\n"+
".danmu-wrap .danmu-ctrl .danmu-mute[muted] {\n"+
"  background: url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI%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(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI%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(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI%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 = ['<object type="application/x-shockwave-flash" ', 'id="jsocket" ', 'name="jsocket" ', 'align="middle" ', 'allowscriptaccess="always" ', 'allowfullscreen="true" ', 'allowfullscreeninteractive="true" ', 'wmode="transparent" ', 'data="' + src + '" ', 'width="100%" ', 'height="100%">', '<param name="src" value="' + src + '">', '<param name="quality" value="high">', '<param name="bgcolor" value="#fff">', '<param name="allowscriptaccess" value="always">', '<param name="allowfullscreen" value="true">', '<param name="wmode" value="transparent">', '<param name="allowFullScreenInteractive" value="true">', '<param name="flashvars" value="">', "</object>"].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) => `<img style="height:1em" src="https://shark.douyucdn.cn/app/douyu/res/page/room-normal/face/dy${i}.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();

})));