agefans Enhance

增强agefans播放功能,实现自动换集、无缝换集、画中画、历史记录、断点续播、弹幕等功能

目前为 2022-05-22 提交的版本。查看 最新版本

// ==UserScript==
// @name         agefans Enhance
// @namespace    https://github.com/IronKinoko/agefans-enhance
// @icon         https://www.agemys.com/favicon.ico
// @version      1.24.0
// @description  增强agefans播放功能,实现自动换集、无缝换集、画中画、历史记录、断点续播、弹幕等功能
// @author       IronKinoko
// @include      https://www.age.tv/*
// @include      https://www.agefans.*
// @include      https://www.agemys.*
// @include      *://*.yhdm.so/*
// @include      *://*.yinghuacd.com/*
// @include      https://www.yhdmp.cc/vp/*
// @include      http://www.imomoe.live/player/*
// @include      http://www.88dmw.com/*
// @include      https://new-ani.me/*
// @include      https://bangumi.online/*
// @include      https://danmu.4dm.cc/m3u8.php*
// @include      http*://www.ntyou.*
// @include      https://www.dm233.*
// @include      https://www.olevod.com*
// @include      https://www.bimiacg4.net*
// @run-at       document-body
// @require      https://unpkg.com/[email protected]/dist/jquery.min.js
// @require      https://unpkg.com/[email protected]/dist/plyr.min.js
// @require      https://unpkg.com/[email protected]/dist/hls.min.js
// @require      https://unpkg.com/@ironkinoko/[email protected]/dist/danmaku.umd.js
// @resource     plyrCSS https://unpkg.com/[email protected]/dist/plyr.css
// @grant        GM_getResourceText
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_xmlhttpRequest
// @connect      api.acplay.net
// @connect      chinacloudsites.cn
// @license      MIT
// ==/UserScript==

/**
 * 权限声明:
 * 1. GM_xmlhttpRequest
 *    脚本会请求有限的网络权限。仅用于访问弹幕查询功能需要链接到的 api.acplay.net 与 chinacloudsites.cn 第三方域名
 *    你可以从 脚本编辑/设置/XHR安全 中管理网络权限
 *
 * 2. GM_getResourceText, GM_addStyle
 *    获取播放器样式文件,用于播放器样式渲染
 *
 * 3. GM_getValue, GM_setValue
 *    脚本会使用本地存储功能,用于在不同页面间保存“播放器配置”与“agefans 历史浏览记录”。
 *
 * 4. @include
 *    脚本还匹配了 agefans 以外的一些链接,用于提供相同视频资源搜索功能
 */

(function() {
    let plyrCSS = GM_getResourceText('plyrCSS')  
    GM_addStyle(plyrCSS)
})();

(function (Hls, Plyr, Danmaku) {
  'use strict';

  function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }

  var Hls__default = /*#__PURE__*/_interopDefaultLegacy(Hls);
  var Plyr__default = /*#__PURE__*/_interopDefaultLegacy(Plyr);
  var Danmaku__default = /*#__PURE__*/_interopDefaultLegacy(Danmaku);

  var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}}

  var css$f = ":root {\n  --k-player-background-highlight: rgba(95, 95, 95, 0.65);\n  --k-player-background: rgba(0, 0, 0, 0.65);\n  --k-player-color: white;\n  --k-player-primary-color: #00b3ff;\n}\n\n.k-menu {\n  list-style: none;\n  margin: 0;\n  padding: 0;\n  border-radius: 4px;\n  overflow: hidden;\n}\n.k-menu-item {\n  padding: 0 16px;\n  line-height: 36px;\n  height: 36px;\n  cursor: pointer;\n  width: 100%;\n  white-space: nowrap;\n  color: white;\n  transition: all 0.3s;\n  text-align: center;\n}\n.k-menu-item:hover {\n  background: var(--k-player-background-highlight);\n}\n\n.k-menu-item.k-menu-active {\n  color: var(--k-player-primary-color);\n}\n\n.k-settings-list {\n  margin: 0;\n  padding: 8px;\n  text-align: left;\n}\n.k-settings-item {\n  width: 100%;\n  white-space: nowrap;\n  color: white;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n}\n.k-settings-list > .k-settings-item + .k-settings-item {\n  margin-top: 8px;\n}";
  n(css$f,{});

  function createTest(target) {
      return (test) => typeof test === 'string'
          ? target.includes(test) || test === '*'
          : test.test(target);
  }
  class Runtime {
      constructor() {
          this.list = [];
      }
      register(item) {
          this.list.push(item);
      }
      async getSearchActions() {
          const isInIframe = parent !== self;
          const searchs = this.list
              .map((o) => o.search)
              .filter(Boolean)
              .filter((o) => !(isInIframe && o.disabledInIframe));
          const register = this.getActiveRegister();
          const info = await this.getCurrentVideoNameAndEpisode();
          if (!(info === null || info === void 0 ? void 0 : info.name))
              return [];
          let name = info.name;
          return searchs
              .filter((search) => search !== register.search && search.search)
              .map((search) => ({
              name: search.name,
              search: () => {
                  const url = search.search(encodeURIComponent(name));
                  if (!url)
                      return;
                  if (isInIframe)
                      parent.postMessage({ key: 'openLink', url }, '*');
                  else
                      window.open(url);
              },
          }));
      }
      async getCurrentVideoNameAndEpisode() {
          var _a, _b, _c;
          const register = this.getActiveRegister();
          if (!((_a = register.search) === null || _a === void 0 ? void 0 : _a.getSearchName))
              return;
          let rawName = await register.search.getSearchName();
          let episode = (await ((_c = (_b = register.search).getEpisode) === null || _c === void 0 ? void 0 : _c.call(_b))) || '';
          if (!rawName)
              return;
          let name = rawName
              .replace(/第.季/, '')
              .replace(/[<>《》''‘’""“”\[\]]/g, '')
              .trim();
          episode = episode.replace(/[第集话]/g, '').replace(/^0+/, '');
          return { name, rawName, episode };
      }
      getActiveRegister() {
          const registers = this.list.filter(({ domains }) => domains.some(createTest(location.origin)));
          if (registers.length !== 1)
              throw new Error('激活的域名应该就一个');
          return registers[0];
      }
      getActiveOpts() {
          const register = this.getActiveRegister();
          return register.opts.filter(({ test }) => {
              const testArr = Array.isArray(test) ? test : [test];
              return testArr.some(createTest(location.pathname + location.search));
          });
      }
      run() {
          let setupList = [];
          let runList = [];
          const opts = this.getActiveOpts();
          opts.forEach(({ run, runInIframe, setup }) => {
              let needRun = runInIframe ? parent !== self : parent === self;
              if (needRun) {
                  setup && setupList.push(setup);
                  runList.push(run);
              }
          });
          setupList.forEach((setup) => setup());
          window.addEventListener('DOMContentLoaded', () => {
              runList.forEach((run) => run());
          });
      }
  }
  const runtime = new Runtime();

  var css$e = ".agefans-wrapper .loginout a {\n  cursor: pointer;\n  text-decoration: underline;\n}\n.agefans-wrapper .loginout a + a {\n  margin-left: 8px;\n}\n.agefans-wrapper .nav_button {\n  cursor: pointer;\n}\n.agefans-wrapper .res_links {\n  word-break: break-all;\n  word-wrap: break-word;\n}\n\n@media (max-width: 480px) {\n  .nav_button:nth-child(n+6) {\n    display: inline-block;\n  }\n\n  #nav {\n    position: relative;\n    overflow-x: auto;\n    white-space: nowrap;\n    height: 91px;\n  }\n  #nav::-webkit-scrollbar {\n    display: none;\n  }\n  #nav .nav_button {\n    white-space: nowrap;\n  }\n\n  #top_search_from {\n    width: calc(100% - 16px);\n    float: left;\n    margin-top: 10px;\n    position: sticky;\n    left: 8px;\n    margin: 8px;\n  }\n\n  #new_tip1 {\n    margin-top: 10px !important;\n  }\n}";
  n(css$e,{});

  var css$d = ".agefans-wrapper .page-preview-trigger .page-preview {\n  position: fixed;\n  pointer-events: none;\n  background-color: rgb(32, 32, 32);\n  border: 1px solid rgb(64, 64, 65);\n  z-index: 1000;\n  display: flex;\n  border-radius: 4px;\n  overflow: hidden;\n}\n.agefans-wrapper .page-preview-trigger .page-preview-center {\n  display: flex;\n  justify-content: center;\n  align-items: center;\n}\n.agefans-wrapper .page-preview-trigger .page-preview .baseblock2 {\n  border: none;\n  border-left: 1px solid #404041;\n  border-radius: 0;\n}\n.agefans-wrapper .page-preview-trigger .blocktitle.detail_title1 {\n  color: #e0e0e0;\n  border-bottom: 1px solid #404041;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_tag {\n  display: inline-block;\n  color: #808081;\n  min-width: 5em;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_value {\n  color: #e0e0e0;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_show_full {\n  display: none;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_kv {\n  min-width: 200px;\n  max-width: 256px;\n  display: inline-block;\n  margin: 3px 0px;\n  word-break: break-all;\n  word-wrap: break-word;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_desc_pre {\n  font-size: 15px;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_desc_pre * {\n  color: #e0e0e0;\n}\n.agefans-wrapper .page-preview-trigger .detail_imform_name {\n  margin: 0px;\n  color: #d0e0f0;\n  font-size: 1.2em;\n  font-weight: bold;\n  display: inline-block;\n}";
  n(css$d,{});

  function createStorage$1(storage) {
      function getItem(key, defaultValue) {
          try {
              const value = storage.getItem(key);
              if (value)
                  return JSON.parse(value);
              return defaultValue;
          }
          catch (error) {
              return defaultValue;
          }
      }
      return {
          getItem,
          setItem(key, value) {
              storage.setItem(key, JSON.stringify(value));
          },
          removeItem: storage.removeItem.bind(storage),
          clear: storage.clear.bind(storage),
      };
  }
  const session = createStorage$1(window.sessionStorage);
  const local = createStorage$1(window.localStorage);
  const gm = {
      getItem: GM_getValue,
      setItem: GM_setValue,
  };

  function parseToURL(url, count = 0) {
      if (count > 4)
          throw new Error('url解析失败 ' + url);
      try {
          url = new URL(url);
      }
      catch (error) {
          url = decodeURIComponent(url);
          url = parseToURL(url, ++count);
      }
      return url.toString();
  }

  function ageBlock(params) {
      const { title, content } = params;
      return `<div class="baseblock">
  <div class="blockcontent">
    <div class="baseblock2">
      <div class="blocktitle">${title}</div>
      <div class="blockcontent">${content}</div>
    </div>
  </div>
</div>`;
  }

  function set(name, value, _in_days = 1) {
      var Days = _in_days;
      var exp = new Date();
      exp.setTime(exp.getTime() + Days * 24 * 60 * 60 * 1000);
      document.cookie =
          name +
              '=' +
              escape(String(value)) +
              ';expires=' +
              exp.toUTCString() +
              ';path=/';
  }
  function get(name) {
      let reg = new RegExp('(^| )' + name + '=([^;]*)(;|$)');
      let arr = document.cookie.match(reg);
      if (arr) {
          return decodeURIComponent(arr[2]);
      }
      else {
          return null;
      }
  }
  const Cookie = {
      get,
      set,
      remove: function (name) {
          set(name, '', 0);
      },
  };

  /**
   * agefans 安全机制:
   * 1. 从服务端获取cookie `t1` `k1`
   * 2. 本地根据规则生成cookie `t2` `k2`
   * 3. 获取链接时候生成cookie `fa_t` `fa_c`
   *
   * t1 t2 fa_t 均为时间,相差太多就报错超时
   * k1 k2 类似密钥
   * fa_c 不重要
   */
  /**
   * 获取视频链接的请求地址
   */
  function getPlayUrl(_url) {
      const _rand = Math.random();
      var _getplay_url = _url.replace(/.*\/play\/(\d+?)\?playid=(\d+)_(\d+).*/, '/_getplay?aid=$1&playindex=$2&epindex=$3') +
          '&r=' +
          _rand;
      /**
       * fa_t 取当前时间
       * fa_c 1-9之间随便取 固定1就行
       */
      Cookie.set('fa_t', Date.now(), 1);
      Cookie.set('fa_c', 1, 1);
      return _getplay_url;
  }
  /**
   * 因为agefans的安全策略,需要刷新下cookie才能正常访问
   *
   * 这个方法实现了 t1 k1 t2 k2 全部刷新
   */
  function updateCookie(href) {
      href = href ? location.origin + href : location.href;
      return new Promise((resolve, reject) => {
          var _a, _b, _c;
          const doneFn = () => {
              resolve();
              dom.remove();
          };
          // DOMContentLoaded is faster than load
          const dom = document.createElement('iframe');
          dom.style.display = 'none';
          dom.src = href;
          document.body.append(dom);
          (_a = dom.contentWindow) === null || _a === void 0 ? void 0 : _a.addEventListener('DOMContentLoaded', doneFn);
          (_b = dom.contentWindow) === null || _b === void 0 ? void 0 : _b.addEventListener('load', doneFn);
          (_c = dom.contentWindow) === null || _c === void 0 ? void 0 : _c.addEventListener('error', reject);
      });
  }

  const LOCAL_PLAY_URL_KEY = 'play-url-key';
  function insertBTSites() {
      const title = $('#detailname a').text();
      const encodedTitle = encodeURIComponent(title);
      const sites = [
          {
              title: '蜜柑计划',
              url: `https://mikanani.me/Home/Search?searchstr=${encodedTitle}`,
          },
      ];
      $(ageBlock({
          title: '种子资源:',
          content: sites
              .map((site) => `<a href="${site.url}" rel="noreferrer" target="_blank" class="res_links_a">${site.title}</a>`)
              .join(''),
      })).insertAfter('.baseblock:contains(网盘资源)');
  }
  const loadingIcon = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" style="margin-right:4px;" width="1em" height="1em" viewBox="0 0 100 100" preserveAspectRatio="xMidYMid">
  <circle cx="50" cy="50" fill="none" stroke="#5699d2" stroke-width="10" r="40" stroke-dasharray="164.93361431346415 56.97787143782138">
    <animateTransform attributeName="transform" type="rotate" repeatCount="indefinite" dur="0.6s" values="0 50 50;360 50 50" keyTimes="0;1"></animateTransform>
  </circle>
</svg>`;
  function getLocal(href) {
      const map = session.getItem(LOCAL_PLAY_URL_KEY, {});
      if (href) {
          const item = map[href];
          if (!item)
              return null;
          return item.url;
      }
      return map;
  }
  function saveLocal(href, url) {
      const map = getLocal();
      map[href] = { url };
      session.setItem(LOCAL_PLAY_URL_KEY, map);
  }
  function removeLocal(href) {
      const map = getLocal();
      delete map[href];
      session.setItem(LOCAL_PLAY_URL_KEY, map);
  }
  class AGEfansError extends Error {
      constructor(message) {
          super(message);
          this.name = 'AGEfans Enhance Exception';
      }
  }
  async function getVurl(href) {
      const res = await fetch(getPlayUrl(href), {
          referrerPolicy: 'strict-origin-when-cross-origin',
      });
      const text = await res.text();
      if (text.includes('ipchk')) {
          throw new AGEfansError(`你被限流了,请5分钟后重试(${text})`);
      }
      if (text.includes('timeout')) {
          throw new AGEfansError(`Cookie过期,请刷新页面重试(${text})`);
      }
      function __qpic_chkvurl_converting(_in_vurl) {
          const vurl = decodeURIComponent(_in_vurl);
          const match_resl = vurl.match(/^http.+\.f20\.mp4\?ptype=http\?w5=0&h5=0&state=1$/);
          return !!match_resl;
      }
      const _json_obj = JSON.parse(text);
      const _purl = _json_obj['purl'];
      const _vurl = _json_obj['vurl'];
      const _playid = _json_obj['playid'];
      if (__qpic_chkvurl_converting(_vurl)) {
          throw new AGEfansError('视频转码中,请稍后再试');
      }
      if (_playid === '<play>QLIVE</play>') {
          throw new AGEfansError('脚本不支持QLIVE模式,请使用关闭脚本使用原生播放');
      }
      let _url = _purl + _vurl;
      let url = new URL(_url, location.origin);
      const vurl = url.searchParams.get('url');
      return parseToURL(vurl);
  }
  async function getVurlWithLocal(href) {
      let vurl = getLocal(href);
      if (vurl) {
          return vurl;
      }
      await updateCookie(href);
      vurl = await getVurl(href);
      saveLocal(href, vurl);
      return vurl;
  }
  function initGetAllVideoURL() {
      insertBTSites();
  }

  var css$c = ".k-modal {\n  position: fixed;\n  left: 0;\n  right: 0;\n  top: 0;\n  bottom: 0;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  z-index: 1000;\n  text-align: left;\n  animation: fadeIn 0.3s ease forwards;\n  color: rgba(0, 0, 0, 0.85);\n}\n@keyframes fadeIn {\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n}\n.k-modal * {\n  color: inherit;\n}\n.k-modal-mask {\n  position: fixed;\n  left: 0;\n  right: 0;\n  bottom: 0;\n  top: 0;\n  background: rgba(0, 0, 0, 0.45);\n  cursor: pointer;\n}\n.k-modal-wrap {\n  position: fixed;\n  left: 0;\n  right: 0;\n  bottom: 0;\n  top: 0;\n  overflow: auto;\n  text-align: center;\n  user-select: none;\n}\n.k-modal-wrap::before {\n  content: \"\";\n  display: inline-block;\n  width: 0;\n  height: 100%;\n  vertical-align: middle;\n}\n.k-modal-container {\n  margin: 20px 0;\n  display: inline-block;\n  vertical-align: middle;\n  text-align: left;\n  position: relative;\n  width: 520px;\n  min-height: 100px;\n  background: white;\n  border-radius: 2px;\n  user-select: text;\n}\n.k-modal-header {\n  font-size: 16px;\n  padding: 16px;\n  border-bottom: 1px solid #f1f1f1;\n  display: flex;\n  justify-content: space-between;\n  align-items: center;\n}\n.k-modal-close {\n  cursor: pointer;\n  height: 55px;\n  width: 55px;\n  position: absolute;\n  right: 0;\n  top: 0;\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  user-select: none;\n}\n.k-modal-close * {\n  color: rgba(0, 0, 0, 0.45);\n  transition: color 0.15s ease;\n}\n.k-modal-close:hover * {\n  color: rgba(0, 0, 0, 0.85);\n}\n.k-modal-body {\n  padding: 16px;\n  font-size: 14px;\n}\n.k-modal-footer {\n  padding: 10px 16px;\n  font-size: 14px;\n  border-top: 1px solid #f1f1f1;\n  display: flex;\n  justify-content: flex-end;\n}\n.k-modal-btn {\n  user-select: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 32px;\n  line-height: 32px;\n  border-radius: 2px;\n  border: 1px solid #1890ff;\n  background: #1890ff;\n  color: white;\n  min-width: 64px;\n  cursor: pointer;\n  padding: 0 8px;\n}";
  n(css$c,{});

  function modal(opts) {
      const { title, content, onClose, onOk, okText = '确 定' } = opts;
      const store = {
          width: document.body.style.width,
          overflow: document.body.style.overflow,
      };
      const ID = Math.random().toString(16).slice(2);
      $(`
<div class="k-modal" role="dialog" id="${ID}">
  <div class="k-modal-mask"></div>
  <div class="k-modal-wrap">
    <div class="k-modal-container">
      <div class="k-modal-header">
        <div class="k-modal-title"></div>
        <a class="k-modal-close">
          <svg viewBox="64 64 896 896" focusable="false" data-icon="close" width="1em" height="1em" fill="currentColor" aria-hidden="true"><path d="M563.8 512l262.5-312.9c4.4-5.2.7-13.1-6.1-13.1h-79.8c-4.7 0-9.2 2.1-12.3 5.7L511.6 449.8 295.1 191.7c-3-3.6-7.5-5.7-12.3-5.7H203c-6.8 0-10.5 7.9-6.1 13.1L459.4 512 196.9 824.9A7.95 7.95 0 00203 838h79.8c4.7 0 9.2-2.1 12.3-5.7l216.5-258.1 216.5 258.1c3 3.6 7.5 5.7 12.3 5.7h79.8c6.8 0 10.5-7.9 6.1-13.1L563.8 512z"></path></svg>
        </a>
      </div>
      <div class="k-modal-body">
      </div>
    </div>
  </div>
</div>`).appendTo('body');
      // init css
      $('body').css({
          width: `calc(100% - ${window.innerWidth - document.body.clientWidth}px)`,
          overflow: 'hidden',
      });
      $(`#${ID} .k-modal-title`).append(title);
      $(`#${ID} .k-modal-body`).append(content);
      $(`#${ID} .k-modal-close`).on('click', () => {
          handleClose();
      });
      $(`#${ID} .k-modal-container`).on('click', (e) => {
          e.stopPropagation();
      });
      $(`#${ID} .k-modal-wrap`).on('click', () => {
          handleClose();
      });
      function reset() {
          $(`#${ID}`).remove();
          $('body').css(store);
          window.removeEventListener('keydown', fn, { capture: true });
      }
      function handleClose() {
          onClose === null || onClose === void 0 ? void 0 : onClose();
          reset();
      }
      function handleOk() {
          onOk === null || onOk === void 0 ? void 0 : onOk();
          reset();
      }
      function fn(e) {
          if (['Escape', '?', '?'].includes(e.key)) {
              handleClose();
          }
          e.stopPropagation();
      }
      window.addEventListener('keydown', fn, { capture: true });
      if (onOk) {
          $(`#${ID} .k-modal-container`).append(`
      <div class="k-modal-footer">
        <button class="k-modal-btn k-modal-ok">${okText}</button>
      </div>
    `);
          $(`#${ID} .k-modal-ok`).on('click', () => {
              handleOk();
          });
      }
  }

  var css$b = ".k-alert {\n  margin-bottom: 16px;\n  box-sizing: border-box;\n  color: rgba(0, 0, 0, 0.8509803922);\n  font-size: 14px;\n  font-variant: tabular-nums;\n  line-height: 1.5715;\n  list-style: none;\n  font-feature-settings: \"tnum\";\n  position: relative;\n  display: flex;\n  align-items: center;\n  padding: 8px 15px;\n  word-wrap: break-word;\n  border-radius: 2px;\n}\n.k-alert-icon {\n  margin-right: 8px;\n  display: inline-block;\n  color: inherit;\n  font-style: normal;\n  line-height: 0;\n  text-align: center;\n  text-transform: none;\n  vertical-align: -0.125em;\n  text-rendering: optimizeLegibility;\n  -webkit-font-smoothing: antialiased;\n  color: #1890ff;\n}\n.k-alert-content {\n  flex: 1;\n  min-width: 0;\n}\n.k-alert-info {\n  background-color: #e6f7ff;\n  border: 1px solid #91d5ff;\n}";
  n(css$b,{});

  function alert(html) {
      return `<div class="k-alert k-alert-info">
  <span class="k-alert-icon">
    <svg viewBox="64 64 896 896" focusable="false" data-icon="info-circle" width="1em" height="1em" fill="currentColor" aria-hidden="true">
      <path d="M512 64C264.6 64 64 264.6 64 512s200.6 448 448 448 448-200.6 448-448S759.4 64 512 64zm32 664c0 4.4-3.6 8-8 8h-48c-4.4 0-8-3.6-8-8V456c0-4.4 3.6-8 8-8h48c4.4 0 8 3.6 8 8v272zm-32-344a48.01 48.01 0 010-96 48.01 48.01 0 010 96z"></path>
    </svg>
  </span>
  <div class="k-alert-content">
    <div class="k-alert-message">${html}</div>
  </div>
</div>`;
  }

  var css$a = ".agefans-setting-item {\n  cursor: pointer;\n  width: 100%;\n  white-space: nowrap;\n  display: flex;\n  align-items: center;\n  line-height: 1;\n}\n.agefans-setting-item + .agefans-setting-item {\n  margin-top: 12px;\n}\n.agefans-setting-item input {\n  margin-right: 4px;\n}";
  n(css$a,{});

  const LOCAL_SETTING_KEY = 'agefans-setting';
  const defaultSetting = {
      usePreview: true,
  };
  function ensureDefaultSetting() {
      let setting = gm.getItem(LOCAL_SETTING_KEY, defaultSetting);
      setting = Object.assign({}, defaultSetting, setting);
      gm.setItem(LOCAL_SETTING_KEY, setting);
  }
  function setSetting(key, value) {
      const setting = gm.getItem(LOCAL_SETTING_KEY);
      setting[key] = value;
      gm.setItem(LOCAL_SETTING_KEY, setting);
  }
  function getSetting(key) {
      const setting = gm.getItem(LOCAL_SETTING_KEY);
      if (key)
          return setting[key];
      return setting;
  }
  function showSetting() {
      const setting = getSetting();
      const $usePreview = $(`<label class="agefans-setting-item"><input type="checkbox" />启用番剧信息预览(鼠标悬浮时预览番剧基础信息)</label>`);
      $usePreview
          .find('input')
          .prop('checked', setting.usePreview)
          .on('change', (e) => {
          setSetting('usePreview', e.target.checked);
      });
      const $stopUseKPlayer = $(`<label class="agefans-setting-item"><input type="checkbox" /><b>暂时</b>使用原生播放器</label>`);
      $stopUseKPlayer
          .find('input')
          .prop('checked', session.getItem('stop-use'))
          .on('change', (e) => {
          session.setItem('stop-use', e.target.checked);
      });
      modal({
          title: '脚本设置',
          okText: '刷新页面',
          onOk: () => location.reload(),
          content: $('<div></div>').append(alert('这些配置需要刷新页面才能生效'), $usePreview, $stopUseKPlayer),
      });
  }
  function settingModule() {
      ensureDefaultSetting();
      $('<a>设置</a>').on('click', showSetting).insertBefore('.loginout a');
  }

  function pagePreview(trigger, previewURL) {
      if (!getSetting('usePreview'))
          return;
      const $popover = $(`<div class='page-preview' style="display:none">
       <div class="page-preview-center">${loadingIcon}加载中...</div>
     </div>`).appendTo($(trigger));
      function caclPosition(e) {
          const safeArea = 16;
          const offset = 20;
          const width = $popover.width() || 0;
          const height = $popover.height() || 0;
          const { innerWidth, innerHeight } = window;
          const { clientX, clientY } = e;
          const maxLeft = innerWidth - width - safeArea * 2;
          const maxTop = innerHeight - height - safeArea;
          const left = Math.min(clientX + offset, maxLeft);
          const top = 
          // 当指针与 popover 重叠时,切换位置
          clientX + offset > maxLeft && clientY + offset > maxTop
              ? clientY - offset - height
              : Math.min(clientY + offset, maxTop);
          $popover.css({ left, top });
      }
      let isLoaded = false;
      let timeId;
      $(trigger)
          .addClass('page-preview-trigger')
          .on('mouseenter', (e) => {
          $popover.show();
          caclPosition(e);
          if (isLoaded)
              return;
          clearTimeout(timeId);
          timeId = window.setTimeout(async () => {
              if (isLoaded)
                  return;
              isLoaded = true;
              let { img, info } = session.getItem(previewURL, { img: '', info: '' });
              if (!info) {
                  const $root = $(await fetch(previewURL).then((r) => r.text()));
                  img = $root
                      .find('#container > div.div_left > div:nth-child(1) > div > img')
                      .css({
                      display: 'block',
                      width: 256,
                      height: 356,
                  })
                      .prop('outerHTML');
                  const $info = $root
                      .find('#container > div.div_left > div:nth-child(2) > div > div')
                      .width(256);
                  $info
                      .find('.blocktitle.detail_title1')
                      .text($root.find('.detail_imform_name').text());
                  info = $info.prop('outerHTML');
                  session.setItem(previewURL, { img, info });
              }
              $popover.empty().append(img, info);
              caclPosition(e);
          }, 100);
      })
          .on('mousemove', (e) => {
          caclPosition(e);
      })
          .on('mouseleave', () => {
          clearTimeout(timeId);
          $popover.hide();
      });
  }

  function renderHistroyStyle() {
      $('<style/>').html(`.movurl li a:visited { color: red; }`).appendTo('head');
  }
  function detailModule() {
      renderHistroyStyle();
      $('.div_left li > a:nth-child(1), .ul_li_a4 li > a:nth-child(1)').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
  }

  var css$9 = ".agefans-wrapper #history {\n  background: #202020;\n  border: 4px solid #303030;\n}\n.agefans-wrapper #history .history-list {\n  padding: 8px;\n  display: flex;\n  flex-wrap: wrap;\n}\n.agefans-wrapper #history .history-item {\n  width: 115px;\n  display: inline-block;\n  margin: 4px;\n}\n.agefans-wrapper #history .history-item img {\n  width: 100%;\n  border-radius: 2px;\n}\n.agefans-wrapper #history .history-item .desc .title {\n  overflow: hidden;\n  white-space: nowrap;\n  text-overflow: ellipsis;\n  font-size: 14px;\n  margin: 4px 0;\n}\n.agefans-wrapper #history .history-item .desc .position {\n  font-size: 14px;\n}\n@media (max-width: 480px) {\n  .agefans-wrapper #history .history-list {\n    display: grid;\n    grid-template-columns: repeat(3, 1fr);\n    grid-gap: 8px;\n  }\n  .agefans-wrapper #history .history-item {\n    width: auto;\n    margin: 0;\n    display: block;\n    min-width: 0;\n  }\n}";
  n(css$9,{});

  function parseTime(time = 0) {
      return `${Math.floor(time / 60)
        .toString()
        .padStart(2, '0')}:${(time % 60).toString().padStart(2, '0')}`;
  }

  const LOCAL_HISTORY_KEY = 'v-his';
  const MAX_HISTORY_LENGTH = 100;
  class History {
      get his() {
          return gm.getItem(LOCAL_HISTORY_KEY, []);
      }
      set his(value) {
          if (Array.isArray(value)) {
              gm.setItem(LOCAL_HISTORY_KEY, value.slice(0, MAX_HISTORY_LENGTH));
          }
      }
      getAll() {
          return this.his;
      }
      get(id) {
          return this.his.find((o) => o.id === id);
      }
      setTime(id, time = 0) {
          const his = this.his;
          his.find((o) => o.id === id).time = time;
          this.his = his;
      }
      log(item) {
          const his = this.his;
          his.unshift(item);
          this.his = his;
      }
      refresh(id, data) {
          const his = this.his;
          const index = his.findIndex((o) => o.id === id);
          const item = his.splice(index, 1)[0];
          his.unshift(data || item);
          this.his = his;
      }
      has(id) {
          return Boolean(this.his.find((o) => o.id === id));
      }
      logHistory() {
          var _a;
          const id = (_a = location.pathname.match(/\/play\/(\d*)/)) === null || _a === void 0 ? void 0 : _a[1];
          if (!id)
              return;
          const hisItem = {};
          hisItem.id = id;
          hisItem.title = $('#detailname a').text();
          hisItem.href = location.pathname + location.search;
          hisItem.section = $('li a[style*="color: rgb(238, 0, 0);"]').text();
          hisItem.time = 0;
          hisItem.logo = $('#play_poster_img').attr('src');
          if (this.has(id)) {
              const oldItem = this.get(id);
              if (oldItem.href !== hisItem.href) {
                  this.refresh(id, hisItem);
              }
              else {
                  this.refresh(id);
              }
          }
          else {
              this.log(hisItem);
          }
      }
  }
  const his = new History();
  function renderHistoryList() {
      $('#history')
          .html('')
          .append(() => {
          /** @type {any[]} */
          const histories = his.getAll();
          let html = '';
          histories.forEach((o) => {
              html += `<a class="history-item" href="${o.href}" data-id="${o.id}" data-detail-href="/detail/${o.id}">
          <img
            referrerpolicy="no-referrer"
            src="${o.logo}"
            alt="${o.title}"
            title="${o.title}"
          />
          <div class="desc">
            <div class="title">${o.title}</div>
            <div class="position">${o.section} ${parseTime(o.time)}</div>
          </div>
        </a>
      `;
          });
          return `<div class="history-list">${html || '<center>暂无数据</center>'}</div>`;
      })
          .find('a')
          .each((_, anchor) => pagePreview(anchor, anchor.dataset.detailHref));
  }
  function changeHash(hash) {
      if (hash) {
          history.replaceState(null, '', `#${hash}`);
      }
      else {
          const url = new URL(location.href);
          url.hash = '';
          history.replaceState(null, '', url);
      }
  }
  function refreshHistoryList() {
      const list = his.getAll();
      const $doms = $('#history .history-item');
      if (list.length !== $doms.length)
          return renderHistoryList();
      list.forEach((item) => {
          const $dom = $(`#history a[data-id='${item.id}']`);
          $dom.attr('href', item.href);
          $dom.find('.title').text(item.title);
          $dom.find('.position').text(`${item.section} ${parseTime(item.time)}`);
      });
  }
  const { startRefresh, stopRefresh } = (function () {
      let time;
      return {
          startRefresh: () => {
              clearInterval(time);
              time = window.setInterval(refreshHistoryList, 1000);
          },
          stopRefresh: () => {
              clearInterval(time);
          },
      };
  })();
  function renderHistoryPage() {
      const currentDom = $('.nav_button_current');
      $('<div id="history"></div>').insertAfter('#container').hide();
      const $hisNavBtn = $(`<a class="nav_button">历史</a>`)
          .insertBefore('#nav form')
          .on('click', (e) => {
          if ($('#history').is(':visible')) {
              $('#container').show();
              $('#history').hide();
              stopRefresh();
              changeHash();
              changeActive(currentDom);
          }
          else {
              refreshHistoryList();
              $('#container').hide();
              $('#history').show();
              startRefresh();
              changeHash('/history');
              changeActive($(e.currentTarget));
          }
      });
      // 移除默认激活的 nav 上的 href 与增加点击事件
      $('.nav_button_current')
          .on('click', (e) => {
          $('#container').show();
          $('#history').hide();
          stopRefresh();
          changeActive(e.currentTarget);
          changeHash();
      })
          .removeAttr('href');
      if (window.location.hash === '#/history') {
          $('#container').hide();
          $('#history').show();
          startRefresh();
          changeActive($hisNavBtn);
      }
  }
  function changeActive(dom) {
      $('.nav_button_current').removeClass('nav_button_current');
      $(dom).addClass('nav_button_current');
  }
  function historyModule() {
      renderHistoryPage();
      renderHistoryList();
  }

  function homeModule() {
      $('#container li > a:nth-child(1)').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
      $('#new_anime_btns li').on('click', () => {
          $('#new_anime_page > li > a.one_new_anime_name').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
      });
  }

  /** Detect free variable `global` from Node.js. */
  var freeGlobal = typeof global == 'object' && global && global.Object === Object && global;

  /** Detect free variable `self`. */
  var freeSelf = typeof self == 'object' && self && self.Object === Object && self;

  /** Used as a reference to the global object. */
  var root = freeGlobal || freeSelf || Function('return this')();

  /** Built-in value references. */
  var Symbol = root.Symbol;

  /** Used for built-in method references. */
  var objectProto$1 = Object.prototype;

  /** Used to check objects for own properties. */
  var hasOwnProperty = objectProto$1.hasOwnProperty;

  /**
   * Used to resolve the
   * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
   * of values.
   */
  var nativeObjectToString$1 = objectProto$1.toString;

  /** Built-in value references. */
  var symToStringTag$1 = Symbol ? Symbol.toStringTag : undefined;

  /**
   * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
   *
   * @private
   * @param {*} value The value to query.
   * @returns {string} Returns the raw `toStringTag`.
   */
  function getRawTag(value) {
    var isOwn = hasOwnProperty.call(value, symToStringTag$1),
        tag = value[symToStringTag$1];

    try {
      value[symToStringTag$1] = undefined;
      var unmasked = true;
    } catch (e) {}

    var result = nativeObjectToString$1.call(value);
    if (unmasked) {
      if (isOwn) {
        value[symToStringTag$1] = tag;
      } else {
        delete value[symToStringTag$1];
      }
    }
    return result;
  }

  /** Used for built-in method references. */
  var objectProto = Object.prototype;

  /**
   * Used to resolve the
   * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
   * of values.
   */
  var nativeObjectToString = objectProto.toString;

  /**
   * Converts `value` to a string using `Object.prototype.toString`.
   *
   * @private
   * @param {*} value The value to convert.
   * @returns {string} Returns the converted string.
   */
  function objectToString(value) {
    return nativeObjectToString.call(value);
  }

  /** `Object#toString` result references. */
  var nullTag = '[object Null]',
      undefinedTag = '[object Undefined]';

  /** Built-in value references. */
  var symToStringTag = Symbol ? Symbol.toStringTag : undefined;

  /**
   * The base implementation of `getTag` without fallbacks for buggy environments.
   *
   * @private
   * @param {*} value The value to query.
   * @returns {string} Returns the `toStringTag`.
   */
  function baseGetTag(value) {
    if (value == null) {
      return value === undefined ? undefinedTag : nullTag;
    }
    return (symToStringTag && symToStringTag in Object(value))
      ? getRawTag(value)
      : objectToString(value);
  }

  /**
   * Checks if `value` is object-like. A value is object-like if it's not `null`
   * and has a `typeof` result of "object".
   *
   * @static
   * @memberOf _
   * @since 4.0.0
   * @category Lang
   * @param {*} value The value to check.
   * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
   * @example
   *
   * _.isObjectLike({});
   * // => true
   *
   * _.isObjectLike([1, 2, 3]);
   * // => true
   *
   * _.isObjectLike(_.noop);
   * // => false
   *
   * _.isObjectLike(null);
   * // => false
   */
  function isObjectLike(value) {
    return value != null && typeof value == 'object';
  }

  /** `Object#toString` result references. */
  var symbolTag = '[object Symbol]';

  /**
   * Checks if `value` is classified as a `Symbol` primitive or object.
   *
   * @static
   * @memberOf _
   * @since 4.0.0
   * @category Lang
   * @param {*} value The value to check.
   * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
   * @example
   *
   * _.isSymbol(Symbol.iterator);
   * // => true
   *
   * _.isSymbol('abc');
   * // => false
   */
  function isSymbol(value) {
    return typeof value == 'symbol' ||
      (isObjectLike(value) && baseGetTag(value) == symbolTag);
  }

  /** Used to match a single whitespace character. */
  var reWhitespace = /\s/;

  /**
   * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace
   * character of `string`.
   *
   * @private
   * @param {string} string The string to inspect.
   * @returns {number} Returns the index of the last non-whitespace character.
   */
  function trimmedEndIndex(string) {
    var index = string.length;

    while (index-- && reWhitespace.test(string.charAt(index))) {}
    return index;
  }

  /** Used to match leading whitespace. */
  var reTrimStart = /^\s+/;

  /**
   * The base implementation of `_.trim`.
   *
   * @private
   * @param {string} string The string to trim.
   * @returns {string} Returns the trimmed string.
   */
  function baseTrim(string) {
    return string
      ? string.slice(0, trimmedEndIndex(string) + 1).replace(reTrimStart, '')
      : string;
  }

  /**
   * Checks if `value` is the
   * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
   * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
   *
   * @static
   * @memberOf _
   * @since 0.1.0
   * @category Lang
   * @param {*} value The value to check.
   * @returns {boolean} Returns `true` if `value` is an object, else `false`.
   * @example
   *
   * _.isObject({});
   * // => true
   *
   * _.isObject([1, 2, 3]);
   * // => true
   *
   * _.isObject(_.noop);
   * // => true
   *
   * _.isObject(null);
   * // => false
   */
  function isObject(value) {
    var type = typeof value;
    return value != null && (type == 'object' || type == 'function');
  }

  /** Used as references for various `Number` constants. */
  var NAN = 0 / 0;

  /** Used to detect bad signed hexadecimal string values. */
  var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;

  /** Used to detect binary string values. */
  var reIsBinary = /^0b[01]+$/i;

  /** Used to detect octal string values. */
  var reIsOctal = /^0o[0-7]+$/i;

  /** Built-in method references without a dependency on `root`. */
  var freeParseInt = parseInt;

  /**
   * Converts `value` to a number.
   *
   * @static
   * @memberOf _
   * @since 4.0.0
   * @category Lang
   * @param {*} value The value to process.
   * @returns {number} Returns the number.
   * @example
   *
   * _.toNumber(3.2);
   * // => 3.2
   *
   * _.toNumber(Number.MIN_VALUE);
   * // => 5e-324
   *
   * _.toNumber(Infinity);
   * // => Infinity
   *
   * _.toNumber('3.2');
   * // => 3.2
   */
  function toNumber(value) {
    if (typeof value == 'number') {
      return value;
    }
    if (isSymbol(value)) {
      return NAN;
    }
    if (isObject(value)) {
      var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
      value = isObject(other) ? (other + '') : other;
    }
    if (typeof value != 'string') {
      return value === 0 ? value : +value;
    }
    value = baseTrim(value);
    var isBinary = reIsBinary.test(value);
    return (isBinary || reIsOctal.test(value))
      ? freeParseInt(value.slice(2), isBinary ? 2 : 8)
      : (reIsBadHex.test(value) ? NAN : +value);
  }

  /**
   * Gets the timestamp of the number of milliseconds that have elapsed since
   * the Unix epoch (1 January 1970 00:00:00 UTC).
   *
   * @static
   * @memberOf _
   * @since 2.4.0
   * @category Date
   * @returns {number} Returns the timestamp.
   * @example
   *
   * _.defer(function(stamp) {
   *   console.log(_.now() - stamp);
   * }, _.now());
   * // => Logs the number of milliseconds it took for the deferred invocation.
   */
  var now = function() {
    return root.Date.now();
  };

  /** Error message constants. */
  var FUNC_ERROR_TEXT$1 = 'Expected a function';

  /* Built-in method references for those with the same name as other `lodash` methods. */
  var nativeMax = Math.max,
      nativeMin = Math.min;

  /**
   * Creates a debounced function that delays invoking `func` until after `wait`
   * milliseconds have elapsed since the last time the debounced function was
   * invoked. The debounced function comes with a `cancel` method to cancel
   * delayed `func` invocations and a `flush` method to immediately invoke them.
   * Provide `options` to indicate whether `func` should be invoked on the
   * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
   * with the last arguments provided to the debounced function. Subsequent
   * calls to the debounced function return the result of the last `func`
   * invocation.
   *
   * **Note:** If `leading` and `trailing` options are `true`, `func` is
   * invoked on the trailing edge of the timeout only if the debounced function
   * is invoked more than once during the `wait` timeout.
   *
   * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
   * until to the next tick, similar to `setTimeout` with a timeout of `0`.
   *
   * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
   * for details over the differences between `_.debounce` and `_.throttle`.
   *
   * @static
   * @memberOf _
   * @since 0.1.0
   * @category Function
   * @param {Function} func The function to debounce.
   * @param {number} [wait=0] The number of milliseconds to delay.
   * @param {Object} [options={}] The options object.
   * @param {boolean} [options.leading=false]
   *  Specify invoking on the leading edge of the timeout.
   * @param {number} [options.maxWait]
   *  The maximum time `func` is allowed to be delayed before it's invoked.
   * @param {boolean} [options.trailing=true]
   *  Specify invoking on the trailing edge of the timeout.
   * @returns {Function} Returns the new debounced function.
   * @example
   *
   * // Avoid costly calculations while the window size is in flux.
   * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
   *
   * // Invoke `sendMail` when clicked, debouncing subsequent calls.
   * jQuery(element).on('click', _.debounce(sendMail, 300, {
   *   'leading': true,
   *   'trailing': false
   * }));
   *
   * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
   * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
   * var source = new EventSource('/stream');
   * jQuery(source).on('message', debounced);
   *
   * // Cancel the trailing debounced invocation.
   * jQuery(window).on('popstate', debounced.cancel);
   */
  function debounce(func, wait, options) {
    var lastArgs,
        lastThis,
        maxWait,
        result,
        timerId,
        lastCallTime,
        lastInvokeTime = 0,
        leading = false,
        maxing = false,
        trailing = true;

    if (typeof func != 'function') {
      throw new TypeError(FUNC_ERROR_TEXT$1);
    }
    wait = toNumber(wait) || 0;
    if (isObject(options)) {
      leading = !!options.leading;
      maxing = 'maxWait' in options;
      maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
      trailing = 'trailing' in options ? !!options.trailing : trailing;
    }

    function invokeFunc(time) {
      var args = lastArgs,
          thisArg = lastThis;

      lastArgs = lastThis = undefined;
      lastInvokeTime = time;
      result = func.apply(thisArg, args);
      return result;
    }

    function leadingEdge(time) {
      // Reset any `maxWait` timer.
      lastInvokeTime = time;
      // Start the timer for the trailing edge.
      timerId = setTimeout(timerExpired, wait);
      // Invoke the leading edge.
      return leading ? invokeFunc(time) : result;
    }

    function remainingWait(time) {
      var timeSinceLastCall = time - lastCallTime,
          timeSinceLastInvoke = time - lastInvokeTime,
          timeWaiting = wait - timeSinceLastCall;

      return maxing
        ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke)
        : timeWaiting;
    }

    function shouldInvoke(time) {
      var timeSinceLastCall = time - lastCallTime,
          timeSinceLastInvoke = time - lastInvokeTime;

      // Either this is the first call, activity has stopped and we're at the
      // trailing edge, the system time has gone backwards and we're treating
      // it as the trailing edge, or we've hit the `maxWait` limit.
      return (lastCallTime === undefined || (timeSinceLastCall >= wait) ||
        (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));
    }

    function timerExpired() {
      var time = now();
      if (shouldInvoke(time)) {
        return trailingEdge(time);
      }
      // Restart the timer.
      timerId = setTimeout(timerExpired, remainingWait(time));
    }

    function trailingEdge(time) {
      timerId = undefined;

      // Only invoke if we have `lastArgs` which means `func` has been
      // debounced at least once.
      if (trailing && lastArgs) {
        return invokeFunc(time);
      }
      lastArgs = lastThis = undefined;
      return result;
    }

    function cancel() {
      if (timerId !== undefined) {
        clearTimeout(timerId);
      }
      lastInvokeTime = 0;
      lastArgs = lastCallTime = lastThis = timerId = undefined;
    }

    function flush() {
      return timerId === undefined ? result : trailingEdge(now());
    }

    function debounced() {
      var time = now(),
          isInvoking = shouldInvoke(time);

      lastArgs = arguments;
      lastThis = this;
      lastCallTime = time;

      if (isInvoking) {
        if (timerId === undefined) {
          return leadingEdge(lastCallTime);
        }
        if (maxing) {
          // Handle invocations in a tight loop.
          clearTimeout(timerId);
          timerId = setTimeout(timerExpired, wait);
          return invokeFunc(lastCallTime);
        }
      }
      if (timerId === undefined) {
        timerId = setTimeout(timerExpired, wait);
      }
      return result;
    }
    debounced.cancel = cancel;
    debounced.flush = flush;
    return debounced;
  }

  /** Error message constants. */
  var FUNC_ERROR_TEXT = 'Expected a function';

  /**
   * Creates a throttled function that only invokes `func` at most once per
   * every `wait` milliseconds. The throttled function comes with a `cancel`
   * method to cancel delayed `func` invocations and a `flush` method to
   * immediately invoke them. Provide `options` to indicate whether `func`
   * should be invoked on the leading and/or trailing edge of the `wait`
   * timeout. The `func` is invoked with the last arguments provided to the
   * throttled function. Subsequent calls to the throttled function return the
   * result of the last `func` invocation.
   *
   * **Note:** If `leading` and `trailing` options are `true`, `func` is
   * invoked on the trailing edge of the timeout only if the throttled function
   * is invoked more than once during the `wait` timeout.
   *
   * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
   * until to the next tick, similar to `setTimeout` with a timeout of `0`.
   *
   * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
   * for details over the differences between `_.throttle` and `_.debounce`.
   *
   * @static
   * @memberOf _
   * @since 0.1.0
   * @category Function
   * @param {Function} func The function to throttle.
   * @param {number} [wait=0] The number of milliseconds to throttle invocations to.
   * @param {Object} [options={}] The options object.
   * @param {boolean} [options.leading=true]
   *  Specify invoking on the leading edge of the timeout.
   * @param {boolean} [options.trailing=true]
   *  Specify invoking on the trailing edge of the timeout.
   * @returns {Function} Returns the new throttled function.
   * @example
   *
   * // Avoid excessively updating the position while scrolling.
   * jQuery(window).on('scroll', _.throttle(updatePosition, 100));
   *
   * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
   * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });
   * jQuery(element).on('click', throttled);
   *
   * // Cancel the trailing throttled invocation.
   * jQuery(window).on('popstate', throttled.cancel);
   */
  function throttle(func, wait, options) {
    var leading = true,
        trailing = true;

    if (typeof func != 'function') {
      throw new TypeError(FUNC_ERROR_TEXT);
    }
    if (isObject(options)) {
      leading = 'leading' in options ? !!options.leading : leading;
      trailing = 'trailing' in options ? !!options.trailing : trailing;
    }
    return debounce(func, wait, {
      'leading': leading,
      'maxWait': wait,
      'trailing': trailing
    });
  }

  function genIssueURL({ title, body }) {
      const url = new URL(`https://github.com/IronKinoko/agefans-enhance/issues/new`);
      url.searchParams.set('title', title);
      url.searchParams.set('body', body);
      return url.toString();
  }

  const SHIFT_KEY = '~!@#$%^&*()_+{}|:"<>?' + '~!@#¥%…&*()——+「」|:“《》?';
  function keybind(keys, cb) {
      const isMac = /macintosh|mac os x/i.test(navigator.userAgent);
      keys = keys.filter((key) => !key.includes(isMac ? 'ctrl' : 'meta'));
      $(window).on('keydown', (e) => {
          var _a;
          if (((_a = document.activeElement) === null || _a === void 0 ? void 0 : _a.tagName) === 'INPUT')
              return;
          let keyArr = [];
          e.ctrlKey && keyArr.push('ctrl');
          e.metaKey && keyArr.push('meta');
          e.shiftKey && !SHIFT_KEY.includes(e.key) && keyArr.push('shift');
          e.altKey && keyArr.push('alt');
          if (!['Control', 'Meta', 'Shift', 'Alt'].includes(e.key)) {
              keyArr.push(e.key);
          }
          keyArr = [...new Set(keyArr)];
          const key = keyArr.join('+');
          if (keys.includes(key)) {
              cb(e.originalEvent, key);
          }
      });
  }

  var css$8 = "#k-player-message {\n  z-index: 999;\n  position: absolute;\n  left: 20px;\n  bottom: 60px;\n}\n#k-player-message .k-player-message-item {\n  display: block;\n  width: max-content;\n  padding: 8px 16px;\n  background: var(--k-player-background);\n  border-radius: 4px;\n  color: white;\n  font-size: 14px;\n  white-space: nowrap;\n  overflow: hidden;\n  box-sizing: border-box;\n  margin-top: 4px;\n}\n#k-player-message .k-player-message-item:hover {\n  background: var(--k-player-background-highlight);\n  transition: all 0.3s;\n}";
  n(css$8,{});

  class Message {
      constructor(selector) {
          this.$message = $('<div id="k-player-message">');
          this.$message.appendTo($(selector));
      }
      info(message, ms = 1500) {
          return new Promise((resolve) => {
              $(`<div class="k-player-message-item"></div>`)
                  .append(message)
                  .hide()
                  .appendTo(this.$message)
                  .show(150)
                  .delay(ms)
                  .hide(150, function () {
                  $(this).remove();
                  resolve();
              });
          });
      }
      destroy() {
          this.$message.empty();
      }
  }

  var css$7 = ".k-popover {\n  position: relative;\n}\n.k-popover-overlay {\n  position: absolute;\n  display: none;\n  bottom: 100%;\n  left: 50%;\n  transform: translateX(-50%);\n  z-index: 100;\n  padding-bottom: 20px;\n}\n.k-popover-content {\n  background: var(--k-player-background);\n  border-radius: 4px;\n  overflow: hidden;\n}";
  n(css$7,{});

  function popover(target, overlay) {
      const $target = $(target);
      const $content = $(`<div class="k-popover-overlay"><div class="k-popover-content"></div></div>`);
      $content.on('click', (e) => e.stopPropagation());
      $content.find('.k-popover-content').append(overlay);
      let timeID;
      $target.addClass('k-popover');
      $target.on('mouseenter', () => {
          clearTimeout(timeID);
          timeID = window.setTimeout(() => {
              $content.fadeIn('fast');
          }, 100);
      });
      $target.on('mouseleave', () => {
          clearTimeout(timeID);
          timeID = window.setTimeout(() => {
              $content.fadeOut('fast');
          }, 100);
      });
      $target.append($content);
      return $target;
  }

  const isMac = /macintosh|mac os x/i.test(navigator.userAgent);
  const macKeyMap = {
      ctrl: '⌘',
      alt: '⌥',
      shift: '⇧',
  };
  function renderKey(key) {
      const lowerCaseKey = key.toLowerCase();
      if (isMac && Reflect.has(macKeyMap, lowerCaseKey)) {
          return macKeyMap[lowerCaseKey];
      }
      return key;
  }

  const icons = `
<svg
xmlns="http://www.w3.org/2000/svg"
style="position: absolute; width: 0px; height: 0px; overflow: hidden"
aria-hidden="true"
>
  <symbol id="next" viewBox="0 0 22 22">
    <path
      d="M16 5a1 1 0 00-1 1v4.615a1.431 1.431 0 00-.615-.829L7.21 5.23A1.439 1.439 0 005 6.445v9.11a1.44 1.44 0 002.21 1.215l7.175-4.555a1.436 1.436 0 00.616-.828V16a1 1 0 002 0V6C17 5.448 16.552 5 16 5z"
    ></path>
  </symbol>

  <symbol
    id="widescreen"
    viewBox="0 0 88 88"
    preserveAspectRatio="xMidYMid meet"
  >
    <defs>
      <clipPath id="__lottie_element_127">
        <rect width="88" height="88" x="0" y="0"></rect>
      </clipPath>
    </defs>
    <g clip-path="url(#__lottie_element_127)">
      <g
        transform="matrix(1,0,0,1,44,44)"
        opacity="1"
        style="display: block"
      >
        <g opacity="1" transform="matrix(1,0,0,1,0,0)">
          <path
            fill="rgb(255,255,255)"
            fill-opacity="1"
            d=" M-14,-20 C-14,-20 -26,-20 -26,-20 C-27.049999237060547,-20 -27.920000076293945,-19.18000030517578 -27.989999771118164,-18.149999618530273 C-27.989999771118164,-18.149999618530273 -28,-18 -28,-18 C-28,-18 -28,-6 -28,-6 C-28,-4.949999809265137 -27.18000030517578,-4.079999923706055 -26.149999618530273,-4.010000228881836 C-26.149999618530273,-4.010000228881836 -26,-4 -26,-4 C-26,-4 -22,-4 -22,-4 C-20.950000762939453,-4 -20.079999923706055,-4.820000171661377 -20.010000228881836,-5.849999904632568 C-20.010000228881836,-5.849999904632568 -20,-6 -20,-6 C-20,-6 -20,-12 -20,-12 C-20,-12 -14,-12 -14,-12 C-12.949999809265137,-12 -12.079999923706055,-12.819999694824219 -12.010000228881836,-13.850000381469727 C-12.010000228881836,-13.850000381469727 -12,-14 -12,-14 C-12,-14 -12,-18 -12,-18 C-12,-19.049999237060547 -12.819999694824219,-19.920000076293945 -13.850000381469727,-19.989999771118164 C-13.850000381469727,-19.989999771118164 -14,-20 -14,-20z M26,-20 C26,-20 14,-20 14,-20 C12.949999809265137,-20 12.079999923706055,-19.18000030517578 12.010000228881836,-18.149999618530273 C12.010000228881836,-18.149999618530273 12,-18 12,-18 C12,-18 12,-14 12,-14 C12,-12.949999809265137 12.819999694824219,-12.079999923706055 13.850000381469727,-12.010000228881836 C13.850000381469727,-12.010000228881836 14,-12 14,-12 C14,-12 20,-12 20,-12 C20,-12 20,-6 20,-6 C20,-4.949999809265137 20.81999969482422,-4.079999923706055 21.850000381469727,-4.010000228881836 C21.850000381469727,-4.010000228881836 22,-4 22,-4 C22,-4 26,-4 26,-4 C27.049999237060547,-4 27.920000076293945,-4.820000171661377 27.989999771118164,-5.849999904632568 C27.989999771118164,-5.849999904632568 28,-6 28,-6 C28,-6 28,-18 28,-18 C28,-19.049999237060547 27.18000030517578,-19.920000076293945 26.149999618530273,-19.989999771118164 C26.149999618530273,-19.989999771118164 26,-20 26,-20z M-22,4 C-22,4 -26,4 -26,4 C-27.049999237060547,4 -27.920000076293945,4.820000171661377 -27.989999771118164,5.849999904632568 C-27.989999771118164,5.849999904632568 -28,6 -28,6 C-28,6 -28,18 -28,18 C-28,19.049999237060547 -27.18000030517578,19.920000076293945 -26.149999618530273,19.989999771118164 C-26.149999618530273,19.989999771118164 -26,20 -26,20 C-26,20 -14,20 -14,20 C-12.949999809265137,20 -12.079999923706055,19.18000030517578 -12.010000228881836,18.149999618530273 C-12.010000228881836,18.149999618530273 -12,18 -12,18 C-12,18 -12,14 -12,14 C-12,12.949999809265137 -12.819999694824219,12.079999923706055 -13.850000381469727,12.010000228881836 C-13.850000381469727,12.010000228881836 -14,12 -14,12 C-14,12 -20,12 -20,12 C-20,12 -20,6 -20,6 C-20,4.949999809265137 -20.81999969482422,4.079999923706055 -21.850000381469727,4.010000228881836 C-21.850000381469727,4.010000228881836 -22,4 -22,4z M26,4 C26,4 22,4 22,4 C20.950000762939453,4 20.079999923706055,4.820000171661377 20.010000228881836,5.849999904632568 C20.010000228881836,5.849999904632568 20,6 20,6 C20,6 20,12 20,12 C20,12 14,12 14,12 C12.949999809265137,12 12.079999923706055,12.819999694824219 12.010000228881836,13.850000381469727 C12.010000228881836,13.850000381469727 12,14 12,14 C12,14 12,18 12,18 C12,19.049999237060547 12.819999694824219,19.920000076293945 13.850000381469727,19.989999771118164 C13.850000381469727,19.989999771118164 14,20 14,20 C14,20 26,20 26,20 C27.049999237060547,20 27.920000076293945,19.18000030517578 27.989999771118164,18.149999618530273 C27.989999771118164,18.149999618530273 28,18 28,18 C28,18 28,6 28,6 C28,4.949999809265137 27.18000030517578,4.079999923706055 26.149999618530273,4.010000228881836 C26.149999618530273,4.010000228881836 26,4 26,4z M28,-28 C32.41999816894531,-28 36,-24.420000076293945 36,-20 C36,-20 36,20 36,20 C36,24.420000076293945 32.41999816894531,28 28,28 C28,28 -28,28 -28,28 C-32.41999816894531,28 -36,24.420000076293945 -36,20 C-36,20 -36,-20 -36,-20 C-36,-24.420000076293945 -32.41999816894531,-28 -28,-28 C-28,-28 28,-28 28,-28z"
          ></path>
        </g>
      </g>
    </g>
  </symbol>

  <symbol
    id="widescreen-quit"
    viewBox="0 0 88 88"
    preserveAspectRatio="xMidYMid meet"
  >
    <defs>
      <clipPath id="__lottie_element_132">
        <rect width="88" height="88" x="0" y="0"></rect>
      </clipPath>
    </defs>
    <g clip-path="url(#__lottie_element_132)">
      <g
        transform="matrix(1,0,0,1,44,44)"
        opacity="1"
        style="display: block"
      >
        <g opacity="1" transform="matrix(1,0,0,1,0,0)">
          <path
            fill="rgb(255,255,255)"
            fill-opacity="1"
            d=" M-14,-20 C-14,-20 -18,-20 -18,-20 C-19.049999237060547,-20 -19.920000076293945,-19.18000030517578 -19.989999771118164,-18.149999618530273 C-19.989999771118164,-18.149999618530273 -20,-18 -20,-18 C-20,-18 -20,-12 -20,-12 C-20,-12 -26,-12 -26,-12 C-27.049999237060547,-12 -27.920000076293945,-11.180000305175781 -27.989999771118164,-10.149999618530273 C-27.989999771118164,-10.149999618530273 -28,-10 -28,-10 C-28,-10 -28,-6 -28,-6 C-28,-4.949999809265137 -27.18000030517578,-4.079999923706055 -26.149999618530273,-4.010000228881836 C-26.149999618530273,-4.010000228881836 -26,-4 -26,-4 C-26,-4 -14,-4 -14,-4 C-12.949999809265137,-4 -12.079999923706055,-4.820000171661377 -12.010000228881836,-5.849999904632568 C-12.010000228881836,-5.849999904632568 -12,-6 -12,-6 C-12,-6 -12,-18 -12,-18 C-12,-19.049999237060547 -12.819999694824219,-19.920000076293945 -13.850000381469727,-19.989999771118164 C-13.850000381469727,-19.989999771118164 -14,-20 -14,-20z M18,-20 C18,-20 14,-20 14,-20 C12.949999809265137,-20 12.079999923706055,-19.18000030517578 12.010000228881836,-18.149999618530273 C12.010000228881836,-18.149999618530273 12,-18 12,-18 C12,-18 12,-6 12,-6 C12,-4.949999809265137 12.819999694824219,-4.079999923706055 13.850000381469727,-4.010000228881836 C13.850000381469727,-4.010000228881836 14,-4 14,-4 C14,-4 26,-4 26,-4 C27.049999237060547,-4 27.920000076293945,-4.820000171661377 27.989999771118164,-5.849999904632568 C27.989999771118164,-5.849999904632568 28,-6 28,-6 C28,-6 28,-10 28,-10 C28,-11.050000190734863 27.18000030517578,-11.920000076293945 26.149999618530273,-11.989999771118164 C26.149999618530273,-11.989999771118164 26,-12 26,-12 C26,-12 20,-12 20,-12 C20,-12 20,-18 20,-18 C20,-19.049999237060547 19.18000030517578,-19.920000076293945 18.149999618530273,-19.989999771118164 C18.149999618530273,-19.989999771118164 18,-20 18,-20z M-14,4 C-14,4 -26,4 -26,4 C-27.049999237060547,4 -27.920000076293945,4.820000171661377 -27.989999771118164,5.849999904632568 C-27.989999771118164,5.849999904632568 -28,6 -28,6 C-28,6 -28,10 -28,10 C-28,11.050000190734863 -27.18000030517578,11.920000076293945 -26.149999618530273,11.989999771118164 C-26.149999618530273,11.989999771118164 -26,12 -26,12 C-26,12 -20,12 -20,12 C-20,12 -20,18 -20,18 C-20,19.049999237060547 -19.18000030517578,19.920000076293945 -18.149999618530273,19.989999771118164 C-18.149999618530273,19.989999771118164 -18,20 -18,20 C-18,20 -14,20 -14,20 C-12.949999809265137,20 -12.079999923706055,19.18000030517578 -12.010000228881836,18.149999618530273 C-12.010000228881836,18.149999618530273 -12,18 -12,18 C-12,18 -12,6 -12,6 C-12,4.949999809265137 -12.819999694824219,4.079999923706055 -13.850000381469727,4.010000228881836 C-13.850000381469727,4.010000228881836 -14,4 -14,4z M26,4 C26,4 14,4 14,4 C12.949999809265137,4 12.079999923706055,4.820000171661377 12.010000228881836,5.849999904632568 C12.010000228881836,5.849999904632568 12,6 12,6 C12,6 12,18 12,18 C12,19.049999237060547 12.819999694824219,19.920000076293945 13.850000381469727,19.989999771118164 C13.850000381469727,19.989999771118164 14,20 14,20 C14,20 18,20 18,20 C19.049999237060547,20 19.920000076293945,19.18000030517578 19.989999771118164,18.149999618530273 C19.989999771118164,18.149999618530273 20,18 20,18 C20,18 20,12 20,12 C20,12 26,12 26,12 C27.049999237060547,12 27.920000076293945,11.180000305175781 27.989999771118164,10.149999618530273 C27.989999771118164,10.149999618530273 28,10 28,10 C28,10 28,6 28,6 C28,4.949999809265137 27.18000030517578,4.079999923706055 26.149999618530273,4.010000228881836 C26.149999618530273,4.010000228881836 26,4 26,4z M28,-28 C32.41999816894531,-28 36,-24.420000076293945 36,-20 C36,-20 36,20 36,20 C36,24.420000076293945 32.41999816894531,28 28,28 C28,28 -28,28 -28,28 C-32.41999816894531,28 -36,24.420000076293945 -36,20 C-36,20 -36,-20 -36,-20 C-36,-24.420000076293945 -32.41999816894531,-28 -28,-28 C-28,-28 28,-28 28,-28z"
          ></path>
        </g>
      </g>
    </g>
  </symbol>

  <symbol id="question" width="1em" height="1em" viewBox="0 0 22 22">
    <path fill="currentColor" d="M6 16l-3 3V5a2 2 0 012-2h12a2 2 0 012 2v9a2 2 0 01-2 2H6zm4-4v2h2v-2h-2zm2-.998c0-.34.149-.523.636-.925.022-.018.296-.24.379-.31a5.81 5.81 0 00.173-.152C13.705 9.145 14 8.656 14 8a3 3 0 00-5.698-1.314c-.082.17-.153.41-.213.72A.5.5 0 008.581 8h1.023a.5.5 0 00.476-.348.851.851 0 01.114-.244A.999.999 0 0112 8c0 1.237-2 1.16-2 3h2z"></path>
  </symbol>
</svg>

<template id="plyr__next">
  <button
    class="plyr__controls__item plyr__control plyr__next plyr__custom"
    type="button"
    data-plyr="next"
    aria-label="Next"
  >
    <svg focusable="false">
      <use xlink:href="#next"></use>
    </svg>
    <span class="plyr__tooltip">下一集(N)</span>
  </button>
</template>

<template id="plyr__widescreen">
  <button
    class="plyr__controls__item plyr__control plyr__widescreen plyr__custom"
    type="button"
    data-plyr="widescreen"
    aria-label="widescreen"
  >
    <svg class="icon--not-pressed" focusable="false">
      <use xlink:href="#widescreen"></use>
    </svg>
    <svg class="icon--pressed" focusable="false">
      <use xlink:href="#widescreen-quit"></use>
    </svg>
    <span class="label--not-pressed plyr__tooltip">网页全屏(W)</span>
    <span class="label--pressed plyr__tooltip">退出网页全屏(W)</span>
  </button>
</template>

`;
  $('body').append(icons);
  const loadingHTML = `
<div id="k-player-loading" style="display: none">
  <div class="k-player-center">
    <div class="k-player-tsuma"></div>
    <div class="lds-spinner">
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
      <div></div>
    </div>  
  </div>
</div>
`;
  const errorHTML = `
<div id="k-player-error" style="display: none">
  <div class="k-player-center">
    <div class="k-player-error-img"></div>
    <div class="k-player-tsuma"></div>
    <div class="k-player-error-info"></div>
  </div>
</div>`;
  const pipHTML = `
<div id="k-player-pip" style="display: none">
  <div class="k-player-center">
    <div class="k-player-tsuma"></div>
  </div>
</div>`;
  const speedList = [0.5, 0.75, 1, 1.25, 1.5, 2, 4];
  const speedHTML = popover(`
<div id="k-speed" class="plyr__controls__item k-popover k-text-btn">
  <span id="k-speed-text" class="k-text-btn-text">倍速</span>
</div>
`, `<ul class="k-menu">
${[...speedList]
    .reverse()
    .map((speed) => `<li class="k-menu-item k-speed-item" data-speed="${speed}">${speed}x</li>`)
    .join('')}
</ul>`);
  const settingsHTML = popover(`
<button id="k-settings" type="button" class="plyr__control plyr__controls__item">
  <svg><use href="#plyr-settings" /></svg>
</button>
`, `
<div class="k-settings-list">
  <label class="k-settings-item">
    <input type="checkbox" name="showSearchActions" />
    显示拓展搜索
  </label>
  <label class="k-settings-item">
    <input type="checkbox" name="autoplay" />
    自动播放
  </label>
  <label class="k-settings-item">
    <input type="checkbox" name="autoNext" />
    自动下一集
  </label>
  <label class="k-settings-item">
    <input type="checkbox" name="continuePlay" />
    记忆播放位置
  </label>
  <label class="k-settings-item">
    <input type="checkbox" name="showProgress" />
    显示底部进度条
  </label>
</div>
`);
  const searchActionsHTML = popover(`
<div class="plyr__controls__item k-popover k-text-btn">
  <span class="k-text-btn-text">友链</span>
</div>
`, `<ul class="k-menu"></ul>`);
  const scriptInfo = (video, githubIssueURL) => `
<table class="script-info">
  <tbody>
  <tr><td>脚本版本</td><td>${"1.24.0"}</td></tr>
  <tr>
    <td>脚本源码</td>
    <td>
      <a target="_blank" rel="noreferrer" href="https://github.com/IronKinoko/agefans-enhance">GitHub</a>
      <a target="_blank" rel="noreferrer" href="https://github.com/IronKinoko/agefans-enhance/releases">更新记录</a>
      </td>
  </tr>
  <tr>
    <td>报错/意见</td>
    <td>
      <a target="_blank" rel="noreferrer" href="${githubIssueURL}">GitHub Issues</a>
      <a target="_blank" rel="noreferrer" href="https://gf.qytechs.cn/zh-CN/scripts/424023-agefans-enhance/feedback">Greasy Fork镜像 反馈</a>
    </td>
  </tr>
  ${video
    ? `<tr><td colspan="2" class="info-title">视频信息</td></tr>
     <tr><td>视频链接</td><td>${video.src}</td></tr>
     <tr><td>视频信息</td><td>${video.videoWidth} x ${video.videoHeight}</td></tr>`
    : ''}
  <tr><td colspan="2" class="info-title">快捷键</td></tr>
  <tr>
    <td colspan="2">
      <div class="shortcuts-wrap">
        <table class="shortcuts-table">
          <tbody>
            <tr><td><span class="key">W</span></td><td>宽屏</td></tr>
            <tr><td><span class="key">F</span></td><td>全屏</td></tr>
            <tr><td><span class="key">←</span></td><td>步退5s</td></tr>
            <tr><td><span class="key">→</span></td><td>步进5s</td></tr>
            <tr><td><span class="key">${renderKey('Shift')} ←</span></td><td>步退30s</td></tr>
            <tr><td><span class="key">${renderKey('Shift')} →</span></td><td>步进30s</td></tr>
            <tr><td><span class="key">${renderKey('Alt')} ←</span></td><td>步退60s</td></tr>
            <tr><td><span class="key">${renderKey('Alt')} →</span></td><td>步进60s</td></tr>
            <tr><td><span class="key">${renderKey('Ctrl')} ←</span></td><td>步退90s</td></tr>
            <tr><td><span class="key">${renderKey('Ctrl')} →</span></td><td>步进90s</td></tr>
            <tr><td><span class="key">D</span></td><td>弹幕开关</td></tr>
            </tbody>
        </table>
        <table class="shortcuts-table">
          <tbody>
            <tr><td><span class="key">esc</span></td><td>退出全屏/宽屏</td></tr>
            <tr>
              <td>
                <span class="key carousel">
                  <span>[</span>
                  <span>P</span>
                  <span>PgUp</span>
                </span>
              </td>
              <td>上一集</td>
            </tr>
            <tr>
              <td>
                <span class="key carousel">
                  <span>]</span>
                  <span>N</span>
                  <span>PgDn</span>
                </span>
              </td>
              <td>下一集</td>
            </tr>
            <tr><td><span class="key">Z</span></td><td>原速播放</td></tr>
            <tr><td><span class="key">X</span></td><td>减速播放</td></tr>
            <tr><td><span class="key">C</span></td><td>加速播放</td></tr>
            <tr><td><span class="key">↑</span></td><td>音量+</td></tr>
            <tr><td><span class="key">↓</span></td><td>音量-</td></tr>
            <tr><td><span class="key">M</span></td><td>静音</td></tr>
            <tr><td><span class="key">I</span></td><td>画中画</td></tr>
            <tr><td><span class="key">?</span></td><td>脚本信息</td></tr>
          </tbody>
        </table>
      </div>
    </td>
  </tr>
  </tbody>
</table>
`;
  const issueBody = (src = '') => `# 文字描述
<!-- 如果有需要额外描述,或者提意见可以写在下面空白处 -->


# 网址链接
${window.location.href}

# 视频链接
${src}

# 环境
userAgent: ${navigator.userAgent}
脚本版本: ${"1.24.0"}
`;
  const progressHTML = `
<div class="k-player-progress">
  <div class="k-player-progress-current"></div>
  <div class="k-player-progress-buffer"></div>
</div>
`;

  var css$6 = "#k-player-wrapper {\n  position: relative;\n  width: 100%;\n  height: 100%;\n  background: #000;\n  overflow: hidden;\n  font-size: 14px;\n  --k-player-error-background: url();\n  --k-player-tsuma-length: 2;\n  --plyr-line-height: 1;\n  --plyr-tooltip-background: var(--k-player-background);\n  --plyr-tooltip-color: var(--k-player-color);\n  --plyr-range-thumb-background: url() no-repeat\n    center/contain;\n  --plyr-range-thumb-width: 18px;\n  --plyr-range-thumb-height: 18px;\n  --plyr-color-main: var(--k-player-primary-color);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range] {\n  cursor: pointer;\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]::-webkit-slider-thumb {\n  transform: scale(0);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]:hover::-webkit-slider-thumb {\n  transform: scale(1);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]:active::-webkit-slider-thumb {\n  transition: all 0.1s linear;\n  box-shadow: none;\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range].shake-0:active::-webkit-slider-thumb {\n  transform: scale(1.3) rotate(15deg);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range].shake-1:active::-webkit-slider-thumb {\n  transform: scale(1.3) rotate(-15deg);\n}\n#k-player-wrapper.k-player-widescreen {\n  position: fixed;\n  left: 0;\n  top: 0;\n  z-index: 10000;\n}\n#k-player-wrapper .k-player-contianer {\n  width: 100%;\n  height: 100%;\n}\n#k-player-wrapper #k-player-loading,\n#k-player-wrapper #k-player-error {\n  position: absolute;\n  left: 0;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  z-index: 10;\n  font-size: 66px;\n  color: white;\n  pointer-events: none;\n  background: black;\n}\n#k-player-wrapper .k-player-error-img {\n  background: var(--k-player-error-background) no-repeat center/contain;\n  width: 200px;\n  height: 200px;\n  opacity: 0.4;\n}\n#k-player-wrapper .k-player-error-info {\n  text-align: center;\n  padding: 24px;\n  font-size: 18px;\n}\n#k-player-wrapper #k-player-pip {\n  position: absolute;\n  left: 0;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  z-index: 10;\n  pointer-events: none;\n}\n#k-player-wrapper .k-player-tsuma {\n  width: 200px;\n  height: 200px;\n  position: absolute;\n  bottom: 0;\n  right: 0;\n  background: no-repeat center/contain;\n  opacity: 0.1;\n  z-index: -1;\n  pointer-events: none;\n}\n#k-player-wrapper .k-player-tsuma[data-bg-idx=\"0\"] {\n  background-image: url();\n}\n#k-player-wrapper .k-player-tsuma[data-bg-idx=\"1\"] {\n  background-image: url();\n}\n#k-player-wrapper .k-player-center {\n  width: 100%;\n  height: 100%;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  justify-content: center;\n}\n#k-player-wrapper #k-player-header {\n  transform: translateY(0);\n  transition: transform 0.3s;\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  padding: 8px;\n  text-align: right;\n}\n#k-player-wrapper #k-player-header .k-player-question-icon {\n  font-size: 24px;\n  width: 1em;\n  height: 1em;\n  color: white;\n  cursor: pointer;\n}\n#k-player-wrapper .plyr--hide-controls #k-player-header {\n  transform: translateY(-100%);\n}\n#k-player-wrapper .plyr {\n  width: 100%;\n  height: 100%;\n}\n#k-player-wrapper .plyr__control svg {\n  font-size: 18px;\n}\n#k-player-wrapper video {\n  display: block;\n}\n#k-player-wrapper .plyr__next svg {\n  transform: scale(1.7);\n}\n#k-player-wrapper .plyr__widescreen svg {\n  transform: scale(1.3);\n}\n#k-player-wrapper .plyr--hide-cursor {\n  cursor: none;\n}\n#k-player-wrapper .plyr__control span:not(.plyr__tooltip) {\n  color: inherit;\n}\n#k-player-wrapper .plyr--hide-controls .k-player-progress {\n  opacity: 1;\n  transition: opacity 0.3s ease-in 0.2s;\n}\n#k-player-wrapper .k-player-fullscreen .k-player-progress,\n#k-player-wrapper .k-player-fullscreen [data-plyr=widescreen] {\n  display: none;\n}\n#k-player-wrapper .k-player-progress {\n  opacity: 0;\n  transition: opacity 0.2s ease-out;\n  height: 2px;\n  width: 100%;\n  position: absolute;\n  bottom: 0;\n}\n#k-player-wrapper .k-player-progress .k-player-progress-current {\n  position: absolute;\n  left: 0;\n  top: 0;\n  height: 100%;\n  z-index: 2;\n  background-color: var(--k-player-primary-color);\n}\n#k-player-wrapper .k-player-progress .k-player-progress-buffer {\n  position: absolute;\n  left: 0;\n  top: 0;\n  z-index: 1;\n  height: 100%;\n  background-color: var(--plyr-video-progress-buffered-background, rgba(255, 255, 255, 0.25));\n}\n#k-player-wrapper .plyr__controls {\n  z-index: 20;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item:first-child {\n  margin-right: 0;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item.plyr__progress__container {\n  position: absolute;\n  top: 15px;\n  left: 10px;\n  right: 10px;\n  --plyr-range-track-height: 2px;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item.plyr__progress__container:hover {\n  --plyr-range-track-height: 4px;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item.plyr__time--duration.plyr__time {\n  margin-right: auto;\n}\n#k-player-wrapper .plyr__controls .k-text-btn {\n  display: inline-block;\n  padding: 0 8px;\n  text-align: center;\n}\n#k-player-wrapper .plyr__controls .k-text-btn-text {\n  line-height: 32px;\n  user-select: none;\n}\n@media (max-width: 480px) {\n  #k-player-wrapper .plyr__controls {\n    padding-top: 30px;\n  }\n  #k-player-wrapper [data-plyr=pip],\n#k-player-wrapper [data-plyr=snapshot],\n#k-player-wrapper [data-plyr=widescreen] {\n    display: none;\n  }\n}\n\n.lds-spinner {\n  color: official;\n  display: inline-block;\n  position: relative;\n  width: 80px;\n  height: 80px;\n}\n.lds-spinner div {\n  transform-origin: 40px 40px;\n  animation: lds-spinner 1.2s linear infinite;\n}\n.lds-spinner div::after {\n  content: \" \";\n  display: block;\n  position: absolute;\n  top: 3px;\n  left: 37px;\n  width: 6px;\n  height: 18px;\n  border-radius: 20%;\n  background: #fff;\n}\n.lds-spinner div:nth-child(1) {\n  transform: rotate(0deg);\n  animation-delay: -1.1s;\n}\n.lds-spinner div:nth-child(2) {\n  transform: rotate(30deg);\n  animation-delay: -1s;\n}\n.lds-spinner div:nth-child(3) {\n  transform: rotate(60deg);\n  animation-delay: -0.9s;\n}\n.lds-spinner div:nth-child(4) {\n  transform: rotate(90deg);\n  animation-delay: -0.8s;\n}\n.lds-spinner div:nth-child(5) {\n  transform: rotate(120deg);\n  animation-delay: -0.7s;\n}\n.lds-spinner div:nth-child(6) {\n  transform: rotate(150deg);\n  animation-delay: -0.6s;\n}\n.lds-spinner div:nth-child(7) {\n  transform: rotate(180deg);\n  animation-delay: -0.5s;\n}\n.lds-spinner div:nth-child(8) {\n  transform: rotate(210deg);\n  animation-delay: -0.4s;\n}\n.lds-spinner div:nth-child(9) {\n  transform: rotate(240deg);\n  animation-delay: -0.3s;\n}\n.lds-spinner div:nth-child(10) {\n  transform: rotate(270deg);\n  animation-delay: -0.2s;\n}\n.lds-spinner div:nth-child(11) {\n  transform: rotate(300deg);\n  animation-delay: -0.1s;\n}\n.lds-spinner div:nth-child(12) {\n  transform: rotate(330deg);\n  animation-delay: 0s;\n}\n\n@keyframes lds-spinner {\n  0% {\n    opacity: 1;\n  }\n  100% {\n    opacity: 0;\n  }\n}\n.script-info {\n  width: 100%;\n}\n.script-info * {\n  box-sizing: border-box;\n  font-size: 14px;\n  font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Oxygen, Ubuntu, Cantarell, \"Open Sans\", \"Helvetica Neue\", sans-serif;\n}\n.script-info tbody tr td:first-child {\n  white-space: nowrap;\n  width: 77px;\n}\n.script-info td {\n  padding: 8px;\n  border-bottom: 1px solid #f1f1f1;\n  word-wrap: break-word;\n  word-break: break-all;\n}\n.script-info .info-title {\n  font-weight: 600;\n  padding-top: 24px;\n}\n.script-info a {\n  color: var(--k-player-primary-color);\n  padding: 4px 8px;\n  border-radius: 4px;\n  text-decoration: none;\n}\n.script-info a:hover {\n  text-decoration: underline;\n  background-color: #f1f1f1;\n}\n.script-info .shortcuts-wrap {\n  display: flex;\n  width: 100%;\n  margin: -8px;\n}\n.script-info .shortcuts-table {\n  flex: 1;\n}\n.script-info .key {\n  display: inline-block;\n  position: relative;\n  background: #333;\n  text-align: center;\n  color: #eee;\n  border-radius: 4px;\n  padding: 2px 0;\n  width: 56px;\n  box-sizing: border-box;\n  border: 1px solid #444;\n  box-shadow: 0 2px 0 1px #222;\n  border-bottom-color: #555;\n  user-select: none;\n}\n.script-info .carousel {\n  position: relative;\n  display: flex;\n  flex-wrap: nowrap;\n  overflow: hidden;\n}\n.script-info .carousel span {\n  display: block;\n  width: 100%;\n  height: 100%;\n  flex-basis: 100%;\n  flex-shrink: 0;\n  animation: carousel-3 6s infinite alternate;\n}\n\n@keyframes carousel-3 {\n  0% {\n    transform: translateX(0);\n  }\n  20% {\n    transform: translateX(0);\n  }\n  40% {\n    transform: translateX(-100%);\n  }\n  60% {\n    transform: translateX(-100%);\n  }\n  80% {\n    transform: translateX(-200%);\n  }\n  100% {\n    transform: translateX(-200%);\n  }\n}";
  n(css$6,{});

  const i18n = {
      restart: '重播',
      rewind: '快退 {seektime}s',
      play: '播放(空格键)',
      pause: '暂停(空格键)',
      fastForward: '快进 {seektime}s',
      seek: 'Seek',
      seekLabel: '{currentTime} / {duration}',
      played: '已播放',
      buffered: '已缓冲',
      currentTime: '当前时间',
      duration: '片长',
      volume: '音量',
      mute: '静音(M)',
      unmute: '取消静音(M)',
      enableCaptions: '显示字幕',
      disableCaptions: '隐藏字幕',
      download: '下载',
      enterFullscreen: '进入全屏(F)',
      exitFullscreen: '退出全屏(F)',
      frameTitle: '标题名称: {title}',
      captions: '字幕',
      settings: '设置',
      pip: '画中画(I)',
      menuBack: '返回上级',
      speed: '倍速',
      normal: '1.0x',
      quality: '分辨率',
      loop: '循环',
      start: '开始',
      end: '结束',
      all: '全部',
      reset: '重置',
      disabled: '禁用',
      enabled: '启用',
      advertisement: '广告',
      qualityBadge: {
          2160: '4K',
          1440: 'HD',
          1080: 'HD',
          720: 'HD',
          576: 'SD',
          480: 'SD',
      },
  };
  const MediaErrorMessage = {
      1: '你中止了媒体播放',
      2: '网络错误',
      3: '文件损坏',
      4: '资源有问题看不了',
      5: '资源被加密了',
  };
  const defaultConfig = {
      speed: 1,
      continuePlay: true,
      autoNext: true,
      showProgress: true,
      volume: 1,
      showSearchActions: true,
      autoplay: true,
  };
  class KPlayer {
      /**
       * @typedef {Object} EnhanceOpts
       * @property {HTMLVideoElement} [video]
       * @property {boolean} [eventToParentWindow]
       *
       * Creates an instance of KPlayer.
       * @param {string|Element} selector
       * @param {Plyr.Options & EnhanceOpts} [opts]
       */
      constructor(selector, opts = {}) {
          this.isHoverControls = false;
          this.setCurrentTimeLogThrottled = throttle(() => {
              this.setCurrentTimeLog();
          }, 1000);
          this.hideControlsDebounced = debounce(() => {
              const dom = document.querySelector('.plyr');
              if (!this.isHoverControls)
                  dom === null || dom === void 0 ? void 0 : dom.classList.add('plyr--hide-controls');
          }, 1000);
          this.hideCursorDebounced = debounce(() => {
              const dom = document.querySelector('.plyr');
              dom === null || dom === void 0 ? void 0 : dom.classList.add('plyr--hide-cursor');
          }, 1000);
          this.isJumped = false;
          this.jumpToLogTime = throttle(() => {
              if (this.isJumped)
                  return;
              if (this.currentTime < 3) {
                  this.isJumped = true;
                  const logTime = this.getCurrentTimeLog();
                  if (logTime && this.plyr.duration - logTime > 10) {
                      this.message.info(`已自动跳转至历史播放位置 ${parseTime(logTime)}`);
                      this.currentTime = logTime;
                  }
              }
          }, 1000);
          this.$wrapper = $('<div id="k-player-wrapper"/>').replaceAll(selector);
          this.$loading = $(loadingHTML);
          this.$error = $(errorHTML);
          this.$pip = $(pipHTML);
          this.$video = (opts.video ? $(opts.video) : $('<video />')).attr('id', 'k-player');
          this.$progress = $(progressHTML);
          this.$header = $('<div id="k-player-header"/>');
          this.$wrapper.append(this.$video);
          this.localConfigKey = 'kplayer';
          this.statusSessionKey = 'k-player-status';
          this.localPlayTimeKey = 'k-player-play-time';
          this.localConfig = Object.assign({}, defaultConfig, gm.getItem(this.localConfigKey));
          this.plyr = new Plyr__default["default"]('#k-player', Object.assign({ autoplay: this.localConfig.autoplay, keyboard: { global: true }, controls: [
                  'play',
                  'progress',
                  'current-time',
                  'duration',
                  'mute',
                  'volume',
                  'pip',
                  'fullscreen',
              ], storage: { enabled: false }, seekTime: 5, volume: this.localConfig.volume, speed: { options: speedList, selected: 1 }, i18n, tooltips: {
                  controls: true,
                  seek: true,
              } }, opts));
          this.$videoWrapper = this.$wrapper.find('.plyr');
          this.$videoWrapper.append(this.$loading, this.$error, this.$pip, this.$progress, this.$header);
          this.message = new Message(this.$videoWrapper);
          this.eventMap = {};
          this.isWideScreen = false;
          this.wideScreenBodyStyles = {};
          this.tsumaLength = +getComputedStyle(this.$wrapper[0])
              .getPropertyValue('--k-player-tsuma-length')
              .trim();
          this.curentTsuma = -1;
          this.injectSettings();
          this.injectSpeed();
          this.injectQuestion();
          this.injectNext();
          this.injectSreen();
          this.injectSearchActions();
          KPlayer.plguinList.forEach((setup) => setup(this));
          this.initEvent();
          if (opts.eventToParentWindow) {
              this.eventToParentWindow();
          }
          const status = session.getItem(this.statusSessionKey);
          if (status) {
              session.removeItem(this.statusSessionKey);
              this.toggleWidescreen(status);
          }
      }
      static register(setup) {
          this.plguinList.push(setup);
      }
      setCurrentTimeLog(time) {
          const store = local.getItem(this.localPlayTimeKey, {});
          store[this.playTimeStoreKey] = Math.floor(time !== null && time !== void 0 ? time : this.plyr.currentTime);
          local.setItem(this.localPlayTimeKey, store);
      }
      getCurrentTimeLog() {
          const store = local.getItem(this.localPlayTimeKey, {});
          return store[this.playTimeStoreKey];
      }
      get playTimeStoreKey() {
          if (this.src.startsWith('blob')) {
              return location.origin + location.pathname + location.search;
          }
          else {
              return this.src;
          }
      }
      initEvent() {
          this.on('loadstart', () => {
              this.$loading.show();
              this.hideError();
          });
          this.on('canplay', () => {
              this.$loading.hide();
              if (this.localConfig.autoplay) {
                  this.plyr.play();
              }
              if (this.localConfig.continuePlay) {
                  this.jumpToLogTime();
              }
          });
          this.on('error', () => {
              this.setCurrentTimeLog(0);
              this.$searchActions.show();
              const code = this.media.error.code;
              this.$loading.hide();
              this.showError(MediaErrorMessage[code] || this.src);
              if (code === 3) {
                  const countKey = 'skip-error-retry-count' + window.location.search;
                  let skipErrorRetryCount = parseInt(session.getItem(countKey) || '0');
                  if (skipErrorRetryCount < 3) {
                      skipErrorRetryCount++;
                      const duration = 2 * skipErrorRetryCount;
                      this.message
                          .info(`视频源出现问题,第${skipErrorRetryCount}次尝试跳过${duration}s错误片段`, 4000)
                          .then(() => {
                          this.trigger('skiperror', 2 * skipErrorRetryCount);
                      });
                      session.setItem(countKey, skipErrorRetryCount.toString());
                  }
                  else {
                      this.message
                          .info(`视频源出现问题,多次尝试失败,请手动跳过错误片段`, 4000)
                          .then(() => {
                          this.trigger('skiperror', 0);
                      });
                      session.removeItem(countKey);
                  }
              }
              else {
                  const $dom = $('<div>视频播放失败,点击此处暂时关闭脚本功能,使用原生播放器观看</div>').css('cursor', 'pointer');
                  $dom.on('click', () => {
                      this.message.destroy();
                      session.setItem('stop-use', true);
                      window.location.reload();
                  });
                  this.message.info($dom, 10000);
              }
          });
          this.on('pause', () => {
              this.hideControlsDebounced();
          });
          this.on('enterfullscreen', () => {
              this.$videoWrapper.addClass('k-player-fullscreen');
          });
          this.on('exitfullscreen', () => {
              this.$videoWrapper.removeClass('k-player-fullscreen');
          });
          this.on('volumechange', () => {
              this.configSaveToLocal('volume', this.plyr.volume);
          });
          this.on('timeupdate', () => {
              this.setCurrentTimeLogThrottled();
              this.$progress
                  .find('.k-player-progress-current')
                  .css('width', (this.currentTime / this.plyr.duration || 0) * 100 + '%');
              this.$progress
                  .find('.k-player-progress-buffer')
                  .css('width', this.plyr.buffered * 100 + '%');
          });
          this.on('ended', () => {
              if (this.localConfig.autoNext) {
                  this.trigger('next');
              }
          });
          this.on('enterpictureinpicture', () => {
              this.setRandomTsuma();
              this.$pip.fadeIn();
          });
          this.on('leavepictureinpicture', () => {
              this.$pip.fadeOut();
          });
          keybind([
              // 进退 30s
              'shift+ArrowLeft',
              'shift+ArrowRight',
              // 进退 60s
              'alt+ArrowLeft',
              'alt+ArrowRight',
              // 进退 90s
              'ctrl+ArrowLeft',
              'ctrl+ArrowRight',
              'meta+ArrowLeft',
              'meta+ArrowRight',
              // 下一集
              'n',
              ']',
              '】',
              'PageDown',
              // 上一集
              'p',
              '[',
              '【',
              'PageUp',
              // 切换网页全屏
              'w',
              // 关闭网页全屏
              'Escape',
              // 播放速度
              'z',
              'x',
              'c',
              // 截图
              'ctrl+s',
              'meta+s',
              // 画中画,
              'i',
          ], (e, key) => {
              switch (key) {
                  case 'ctrl+ArrowLeft':
                  case 'meta+ArrowLeft':
                  case 'shift+ArrowLeft':
                  case 'alt+ArrowLeft':
                  case 'ctrl+ArrowRight':
                  case 'meta+ArrowRight':
                  case 'shift+ArrowRight':
                  case 'alt+ArrowRight': {
                      e.stopPropagation();
                      e.preventDefault();
                      const time = {
                          'ctrl+ArrowLeft': 90,
                          'meta+ArrowLeft': 90,
                          'shift+ArrowLeft': 30,
                          'alt+ArrowLeft': 60,
                          'ctrl+ArrowRight': 90,
                          'meta+ArrowRight': 90,
                          'shift+ArrowRight': 30,
                          'alt+ArrowRight': 60,
                      }[key];
                      this.message.destroy();
                      if (e.key === 'ArrowLeft') {
                          this.currentTime = Math.max(0, this.currentTime - time);
                          this.message.info(`步退${time}s`);
                      }
                      else {
                          this.currentTime = Math.min(this.currentTime + time, this.plyr.duration);
                          this.message.info(`步进${time}s`);
                      }
                      break;
                  }
                  case 'n':
                  case ']':
                  case '】':
                  case 'PageDown':
                      e.preventDefault();
                      this.trigger('next');
                      break;
                  case 'p':
                  case '[':
                  case '【':
                  case 'PageUp':
                      e.preventDefault();
                      this.trigger('prev');
                      break;
                  case 'w':
                      if (this.plyr.fullscreen.active)
                          break;
                      this.toggleWidescreen();
                      break;
                  case 'Escape':
                      if (this.plyr.fullscreen.active || !this.isWideScreen)
                          break;
                      this.toggleWidescreen(false);
                      break;
                  case 'z':
                      this.speed = 1;
                      break;
                  case 'x':
                  case 'c': {
                      let idx = speedList.indexOf(this.speed);
                      const newIdx = key === 'x'
                          ? Math.max(0, idx - 1)
                          : Math.min(speedList.length - 1, idx + 1);
                      if (newIdx === idx)
                          break;
                      const speed = speedList[newIdx];
                      this.speed = speed;
                      break;
                  }
                  case 'i':
                      this.plyr.pip = !this.plyr.pip;
                      break;
              }
          });
          $('.plyr__controls button,.plyr__controls input').on('mouseleave', (e) => {
              e.target.blur();
          });
          const playerEl = document.querySelector('.plyr');
          playerEl.addEventListener('mousemove', () => {
              playerEl.classList.remove('plyr--hide-cursor');
              this.hideCursorDebounced();
              if (this.plyr.paused) {
                  this.hideControlsDebounced();
              }
          });
          const controlsEl = document.querySelector('.plyr__controls');
          controlsEl.addEventListener('mouseenter', () => {
              this.isHoverControls = true;
          });
          controlsEl.addEventListener('mouseleave', () => {
              this.isHoverControls = false;
          });
          this.initInputEvent();
      }
      initInputEvent() {
          let timeId;
          const $dom = $("#k-player-wrapper input[type='range']");
          $dom.trigger('mouseup').off('mousedown').off('mouseup');
          $dom.on('mousedown', function () {
              clearInterval(timeId);
              let i = 0;
              timeId = window.setInterval(() => {
                  $(this)
                      .removeClass()
                      .addClass(`shake-${i++ % 2}`);
              }, 100);
          });
          $dom.on('mouseup', function () {
              clearInterval(timeId);
              $(this).removeClass();
          });
      }
      on(event, callback) {
          if ([
              'prev',
              'next',
              'enterwidescreen',
              'exitwidescreen',
              'skiperror',
          ].includes(event)) {
              if (!this.eventMap[event])
                  this.eventMap[event] = [];
              this.eventMap[event].push(callback);
          }
          else {
              this.plyr.on(event, callback);
          }
      }
      trigger(event, params) {
          const fnList = this.eventMap[event] || [];
          fnList.forEach((fn) => {
              fn(this, params);
          });
      }
      injectSettings() {
          this.$settings = $(settingsHTML);
          this.$settings
              .find('[name=showSearchActions]')
              .prop('checked', this.localConfig.showSearchActions)
              .on('change', (e) => {
              const checked = e.target.checked;
              this.configSaveToLocal('showSearchActions', checked);
              this.$searchActions.toggle(checked);
          });
          this.$settings
              .find('[name=autoNext]')
              .prop('checked', this.localConfig.autoNext)
              .on('change', (e) => {
              const checked = e.target.checked;
              this.configSaveToLocal('autoNext', checked);
          });
          this.$settings
              .find('[name=showProgress]')
              .prop('checked', this.localConfig.showProgress)
              .on('change', (e) => {
              const checked = e.target.checked;
              this.configSaveToLocal('showProgress', checked);
              this.$progress.toggle(checked);
          });
          if (!this.localConfig.showProgress) {
              this.$progress.css('display', 'none');
          }
          this.$settings
              .find('[name=autoplay]')
              .prop('checked', this.localConfig.autoplay)
              .on('change', (e) => {
              const checked = e.target.checked;
              this.configSaveToLocal('autoplay', checked);
              this.plyr.autoplay = checked;
          });
          this.$settings
              .find('[name=continuePlay]')
              .prop('checked', this.localConfig.continuePlay)
              .on('change', (e) => {
              const checked = e.target.checked;
              this.configSaveToLocal('continuePlay', checked);
          });
          this.$settings.insertAfter('.plyr__controls__item.plyr__volume');
      }
      configSaveToLocal(key, value) {
          this.localConfig[key] = value;
          gm.setItem(this.localConfigKey, this.localConfig);
      }
      injectSpeed() {
          this.$speed = $(speedHTML);
          const speedItems = this.$speed.find('.k-speed-item');
          const localSpeed = this.localConfig.speed;
          speedItems.each((_, el) => {
              const speed = +el.dataset.speed;
              if (speed === localSpeed) {
                  el.classList.add('k-menu-active');
              }
              $(el).on('click', () => {
                  this.speed = speed;
              });
          });
          this.plyr.speed = localSpeed;
          this.$speed
              .find('#k-speed-text')
              .text(localSpeed === 1 ? '倍速' : localSpeed + 'x');
          this.$speed.insertBefore('.plyr__controls__item.plyr__volume');
      }
      injectQuestion() {
          $(`<svg class="k-player-question-icon"><use xlink:href="#question"/></svg>`)
              .appendTo(this.$header)
              .on('click', () => {
              showInfo();
          });
      }
      injectNext() {
          $($('#plyr__next').html())
              .insertBefore('.plyr__controls__item.plyr__progress__container')
              .on('click', () => {
              this.trigger('next');
          });
      }
      injectSreen() {
          $($('#plyr__widescreen').html())
              .insertBefore('[data-plyr="fullscreen"]')
              .on('click', () => {
              this.toggleWidescreen();
          });
      }
      async injectSearchActions() {
          this.$searchActions = $(searchActionsHTML).toggle(this.localConfig.showSearchActions);
          const actions = await runtime.getSearchActions();
          if (actions.length === 0)
              return;
          this.$searchActions.find('.k-menu').append(actions.map(({ name, search }) => {
              return $(`<li class="k-menu-item k-speed-item">${name}</li>`).on('click', search);
          }));
          this.$searchActions.insertBefore(this.$speed);
      }
      toggleWidescreen(bool = !this.isWideScreen) {
          if (this.isWideScreen === bool)
              return;
          this.isWideScreen = bool;
          session.setItem(this.statusSessionKey, this.isWideScreen);
          if (this.isWideScreen) {
              this.wideScreenBodyStyles = $('body').css(['overflow']);
              $('body').css('overflow', 'hidden');
              this.$wrapper.addClass('k-player-widescreen');
              $('.plyr__widescreen').addClass('plyr__control--pressed');
          }
          else {
              $('body').css(this.wideScreenBodyStyles);
              this.$wrapper.removeClass('k-player-widescreen');
              $('.plyr__widescreen').removeClass('plyr__control--pressed');
          }
          this.trigger(this.isWideScreen ? 'enterwidescreen' : 'exitwidescreen');
      }
      get media() {
          return this.$video[0];
      }
      set src(src) {
          this.isJumped = false;
          if (src.includes('.m3u8')) {
              if (!Hls__default["default"].isSupported())
                  throw new Error('不支持播放 hls 文件');
              const hls = new Hls__default["default"]();
              hls.loadSource(src);
              hls.attachMedia(this.media);
          }
          else {
              this.$video.attr('src', src);
          }
      }
      get src() {
          return this.media.src;
      }
      set currentTime(value) {
          this.plyr.currentTime = value;
      }
      get currentTime() {
          return this.plyr.currentTime;
      }
      get speed() {
          return this.plyr.speed;
      }
      set speed(speed) {
          this.plyr.speed = speed;
          const speedItems = this.$speed.find('.k-speed-item');
          speedItems.each((_, el) => {
              if (speed === +el.dataset.speed) {
                  el.classList.add('k-menu-active');
              }
              else {
                  el.classList.remove('k-menu-active');
              }
          });
          this.$speed.find('#k-speed-text').text(speed === 1 ? '倍速' : speed + 'x');
          this.message.destroy();
          this.message.info(`视频速度:${speed}`);
          this.configSaveToLocal('speed', speed);
      }
      showError(text) {
          this.setRandomTsuma();
          this.$error.show().find('.k-player-error-info').text(text);
      }
      hideError() {
          this.$error.hide();
      }
      setRandomTsuma() {
          this.curentTsuma = ++this.curentTsuma % this.tsumaLength;
          this.$wrapper.find('.k-player-tsuma').attr('data-bg-idx', this.curentTsuma);
      }
      eventToParentWindow() {
          const evnetKeys = [
              'prev',
              'next',
              'enterwidescreen',
              'exitwidescreen',
              'skiperror',
              'progress',
              'playing',
              'play',
              'pause',
              'timeupdate',
              'volumechange',
              'seeking',
              'seeked',
              'ratechange',
              'ended',
              'enterfullscreen',
              'exitfullscreen',
              'captionsenabled',
              'captionsdisabled',
              'languagechange',
              'controlshidden',
              'controlsshown',
              'ready',
              'loadstart',
              'loadeddata',
              'loadedmetadata',
              'canplay',
              'canplaythrough',
              'stalled',
              'waiting',
              'emptied',
              'cuechange',
              'error',
          ];
          evnetKeys.forEach((key) => {
              this.on(key, () => {
                  const video = this.media;
                  const info = {
                      width: video.videoWidth,
                      height: video.videoHeight,
                      currentTime: video.currentTime,
                      src: video.src,
                      duration: video.duration,
                  };
                  window.parent.postMessage({ key, video: info }, '*');
              });
          });
      }
  }
  KPlayer.plguinList = [];
  function addReferrerMeta(content) {
      if ($('meta[name=referrer]').length === 0) {
          $('head').append(`<meta name="referrer" content="${content}">`);
      }
      else {
          const $meta = $('meta[name=referrer]');
          $meta.attr('content', content);
      }
  }
  function showInfo() {
      const video = $('#k-player')[0];
      const githubIssueURL = genIssueURL({
          title: '🐛[Bug]',
          body: issueBody(video === null || video === void 0 ? void 0 : video.src),
      });
      modal({
          title: '脚本信息',
          content: scriptInfo(video, githubIssueURL),
      });
  }
  keybind(['?', '?'], (e) => {
      if (!document.fullscreenElement) {
          e.stopPropagation();
          e.preventDefault();
          showInfo();
      }
  });

  function request(opts) {
      let { url, method, params } = opts;
      if (params) {
          let u = new URL(url);
          Object.keys(params).forEach((key) => {
              const value = params[key];
              if (value !== undefined && value !== null) {
                  u.searchParams.set(key, params[key]);
              }
          });
          url = u.toString();
      }
      return new Promise((resolve, reject) => {
          GM_xmlhttpRequest({
              url,
              method: method || 'GET',
              responseType: 'json',
              onload: (res) => {
                  resolve(res.response);
              },
              onerror: reject,
          });
      });
  }

  function createStorage(storageKey) {
      function storage(key, value) {
          const store = local.getItem(storageKey, {});
          if (value) {
              store[key] = value;
              local.setItem(storageKey, store);
          }
          else {
              return store[key];
          }
      }
      return storage;
  }
  const storageAnimeName = createStorage('k-player-danmaku-anime-name');
  const storageEpisodeName = createStorage('k-player-danmaku-episode-name');
  function createLock() {
      let prev;
      return function check(deps) {
          if (prev === deps)
              return true;
          prev = deps;
          return false;
      };
  }
  const episodeIdLock = createLock();
  const searchAnimeLock = createLock();
  function convert32ToHex(color) {
      return '#' + parseInt(color).toString(16);
  }
  function rangePercent(min, input, max) {
      input = Math.min(max, Math.max(min, input));
      return ((input - min) / (max - min)) * 100;
  }
  function addRangeListener(opts) {
      const { $dom, name, onInput, player, onChange } = opts;
      const $valueDom = $('<div style="width:45px;flex-shrink:0;text-align:right;white-space:nowrap;"></div>');
      $valueDom.insertAfter($dom);
      const min = parseFloat($dom.attr('min'));
      const max = parseFloat($dom.attr('max'));
      const setStyle = () => {
          const value = parseFloat($dom.val());
          player.configSaveToLocal(name, value);
          onInput === null || onInput === void 0 ? void 0 : onInput(value);
          $valueDom.text((value * 100).toFixed(0) + '%');
          $dom.css('--value', rangePercent(min, value, max) + '%');
      };
      $dom.val(player.localConfig[name]);
      $dom.on('input', setStyle);
      $dom.on('change', () => {
          onChange === null || onChange === void 0 ? void 0 : onChange(parseFloat($dom.val()));
      });
      setStyle();
  }
  function getCheckboxGroupValue($dom) {
      const ret = [];
      $dom.each((_, el) => {
          if (el.checked)
              ret.push(el.value);
      });
      return ret;
  }
  function setCheckboxGroupValue($dom, value) {
      $dom.each((_, el) => {
          if (value.includes(el.value)) {
              el.checked = true;
          }
      });
  }

  // https://api.acplay.net/swagger/ui/index#/
  async function getComments(episodeId) {
      const res = await request({
          url: `https://api.acplay.net/api/v2/comment/${episodeId}?withRelated=true&chConvert=1`,
      });
      return res.comments
          .map((o) => {
          const [time, type, color] = o.p.split(',');
          return {
              mode: { 1: 'rtl', 4: 'bottom', 5: 'top' }[type] || 'rtl',
              text: o.m,
              time: parseFloat(time),
              style: { color: convert32ToHex(color) },
          };
      })
          .sort((a, b) => a.time - b.time);
  }
  async function searchAnimeWithEpisode(anime, episode) {
      const res = await request({
          url: 'https://api.acplay.net/api/v2/search/episodes',
          params: { anime, episode },
      });
      return res.animes;
  }

  var css$5 = ".k-tab {\n  flex: 1;\n  white-space: nowrap;\n  cursor: pointer;\n  text-align: center;\n  padding: 8px 16px;\n}\n.k-tabs {\n  display: flex;\n  position: relative;\n  border-bottom: 1px solid rgba(255, 255, 255, 0.2);\n}\n.k-tabs-wrapper {\n  text-align: left;\n}\n.k-tabs-wrapper * {\n  box-sizing: border-box;\n}\n.k-tab-indicator {\n  position: absolute;\n  width: 0;\n  height: 1px;\n  left: 0;\n  bottom: -1px;\n  background-color: var(--k-player-primary-color);\n  transition: all 0.3s;\n}\n.k-tabs-panes {\n  display: flex;\n  flex-wrap: nowrap;\n  transition: all 0.3s;\n}\n.k-tab-pane {\n  flex: 0 0 100%;\n  padding: 8px;\n}";
  n(css$5,{});

  function tabs(opts) {
      const tabsHTML = [];
      const tabsContentHTML = [];
      opts.forEach((tab, idx) => {
          const tabHTML = `<div class="k-tab" data-idx="${idx}">${tab.name}</div>`;
          const contentHTML = `<div class="k-tab-pane">${tab.content}</div>`;
          tabsHTML.push(tabHTML);
          tabsContentHTML.push(contentHTML);
      });
      const $root = $(`<div class="k-tabs-wrapper">
    <div class="k-tabs">
      ${tabsHTML.join('')}
      <div class="k-tab-indicator"></div>
    </div>
    <div class="k-tabs-panes">${tabsContentHTML.join('')}</div>
  </div>`);
      const $indicator = $root.find('.k-tab-indicator');
      $root.find('.k-tab').on('click', (e) => {
          $root.find('.k-tab').removeClass('active');
          const $tab = $(e.target).addClass('active');
          const idx = parseInt($tab.attr('data-idx'));
          $root.find('.k-tabs-panes').css('transform', `translateX(-${idx * 100}%)`);
          function updateIndictor() {
              const width = $tab.outerWidth();
              if (width)
                  $indicator.css({ width, left: idx * width });
              else
                  requestAnimationFrame(updateIndictor);
          }
          updateIndictor();
      });
      $root.find('.k-tab:first').trigger('click');
      return $root;
  }

  const $danmakuOverlay = tabs([
      {
          name: '搜索',
          content: `<div id="k-player-danmaku-search-form">
      <label>
        <span>搜索番剧名称</span>
        <input type="text" id="animeName" />
      </label>
      <div style="min-height:24px; padding-top:4px">
        <span id="tips"></span>
      </div>
      <label>
        <span>番剧名称</span>
        <select id="animes"></select>
      </label>
      <label>
        <span>章节</span>
        <select id="episodes"></select>
      </label>
    </div>`,
      },
      {
          name: '设置',
          content: `
    <div id="k-player-danmaku-setting-form" class="k-settings-list">
      <label class="k-settings-item">
        <input type="checkbox" name="showDanmaku" />
        <span>显示弹幕(D)</span>
        </label>
      <label class="k-settings-item">
        <input type="checkbox" name="showPbp" />
        <span>显示高能进度条</span>
      </label>
      <label class="k-settings-item">
        <span>透明度&#12288;</span>
        <input type="range" name="opacity" step="0.01" min="0" max="1" />
      </label>
      <label class="k-settings-item">
        <span>弹幕速度</span>
        <input type="range" name="danmakuSpeed" step="0.01" min="0.5" max="1.5" />
      </label>
      <label class="k-settings-item" title="基准为 24 分钟 3000 条弹幕">
        <span>弹幕密度</span>
        <input type="range" name="danmakuDensity" step="0.01" min="0.5" max="2" />
      </label>
      <div class="k-settings-item" style="height:24px">
        <div>弹幕类型</div>
        <label class="k-settings-item" title="顶部弹幕">
          <input type="checkbox" name="danmakuMode" value="top"/>
          <span>顶</span>
        </label>
        <label class="k-settings-item" title="底部弹幕">
          <input type="checkbox" name="danmakuMode" value="bottom"/>
          <span>底</span>
        </label>
        <label class="k-settings-item" title="彩色弹幕">
          <input type="checkbox" name="danmakuMode" value="color" />
          <span>彩</span>
        </label>
      </div>
    </div>
    `,
      },
      {
          name: '过滤',
          content: `
    <div id="k-player-danmaku-filter-form" class="k-settings-list">
      <input name="filter-input" placeholder="回车添加屏蔽词"/>

      <div id="k-player-danmaku-filter-table">
        <div class="ft-row" style="pointer-events:none;">
          <div class="ft-content">内容(<span id="filter-count"></span>)</div>
          <div class="ft-op">操作</div>
        </div>
        <div class="ft-body"></div>
      </div>
    </div>
    `,
      },
  ]);
  $danmakuOverlay.attr('id', 'k-player-danmaku-overlay');
  const $danmakuSwitch = $(`<div class="k-switch k-danmaku-switch">
<input class="k-switch-input" type="checkbox" />
<div class="k-switch-label">
  <div class="k-switch-dot">
    <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 10 10" width="1em" height="1em"><path fill="currentColor" d="M1.311 3.759l-.153 1.438h2.186c0 1.832-.066 3.056-.175 3.674-.131.618-.688.959-1.683 1.023-.284 0-.568-.021-.874-.043L.317 8.818c.284.032.59.053.896.053.546 0 .852-.17.929-.511.077-.341.12-1.076.12-2.204H0l.306-3.344h1.847V1.427H.098V.479h3.18v3.28H1.311zM4 1.747h1.311A8.095 8.095 0 004.492.426L5.53.085c.306.426.579.873.809 1.363l-.689.299h1.508c.306-.544.569-1.129.809-1.747l1.082.373c-.219.511-.47.969-.743 1.374h1.268V6.23H7.322v.82H10v1.044H7.322V10H6.208V8.094H3.607V7.05h2.601v-.82H4V1.747zm4.568 3.557v-.831H7.322v.831h1.246zm-2.36 0v-.831H5.016v.831h1.192zM5.016 3.557h1.191v-.873H5.016v.873zm2.306-.873v.873h1.246v-.873H7.322z"></path></svg>
  </div>
</div>
</div>`);
  const $danmaku = popover($danmakuSwitch, $danmakuOverlay);
  const $danmakuContainer = $('<div id="k-player-danmaku"></div>');
  const $pbp = $(`
<svg
  viewBox="0 0 1000 100"
  preserveAspectRatio="none"
  id="k-player-pbp"
>
  <defs>
    <clipPath id="k-player-pbp-curve-path" clipPathUnits="userSpaceOnUse">
      <path d=""></path>
    </clipPath>
  </defs>

  <g
    fill-opacity="0.2"
    clip-path="url(#k-player-pbp-curve-path)"
    hover-bind="1"
  >
    <rect x="0" y="0" width="100%" height="100%" fill="rgb(255,255,255)"></rect>
    <rect id="k-player-pbp-played-path" x="0" y="0" width="0" height="100%" fill="currentColor"></rect>
  </g>
</svg>
`);

  var css$4 = "#k-player-danmaku {\n  position: absolute;\n  left: 0;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  z-index: 10;\n  pointer-events: none;\n}\n#k-player-danmaku * {\n  font-size: 25px;\n  font-family: SimHei, \"Microsoft JhengHei\", Arial, Helvetica, sans-serif;\n  font-weight: bold;\n  text-shadow: black 1px 0px 1px, black 0px 1px 1px, black 0px -1px 1px, black -1px 0px 1px;\n  line-height: 1.3;\n}\n#k-player-danmaku-overlay {\n  width: 210px;\n}\n#k-player-danmaku-search-form > * {\n  font-size: 14px;\n  box-sizing: border-box;\n  text-align: left;\n}\n#k-player-danmaku-search-form input,\n#k-player-danmaku-search-form select {\n  display: block;\n  margin-top: 4px;\n  width: 100%;\n}\n#k-player-danmaku-search-form label {\n  display: block;\n}\n#k-player-danmaku-search-form label span {\n  line-height: 1.4;\n}\n#k-player-danmaku-search-form label + label {\n  margin-top: 8px;\n}\n#k-player-danmaku-setting-form {\n  padding: 0;\n}\n#k-player-danmaku-setting-form input {\n  margin: 0;\n}\n#k-player-danmaku-filter-form {\n  padding: 0;\n}\n#k-player-danmaku-filter-form input {\n  width: 100%;\n}\n#k-player-danmaku-filter-table {\n  margin-top: 8px;\n}\n#k-player-danmaku-filter-table .ft-body {\n  height: 114px;\n  overflow: auto;\n}\n#k-player-danmaku-filter-table .ft-body::-webkit-scrollbar {\n  display: none;\n}\n#k-player-danmaku-filter-table .ft-row {\n  display: flex;\n  border-radius: 4px;\n  transition: all 0.15s;\n}\n#k-player-danmaku-filter-table .ft-row:hover {\n  background: var(--k-player-background-highlight);\n}\n#k-player-danmaku-filter-table .ft-content {\n  padding: 4px 8px;\n  flex: 1px;\n  min-width: 0;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  white-space: nowrap;\n}\n#k-player-danmaku-filter-table .ft-op {\n  flex-shrink: 0;\n  padding: 4px 8px;\n}\n#k-player-danmaku-filter-table a {\n  color: white;\n  cursor: pointer;\n  transition: color 0.15s;\n}\n#k-player-danmaku-filter-table a:hover {\n  color: var(--k-player-primary-color);\n}\n\n#k-player-pbp {\n  position: absolute;\n  top: -17px;\n  height: 28px;\n  -webkit-appearance: none;\n  left: 0;\n  position: absolute;\n  margin-left: calc(var(--plyr-range-thumb-height, 13px) * -0.5);\n  margin-right: calc(var(--plyr-range-thumb-height, 13px) * -0.5);\n  width: calc(100% + var(--plyr-range-thumb-height, 13px));\n  pointer-events: none;\n}\n\n#k-player-pbp-played-path {\n  color: var(--k-player-primary-color);\n}\n\n.plyr__controls__item.plyr__progress__container:hover #k-player-pbp {\n  top: -18px;\n}\n\n.k-danmaku-switch {\n  margin: 0 8px;\n}\n\n.k-switch {\n  position: relative;\n  width: 30px;\n  height: 20px;\n  display: flex;\n  align-items: center;\n}\n.k-switch-input {\n  position: absolute;\n  width: 100%;\n  height: 100%;\n  left: 0;\n  top: 0;\n  opacity: 0;\n  margin: 0;\n  cursor: pointer;\n  z-index: 1;\n}\n.k-switch-input + .k-switch-label {\n  --color: #757575;\n  --left: 2px;\n}\n.k-switch-input:checked + .k-switch-label {\n  --color: var(--k-player-primary-color);\n  --left: 12px;\n}\n.k-switch-label {\n  width: 30px;\n  height: 20px;\n  position: relative;\n  background: var(--color);\n  transition: all 0.3s;\n  border-radius: 15px;\n}\n.k-switch-dot {\n  position: absolute;\n  color: var(--color);\n  transition: all 0.3s;\n  height: 16px;\n  width: 16px;\n  font-size: 10px;\n  background: white;\n  border-radius: 50%;\n  left: var(--left);\n  top: 50%;\n  transform: translateY(-50%);\n  display: flex;\n  justify-content: center;\n  align-items: center;\n}";
  n(css$4,{});

  /**
   * 逻辑源自 bilibili 高能进度条 svg path 规则
   */
  function createProgressBarPower(duration, comments) {
      const data = comments.map((cmt) => cmt.time);
      // svg总长度
      const svgMaxLength = 1000;
      // 分成 size 份
      const size = 100;
      // svg步长
      const stepLength = svgMaxLength / size;
      // 每份时间多长
      const stepTime = duration / size;
      const counts = [];
      let i = 0, j = 0;
      // 统计的每份基础时间里弹幕数量
      while (i++ <= size) {
          const base = stepTime * i;
          let count = 0;
          while (data[j++] < base) {
              count++;
          }
          counts.push(count);
      }
      // 为了美观,100 高度的 svg,底部 20 预留显示出来,其他从 80 开始算
      let start = 'M 0 100, L ';
      let end = ' 1000.0 80.0 L 1000 100 Z';
      // 巅峰弹幕数量
      const maxCount = Math.max(Math.max(...counts), 1);
      const points = [];
      counts.forEach((count, i) => {
          const x = i * stepLength;
          const y = (1 - count / maxCount) * 80;
          // 做一份相同的数据,让曲线在两侧顺滑变化
          if (i !== 0)
              points.push({ x: (x - stepLength / 2).toFixed(2), y: y.toFixed(2) });
          if (i !== counts.length - 1)
              points.push({ x: x.toFixed(2), y: y.toFixed(2) });
      });
      for (let i = 0; i < points.length;) {
          const p1 = points[i++]; // data point
          const p2 = points[i++]; // mid point
          start += `${p1.x} ${p1.y} C ${p2.x} ${p1.y}, ${p2.x} ${p2.y},`;
      }
      $pbp.find('path').attr('d', start + end);
      $('.plyr__controls__item.plyr__progress__container .plyr__progress').append($pbp);
  }

  function createFilter(player, refreshDanmaku) {
      const $filter = $('#k-player-danmaku-filter-form');
      const $input = $filter.find('[name="filter-input"]');
      $input.on('keypress', (e) => {
          if (e.key === 'Enter')
              addFilter($input.val());
      });
      function refreshFilterDom() {
          const filters = player.localConfig.danmakuFilter;
          let html = '';
          filters.forEach((filter, idx) => {
              html += `<div class="ft-row">
    <div class="ft-content">${filter}</div>
    <div class="ft-op"><a key="delete" data-idx="${idx}">删除</a></div>
    </div>`;
          });
          $filter.find('.ft-body').empty().append(html);
          $filter.find('[key=delete]').on('click', (e) => {
              const idx = parseInt($(e.target).attr('data-idx'));
              deleteFilter(idx);
          });
          $filter.find('#filter-count').text(filters.length);
      }
      function deleteFilter(idx) {
          player.localConfig.danmakuFilter.splice(idx, 1);
          player.configSaveToLocal('danmakuFilter', player.localConfig.danmakuFilter);
          refreshDanmaku();
          refreshFilterDom();
      }
      function addFilter(text) {
          const filters = player.localConfig.danmakuFilter;
          $input.val('');
          if (!text || filters.includes(text))
              return;
          filters.push(text);
          player.configSaveToLocal('danmakuFilter', filters);
          refreshFilterDom();
          refreshDanmaku();
      }
      refreshFilterDom();
  }

  Object.assign(defaultConfig, {
      showDanmaku: false,
      opacity: 0.6,
      showPbp: false,
      danmakuSpeed: 1,
      danmakuMode: ['top', 'color'],
      danmakuFilter: [],
  });
  var State;
  (function (State) {
      State[State["unSearched"] = 0] = "unSearched";
      State[State["searched"] = 1] = "searched";
      State[State["findEpisode"] = 2] = "findEpisode";
      State[State["getComments"] = 3] = "getComments";
  })(State || (State = {}));
  const baseDanmkuSpeed = 130;
  let state = State.unSearched;
  const $animeName = $danmaku.find('#animeName');
  const $animes = $danmaku.find('#animes');
  const $episodes = $danmaku.find('#episodes');
  const $tips = $danmaku.find('#tips');
  const $danmakuSwitcher = $danmakuSwitch.find('.k-switch-input');
  const $showDanmaku = $danmaku.find("[name='showDanmaku']");
  const $showPbp = $danmaku.find("[name='showPbp']");
  const $opacity = $danmaku.find("[name='opacity']");
  const $danmakuSpeed = $danmaku.find("[name='danmakuSpeed']");
  const $danmakuDensity = $danmaku.find("[name='danmakuDensity']");
  const $danmakuMode = $danmaku.find("[name='danmakuMode']");
  let core;
  let comments;
  let player$5;
  let videoInfo;
  function refreshDanmaku() {
      stop();
      autoStart();
  }
  const showTips = (message) => {
      $tips.text(message).fadeIn('fast').delay(1500).fadeOut('fast');
  };
  const stop = () => {
      core === null || core === void 0 ? void 0 : core.hide();
  };
  const start = () => {
      function run() {
          if (!player$5.media.duration)
              return requestAnimationFrame(run);
          if (!comments)
              return;
          if (player$5.localConfig.showDanmaku) {
              if (!core) {
                  core = new Danmaku__default["default"]({
                      container: $danmakuContainer[0],
                      media: player$5.media,
                      comments: adjustCommentCount(comments),
                  });
              }
              else {
                  core.reload(adjustCommentCount(comments));
                  core.show();
              }
              core.speed = baseDanmkuSpeed * player$5.localConfig.danmakuSpeed;
          }
          if (player$5.localConfig.showPbp) {
              createProgressBarPower(player$5.media.duration, comments);
          }
      }
      requestAnimationFrame(run);
  };
  const adjustCommentCount = (comments) => {
      if (!comments)
          return;
      let ret = comments;
      // 过滤弹幕
      ret = ret.filter((cmt) => !player$5.localConfig.danmakuFilter.some((o) => cmt.text.includes(o)));
      // 过滤弹幕类型
      const mode = player$5.localConfig.danmakuMode;
      if (!mode.includes('color')) {
          ret = ret.filter((cmt) => cmt.style.color === '#ffffff');
      }
      if (!mode.includes('bottom')) {
          ret = ret.filter((cmt) => cmt.mode !== 'bottom');
      }
      if (!mode.includes('top')) {
          ret = ret.filter((cmt) => cmt.mode !== 'top');
      }
      // 24 分钟 3000 弹幕,按比例缩放
      const maxLength = Math.round((3000 / (24 * 60)) *
          player$5.media.duration *
          player$5.localConfig.danmakuDensity);
      // 均分
      if (ret.length > maxLength) {
          let ratio = ret.length / maxLength;
          ret = [...new Array(maxLength)].map((_, i) => ret[Math.floor(i * ratio)]);
      }
      return ret;
  };
  const loadEpisode = async (episodeId) => {
      if (episodeIdLock(episodeId))
          return;
      stop();
      comments = await getComments(episodeId);
      state = State.getComments;
      start();
      player$5.message.info(`番剧:${$animes.find(':selected').text()}`, 2000);
      player$5.message.info(`章节:${$episodes.find(':selected').text()}`, 2000);
      player$5.message.info(`已加载 ${comments.length} 条弹幕`, 2000);
  };
  const searchAnime = async (name) => {
      state = State.searched;
      name || (name = $animeName.val());
      if (!name || name.length < 2)
          return showTips('番剧名称不少于2个字');
      if (searchAnimeLock(name))
          return;
      const animes = await searchAnimeWithEpisode(name);
      if (animes.length === 0)
          return showTips('未搜索到番剧');
      updateAnimes(animes);
      findEpisode(animes);
  };
  const findEpisode = async (animes) => {
      if (!animes)
          return;
      const anime = animes.find((anime) => anime.animeTitle ===
          (storageAnimeName(videoInfo.rawName) || videoInfo.rawName));
      if (anime) {
          let episodeName = videoInfo.episode;
          let episode;
          let storedEpisodeId = storageEpisodeName(`${videoInfo.rawName}.${videoInfo.episode}`);
          if (storedEpisodeId) {
              episode = anime.episodes.find((episode) => String(episode.episodeId) === storedEpisodeId);
          }
          if (!episode && !isNaN(+episodeName)) {
              episode = anime.episodes.find((episode) => episode.episodeTitle.includes(episodeName));
          }
          if (episode) {
              state = State.findEpisode;
              $animeName.val(anime.animeTitle);
              $animes.val(anime.animeId);
              $animes.trigger('change');
              $episodes.val(episode.episodeId);
              $episodes.trigger('change');
              return;
          }
      }
      player$5.message.info('弹幕未能自动匹配数据源,请手动搜索');
  };
  const initEvents = (name) => {
      $animeName.val(name);
      $animeName.on('keypress', (e) => {
          if (e.key === 'Enter')
              searchAnime($animeName.val());
      });
      $animeName.on('blur', (e) => {
          searchAnime($animeName.val());
      });
      $animes.on('change', (e) => {
          const animeId = $(e.target).val();
          const animes = $animes.data('animes');
          const anime = animes.find((anime) => String(anime.animeId) === animeId);
          if (!anime)
              return;
          storageAnimeName(videoInfo.rawName, anime.animeTitle);
          updateEpisodes(anime);
      });
      $episodes.on('change', (e) => {
          const episodeId = $(e.target).val();
          storageAnimeName(videoInfo.rawName, $episodes.data('anime').animeTitle);
          storageEpisodeName(`${videoInfo.rawName}.${videoInfo.episode}`, episodeId);
          loadEpisode(episodeId);
      });
      $danmakuSwitcher
          .prop('checked', player$5.localConfig.showDanmaku)
          .on('click', () => {
          switchDanmaku();
      });
      const resizeOb = new ResizeObserver(() => {
          core === null || core === void 0 ? void 0 : core.resize();
      });
      resizeOb.observe(player$5.$videoWrapper[0]);
      const mutationOb = new MutationObserver(async () => {
          searchAnimeLock(Math.random());
          Object.assign(videoInfo, await runtime.getCurrentVideoNameAndEpisode());
          state = State.searched;
          autoStart();
      });
      mutationOb.observe(player$5.media, { attributeFilter: ['src'] });
      player$5.initInputEvent();
      // 绑定快捷键
      keybind(['d'], () => switchDanmaku());
      $showDanmaku
          .prop('checked', player$5.localConfig.showDanmaku)
          .on('change', (e) => {
          switchDanmaku(e.target.checked);
      });
      // 绑定 pbp 相关事件
      $showPbp.prop('checked', player$5.localConfig.showPbp).on('change', (e) => {
          const chekced = e.target.checked;
          $pbp.toggle(chekced);
          player$5.configSaveToLocal('showPbp', chekced);
          if (chekced)
              autoStart();
      });
      $pbp.toggle(player$5.localConfig.showPbp || false);
      const $pbpPlayed = $pbp.find('#k-player-pbp-played-path');
      player$5.on('timeupdate', () => {
          $pbpPlayed.attr('width', (player$5.currentTime / player$5.plyr.duration || 0) * 100 + '%');
      });
      addRangeListener({
          $dom: $opacity,
          name: 'opacity',
          onInput: (v) => {
              $danmakuContainer.css({ opacity: v });
          },
          player: player$5,
      });
      addRangeListener({
          $dom: $danmakuSpeed,
          name: 'danmakuSpeed',
          onChange: (v) => {
              if (core)
                  core.speed = baseDanmkuSpeed * v;
          },
          player: player$5,
      });
      addRangeListener({
          $dom: $danmakuDensity,
          name: 'danmakuDensity',
          onChange: refreshDanmaku,
          player: player$5,
      });
      setCheckboxGroupValue($danmakuMode, player$5.localConfig.danmakuMode);
      $danmakuMode.on('change', () => {
          const modes = getCheckboxGroupValue($danmakuMode);
          player$5.configSaveToLocal('danmakuMode', modes);
          if (core) {
              refreshDanmaku();
          }
      });
      createFilter(player$5, refreshDanmaku);
  };
  function switchDanmaku(bool) {
      bool !== null && bool !== void 0 ? bool : (bool = !player$5.localConfig.showDanmaku);
      player$5.configSaveToLocal('showDanmaku', bool);
      $danmakuSwitcher.prop('checked', bool);
      $showDanmaku.prop('checked', bool);
      player$5.message.info(`弹幕${bool ? '开启' : '关闭'}`);
      if (bool) {
          autoStart();
      }
      else {
          stop();
      }
  }
  // 更新 anime select
  const updateAnimes = (animes) => {
      const html = animes.reduce((html, anime) => html + `<option value="${anime.animeId}">${anime.animeTitle}</option>`, '');
      $animes.data('animes', animes);
      $animes.html(html);
      updateEpisodes(animes[0]);
      showTips(`找到 ${animes.length} 部番剧`);
  };
  // 更新 episode select
  const updateEpisodes = (anime) => {
      const { episodes } = anime;
      const html = episodes.reduce((html, episode) => html +
          `<option value="${episode.episodeId}">${episode.episodeTitle}</option>`, '');
      $episodes.data('anime', anime);
      $episodes.html(html);
      $episodes.val('');
  };
  function autoStart() {
      if (!(player$5.localConfig.showDanmaku || player$5.localConfig.showPbp))
          return;
      switch (state) {
          case State.unSearched:
              searchAnime();
              break;
          case State.searched:
              findEpisode($animes.data('animes'));
              break;
          case State.findEpisode:
              $episodes.trigger('change');
              break;
          case State.getComments:
              start();
              break;
      }
  }
  async function setup(_player) {
      player$5 = _player;
      const info = await runtime.getCurrentVideoNameAndEpisode();
      if (!info)
          return;
      videoInfo = info;
      player$5.$videoWrapper.append($danmakuContainer);
      $danmaku.insertBefore(player$5.$searchActions);
      let defaultSearchName = storageAnimeName(videoInfo.rawName) || videoInfo.name;
      initEvents(defaultSearchName);
      autoStart();
  }

  KPlayer.register(setup);

  var css$3 = ".agefans-wrapper #relates-series .relates_series {\n  display: block;\n  padding: 4px 0px;\n}";
  n(css$3,{});

  let player$4;
  function replacePlayer$6() {
      const dom = document.querySelector('#age_playfram');
      const fn = () => {
          if (!dom.src)
              return;
          let url = new URL(dom.src);
          if (url.origin === location.origin) {
              let videoURL = url.searchParams.get('url');
              if (videoURL) {
                  addReferrerMeta('same-origin');
                  initPlayer(parseToURL(videoURL));
                  mutationOb.disconnect();
              }
          }
          else {
              const message = new Message('#ageframediv');
              message.info('这个视频似乎是第三方链接,并非由agefans自身提供,将使用默认播放器播放', 3000);
              mutationOb.disconnect();
          }
      };
      const mutationOb = new MutationObserver(fn);
      mutationOb.observe(dom, { attributes: true });
      fn();
  }
  function showCurrentLink(vurl) {
      const decodeVurl = parseToURL(vurl);
      const title = [$('#detailname a').text(), getActivedom$2().text()].join(' ');
      if ($('#current-link').length) {
          $('#current-link').text(decodeVurl);
          $('#current-link').attr('href', decodeVurl);
          return;
      }
      $(ageBlock({
          title: '本集链接:',
          content: `<a class="res_links" id="current-link" download="${title}" rel="noreferrer" href="${decodeVurl}">${decodeVurl}</a>`,
      })).insertBefore($('.baseblock:contains(网盘资源)'));
  }
  function gotoPrevPart() {
      const dom = getActivedom$2().parent().prev().find('a');
      if (dom.length) {
          switchPart$8(dom.data('href'), dom);
      }
  }
  function gotoNextPart() {
      const dom = getActivedom$2().parent().next().find('a');
      if (dom.length) {
          switchPart$8(dom.data('href'), dom);
      }
  }
  function getActivedom$2() {
      return $("li a[style*='color: rgb(238, 0, 0)']");
  }
  // switch part retry count
  let retryCount = 0;
  let switchLoading = false;
  /**
   *
   * @param {string} href
   * @param {JQuery<HTMLAnchorElement>} $dom
   * @param {boolean} [push]
   */
  async function switchPart$8(href, $dom, push = true) {
      try {
          if (switchLoading === true)
              return;
          switchLoading = true;
          retryCount++;
          push && player$4.message.info(`即将播放${$dom.text()}`);
          const vurl = await getVurlWithLocal(href);
          push && player$4.message.destroy();
          const speed = player$4.plyr.speed;
          player$4.src = vurl;
          player$4.plyr.speed = speed;
          const $active = getActivedom$2();
          $active.css({ color: '', border: '' });
          $dom.css({ color: 'rgb(238, 0, 0)', border: '1px solid rgb(238, 0, 0)' });
          const title = document.title.replace($active.text(), $dom.text());
          push && history.pushState({}, title, href);
          document.title = title;
          showCurrentLink(vurl);
          his.logHistory();
          retryCount = 0;
          switchLoading = false;
      }
      catch (error) {
          switchLoading = false;
          if (retryCount > 3) {
              console.error(error);
              window.location.href = href.toString();
          }
          else {
              switchPart$8(href, $dom, push);
          }
      }
  }
  function resetVideoHeight() {
      const $root = $('#ageframediv');
      /** @type {HTMLVideoElement} */
      const video = player$4.media;
      const ratio = video.videoWidth / video.videoHeight;
      const width = $root.width();
      $root.height(width / ratio);
  }
  function updateTime(time = 0) {
      var _a;
      const id = (_a = location.pathname.match(/\/play\/(\d*)/)) === null || _a === void 0 ? void 0 : _a[1];
      if (!id)
          return;
      his.setTime(id, Math.floor(time));
  }
  function addListener() {
      player$4.on('next', () => {
          gotoNextPart();
      });
      player$4.on('prev', () => {
          gotoPrevPart();
      });
      player$4.plyr.once('canplay', () => {
          resetVideoHeight();
      });
      player$4.on('error', () => {
          removeLocal(getActivedom$2().data('href'));
      });
      const update = throttle(() => {
          updateTime(player$4.currentTime);
      }, 1000);
      player$4.on('timeupdate', () => {
          update();
      });
      player$4.on('skiperror', (_, duration) => {
          if (duration === 0) {
              updateTime(0);
          }
          else {
              updateTime(player$4.currentTime + duration);
          }
          window.location.reload();
      });
      window.addEventListener('popstate', () => {
          const href = location.pathname + location.search;
          const $dom = $(`[data-href='${href}']`);
          if ($dom.length) {
              switchPart$8(href, $dom, false);
          }
          else {
              window.location.reload();
          }
      });
  }
  function replaceHref() {
      $('.movurl:visible li a').each(function () {
          const href = $(this).attr('href');
          $(this)
              .removeAttr('href')
              .attr('data-href', href)
              .css('cursor', 'pointer')
              .on('click', (e) => {
              e.preventDefault();
              switchPart$8(href, $(this));
          });
      });
  }
  function initPlayer(vurl) {
      player$4 = new KPlayer('#age_playfram');
      showCurrentLink(vurl);
      addListener();
      player$4.src = vurl;
      saveLocal(getActivedom$2().data('href'), vurl);
  }
  function useOriginPlayer() {
      const message = new Message('#ageframediv');
      message.info('脚本功能已暂时禁用,使用原生播放器观看,右下角可启动脚本', 3000);
      const $dom = $(`<span>启用脚本</span>`)
          .css({ color: '#60b8cc', cursor: 'pointer' })
          .on('click', () => {
          session.removeItem('stop-use');
          window.location.reload();
      });
      $('#wangpan-div .blocktitle')
          .css({ display: 'flex', justifyContent: 'space-between' })
          .append($dom);
  }
  async function showRelatesSeries() {
      const info = await fetch(location.pathname.replace('play', 'detail')).then((r) => r.text());
      const $series = $(info).find('li.relates_series');
      $series.find('a').each((_, anchor) => pagePreview(anchor, anchor.href));
      $(ageBlock({ title: '相关动画:', content: '<ul id="relates-series"></ul>' }))
          .insertAfter('.baseblock:contains(种子资源)')
          .find('ul')
          .append($series);
  }
  function playModule$a() {
      $('#cpraid').remove();
      if (session.getItem('stop-use')) {
          useOriginPlayer();
          return;
      }
      his.logHistory();
      $('.fullscn').remove();
      replaceHref();
      replacePlayer$6();
      initGetAllVideoURL();
      showRelatesSeries();
      $('.ul_li_a8 > .anime_icon1 > a:nth-child(1)').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
  }

  function rankModule() {
      $('.div_right_r_3 ul > li > a').each((_, anchor) => pagePreview(anchor, anchor.href));
  }

  function recommendModule() {
      $('ul.ul_li_a6 > li > a').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
  }

  function updateModule() {
      $('ul.ul_li_a6 > li > a').each((_, anchor) => pagePreview(anchor.parentElement, anchor.href));
  }

  runtime.register({
      domains: ['age.tv', 'agemys', 'agefans'],
      opts: [
          {
              test: '*',
              run: () => {
                  $('body').addClass('agefans-wrapper');
                  settingModule();
                  historyModule();
              },
          },
          { test: '/play', run: playModule$a },
          { test: '/detail', run: detailModule },
          { test: '/recommend', run: recommendModule },
          { test: '/update', run: updateModule },
          { test: '/rank', run: rankModule },
          { test: /^\/$/, run: homeModule },
      ],
      search: {
          name: 'agefans',
          search: (name) => `https://www.agemys.com/search?query=${name}&page=1`,
          getSearchName: () => $('#detailname a').text(),
          getEpisode: () => $("li a[style*='color: rgb(238, 0, 0)']").text(),
      },
  });

  let player$3;
  function replacePlayer$5() {
      const dom = document.querySelector('#playleft iframe[allowfullscreen="true"]');
      const fn = () => {
          if (!dom.src)
              return;
          let url = new URL(dom.src);
          let videoURL = url.searchParams.get('url');
          if (videoURL) {
              player$3 = new KPlayer('#beyond-play-box');
              player$3.src = parseToURL(videoURL);
              initEvent$2();
              mutationOb.disconnect();
          }
      };
      const mutationOb = new MutationObserver(fn);
      mutationOb.observe(dom, { attributes: true });
      fn();
  }
  function initEvent$2() {
      player$3.on('prev', () => unsafeWindow.MacPlayer.GoPreUrl());
      player$3.on('next', () => unsafeWindow.MacPlayer.GoNextUrl());
  }
  function playModule$9() {
      $('body').addClass('www88dmw-wrapper');
      $('.kp_flash_box .mb').remove();
      replacePlayer$5();
  }

  var css$2 = ".www88dmw-wrapper .menuBoxbg {\n  z-index: 999;\n}\n.www88dmw-wrapper #k-player-wrapper div:not(.plyr__progress) {\n  margin: initial;\n}";
  n(css$2,{});

  function searchAction(name) {
      const $form = $(`
  <form action="http://www.88dmw.com/index.php?m=vod-search" method="post">
    <input type="text" id="wd" name="wd">
  </form>
  `);
      $form.hide();
      $form.find('#wd').val(decodeURIComponent(name));
      $form.appendTo('body');
      $form.trigger('submit');
  }

  function www88dmwSetup() {
      try {
          Object.defineProperty(unsafeWindow, 'devtoolsDetector', {
              writable: false,
              value: null,
          });
          document.oncontextmenu = null;
          // eslint-disable-next-line no-empty
      }
      catch (error) { }
  }
  runtime.register({
      domains: ['88dmw'],
      opts: [{ test: '/play', setup: www88dmwSetup, run: playModule$9 }],
      search: {
          name: '动漫岛',
          search: searchAction,
          getSearchName: () => $('.play_menu a:last').text(),
          disabledInIframe: true,
      },
  });

  var css$1 = ".yhdm-wrapper {\n  font-family: -apple-system, BlinkMacSystemFont, Segoe UI, Roboto, Helvetica Neue, Arial, Noto Sans, sans-serif, Apple Color Emoji, Segoe UI Emoji, Segoe UI Symbol, Noto Color Emoji;\n}\n.yhdm-wrapper .play,\n.yhdm-wrapper #playbox,\n.yhdm-wrapper .bofang,\n.yhdm-wrapper .pp .player {\n  height: 540px;\n}";
  n(css$1,{});

  let player$2;
  function replacePlayer$4() {
      const dom = document.querySelector('#play2');
      const fn = () => {
          if (!dom.src)
              return;
          let url = new URL(dom.src);
          let videoURL = url.searchParams.get('vid');
          if (videoURL) {
              player$2 = new KPlayer('#play2');
              player$2.src = parseToURL(videoURL);
              initEvent$1();
              mutationOb.disconnect();
          }
      };
      const mutationOb = new MutationObserver(fn);
      mutationOb.observe(dom, { attributes: true });
      fn();
  }
  function switchPart$7(next) {
      var _a;
      (_a = getActivedom$1().parent()[next ? 'next' : 'prev']().find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  function getActivedom$1() {
      return $(`.movurls:visible li a[href='${location.pathname}']`);
  }
  function initEvent$1() {
      player$2.on('prev', () => switchPart$7(false));
      player$2.on('next', () => switchPart$7(true));
  }
  function playModule$8() {
      $('body').addClass('yhdm-wrapper');
      $('#adl').remove();
      $('#adr').remove();
      $('#adv').remove();
      $('.fullscn').remove();
      replacePlayer$4();
  }

  runtime.register({
      domains: ['imomoe.live'],
      opts: [{ test: '/player', run: playModule$8 }],
  });

  function replacePlayer$3() {
      new KPlayer('#dplayer', {
          video: $('video')[0],
          eventToParentWindow: true,
      });
  }
  function switchPart$6(next) {
      var _a;
      let directionRight = true;
      const re = /\/v\/\d+-(\d+)/;
      let prevID;
      Array.from($('.movurls a')).forEach((a) => {
          if (re.test(a.href)) {
              const [, id] = a.href.match(re);
              if (prevID)
                  directionRight = +prevID < +id;
              prevID = id;
          }
      });
      let direction = ['prev', 'next'];
      if (!next)
          direction.reverse();
      if (!directionRight)
          direction.reverse();
      (_a = $('.movurls .sel')[direction[1]]().find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  function playModule$7() {
      $('body').addClass('yhdm-wrapper');
      window.addEventListener('message', (e) => {
          var _a;
          if (!Reflect.has(e.data, 'key'))
              return;
          const key = e.data.key;
          if (key === 'prev')
              switchPart$6(false);
          if (key === 'next')
              switchPart$6(true);
          if (key === 'enterwidescreen') {
              $('body').css('overflow', 'hidden');
              $('#playbox iframe').css({
                  position: 'fixed',
                  left: 0,
                  right: 0,
                  bottom: 0,
                  top: 0,
              });
          }
          if (key === 'exitwidescreen') {
              $('body').css('overflow', '');
              $('#playbox iframe').removeAttr('style');
          }
          if (key === 'getSearchName') {
              const iframe = $('#playbox iframe')[0];
              (_a = iframe.contentWindow) === null || _a === void 0 ? void 0 : _a.postMessage({ key: 'getSearchName', name: $('.gohome.l > h1 > a').text() }, '*');
          }
          if (key === 'openLink') {
              window.open(e.data.url);
          }
      });
      window.addEventListener('keydown', (e) => {
          if (document.activeElement !== document.body)
              return;
          $('#playbox iframe')[0].focus();
          if (e.key === ' ')
              e.preventDefault();
      });
  }
  function playInIframeModule$1() {
      if (location.search.includes('vid')) {
          replacePlayer$3();
      }
  }

  runtime.register({
      domains: ['yhdm.so', 'yinghuacd.com'],
      opts: [
          { test: ['/v'], run: playModule$7 },
          { test: ['vid'], runInIframe: true, run: playInIframeModule$1 },
      ],
      search: {
          name: '樱花动漫1',
          search: (name) => `http://www.yinghuacd.com/search/${name}/`,
          getSearchName: async () => {
              return new Promise((resolve) => {
                  const fn = (e) => {
                      if (e.data.key === 'getSearchName') {
                          resolve(e.data.name);
                          window.removeEventListener('message', fn);
                      }
                  };
                  window.addEventListener('message', fn);
                  parent.postMessage({ key: 'getSearchName' }, '*');
              });
          },
      },
  });

  let player$1;
  function replacePlayer$2() {
      const dom = document.querySelector('#yh_playfram');
      const fn = () => {
          if (!dom.src)
              return;
          let url = new URL(dom.src);
          let videoURL = url.searchParams.get('url');
          if (videoURL) {
              player$1 = new KPlayer('#yh_playfram');
              player$1.src = parseToURL(videoURL);
              initEvent();
              mutationOb.disconnect();
          }
      };
      const mutationOb = new MutationObserver(fn);
      mutationOb.observe(dom, { attributes: true });
      fn();
  }
  function switchPart$5(next) {
      var _a;
      (_a = getActivedom().parent()[next ? 'next' : 'prev']().find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  function getActivedom() {
      return $(".movurl:visible li a[style*='color: rgb(255, 255, 255)']");
  }
  function initEvent() {
      player$1.on('prev', () => switchPart$5(false));
      player$1.on('next', () => switchPart$5(true));
  }
  function playModule$6() {
      $('body').addClass('yhdm-wrapper');
      $('#ipchk_getplay').remove();
      $('.fullscn').remove();
      replacePlayer$2();
  }

  runtime.register({
      domains: ['yhdmp.cc'],
      opts: [{ test: '/vp', run: playModule$6 }],
      search: {
          name: '樱花动漫2',
          search: (name) => `https://www.yhdmp.cc/s_all?ex=1&kw=${name}`,
          getSearchName: () => $('.gohome > a:last').text(),
      },
  });

  function queryDom(selector) {
      return new Promise((resolve) => {
          let dom;
          function search() {
              dom = $(selector);
              if (dom.length === 0) {
                  requestAnimationFrame(search);
              }
              else {
                  resolve(dom[0]);
              }
          }
          search();
      });
  }

  let player;
  function switchPart$4(next) {
      player.on('prev', () => {
          var _a;
          (_a = $('.meida-content-main-window-right-series-list-volume-active')[next ? 'next' : 'prev']()
              .prev()
              .find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
      });
  }
  function injectEvent() {
      player.on('prev', () => switchPart$4(false));
      player.on('next', () => switchPart$4(true));
  }
  function replacePlayer$1(video) {
      const fn = () => {
          if (!video.src || video.src === location.href)
              return;
          player = new KPlayer('.player.meida-content-main-window-left', { video });
          injectEvent();
      };
      const ob = new MutationObserver(fn);
      ob.observe(video, { attributes: true, attributeFilter: ['src'] });
      fn();
  }
  async function playModule$5() {
      const video = await queryDom('video');
      replacePlayer$1(video);
  }

  runtime.register({
      domains: ['new-ani.me', 'bangumi.online'],
      opts: [{ test: '/watch', run: playModule$5 }],
  });

  async function playModule$4() {
      const video = await queryDom('video');
      new KPlayer('#player', { video, eventToParentWindow: true });
  }

  // 这是个解析器网站,里面只有一个播放器。将其替换成 KPlayer
  runtime.register({
      domains: ['danmu.4dm.cc'],
      opts: [{ test: '/m3u8.php', runInIframe: true, run: playModule$4 }],
      search: {
          getSearchName: () => {
              return new Promise((resolve) => {
                  const fn = (e) => {
                      if (e.data.key === 'getSearchName') {
                          resolve(e.data.name);
                          window.removeEventListener('message', fn);
                      }
                  };
                  window.addEventListener('message', fn);
                  parent.postMessage({ key: 'getSearchName' }, '*');
              });
          },
          getEpisode: () => {
              return new Promise((resolve) => {
                  const fn = (e) => {
                      if (e.data.key === 'getEpisode') {
                          resolve(e.data.name);
                          window.removeEventListener('message', fn);
                      }
                  };
                  window.addEventListener('message', fn);
                  parent.postMessage({ key: 'getEpisode' }, '*');
              });
          },
      },
  });

  function switchPart$3(next) {
      var _a;
      (_a = $('.active-play').parent()[next ? 'next' : 'prev']().find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  const iframeSelector = '#playleft iframe';
  function playModule$3() {
      window.addEventListener('message', (e) => {
          var _a, _b;
          if (!Reflect.has(e.data, 'key'))
              return;
          const { key, video } = e.data;
          if (key === 'prev')
              switchPart$3(false);
          if (key === 'next')
              switchPart$3(true);
          if (key === 'enterwidescreen') {
              $('body').css('overflow', 'hidden');
              $(iframeSelector).css({
                  position: 'fixed',
                  left: 0,
                  right: 0,
                  bottom: 0,
                  top: 0,
              });
          }
          if (key === 'exitwidescreen') {
              $('body').css('overflow', '');
              $(iframeSelector).removeAttr('style');
          }
          if (key === 'canplay') {
              const width = $('#ageframediv').width();
              if (width)
                  $('#ageframediv').height((video.height / video.width) * width);
          }
          // 这里的事件由 danmu.3dm.cc 触发
          if (key === 'getSearchName') {
              const iframe = $(iframeSelector)[0];
              (_a = iframe.contentWindow) === null || _a === void 0 ? void 0 : _a.postMessage({ key: 'getSearchName', name: $('#detailname').text() }, '*');
          }
          if (key === 'getEpisode') {
              const iframe = $(iframeSelector)[0];
              (_b = iframe.contentWindow) === null || _b === void 0 ? void 0 : _b.postMessage({ key: 'getEpisode', name: $('.movurl .active-play').text() }, '*');
          }
      });
      window.addEventListener('keydown', (e) => {
          if (document.activeElement !== document.body)
              return;
          $(iframeSelector)[0].focus();
          if (e.key === ' ')
              e.preventDefault();
      });
      $(iframeSelector).attr({ gesture: 'media', allow: 'autoplay; fullscreen' });
  }

  runtime.register({
      domains: ['.ntyou.'],
      opts: [{ test: '/play', run: playModule$3 }],
      search: {
          name: 'NT动漫',
          search: (name) => `https://www.ntyou.com/search/-------------.html?wd=${name}&page=1`,
      },
  });

  function switchPart$2(next) {
      var _a;
      (_a = $('.eplist-eppic li[style]')[next ? 'next' : 'prev']().find('a')[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  async function playModule$2() {
      const iframe = await queryDom('#id_main_playiframe');
      const fn = () => {
          if (!iframe.src)
              return;
          const url = new URL(iframe.src);
          const vurl = url.searchParams.get('url');
          if (!vurl)
              return;
          const player = new KPlayer('#player_back');
          player.src = vurl;
          player.on('prev', () => switchPart$2(false));
          player.on('next', () => switchPart$2(true));
      };
      const ob = new MutationObserver(fn);
      ob.observe(iframe, { attributes: true, attributeFilter: ['src'] });
      fn();
  }

  runtime.register({
      domains: ['.dm233.'],
      opts: [{ test: '/play', run: playModule$2 }],
      search: {
          name: '233动漫网',
          search: (name) => `https://www.dm233.org/search?keyword=${name}&seaex=1`,
          getSearchName: () => $('.playtitle span a').text(),
      },
  });

  function switchPart$1(next) {
      var _a;
      (_a = $(`.play_but.bline a:contains(${next ? '下集' : '上集'})`)[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  function playModule$1() {
      const url = unsafeWindow.MacPlayer.PlayUrl;
      const player = new KPlayer('.MacPlayer');
      player.src = url;
      player.on('prev', () => switchPart$1(false));
      player.on('next', () => switchPart$1(true));
      function toggle(bool) {
          $('.hot_banner').toggle(bool);
          $('#play_page > div.foot.foot_nav.clearfix').toggle(bool);
      }
      if (player.isWideScreen) {
          toggle(false);
      }
      player.on('enterwidescreen', () => toggle(false));
      player.on('exitwidescreen', () => toggle(true));
      $('#play_page > div.hidden_xs.hidden_mi.pannel.clearfix').remove();
  }

  runtime.register({
      domains: ['.olevod.'],
      opts: [{ test: '/play', run: playModule$1 }],
      search: {
          name: '欧乐影院',
          search: (name) => `https://www.olevod.com/index.php/vod/search.html?wd=${name}&submit=`,
          getSearchName: () => $('.video_title > .title').text(),
      },
  });

  var css = ".bimi-wrapper .play-full,\n.bimi-wrapper #bkcl,\n.bimi-wrapper marquee {\n  display: none !important;\n}";
  n(css,{});

  function replacePlayer() {
      new KPlayer('#player', {
          video: $('video')[0],
          eventToParentWindow: true,
      });
  }
  function switchPart(next) {
      var _a;
      (_a = $(`.player-info .play-qqun .${next ? 'next' : 'pre'}:not(.btns_disad)`)[0]) === null || _a === void 0 ? void 0 : _a.click();
  }
  async function playModule() {
      var _a;
      $('#bkcl').remove();
      const iframe = await queryDom(`#playleft iframe[src*='url=']`);
      window.addEventListener('message', (e) => {
          var _a, _b, _c;
          if (!Reflect.has(e.data, 'key'))
              return;
          const key = e.data.key;
          const video = e.data.video;
          if (key === 'initDone') {
              (_a = iframe.contentWindow) === null || _a === void 0 ? void 0 : _a.postMessage({ key: 'initDone' }, '*');
          }
          if (key === 'prev')
              switchPart(false);
          if (key === 'next')
              switchPart(true);
          if (key === 'enterwidescreen') {
              $('body').css('overflow', 'hidden');
              $(iframe).css({
                  position: 'fixed',
                  left: 0,
                  right: 0,
                  bottom: 0,
                  top: 0,
                  zIndex: 99999,
              });
          }
          if (key === 'exitwidescreen') {
              $('body').css('overflow', '');
              $(iframe).removeAttr('style');
          }
          if (key === 'getSearchName') {
              (_b = iframe.contentWindow) === null || _b === void 0 ? void 0 : _b.postMessage({ key: 'getSearchName', name: $('.v_path a.current').text() }, '*');
          }
          if (key === 'getEpisode') {
              let name = '';
              let pre = $('.player-info .play-qqun .pre').attr('href');
              let next = $('.player-info .play-qqun .next').attr('href');
              if (pre) {
                  name = $(`.player_list a[href='${pre}']`)
                      .parent()
                      .next()
                      .find('a')
                      .text();
              }
              else if (next) {
                  name = $(`.player_list a[href='${next}']`)
                      .parent()
                      .prev()
                      .find('a')
                      .text();
              }
              else {
                  name = $(`.player_list a[href='${location.pathname}']`).text();
              }
              (_c = iframe.contentWindow) === null || _c === void 0 ? void 0 : _c.postMessage({ key: 'getEpisode', name }, '*');
          }
          if (key === 'openLink') {
              window.open(e.data.url);
          }
          if (key === 'canplay') {
              const height = ($('#video').width() / video.width) * video.height;
              $('#video').height(height);
          }
      });
      (_a = iframe.contentWindow) === null || _a === void 0 ? void 0 : _a.postMessage({ key: 'initDone' }, '*');
      iframe.focus();
      window.addEventListener('keydown', (e) => {
          if (document.activeElement !== document.body)
              return;
          iframe.focus();
          if (e.key === ' ')
              e.preventDefault();
      });
  }
  function playInIframeModule() {
      const fn = (e) => {
          if (!Reflect.has(e.data, 'key'))
              return;
          if (e.data.key === 'initDone') {
              replacePlayer();
              window.removeEventListener('message', fn);
          }
      };
      window.addEventListener('message', fn);
      parent.postMessage({ key: 'initDone' }, '*');
  }

  runtime.register({
      domains: ['bimiacg4.net'],
      opts: [
          {
              test: ['/play/'],
              setup: () => $('body').addClass('bimi-wrapper'),
              run: playModule,
          },
          { test: [/.*/], runInIframe: true, run: playInIframeModule },
      ],
      search: {
          name: 'BIMI动漫',
          search: (name) => `https://www.bimiacg4.net/vod/search/wd/${name}/`,
          getSearchName: () => {
              return new Promise((resolve) => {
                  const fn = (e) => {
                      if (e.data.key === 'getSearchName') {
                          resolve(e.data.name);
                          window.removeEventListener('message', fn);
                      }
                  };
                  window.addEventListener('message', fn);
                  parent.postMessage({ key: 'getSearchName' }, '*');
              });
          },
          getEpisode: () => {
              return new Promise((resolve) => {
                  const fn = (e) => {
                      if (e.data.key === 'getEpisode') {
                          resolve(e.data.name);
                          window.removeEventListener('message', fn);
                      }
                  };
                  window.addEventListener('message', fn);
                  parent.postMessage({ key: 'getEpisode' }, '*');
              });
          },
      },
  });

  runtime.run();

})(Hls, Plyr, Danmaku);

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址