biliscope - B站AI总结+热评外显

悬停视频封面,看B站AI总结+热评

安装此脚本?
作者推荐脚本

您可能也喜欢BiliFold - 收起动态评论

安装此脚本
// ==UserScript==
// @name         biliscope - B站AI总结+热评外显
// @namespace    https://github.com/F-park/biliscope-tampermonkey/blob/master/video
// @version      0.1.0
// @author       F-park
// @description  悬停视频封面,看B站AI总结+热评
// @license      MIT
// @icon         
// @supportURL   https://github.com/F-park/biliscope-tampermonkey/issues
// @match        https://*.bilibili.com/*
// @grant        GM_addStyle
// @run-at       document-start
// ==/UserScript==

(i=>{if(typeof GM_addStyle=="function"){GM_addStyle(i);return}const e=document.createElement("style");e.textContent=i,document.head.append(e)})(' #biliscope-video-card{position:absolute;z-index:1002}#biliscope-video-card *{box-sizing:content-box}#biliscope-video-card .d-none{display:none}#biliscope-video-card-inner{font-size:14px;display:flex;flex-direction:column;border:1px solid #e3e5e7;background-color:#fff;border-radius:8px;max-height:50vh}#biliscope-ai-summary-none{border:solid 1px;font-size:13px;color:#444;width:max-content;padding:0 5px;background-color:#fff}#biliscope-hot-comment-wrapper{display:flex;width:376px;flex-shrink:0;background-color:#fff;border-top:1px solid #e3e5e7;border-radius:0 0 8px 8px}#biliscope-hot-comment{margin-bottom:10px;padding:10px 15px 0;font-weight:400;background-color:#fff;border-radius:8px;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow-y:hidden;font-size:13px;color:#9499a0}#biliscope-hot-comment:hover{display:block;max-height:20vh;overflow-y:auto}#biliscope-hot-comment-text{color:#9499a0;font-size:13px}#biliscope-hot-comment-text img{width:1.4em;height:1.4em;vertical-align:text-bottom}#biliscope-hot-comment-text a{color:#008ac5;cursor:pointer}#biliscope-hot-comment-text a:hover{color:#00aeec}#biliscope-hot-comment-author{color:#61666d;cursor:pointer}#biliscope-hot-comment-author:hover{color:#00aeec}#biliscope-hot-comment-icon{flex-shrink:0;position:relative;height:14px;left:10px;top:15px;width:14px;background:url() 50%/contain no-repeat}.biliscope-ai-summary-popup{width:375px;border-radius:8px 8px 0 0;background:#fff;box-shadow:0 0 30px #0000001a;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-user-select:none;-webkit-user-select:none;user-select:none;min-height:0}.biliscope-ai-summary-popup-header{position:relative;border-radius:8px 8px 0 0;-ms-flex-negative:0;flex-shrink:0;height:32px;padding:18px 14px 10px;-ms-flex-pack:justify;justify-content:space-between;background:url() 0 0/900px 52px no-repeat}.biliscope-ai-summary-popup-header,.biliscope-ai-summary-popup-header-left{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.biliscope-ai-summary-popup-header-left .biliscope-ai-summary-popup-icon{display:-ms-flexbox;display:flex;margin-right:8px}.biliscope-ai-summary-popup-header-left .biliscope-ai-summary-popup-tips{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.biliscope-ai-summary-popup-header-left .biliscope-ai-summary-popup-tips-text{color:#18191c;font-size:14px;font-weight:700}.biliscope-ai-summary-popup-body{-ms-flex:1;flex:1;max-height:50vh;overflow:auto;color:#18191c;box-sizing:border-box;padding:8px 8px 6px}.biliscope-ai-summary-popup-body-abstracts{padding:8px 10px;user-select:text;-webkit-user-select:text;font-size:14px;font-weight:700}.biliscope-ai-summary-popup-body-outline .ai-summary-section{line-height:20px;padding:12px 8px;font-size:14px}.biliscope-ai-summary-popup-body-outline .ai-summary-section .ai-summary-section-title{display:-ms-flexbox;display:flex;font-weight:700;font-size:14px;cursor:pointer}.biliscope-ai-summary-popup-body-outline .ai-summary-section .ai-summary-section-title:before{-ms-flex-negative:0;flex-shrink:0;content:"";display:block;border-radius:2px;width:4px;height:4px;background-color:#000;margin:8px 12px 8px 6px}.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet{display:-ms-flexbox;display:flex;margin-top:8px;margin-left:22px;transition:color .3s;cursor:pointer}.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet .timestamp{font-size:13px;color:#9499a0;-ms-flex:0 0 56px;flex:0 0 56px;text-align:left}.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet .timestamp-inner{transition:color .3s}.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet:first-child{padding-top:0}.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet:hover,.biliscope-ai-summary-popup-body-outline .ai-summary-section .bullet:hover .timestamp{color:#00aeec}.biliscope-ai-summary-popup-body-outline .ai-summary-section:hover{border-radius:8px;background-color:#f6f7f8}.biliscope-ai-summary-popup-body::-webkit-scrollbar{width:4px}.biliscope-ai-summary-popup-body::-webkit-scrollbar-thumb{border-radius:4px;background:#999} ');

(function () {
  'use strict';

  const BILIBILI_API_URL = "https://api.bilibili.com";
  var biliMixin = null;
  async function getBiliMixin() {
    const OE = [
      46,
      47,
      18,
      2,
      53,
      8,
      23,
      32,
      15,
      50,
      10,
      31,
      58,
      3,
      45,
      35,
      27,
      43,
      5,
      49,
      33,
      9,
      42,
      19,
      29,
      28,
      14,
      39,
      12,
      38,
      41,
      13,
      37,
      48,
      7,
      16,
      24,
      55,
      40,
      61,
      26,
      17,
      0,
      1,
      60,
      51,
      30,
      4,
      22,
      25,
      54,
      21,
      56,
      59,
      6,
      63,
      57,
      62,
      11,
      36,
      20,
      34,
      44,
      52
    ];
    return fetch("https://api.bilibili.com/x/web-interface/nav").then((response) => response.json()).then((data) => {
      let img_val = data["data"]["wbi_img"]["img_url"].split("/").pop().split(".")[0];
      let sub_val = data["data"]["wbi_img"]["sub_url"].split("/").pop().split(".")[0];
      let val = img_val + sub_val;
      return OE.reduce((s, v) => s + val[v], "").substring(0, 32);
    });
  }
  async function biliGet(url, params, retry = 5) {
    const origUrl = url;
    if (biliMixin === null) {
      biliMixin = await getBiliMixin();
    }
    if (url.indexOf("/wbi/") != -1 || url.indexOf("/conclusion/get") != -1) {
      params["wts"] = Math.floor(Date.now() / 1e3);
      let keys = Object.keys(params).sort();
      let paramsStr = keys.map((key) => `${key}=${params[key]}`).join("&");
      let sign = md5(paramsStr + biliMixin);
      url = `${url}?${paramsStr}&w_rid=${sign}`;
    } else {
      let keys = Object.keys(params).sort();
      let paramsStr = keys.map((key) => `${key}=${params[key]}`).join("&");
      url = `${url}?${paramsStr}`;
    }
    return fetch(url, { "credentials": "include", "mode": "cors" }).then((response) => response.json()).then((data) => {
      if (data["code"] == -403) {
        biliMixin = null;
      }
      if (data["code"] == -799 && retry > 0) {
        return new Promise((resolve) => setTimeout(resolve, 1e3)).then(() => biliGet(origUrl, params, retry - 1));
      }
      return data;
    });
  }
  var videoInfoCache = /* @__PURE__ */ new Map();
  function cacheValidVideo(cache) {
    if (!cache) {
      return false;
    }
    return ["conclusion"].every((key) => cache[key]);
  }
  function cacheAndUpdateVideo(callback, videoId, api, payload) {
    let cache = videoInfoCache.get(videoId) ?? {};
    cache[api] = payload;
    videoInfoCache.set(videoId, cache);
    callback({ "bvid": videoId, "api": api, "payload": payload });
  }
  function updateVideoInfo(videoId, callback) {
    let cache = videoInfoCache.get(videoId);
    if (cacheValidVideo(cache)) {
      for (let api in cache) {
        callback({ "videoId": videoId, "api": api, "payload": cache[api] });
      }
      return;
    }
    biliGet(`${BILIBILI_API_URL}/x/web-interface/view`, {
      bvid: videoId
    }).then((data) => {
      if (data["code"] == 0) {
        cacheAndUpdateVideo(callback, videoId, "view", data["data"]);
        biliGet(`${BILIBILI_API_URL}/x/web-interface/view/conclusion/get`, {
          bvid: videoId,
          cid: data["data"]["cid"],
          up_mid: data["data"]["owner"]["mid"]
        }).then((data2) => {
          cacheAndUpdateVideo(callback, videoId, "conclusion", data2["data"]);
        });
        biliGet(`${BILIBILI_API_URL}/x/v2/reply/wbi/main`, {
          type: 1,
          oid: data["data"]["aid"]
        }).then((data2) => {
          cacheAndUpdateVideo(callback, videoId, "reply", data2["data"]);
        });
      }
    });
  }
  function showProfile(event, target) {
    const videoId = target.getAttribute("biliscope-videoid");
    if (videoProfileCard == null ? void 0 : videoProfileCard.enable()) {
      const updated = videoProfileCard.updateVideoId(videoId);
      videoProfileCard.updateTarget(target);
      videoProfileCard.updatePosition();
      if (updated) {
        updateVideoInfo(videoId, (data) => {
          videoProfileCard.updateData(data);
        });
      }
    }
  }
  function showProfileDebounce(event) {
    clearTimeout(this.timer);
    const target = event.target.closest("[biliscope-videoid]");
    if (!target) {
      return;
    }
    target.addEventListener("mouseout", () => clearTimeout(this.timer));
    this.timer = setTimeout(() => {
      showProfile(event, target);
    }, 800);
  }
  window.addEventListener("load", function() {
    videoProfileCard = new VideoProfileCard();
    this.document.addEventListener("mouseover", showProfileDebounce);
    installIdHooks();
  });
  const BILIBILI_DYNAMIC_URL = "https://t.bilibili.com";
  const BILIBILI_DYNAMIC_DETAIL_URL = "https://www.bilibili.com/opus";
  const BILIBILI_SPACE_URL = "https://space.bilibili.com";
  const BILIBILI_POPULAR_URL = "https://www.bilibili.com/v/popular";
  const BILIBILI_VIDEO_URL = "https://www.bilibili.com/video";
  const BILIBILI_WATCH_LATER_URL = "https://www.bilibili.com/list/watchlater";
  var videoProfileCard = null;
  function getVideoIdFromLink(link) {
    var _a;
    const regexBV = /(BV[1-9a-zA-Z]{10})/g;
    return (_a = link.match(regexBV)) == null ? void 0 : _a[0];
  }
  function elementImageChildren(el) {
    return el.querySelector("img") || el.querySelector("picture");
  }
  function labelLinks() {
    for (let el of document.getElementsByTagName("a")) {
      if (el.href.startsWith(BILIBILI_VIDEO_URL) || el.href.startsWith(BILIBILI_WATCH_LATER_URL)) {
        const videoId = getVideoIdFromLink(el.href);
        if (videoId && elementImageChildren(el)) {
          el.setAttribute("biliscope-videoid", videoId);
        }
      }
    }
  }
  function installIdHooks() {
    let pageObserver = new MutationObserver((mutationList, observer) => {
      labelLinks();
    });
    pageObserver.observe(document.body, {
      childList: true,
      subtree: true
    });
  }
  const md5 = function(r) {
    function n(o) {
      if (t[o]) return t[o].exports;
      var e = t[o] = { i: o, l: false, exports: {} };
      return r[o].call(e.exports, e, e.exports, n), e.l = true, e.exports;
    }
    var t = {};
    return n.m = r, n.c = t, n.i = function(r2) {
      return r2;
    }, n.d = function(r2, t2, o) {
      n.o(r2, t2) || Object.defineProperty(r2, t2, { configurable: false, enumerable: true, get: o });
    }, n.n = function(r2) {
      var t2 = r2 && r2.__esModule ? function() {
        return r2.default;
      } : function() {
        return r2;
      };
      return n.d(t2, "a", t2), t2;
    }, n.o = function(r2, n2) {
      return Object.prototype.hasOwnProperty.call(r2, n2);
    }, n.p = "", n(n.s = 4);
  }([function(r, n) {
    var t = { utf8: { stringToBytes: function(r2) {
      return t.bin.stringToBytes(unescape(encodeURIComponent(r2)));
    }, bytesToString: function(r2) {
      return decodeURIComponent(escape(t.bin.bytesToString(r2)));
    } }, bin: { stringToBytes: function(r2) {
      for (var n2 = [], t2 = 0; t2 < r2.length; t2++) n2.push(255 & r2.charCodeAt(t2));
      return n2;
    }, bytesToString: function(r2) {
      for (var n2 = [], t2 = 0; t2 < r2.length; t2++) n2.push(String.fromCharCode(r2[t2]));
      return n2.join("");
    } } };
    r.exports = t;
  }, function(r, n, t) {
    !function() {
      var n2 = t(2), o = t(0).utf8, e = t(3), u = t(0).bin, i = function(r2, t2) {
        r2.constructor == String ? r2 = t2 && "binary" === t2.encoding ? u.stringToBytes(r2) : o.stringToBytes(r2) : e(r2) ? r2 = Array.prototype.slice.call(r2, 0) : Array.isArray(r2) || (r2 = r2.toString());
        for (var f = n2.bytesToWords(r2), s = 8 * r2.length, c = 1732584193, a = -271733879, l = -1732584194, g = 271733878, h = 0; h < f.length; h++) f[h] = 16711935 & (f[h] << 8 | f[h] >>> 24) | 4278255360 & (f[h] << 24 | f[h] >>> 8);
        f[s >>> 5] |= 128 << s % 32, f[14 + (s + 64 >>> 9 << 4)] = s;
        for (var p = i._ff, y = i._gg, v = i._hh, d = i._ii, h = 0; h < f.length; h += 16) {
          var b = c, T = a, x = l, B = g;
          c = p(c, a, l, g, f[h + 0], 7, -680876936), g = p(g, c, a, l, f[h + 1], 12, -389564586), l = p(l, g, c, a, f[h + 2], 17, 606105819), a = p(a, l, g, c, f[h + 3], 22, -1044525330), c = p(c, a, l, g, f[h + 4], 7, -176418897), g = p(g, c, a, l, f[h + 5], 12, 1200080426), l = p(l, g, c, a, f[h + 6], 17, -1473231341), a = p(a, l, g, c, f[h + 7], 22, -45705983), c = p(c, a, l, g, f[h + 8], 7, 1770035416), g = p(g, c, a, l, f[h + 9], 12, -1958414417), l = p(l, g, c, a, f[h + 10], 17, -42063), a = p(a, l, g, c, f[h + 11], 22, -1990404162), c = p(c, a, l, g, f[h + 12], 7, 1804603682), g = p(g, c, a, l, f[h + 13], 12, -40341101), l = p(l, g, c, a, f[h + 14], 17, -1502002290), a = p(a, l, g, c, f[h + 15], 22, 1236535329), c = y(c, a, l, g, f[h + 1], 5, -165796510), g = y(g, c, a, l, f[h + 6], 9, -1069501632), l = y(l, g, c, a, f[h + 11], 14, 643717713), a = y(a, l, g, c, f[h + 0], 20, -373897302), c = y(c, a, l, g, f[h + 5], 5, -701558691), g = y(g, c, a, l, f[h + 10], 9, 38016083), l = y(l, g, c, a, f[h + 15], 14, -660478335), a = y(a, l, g, c, f[h + 4], 20, -405537848), c = y(c, a, l, g, f[h + 9], 5, 568446438), g = y(g, c, a, l, f[h + 14], 9, -1019803690), l = y(l, g, c, a, f[h + 3], 14, -187363961), a = y(a, l, g, c, f[h + 8], 20, 1163531501), c = y(c, a, l, g, f[h + 13], 5, -1444681467), g = y(g, c, a, l, f[h + 2], 9, -51403784), l = y(l, g, c, a, f[h + 7], 14, 1735328473), a = y(a, l, g, c, f[h + 12], 20, -1926607734), c = v(c, a, l, g, f[h + 5], 4, -378558), g = v(g, c, a, l, f[h + 8], 11, -2022574463), l = v(l, g, c, a, f[h + 11], 16, 1839030562), a = v(a, l, g, c, f[h + 14], 23, -35309556), c = v(c, a, l, g, f[h + 1], 4, -1530992060), g = v(g, c, a, l, f[h + 4], 11, 1272893353), l = v(l, g, c, a, f[h + 7], 16, -155497632), a = v(a, l, g, c, f[h + 10], 23, -1094730640), c = v(c, a, l, g, f[h + 13], 4, 681279174), g = v(g, c, a, l, f[h + 0], 11, -358537222), l = v(l, g, c, a, f[h + 3], 16, -722521979), a = v(a, l, g, c, f[h + 6], 23, 76029189), c = v(c, a, l, g, f[h + 9], 4, -640364487), g = v(g, c, a, l, f[h + 12], 11, -421815835), l = v(l, g, c, a, f[h + 15], 16, 530742520), a = v(a, l, g, c, f[h + 2], 23, -995338651), c = d(c, a, l, g, f[h + 0], 6, -198630844), g = d(g, c, a, l, f[h + 7], 10, 1126891415), l = d(l, g, c, a, f[h + 14], 15, -1416354905), a = d(a, l, g, c, f[h + 5], 21, -57434055), c = d(c, a, l, g, f[h + 12], 6, 1700485571), g = d(g, c, a, l, f[h + 3], 10, -1894986606), l = d(l, g, c, a, f[h + 10], 15, -1051523), a = d(a, l, g, c, f[h + 1], 21, -2054922799), c = d(c, a, l, g, f[h + 8], 6, 1873313359), g = d(g, c, a, l, f[h + 15], 10, -30611744), l = d(l, g, c, a, f[h + 6], 15, -1560198380), a = d(a, l, g, c, f[h + 13], 21, 1309151649), c = d(c, a, l, g, f[h + 4], 6, -145523070), g = d(g, c, a, l, f[h + 11], 10, -1120210379), l = d(l, g, c, a, f[h + 2], 15, 718787259), a = d(a, l, g, c, f[h + 9], 21, -343485551), c = c + b >>> 0, a = a + T >>> 0, l = l + x >>> 0, g = g + B >>> 0;
        }
        return n2.endian([c, a, l, g]);
      };
      i._ff = function(r2, n3, t2, o2, e2, u2, i2) {
        var f = r2 + (n3 & t2 | ~n3 & o2) + (e2 >>> 0) + i2;
        return (f << u2 | f >>> 32 - u2) + n3;
      }, i._gg = function(r2, n3, t2, o2, e2, u2, i2) {
        var f = r2 + (n3 & o2 | t2 & ~o2) + (e2 >>> 0) + i2;
        return (f << u2 | f >>> 32 - u2) + n3;
      }, i._hh = function(r2, n3, t2, o2, e2, u2, i2) {
        var f = r2 + (n3 ^ t2 ^ o2) + (e2 >>> 0) + i2;
        return (f << u2 | f >>> 32 - u2) + n3;
      }, i._ii = function(r2, n3, t2, o2, e2, u2, i2) {
        var f = r2 + (t2 ^ (n3 | ~o2)) + (e2 >>> 0) + i2;
        return (f << u2 | f >>> 32 - u2) + n3;
      }, i._blocksize = 16, i._digestsize = 16, r.exports = function(r2, t2) {
        if (void 0 === r2 || null === r2) throw new Error("Illegal argument " + r2);
        var o2 = n2.wordsToBytes(i(r2, t2));
        return t2 && t2.asBytes ? o2 : t2 && t2.asString ? u.bytesToString(o2) : n2.bytesToHex(o2);
      };
    }();
  }, function(r, n) {
    !function() {
      var n2 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/", t = { rotl: function(r2, n3) {
        return r2 << n3 | r2 >>> 32 - n3;
      }, rotr: function(r2, n3) {
        return r2 << 32 - n3 | r2 >>> n3;
      }, endian: function(r2) {
        if (r2.constructor == Number) return 16711935 & t.rotl(r2, 8) | 4278255360 & t.rotl(r2, 24);
        for (var n3 = 0; n3 < r2.length; n3++) r2[n3] = t.endian(r2[n3]);
        return r2;
      }, randomBytes: function(r2) {
        for (var n3 = []; r2 > 0; r2--) n3.push(Math.floor(256 * Math.random()));
        return n3;
      }, bytesToWords: function(r2) {
        for (var n3 = [], t2 = 0, o = 0; t2 < r2.length; t2++, o += 8) n3[o >>> 5] |= r2[t2] << 24 - o % 32;
        return n3;
      }, wordsToBytes: function(r2) {
        for (var n3 = [], t2 = 0; t2 < 32 * r2.length; t2 += 8) n3.push(r2[t2 >>> 5] >>> 24 - t2 % 32 & 255);
        return n3;
      }, bytesToHex: function(r2) {
        for (var n3 = [], t2 = 0; t2 < r2.length; t2++) n3.push((r2[t2] >>> 4).toString(16)), n3.push((15 & r2[t2]).toString(16));
        return n3.join("");
      }, hexToBytes: function(r2) {
        for (var n3 = [], t2 = 0; t2 < r2.length; t2 += 2) n3.push(parseInt(r2.substr(t2, 2), 16));
        return n3;
      }, bytesToBase64: function(r2) {
        for (var t2 = [], o = 0; o < r2.length; o += 3) for (var e = r2[o] << 16 | r2[o + 1] << 8 | r2[o + 2], u = 0; u < 4; u++) 8 * o + 6 * u <= 8 * r2.length ? t2.push(n2.charAt(e >>> 6 * (3 - u) & 63)) : t2.push("=");
        return t2.join("");
      }, base64ToBytes: function(r2) {
        r2 = r2.replace(/[^A-Z0-9+\/]/gi, "");
        for (var t2 = [], o = 0, e = 0; o < r2.length; e = ++o % 4) 0 != e && t2.push((n2.indexOf(r2.charAt(o - 1)) & Math.pow(2, -2 * e + 8) - 1) << 2 * e | n2.indexOf(r2.charAt(o)) >>> 6 - 2 * e);
        return t2;
      } };
      r.exports = t;
    }();
  }, function(r, n) {
    function t(r2) {
      return !!r2.constructor && "function" == typeof r2.constructor.isBuffer && r2.constructor.isBuffer(r2);
    }
    function o(r2) {
      return "function" == typeof r2.readFloatLE && "function" == typeof r2.slice && t(r2.slice(0, 0));
    }
    r.exports = function(r2) {
      return null != r2 && (t(r2) || o(r2) || !!r2._isBuffer);
    };
  }, function(r, n, t) {
    r.exports = t(1);
  }]);
  function secondsToDisplay(sec) {
    if (!sec) {
      return 0;
    }
    function digitToStr(n) {
      n = Math.floor(n);
      return n < 10 ? "0" + n : n;
    }
    sec = Math.floor(sec);
    if (sec < 60) {
      return `00:${digitToStr(sec)}`;
    } else if (sec < 60 * 60) {
      return `${digitToStr(sec / 60)}:${digitToStr(sec % 60)}`;
    } else {
      return `${digitToStr(sec / 60 / 60)}:${digitToStr(sec / 60) % 60}:${digitToStr(sec % 60)}`;
    }
  }
  function secondsToTimeLink(sec) {
    if (!sec) {
      return 0;
    }
    function digitToStr(n) {
      n = Math.floor(n);
      return n < 10 ? "0" + n : n;
    }
    sec = Math.floor(sec);
    if (sec < 60) {
      return `${digitToStr(sec)}s`;
    } else if (sec < 60 * 60) {
      return `${digitToStr(sec / 60)}m${digitToStr(sec % 60)}s`;
    } else {
      return `${digitToStr(sec / 60 / 60)}h${digitToStr(sec / 60) % 60}m${digitToStr(sec % 60)}s`;
    }
  }
  function displayElOutsideTarget(el, targetBounding, directions, cursorPadding = 10, windowPadding = 20) {
    const cardWidth = el.scrollWidth;
    const cardHeight = el.scrollHeight;
    const { left = 0, right = 0, top = 0, bottom = 0 } = targetBounding;
    for (const direction of directions) {
      switch (direction) {
        case "left":
          if (left - windowPadding - cardWidth < 0) {
            continue;
          }
          el.style.left = `${left - cursorPadding - cardWidth + window.scrollX}px`;
          break;
        case "right":
          if (right + windowPadding + cardWidth > window.innerWidth) {
            continue;
          }
          el.style.left = `${right + cursorPadding + window.scrollX}px`;
          break;
        case "top":
          if (top - cardHeight < 0) {
            continue;
          }
          el.style.top = `${top - cursorPadding - cardHeight + window.scrollY}px`;
          break;
        case "bottom":
          console.log(bottom + cardHeight, window.innerHeight);
          if (bottom + cardHeight > window.innerHeight) {
            continue;
          }
        case "default":
          el.style.top = `${bottom + cursorPadding + window.scrollY}px`;
          break;
      }
      switch (direction) {
        case "left":
        case "right":
          if (top + windowPadding + cardHeight < window.innerHeight) {
            el.style.top = `${top + window.scrollY}px`;
          } else if (bottom - windowPadding - cardHeight > 0) {
            el.style.top = `${bottom - cardHeight + window.scrollY}px`;
          } else {
            const middle = top + (bottom - top) / 2;
            el.style.top = `${middle - cardHeight / 2 + window.scrollY}px`;
          }
          return;
        case "top":
        case "bottom":
        case "default":
          if (left + cardWidth > window.innerWidth) {
            el.style.left = `${right - cardHeight + window.scrollX}px`;
          } else {
            el.style.left = `${left + window.scrollX}px`;
          }
          return;
      }
    }
  }
  function getVideoProfileCardHTML(data) {
    return `
        <div id="biliscope-video-card">
            <div id="biliscope-ai-summary-none">此视频不存在AI总结</div>
            <div id="biliscope-video-card-inner">
                ${getAiSummaryHTML()}
                <div id="biliscope-hot-comment-wrapper">
                    <div id="biliscope-hot-comment-icon">
                    </div>
                    <div id="biliscope-hot-comment">
                        <span id="biliscope-hot-comment-author">
                        </span>
                        <span id="biliscope-hot-comment-text">
                        </span>
                    </div>
                </div>
            </div>
        </div>
    `;
  }
  function getAiSummaryHTML(data) {
    return `
        <div id="biliscope-ai-summary-popup" class="biliscope-ai-summary-popup">
            <div class="biliscope-ai-summary-popup-header">
                <div class="biliscope-ai-summary-popup-header-left">
                    <svg width="30" height="30" viewBox="0 0 30 30" fill="none"
                        xmlns="http://www.w3.org/2000/svg" class="biliscope-ai-summary-popup-icon">
                        <g clip-path="url(#clip0_8728_3421)">
                            <path fill-rule="evenodd" clip-rule="evenodd"
                                d="M7.53976 2.34771C8.17618 1.81736 9.12202 1.90335 9.65237 2.53976L12.1524 5.53976C12.6827 6.17618 12.5967 7.12202 11.9603 7.65237C11.3239 8.18272 10.3781 8.09673 9.84771 7.46031L7.34771 4.46031C6.81736 3.8239 6.90335 2.87805 7.53976 2.34771Z"
                                fill="url(#paint0_linear_8728_3421)"></path>
                            <path fill-rule="evenodd" clip-rule="evenodd"
                                d="M21.9602 2.34771C21.3238 1.81736 20.378 1.90335 19.8476 2.53976L17.3476 5.53976C16.8173 6.17618 16.9033 7.12202 17.5397 7.65237C18.1761 8.18272 19.1219 8.09673 19.6523 7.46031L22.1523 4.46031C22.6826 3.8239 22.5967 2.87805 21.9602 2.34771Z"
                                fill="url(#paint1_linear_8728_3421)"></path>
                            <g opacity="0.2" filter="url(#filter0_d_8728_3421)">
                                <path
                                    d="M27 18.2533C27 25.0206 21.6274 27 15 27C8.37258 27 3 25.0206 3 18.2533C3 11.486 3.92308 6 15 6C26.5385 6 27 11.486 27 18.2533Z"
                                    fill="#D9D9D9"></path>
                            </g>
                            <g filter="url(#filter1_ii_8728_3421)">
                                <path
                                    d="M28 18.9489C28 26.656 22.1797 28 15 28C7.8203 28 2 26.656 2 18.9489C2 10 3 6 15 6C27.5 6 28 10 28 18.9489Z"
                                    fill="url(#paint2_linear_8728_3421)"></path>
                            </g>
                            <path
                                d="M4.78613 14.2091C4.78613 11.9263 6.44484 9.96205 8.71139 9.6903C13.2069 9.1513 16.7678 9.13141 21.3132 9.68091C23.5697 9.95371 25.2147 11.9138 25.2147 14.1868V19.192C25.2147 21.3328 23.7551 23.2258 21.6452 23.5884C16.903 24.4032 13.1705 24.2461 8.55936 23.5137C6.36235 23.1647 4.78613 21.2323 4.78613 19.0078V14.2091Z"
                                fill="#191924"></path>
                            <path d="M19.6426 15.3125L19.6426 18.0982" stroke="#2CFFFF" stroke-width="2.4"
                                stroke-linecap="round"></path>
                            <path d="M10.3574 14.8516L12.2146 16.7087L10.3574 18.5658" stroke="#2CFFFF" stroke-width="1.8"
                                stroke-linecap="round" stroke-linejoin="round"></path>
                        </g>
                        <defs>
                            <filter id="filter0_d_8728_3421" x="1" y="4" width="30" height="27" filterUnits="userSpaceOnUse"
                                color-interpolation-filters="sRGB">
                                <feFlood flood-opacity="0" result="BackgroundImageFix"></feFlood>
                                <feColorMatrix in="SourceAlpha" type="matrix"
                                    values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"></feColorMatrix>
                                <feOffset dx="1" dy="1"></feOffset>
                                <feGaussianBlur stdDeviation="1.5"></feGaussianBlur>
                                <feComposite in2="hardAlpha" operator="out"></feComposite>
                                <feColorMatrix type="matrix"
                                    values="0 0 0 0 0.039545 0 0 0 0 0.0845023 0 0 0 0 0.200107 0 0 0 0.85 0">
                                </feColorMatrix>
                                <feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_8728_3421">
                                </feBlend>
                                <feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_8728_3421" result="shape">
                                </feBlend>
                            </filter>
                            <filter id="filter1_ii_8728_3421" x="0" y="4.14286" width="30.7857" height="26.6429"
                                filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
                                <feFlood flood-opacity="0" result="BackgroundImageFix"></feFlood>
                                <feBlend mode="normal" in="SourceGraphic" in2="BackgroundImageFix" result="shape"></feBlend>
                                <feColorMatrix in="SourceAlpha" type="matrix"
                                    values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"></feColorMatrix>
                                <feOffset dx="2.78571" dy="3.71429"></feOffset>
                                <feGaussianBlur stdDeviation="1.39286"></feGaussianBlur>
                                <feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"></feComposite>
                                <feColorMatrix type="matrix" values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 0.25 0">
                                </feColorMatrix>
                                <feBlend mode="normal" in2="shape" result="effect1_innerShadow_8728_3421"></feBlend>
                                <feColorMatrix in="SourceAlpha" type="matrix"
                                    values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"></feColorMatrix>
                                <feOffset dx="-2" dy="-1.85714"></feOffset>
                                <feGaussianBlur stdDeviation="1.85714"></feGaussianBlur>
                                <feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"></feComposite>
                                <feColorMatrix type="matrix"
                                    values="0 0 0 0 0 0 0 0 0 0.15445 0 0 0 0 0.454264 0 0 0 0.11 0"></feColorMatrix>
                                <feBlend mode="normal" in2="effect1_innerShadow_8728_3421"
                                    result="effect2_innerShadow_8728_3421"></feBlend>
                            </filter>
                            <linearGradient id="paint0_linear_8728_3421" x1="6.80424" y1="2.84927" x2="9.01897" y2="8.29727"
                                gradientUnits="userSpaceOnUse">
                                <stop stop-color="#393946"></stop>
                                <stop offset="0.401159" stop-color="#23232E"></stop>
                                <stop offset="1" stop-color="#191924"></stop>
                            </linearGradient>
                            <linearGradient id="paint1_linear_8728_3421" x1="22.6958" y1="2.84927" x2="20.481" y2="8.29727"
                                gradientUnits="userSpaceOnUse">
                                <stop stop-color="#393946"></stop>
                                <stop offset="0.401159" stop-color="#23232E"></stop>
                                <stop offset="1" stop-color="#191924"></stop>
                            </linearGradient>
                            <linearGradient id="paint2_linear_8728_3421" x1="7.67091" y1="10.8068" x2="19.9309" y2="29.088"
                                gradientUnits="userSpaceOnUse">
                                <stop stop-color="#F4FCFF"></stop>
                                <stop offset="1" stop-color="#EAF5F9"></stop>
                            </linearGradient>
                            <clipPath id="clip0_8728_3421">
                                <rect width="30" height="30" fill="white"></rect>
                            </clipPath>
                        </defs>
                    </svg>
                    <div class="biliscope-ai-summary-popup-tips">
                        <span class="biliscope-ai-summary-popup-tips-text">已为你生成视频总结</span>
                    </div>
                </div>
            </div>
            <div class="biliscope-ai-summary-popup-body">
                <div id="biliscope-ai-summary-abstracts" class="biliscope-ai-summary-popup-body-abstracts">
                </div>
                <div id="biliscope-ai-summary-outline" class="biliscope-ai-summary-popup-body-outline">
                </div>
            </div>
        </div>
    `;
  }
  function VideoProfileCard() {
    this.enabled = false;
    this.data = {};
    this.target = null;
    this.enabled = false;
    this.videoId = null;
    this.el = document.createElement("div");
    this.el.style.position = "absolute";
    this.el.style.display = "none";
    this.el.innerHTML = getVideoProfileCardHTML(this.data);
    document.body.appendChild(this.el);
    const outlineDiv = document.getElementById("biliscope-ai-summary-outline");
    outlineDiv == null ? void 0 : outlineDiv.addEventListener("click", (ev) => {
      const node = ev.target.closest("[biliscope-video-timestamp]");
      if (node) {
        const timestamp = parseInt(node.getAttribute("biliscope-video-timestamp"));
        const tLink = secondsToTimeLink(timestamp);
        window.open(`https://www.bilibili.com/video/${this.videoId}/?t=${tLink}`, "_blank");
      }
    });
    document.body.appendChild(this.el);
    this.disable();
  }
  VideoProfileCard.prototype.enable = function() {
    if (!this.enabled) {
      this.enabled = true;
      return true;
    }
    return false;
  };
  VideoProfileCard.prototype.disable = function() {
    this.videoId = null;
    this.enabled = false;
    this.valid = false;
    this.data = {};
    if (this.el) {
      this.el.style.display = "none";
    }
    return true;
  };
  VideoProfileCard.prototype.setLeaveEvent = function() {
    let validTargets = [this.el, this.target];
    const dispatchToPopover = (event) => {
      const popover = this.target.closest(".v-popover");
      popover == null ? void 0 : popover.dispatchEvent(new Event(event));
    };
    this.leaveCallback = (ev) => {
      if (this.disable()) {
        for (let target of validTargets) {
          target.removeEventListener("mouseleave", this.disableDebounce);
          target.removeEventListener("mouseenter", this.enterCallback);
        }
        if (ev.target == this.el) {
          dispatchToPopover("mouseleave");
        }
      }
    };
    this.enterCallback = (ev) => {
      clearTimeout(this.disableDebounce.timer);
      if (ev.target == this.el) {
        dispatchToPopover("mouseenter");
      }
    };
    this.disableDebounce = (ev) => {
      this.disableDebounce.timer = setTimeout(() => {
        this.leaveCallback(ev);
      }, 400);
    };
    for (let target of validTargets) {
      target.addEventListener("mouseleave", this.disableDebounce);
      target.addEventListener("mouseenter", this.enterCallback);
    }
  };
  VideoProfileCard.prototype.updateVideoId = function(videoId) {
    let updated = this.videoId != videoId;
    this.videoId = videoId;
    return updated;
  };
  VideoProfileCard.prototype.updatePosition = function() {
    const targetBounding = this.target.getBoundingClientRect();
    const { href } = window.location;
    if ((href.startsWith(BILIBILI_DYNAMIC_URL) || href.startsWith(BILIBILI_DYNAMIC_DETAIL_URL) || href.startsWith(BILIBILI_SPACE_URL) && window.location.pathname.endsWith("/dynamic")) && this.target.matches(".bili-dyn-card-video")) {
      displayElOutsideTarget(this.el, targetBounding, ["left", "bottom", "top", "right", "default"]);
    } else if (href.startsWith(BILIBILI_POPULAR_URL) && this.target.matches(".popular-container [biliscope-videoid]")) {
      displayElOutsideTarget(this.el, targetBounding, ["bottom", "top", "left", "default"]);
    } else if ((href.startsWith(BILIBILI_VIDEO_URL) || href.startsWith(BILIBILI_WATCH_LATER_URL)) && this.target.matches("#reco_list [biliscope-videoid]")) {
      displayElOutsideTarget(this.el, targetBounding, ["left", "bottom", "top", "default"]);
    } else {
      displayElOutsideTarget(this.el, targetBounding, ["right", "left", "bottom", "top", "default"]);
    }
  };
  VideoProfileCard.prototype.updateTarget = function(target) {
    if (target != this.target) {
      let node = target;
      let zIndex = 1002;
      while (node && node != document) {
        if (node instanceof ShadowRoot) {
          node = node.host;
        }
        let containerIndex = window.getComputedStyle(node).getPropertyValue("z-index");
        if (containerIndex && containerIndex != "auto" && containerIndex > zIndex) {
          zIndex = containerIndex;
        }
        node = node.parentNode;
      }
      this.el.style.zIndex = zIndex + 1;
    }
    this.target = target;
    this.valid = null;
    this.setLeaveEvent();
  };
  VideoProfileCard.prototype.drawConclusion = function() {
    var _a, _b, _c, _d;
    let summary = (_b = (_a = this.data.conclusion) == null ? void 0 : _a.model_result) == null ? void 0 : _b.summary;
    let outline = (_d = (_c = this.data.conclusion) == null ? void 0 : _c.model_result) == null ? void 0 : _d.outline;
    const summaryDiv = document.getElementById("biliscope-ai-summary-abstracts");
    const outlineDiv = document.getElementById("biliscope-ai-summary-outline");
    if (summary) {
      summaryDiv.classList.remove("d-none");
      summaryDiv.innerHTML = summary;
    } else {
      summaryDiv.classList.add("d-none");
    }
    if (outline && outline.length > 0) {
      let outlineHTML = "";
      outlineDiv.classList.remove("d-none");
      for (let i = 0; i < outline.length; i++) {
        outlineHTML += `<div class="ai-summary-section" biliscope-video-timestamp="${outline[i].timestamp}">`;
        outlineHTML += `<div class="ai-summary-section-title">
                                ${outline[i].title}
                            </div>`;
        for (let j = 0; j < outline[i].part_outline.length; j++) {
          let part = outline[i].part_outline[j];
          outlineHTML += `<div class="bullet" biliscope-video-timestamp="${part.timestamp}">`;
          outlineHTML += `<span class="timestamp">
                                    <span class="timestamp-inner">${secondsToDisplay(part.timestamp)}</span>
                                </span>
                                <span class="content">
                                    ${part.content}
                                </span>`;
          outlineHTML += `</div>`;
        }
        outlineHTML += "</div>";
      }
      outlineDiv.innerHTML = outlineHTML;
    } else {
      outlineDiv.classList.add("d-none");
    }
  };
  VideoProfileCard.prototype.drawHotComment = function() {
    var _a, _b;
    if (!((_a = this.data.replies) == null ? void 0 : _a.length)) {
      document.getElementById("biliscope-hot-comment-wrapper").classList.add("d-none");
      return;
    }
    let hotComment = (_b = this.data.replies.filter(
      (reply) => (reply == null ? void 0 : reply.attr) == 32768
    )) == null ? void 0 : _b[0];
    if (!hotComment) {
      hotComment = this.data.replies.reduce(
        (lReply, rReply) => lReply.like > rReply.like ? lReply : rReply
      );
    }
    if (hotComment) {
      const { content, member } = hotComment;
      const hotCommentText = document.getElementById("biliscope-hot-comment-text");
      const hotCommentAuthor = document.getElementById("biliscope-hot-comment-author");
      hotCommentText.innerHTML = "";
      hotCommentAuthor.innerHTML = `${member.uname}:`;
      hotCommentAuthor.onclick = function() {
        open(`//space.bilibili.com/${member.mid}`);
      };
      const emotes = content == null ? void 0 : content.emote;
      const jump_urls = content == null ? void 0 : content.jump_url;
      const separators = Object.keys(emotes || {}).concat(Object.keys(jump_urls || {}));
      const regexStr = separators.map(
        (s) => s.replace(/[.*+?^${}()|[\]\\]/g, "\\$&")
      ).join("|");
      if (!regexStr) {
        hotCommentText.innerHTML = content.message;
      } else {
        let jump_urls_copy = { ...jump_urls };
        content.message.split(new RegExp(`(${regexStr})`)).map((s) => {
          let hotCommentItem;
          if (emotes == null ? void 0 : emotes[s]) {
            hotCommentItem = document.createElement("img");
            hotCommentItem.src = emotes[s].url;
          } else if (jump_urls_copy == null ? void 0 : jump_urls_copy[s]) {
            hotCommentItem = document.createElement("a");
            hotCommentItem.target = "_blank";
            hotCommentItem.href = jump_urls_copy[s].pc_url;
            hotCommentItem.innerHTML = s;
            delete jump_urls_copy[s];
          } else {
            hotCommentItem = document.createElement("span");
            hotCommentItem.innerHTML = s;
          }
          hotCommentText.appendChild(hotCommentItem);
        });
      }
      document.getElementById("biliscope-hot-comment-wrapper").classList.remove("d-none");
    }
  };
  VideoProfileCard.prototype.updateData = function(data) {
    var _a;
    if (data["api"] == "view") {
      this.data.view = data["payload"];
    } else if (data["api"] == "conclusion") {
      this.data.conclusion = data["payload"];
      if (this.data.conclusion.code == 0) {
        this.valid = true;
      } else {
        this.valid = false;
      }
    } else if (data["api"] == "reply") {
      this.data.replies = (_a = data.payload) == null ? void 0 : _a.replies;
    }
    if (this.enabled && this.el) {
      if (this.valid != null) {
        this.el.style.display = "flex";
        if (this.valid) {
          this.drawConclusion();
          this.drawHotComment();
          document.getElementById("biliscope-video-card-inner").classList.remove("d-none");
          document.getElementById("biliscope-ai-summary-none").classList.add("d-none");
        } else {
          document.getElementById("biliscope-video-card-inner").classList.add("d-none");
          document.getElementById("biliscope-ai-summary-none").classList.remove("d-none");
        }
      }
    }
    this.updatePosition();
  };

})();