Greasy Fork

Greasy Fork is available in English.

Bangumi 年鉴

根据Bangumi的时光机数据生成年鉴

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Bangumi 年鉴
// @namespace    syaro.io
// @version      1.3.17
// @author       神戸小鳥 @vickscarlet
// @description  根据Bangumi的时光机数据生成年鉴
// @license      MIT
// @icon         https://bgm.tv/img/favicon.ico
// @homepage     https://github.com/bangumi/scripts/blob/master/vickscarlet/scripts/report
// @match        *://bgm.tv/user/*
// @match        *://chii.in/user/*
// @match        *://bangumi.tv/user/*
// ==/UserScript==

(function () {
  'use strict';

  function callWhenDone(fn) {
    let done = true;
    return async () => {
      if (!done) return;
      done = false;
      await fn();
      done = true;
    };
  }
  function callNow(fn) {
    fn();
    return fn;
  }
  const svgTags = [
    "svg",
    "rect",
    "circle",
    "ellipse",
    "line",
    "polyline",
    "polygon",
    "path",
    "text",
    "g",
    "defs",
    "use",
    "symbol",
    "image",
    "clipPath",
    "mask",
    "pattern"
  ];
  function setEvents(element, events) {
    for (const [event, listener] of Object.entries(events)) {
      element.addEventListener(event, listener);
    }
    return element;
  }
  function setProps(element, props) {
    if (!props || typeof props !== "object") return element;
    for (const [key, value] of Object.entries(props)) {
      if (value == null) continue;
      if (key === "events") {
        setEvents(element, value);
      } else if (key === "class") {
        addClass(element, value);
      } else if (key === "style" && typeof value === "object") {
        setStyle(element, value);
      } else if (key.startsWith("data-")) {
        element.setAttribute(key, String(value));
      } else {
        element[key] = value;
      }
    }
    return element;
  }
  function addClass(element, value) {
    element.classList.add(...[value].flat());
    return element;
  }
  function setStyle(element, styles) {
    for (let [k, v] of Object.entries(styles)) {
      if (typeof v === "number" && v !== 0 && !["zIndex", "fontWeight"].includes(k)) {
        v = v + "px";
      }
      element.style[k] = v;
    }
    return element;
  }
  function create(name, props, ...childrens) {
    if (name == null) return null;
    const isSVG = name === "svg" || typeof name === "string" && svgTags.includes(name);
    if (isSVG) return createSVG(name, props, ...childrens);
    const element = name instanceof Element ? name : document.createElement(name);
    if (props === void 0) return element;
    if (Array.isArray(props) || props instanceof Node || typeof props !== "object") {
      return append(element, props, ...childrens);
    }
    return append(setProps(element, props), ...childrens);
  }
  function append(element, ...childrens) {
    const tag = element.tagName.toLowerCase();
    if (svgTags.includes(tag)) {
      return appendSVG(element, ...childrens);
    }
    for (const child of childrens) {
      if (Array.isArray(child)) {
        element.append(create(...child));
      } else if (child instanceof Node) {
        element.appendChild(child);
      } else {
        element.append(document.createTextNode(String(child)));
      }
    }
    return element;
  }
  function createSVG(name, props, ...childrens) {
    const element = document.createElementNS("http://www.w3.org/2000/svg", name);
    if (name === "svg") element.setAttribute("xmlns:xlink", "http://www.w3.org/1999/xlink");
    if (props === void 0) return element;
    if (Array.isArray(props) || props instanceof Node || typeof props !== "object") {
      return appendSVG(element, props, ...childrens);
    }
    return appendSVG(setProps(element, props), ...childrens);
  }
  function appendSVG(element, ...childrens) {
    for (const child of childrens) {
      if (Array.isArray(child)) {
        element.append(createSVG(...child));
      } else if (child instanceof Node) {
        element.appendChild(child);
      } else {
        element.append(document.createTextNode(String(child)));
      }
    }
    return element;
  }
  function addStyle(...styles) {
    const style = document.createElement("style");
    style.append(document.createTextNode(styles.join("\n")));
    document.head.appendChild(style);
    return style;
  }
  function removeAllChildren(element) {
    while (element.firstChild) element.removeChild(element.firstChild);
    return element;
  }
  const loadScript = /* @__PURE__ */ (() => {
    const loaded = /* @__PURE__ */ new Set();
    const pedding = /* @__PURE__ */ new Map();
    return async (src) => {
      if (loaded.has(src)) return;
      const list = pedding.get(src) ?? [];
      const p = new Promise((resolve) => list.push(resolve));
      if (!pedding.has(src)) {
        pedding.set(src, list);
        const script = document.createElement("script");
        script.src = src;
        script.type = "text/javascript";
        script.onload = () => {
          loaded.add(src);
          list.forEach((resolve) => resolve());
        };
        document.body.appendChild(script);
      }
      return p;
    };
  })();
  class Event {
    static #listeners = /* @__PURE__ */ new Map();
    static on(event, listener) {
      if (!this.#listeners.has(event)) this.#listeners.set(event, /* @__PURE__ */ new Set());
      this.#listeners.get(event).add(listener);
    }
    static emit(event, ...args) {
      if (!this.#listeners.has(event)) return;
      for (const listener of this.#listeners.get(event).values()) listener(...args);
    }
    static off(event, listener) {
      if (!this.#listeners.has(event)) return;
      this.#listeners.get(event).delete(listener);
    }
  }
  class Cache {
    constructor({ hot, last }) {
      this.#hotLimit = hot ?? 0;
      this.#lastLimit = last ?? 0;
      this.#cacheLimit = this.#hotLimit + this.#lastLimit;
    }
    #hotLimit;
    #lastLimit;
    #cacheLimit;
    #hotList = [];
    #hot = /* @__PURE__ */ new Set();
    #last = /* @__PURE__ */ new Set();
    #pedding = /* @__PURE__ */ new Set();
    #cache = /* @__PURE__ */ new Map();
    #times = /* @__PURE__ */ new Map();
    #cHot(key) {
      if (!this.#hotLimit) return false;
      const counter = this.#times.get(key) || { key, cnt: 0 };
      counter.cnt++;
      this.#times.set(key, counter);
      if (this.#hot.size == 0) {
        this.#hotList.push(counter);
        this.#hot.add(key);
        this.#pedding.delete(key);
        return true;
      }
      const i = this.#hotList.indexOf(counter);
      if (i == 0) return true;
      if (i > 0) {
        const up = this.#hotList[i - 1];
        if (counter.cnt > up.cnt) this.#hotList.sort((a, b) => b.cnt - a.cnt);
        return true;
      }
      if (this.#hot.size < this.#hotLimit) {
        this.#hotList.push(counter);
        this.#hot.add(key);
        this.#pedding.delete(key);
        return true;
      }
      const min = this.#hotList.at(-1);
      if (counter.cnt <= min.cnt) return false;
      this.#hotList.pop();
      this.#hot.delete(min.key);
      if (!this.#last.has(min.key)) this.#pedding.add(min.key);
      this.#hotList.push(counter);
      this.#hot.add(key);
      this.#pedding.delete(key);
      return true;
    }
    #cLast(key) {
      if (!this.#lastLimit) return false;
      this.#last.delete(key);
      this.#last.add(key);
      this.#pedding.delete(key);
      if (this.#last.size <= this.#lastLimit) return true;
      const out = this.#last.values().next().value;
      this.#last.delete(out);
      if (!this.#hot.has(out)) this.#pedding.add(out);
      return true;
    }
    async get(key, query) {
      const data = this.#cache.get(key) ?? await query();
      const inHot = this.#cHot(key);
      const inLast = this.#cLast(key);
      if (inHot || inLast) this.#cache.set(key, data);
      let i = this.#cache.size - this.#cacheLimit;
      if (!i) return data;
      for (const key2 of this.#pedding) {
        if (!i) return data;
        this.#cache.delete(key2);
        this.#pedding.delete(key2);
        i--;
      }
      return data;
    }
    update(key, value) {
      if (!this.#cache.has(key)) this.#cache.set(key, value);
    }
    clear() {
      this.#cache.clear();
    }
  }
  class Collection {
    constructor(master, { collection, options, indexes, cache }) {
      this.#master = master;
      this.#collection = collection;
      this.#options = options;
      this.#indexes = indexes;
      if (options?.keyPath && cache && cache.enabled) {
        this.#cache = new Cache(cache);
      }
    }
    #master;
    #collection;
    #options;
    #indexes;
    #cache;
    get collection() {
      return this.#collection;
    }
    get options() {
      return this.#options;
    }
    get indexes() {
      return this.#indexes;
    }
    async transaction(handler, mode) {
      return this.#master.transaction(
        this.#collection,
        async (store) => {
          const request = await handler(store);
          return new Promise((resolve, reject) => {
            request.addEventListener("error", (e) => reject(e));
            request.addEventListener("success", () => resolve(request.result));
          });
        },
        mode
      );
    }
    #index(store, index = "") {
      if (!index) return store;
      return store.index(index);
    }
    async get(key, index) {
      const handler = () => this.transaction((store) => this.#index(store, index).get(key));
      if (this.#cache && this.#options?.keyPath && !index && typeof key == "string") {
        return this.#cache.get(key, handler);
      }
      return handler();
    }
    async getAll(key, count, index) {
      return this.transaction((store) => this.#index(store, index).getAll(key, count));
    }
    async getAllKeys(key, count, index) {
      return this.transaction((store) => this.#index(store, index).getAllKeys(key, count));
    }
    async put(data) {
      if (this.#cache) {
        let key;
        if (Array.isArray(this.#options.keyPath)) {
          key = [];
          for (const path of this.#options.keyPath) {
            key.push(data[path]);
          }
          key = key.join("/");
        } else {
          key = data[this.#options.keyPath];
        }
        this.#cache.update(key, data);
      }
      return this.transaction((store) => store.put(data), "readwrite").then((_) => true);
    }
    async delete(key) {
      return this.transaction((store) => store.delete(key), "readwrite").then((_) => true);
    }
    async clear() {
      if (this.#cache) this.#cache.clear();
      return this.transaction((store) => store.clear(), "readwrite").then((_) => true);
    }
  }
  class Database {
    constructor({ dbName, version, collections, blocked }) {
      this.#dbName = dbName;
      this.#version = version;
      this.#blocked = blocked || { alert: false };
      for (const options of collections) {
        this.#collections.set(options.collection, new Collection(this, options));
      }
    }
    #dbName;
    #version;
    #collections = /* @__PURE__ */ new Map();
    #db = null;
    #blocked;
    async init() {
      this.#db = await new Promise((resolve, reject) => {
        const request = window.indexedDB.open(this.#dbName, this.#version);
        request.addEventListener(
          "error",
          () => reject({ type: "error", message: request.error })
        );
        request.addEventListener("blocked", () => {
          const message = this.#blocked?.message || "indexedDB is blocked";
          if (this.#blocked?.alert) alert(message);
          reject({ type: "blocked", message });
        });
        request.addEventListener("success", () => resolve(request.result));
        request.addEventListener("upgradeneeded", () => {
          for (const c of this.#collections.values()) {
            const { collection, options, indexes } = c;
            let store;
            if (!request.result.objectStoreNames.contains(collection))
              store = request.result.createObjectStore(collection, options);
            else store = request.transaction.objectStore(collection);
            if (!indexes) continue;
            for (const { name, keyPath, unique } of indexes) {
              if (store.indexNames.contains(name)) continue;
              store.createIndex(name, keyPath, { unique });
            }
          }
        });
      });
      return this;
    }
    async transaction(collection, handler, mode = "readonly") {
      if (!this.#db) await this.init();
      return new Promise(async (resolve, reject) => {
        const transaction = this.#db.transaction(collection, mode);
        const store = transaction.objectStore(collection);
        const result = await handler(store);
        transaction.addEventListener("error", (e) => reject(e));
        transaction.addEventListener("complete", () => resolve(result));
      });
    }
    async get(collection, key, index) {
      return this.#collections.get(collection).get(key, index);
    }
    async getAll(collection, key, count, index) {
      return this.#collections.get(collection).getAll(key, count, index);
    }
    async getAllKeys(collection, key, count, index) {
      return this.#collections.get(collection).getAllKeys(key, count, index);
    }
    async put(collection, data) {
      return this.#collections.get(collection).put(data);
    }
    async delete(collection, key) {
      return this.#collections.get(collection).delete(key);
    }
    async clear(collection) {
      return this.#collections.get(collection).clear();
    }
    async clearAll() {
      for (const c of this.#collections.values()) await c.clear();
      return true;
    }
  }
  const db = new Database({
    dbName: "VReport",
    version: 6,
    collections: [
      {
        collection: "pages",
        options: { keyPath: "url" },
        indexes: [{ name: "url", keyPath: "url", unique: true }]
      },
      {
        collection: "times",
        options: { keyPath: "id" },
        indexes: [{ name: "id", keyPath: "id", unique: true }]
      }
    ]
  });
  const uid = window.location.pathname.split("/")[2] || "";
  const Types = {
    anime: { sort: 1, value: "anime", name: "动画", action: "看", unit: "部" },
    game: { sort: 2, value: "game", name: "游戏", action: "玩", unit: "部" },
    music: { sort: 3, value: "music", name: "音乐", action: "听", unit: "张" },
    book: { sort: 4, value: "book", name: "图书", action: "读", unit: "本" },
    real: { sort: 5, value: "real", name: "三次元", action: "看", unit: "部" }
  };
  const SubTypes = {
    collect: { sort: 1, value: "collect", name: "$过", checked: true },
    do: { sort: 2, value: "do", name: "在$", checked: false },
    dropped: { sort: 3, value: "dropped", name: "抛弃", checked: false },
    on_hold: { sort: 4, value: "on_hold", name: "搁置", checked: false },
    wish: { sort: 5, value: "wish", name: "想$", checked: false }
  };
  const AnimeTypeTimes = {
    WEB: 23 * 60 + 40,
    TV: 23 * 60 + 40,
    OVA: 45 * 60,
    OAD: 45 * 60,
    剧场版: 90 * 60
  };
  function formatSubType(subType, type) {
    const action = Types[type].action;
    return SubTypes[subType].name.replace("$", action);
  }
  function pad02(n) {
    return n.toString().padStart(2, "0");
  }
  function timeFormat(time, day = false) {
    const s = time % 60;
    const m = (time - s) / 60 % 60;
    if (!day) {
      const h2 = (time - s - m * 60) / 3600;
      return `${h2}:${pad02(m)}:${pad02(s)}`;
    }
    const h = (time - s - m * 60) / 3600 % 24;
    const d = (time - s - m * 60 - h * 3600) / 86400;
    if (d) return `${d}天${pad02(h)}:${pad02(m)}:${pad02(s)}`;
    return `${h}:${pad02(m)}:${pad02(s)}`;
  }
  function easeOut(curtime, begin, end, duration) {
    let x = curtime / duration;
    let y = -x * x + 2 * x;
    return begin + (end - begin) * y;
  }
  function countMap(length) {
    return new Map(new Array(length).fill(0).map((_, i) => [i, 0]));
  }
  function groupBy(list, group) {
    const groups = /* @__PURE__ */ new Map();
    for (const item of list) {
      const key = item[group];
      if (groups.has(key)) groups.get(key).push(item);
      else groups.set(key, [item]);
    }
    return groups;
  }
  function groupCount(list, group, groups = /* @__PURE__ */ new Map()) {
    for (const item of list) {
      const key = typeof group == "function" ? group(item) : item[group];
      groups.set(key, (groups.get(key) || 0) + 1);
    }
    return groups;
  }
  async function element2Canvas(element) {
    await loadScript("https://html2canvas.hertzen.com/dist/html2canvas.min.js");
    return html2canvas(element, {
      allowTaint: true,
      logging: false,
      backgroundColor: "#1c1c1c"
    });
  }
  async function f(url) {
    Event.emit("process", { type: "fetch", data: { url } });
    const html = await fetch(window.location.origin + "/" + url).then((res) => res.text());
    if (html.includes("503 Service Temporarily Unavailable")) return null;
    const e = document.createElement("html");
    e.innerHTML = html.replace(/<img (.*)\/?>/g, '<span class="img" $1></span>');
    return e;
  }
  async function fl(type, subType, p = 1, expire = 30) {
    Event.emit("process", { type: "parse", data: { type, subType, p } });
    const url = `${type}/list/${uid}/${subType}?page=${p}`;
    let data = await db.get("pages", url);
    if (data && data.time + expire * 6e4 > Date.now()) return data;
    const e = await f(url);
    if (!e) return null;
    const list = Array.from(e.querySelectorAll("#browserItemList > li")).map(
      (li) => {
        const id = li.querySelector("a").href.split("/").pop();
        const t = li.querySelector("h3");
        const title = t.querySelector("a").innerText;
        const jp_title = t.querySelector("small")?.innerText;
        const img = li.querySelector("span.img")?.getAttribute("src").replace("cover/c", "cover/l") || "//bgm.tv/img/no_icon_subject.png";
        const time2 = new Date(li.querySelector("span.tip_j").innerText);
        const year = time2.getFullYear();
        const month = time2.getMonth();
        const star = parseInt(
          li.querySelector("span.starlight")?.className.match(/stars(\d{1,2})/)[1]
        ) || 0;
        const tags = li.querySelector("span.tip")?.textContent.trim().match(/标签:\s*(.*)/)?.[1].split(/\s+/) || [];
        return { id, subType, title, jp_title, img, time: time2, year, month, star, tags };
      }
    );
    const edge = e.querySelector("span.p_edge");
    let max;
    if (edge) {
      max = Number(/\/\s*(\d+)\s*\)/.exec(edge.textContent)?.[1] || 1);
    } else {
      const ap = e.querySelectorAll("a.p");
      if (ap.length == 0) {
        max = 1;
      } else {
        let cursor = ap[ap.length - 1];
        if (cursor.innerText == "››")
          cursor = cursor.previousElementSibling;
        max = Number(cursor.textContent) || 1;
      }
    }
    const time = Date.now();
    data = { url, list, max, time };
    if (p == 1) {
      const tags = Array.from(e.querySelectorAll("#userTagList > li > a.l")).map(
        (l) => l.childNodes[1].textContent
      );
      data.tags = tags;
    }
    await db.put("pages", data);
    return data;
  }
  async function ft(type) {
    Event.emit("process", { type: "tags", data: { type } });
    const data = await fl(type, "collect");
    return data?.tags;
  }
  function calcTime(s) {
    let m = /[时片]长:\s*(\d{2}):(\d{2}):(\d{2})/.exec(s);
    if (m) return parseInt(m[1]) * 3600 + parseInt(m[2]) * 60 + parseInt(m[3]);
    m = /[时片]长:\s*(\d{2}):(\d{2})/.exec(s);
    if (m) return parseInt(m[1]) * 60 + parseInt(m[2]);
    m = /[时片]长:\s*(\d+)\s*[m分]/.exec(s);
    if (m) return parseInt(m[1]) * 60;
    return 0;
  }
  async function ftime(id) {
    let data = await db.get("times", id);
    if (data) {
      if (data.time) {
        const { time: time2 } = data;
        return { a: true, time: time2 };
      } else {
        const { eps: eps2, type: type2 } = data;
        const time2 = eps2 * AnimeTypeTimes[type2] || 0;
        return { a: false, time: time2 };
      }
    }
    const e = await f(`subject/${id}/ep`);
    const c = (l) => Array.from(l).reduce((a, e2) => a + calcTime(e2.innerText), 0);
    let time = c(e.querySelectorAll("ul.line_list > li > small.grey"));
    if (time) {
      data = { id, time };
      await db.put("times", data);
      return { time, a: true };
    }
    const se = await f(`subject/${id}`);
    time = c(se.querySelectorAll("ul#infobox > li"));
    if (time) {
      data = { id, time };
      await db.put("times", data);
      return { time, a: true };
    }
    const type = se.querySelector("h1.nameSingle > small")?.textContent;
    const eps = e.querySelectorAll("ul.line_list > li > h6").length;
    data = { id, type, eps };
    await db.put("times", data);
    return { time: eps * AnimeTypeTimes[type] || 0, a: false };
  }
  async function totalTime(list) {
    const total = {
      total: { name: "总计", time: 0, count: 0 },
      normal: { name: "精确", time: 0, count: 0 },
      guess: { name: "推测", time: 0, count: 0 },
      unknown: { name: "未知", time: 0, count: 0 }
    };
    Event.emit("process", { type: "totalTime", data: { total: list.length } });
    for (const { id } of list) {
      Event.emit("process", {
        type: "totalTimeItem",
        data: { id, count: total.total.count + 1 }
      });
      const { time, a } = await ftime(id);
      if (a) {
        total.normal.count++;
        total.normal.time += time;
      } else if (time) {
        total.guess.count++;
        total.guess.time += time;
      } else {
        total.unknown.count++;
      }
      total.total.count++;
      total.total.time += time;
    }
    return total;
  }
  async function bsycs(type, subtype, year) {
    const data = await fl(type, subtype);
    if (!data) return [1, 1];
    const { max } = data;
    let startL = 1;
    let startR = 1;
    let endL = max;
    let endR = max;
    let dL = false;
    let dR = false;
    while (startL <= endL && startR <= endR) {
      const mid = startL < endL ? Math.max(Math.min(Math.floor((startL + endL) / 2), endL), startL) : Math.max(Math.min(Math.floor((startR + endR) / 2), endR), startR);
      Event.emit("process", {
        type: "bsycs",
        data: { type, subtype, p: mid }
      });
      const data2 = await fl(type, subtype, mid);
      if (!data2) return [1, 1];
      const { list } = data2;
      if (list.length == 0) return [1, 1];
      const first = list[0].year;
      const last = list[list.length - 1].year;
      if (first > year && last < year) return [mid, mid];
      if (last > year) {
        if (!dL) startL = Math.min(mid + 1, endL);
        if (!dR) startR = Math.min(mid + 1, endR);
      } else if (first < year) {
        if (!dL) endL = Math.max(mid - 1, startL);
        if (!dR) endR = Math.max(mid - 1, startR);
      } else if (first == last) {
        if (!dL) endL = Math.max(mid - 1, startL);
        if (!dR) startR = Math.min(mid + 1, endR);
      } else if (first == year) {
        startR = endR = mid;
        if (!dL) endL = Math.min(mid + 1, endR);
      } else if (last == year) {
        startL = endL = mid;
        if (!dR) startR = Math.min(mid + 1, endR);
      }
      if (startL == endL) dL = true;
      if (startR == endR) dR = true;
      if (dL && dR) return [startL, startR];
    }
    return [1, 1];
  }
  async function cbt(type, subtype, year = 0) {
    if (!year) return cbtAll(type, subtype);
    return cbtYear(type, subtype, year);
  }
  async function cbtYear(type, subtype, year) {
    const [start, end] = await bsycs(type, subtype, year);
    Event.emit("process", { type: "collZone", data: { zone: [start, end] } });
    const ret = [];
    for (let i = start; i <= end; i++) {
      const data = await fl(type, subtype, i);
      if (data) ret.push(data.list);
    }
    return ret.flat();
  }
  async function cbtAll(type, subtype) {
    const data = await fl(type, subtype, 1);
    if (!data) return [];
    const { list, max } = data;
    Event.emit("process", { type: "collZone", data: { zone: [1, max] } });
    const ret = [list];
    for (let i = 2; i <= max; i++) {
      const data2 = await fl(type, subtype, i);
      if (data2) ret.push(data2.list);
    }
    return ret.flat();
  }
  async function collects({ type, subTypes, tag, year }) {
    const ret = [];
    for (const subtype of subTypes) {
      Event.emit("process", { type: "collSubtype", data: { subtype } });
      const list = await cbt(type, subtype, year);
      ret.push(list);
    }
    const fset = /* @__PURE__ */ new Set();
    return ret.flat().filter(({ id, year: y, tags }) => {
      if (year && year != y) return false;
      if (tag && !tags.includes(tag)) return false;
      if (fset.has(id)) return false;
      fset.add(id);
      return true;
    }).sort(({ time: a }, { time: b }) => b.getTime() - a.getTime());
  }
  const css = '.v-report-btn{user-select:none;cursor:pointer}.v-report-btn.primary{background:#fc899488}.v-report-btn.primary:hover{background:#fc8994}.v-report-btn.danger{background:#fc222288}.v-report-btn.danger:hover{background:#fc2222}.v-report-btn.success{background:#22fc2288}.v-report-btn.success:hover{background:#22fc22}.v-report-btn.warning{background:#fcb12288}.v-report-btn.warning:hover{background:#fcb122}#kotori-report-canvas::-webkit-scrollbar,#kotori-report .scroll::-webkit-scrollbar{display:none}#kotori-report-menu:before{position:absolute;content:"菜单";padding:0 20px;top:-1px;right:-1px;left:-1px;height:30px;line-height:30px;background:#fc8994;backdrop-filter:blur(4px);border-radius:10px 10px 0 0}#kotori-report-menu{color:#fff;position:fixed;display:flex;flex-direction:column;top:50%;left:50%;transform:translate(-50%,-50%);padding:50px 20px 20px;background:#0d111788;backdrop-filter:blur(4px);border-radius:10px;box-shadow:2px 2px 10px #0008;border:1px solid #fc899422;min-width:150px;z-index:9999;>li:first-child{margin-top:0}>li{margin-top:10px;>.btn-group{display:flex;gap:10px;>.v-report-btn{width:100%;padding:10px 0;text-align:center;border-radius:5px;transition:all .3s;font-size:16px;font-weight:700}>.v-report-btn:hover{width:100%;padding:10px 0;text-align:center;border-radius:5px;transition:all .3s}}}>li:last-child{height:20px}fieldset{display:flex;gap:5px;min-inline-size:min-content;margin-inline:1px;border-width:1px;border-style:groove;border-color:threedface;border-image:initial;padding-block:.35em .625em;padding-inline:.75em;>div{display:flex;gap:2px;justify-content:center}}}#kotori-report{color:#fff;position:fixed;inset:0;z-index:9999;>.close{position:absolute;inset:0;background:#0000004d;backdrop-filter:blur(2px)}>.save{position:absolute;top:10px;right:10px;width:40px;height:40px;background:#fc8994;border-radius:40px;border:4px solid #fc8994;cursor:pointer;box-shadow:2px 2px 10px #0008;user-select:none;line-height:40px;background-size:40px;background-image:url();opacity:.8;z-index:9999999999999}>.scroll{position:absolute;top:0;bottom:0;left:50%;transform:translate(-50%);overflow:scroll;>.content{display:flex;flex-direction:column;gap:5px;width:1078px;margin:0 auto;.banner{height:110px;background:#fc899488;backdrop-filter:blur(2px);color:#fff;text-shadow:0 0 5px #000;h1{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);font-size:36px;line-height:36px;text-align:center}.uid{position:absolute;top:5px;left:5px;font-size:20px}ul.bars{position:absolute;display:flex;flex-direction:column;justify-content:space-evenly;>li{position:relative;justify-content:center;>div:last-child{position:absolute;width:60px;top:50%;transform:translateY(-50%);height:3px;transition:all .3s;>div{position:absolute;top:0;height:100%;background:#fff}}}}ul.lb{align-items:flex-end;>li{>div:first-child{text-align:left;padding-left:65px}>div:last-child{left:0;>div{right:0}}}}ul.rb{align-items:flex-start;>li{>div:first-child{text-align:right;padding-right:65px}>div:last-child{right:0;>div{left:0}}}}ul.total-time{font-family:consolas,courier new,monospace,courier;bottom:0;left:0;>li>div:first-child{width:150px}}ul.includes{top:0;right:0;>li>div:first-child{width:80px}}}ul.year-cover{display:flex;flex-direction:column;gap:5px;>li{position:relative;>h2{position:relative;padding:2px;text-align:center;background:#fc899488;backdrop-filter:blur(2px);color:#fff;font-weight:700;text-shadow:0 0 4px #000;>span{position:absolute;top:50%;right:10px;transform:translateY(-50%);font-size:14px;color:#ffde20}}}>li:before{content:"";display:block;position:absolute;inset:0;border:1px solid #fc8994;box-sizing:border-box}}>.bar-group{display:flex;justify-content:space-between;align-items:flex-end;ul.bars{display:flex;flex-direction:column;gap:2px;position:relative;width:calc(50% - 1px);>li{display:block;position:relative;width:100%;height:20px;background:#0008;margin:0;line-height:20px;backdrop-filter:blur(2px);>span{position:absolute;left:5px;text-shadow:0 0 2px #000}>span:nth-child(2){position:absolute;left:50%;transform:translate(-50%)}>div{display:inline-block;height:100%;background:#fc8994aa;margin:0}}}}ul.covers[type=music]>li{height:150px}ul.covers{line-height:0;>li{display:inline-block;position:relative;width:150px;height:220px;margin:2px;overflow:hidden;border-width:1px;border-style:solid;border-color:#fc8994;box-sizing:border-box;img{max-height:100%;position:absolute;top:0;left:50%;transform:translate(-50%)}>span{width:50px;height:30px;position:absolute;top:0;left:0;line-height:30px;text-align:center;font-size:18px;background:#8c49548c;backdrop-filter:blur(2px)}.star{display:block;position:absolute;bottom:3px;right:3px;width:20px;height:20px;padding:5px;background:none;>img{opacity:.85}>span{position:absolute;top:50%;left:50%;color:#f4a;font-family:consolas,courier new,monospace,courier;font-size:18px;font-weight:700;text-shadow:0 0 2px #fff;transform:translate(-50%,-50%)}}}}}}}#kotori-report-canvas{color:#fff;position:fixed;inset:0;z-index:9999;background:#0000004d;backdrop-filter:blur(2px);overflow:scroll;padding:30px;scrollbar-width:none;-ms-overflow-style:none;>div{position:absolute;inset:0;background:#0000004d;backdrop-filter:blur(2px)}>canvas{position:absolute;top:0;left:50%;transform:translate(-50%)}}@media screen and (min-width: 616px){#kotori-report .content{width:616px!important}}@media screen and (min-width: 830px){#kotori-report .content{width:770px!important}}@media screen and (min-width: 924px){#kotori-report .content{width:924px!important}}@media screen and (min-width: 1138px){#kotori-report .content{width:1078px!important}}';
  const Star = "data:image/svg+xml,%3csvg%20fill='%23ffb300'%20width='800px'%20height='800px'%20viewBox='43%20159.5%2021%2021'%20version='1.1'%20xmlns='http://www.w3.org/2000/svg'%20xmlns:xlink='http://www.w3.org/1999/xlink'%3e%3cpath%20d='M60.556381,172.206%20C60.1080307,172.639%2059.9043306,173.263%2060.0093306,173.875%20L60.6865811,177.791%20C60.8976313,179.01%2059.9211306,180%2058.8133798,180%20C58.5214796,180%2058.2201294,179.931%2057.9282291,179.779%20L54.3844766,177.93%20C54.1072764,177.786%2053.8038262,177.714%2053.499326,177.714%20C53.1958758,177.714%2052.8924256,177.786%2052.6152254,177.93%20L49.0714729,179.779%20C48.7795727,179.931%2048.4782224,180%2048.1863222,180%20C47.0785715,180%2046.1020708,179.01%2046.3131209,177.791%20L46.9903714,173.875%20C47.0953715,173.263%2046.8916713,172.639%2046.443321,172.206%20L43.575769,169.433%20C42.4480682,168.342%2043.0707186,166.441%2044.6289197,166.216%20L48.5916225,165.645%20C49.211123,165.556%2049.7466233,165.17%2050.0227735,164.613%20L51.7951748,161.051%20C52.143775,160.35%2052.8220755,160%2053.499326,160%20C54.1776265,160%2054.855927,160.35%2055.2045272,161.051%20L56.9769285,164.613%20C57.2530787,165.17%2057.7885791,165.556%2058.4080795,165.645%20L62.3707823,166.216%20C63.9289834,166.441%2064.5516338,168.342%2063.423933,169.433%20L60.556381,172.206%20Z'%3e%3c/path%3e%3c/svg%3e";
  addStyle(css);
  const PRG = ["|", "/", "-", "\\"];
  async function showCanvas(element) {
    const canvas = await element2Canvas(element);
    const close = create("div", { style: { height: canvas.style.height } });
    const main = create("div", { id: "kotori-report-canvas" }, close, canvas);
    close.addEventListener("click", () => main.remove());
    document.body.appendChild(main);
  }
  function pw(v, m) {
    return { style: { width: v * 100 / m + "%" } };
  }
  function buildTotalTime({ total, normal, guess, unknown }) {
    const list = [total, normal, guess, unknown].sort((a, b) => b.time - a.time);
    const format = ({ name, count, time }) => `${timeFormat(time, true)} (${count})${name}`;
    const buildItem = (item) => [
      "li",
      ["div", format(item)],
      ["div", ["div", pw(item.time, total.time)]]
    ];
    return ["ul", { class: ["total-time", "bars", "rb"] }, ...list.map(buildItem)];
  }
  function buildIncludes(list, type) {
    const l = Array.from(list).map(([k, v]) => [formatSubType(k, type), v]);
    const total = l.reduce((sum, [_, v]) => sum + v, 0);
    l.unshift(["总计", total]);
    l.sort((a, b) => b[1] - a[1]);
    const format = (k, v) => k + ":" + ("" + v).padStart(5, " ") + Types[type].unit;
    const buildItem = ([k, v]) => [
      "li",
      ["div", format(k, v)],
      ["div", ["div", pw(v, total)]]
    ];
    return ["ul", { class: ["includes", "bars", "lb"] }, ...l.map(buildItem)];
  }
  function buildBarList(list) {
    const l = Array.from(list).sort(([, , a], [, , b]) => a - b);
    const m = Math.max(...l.map(([v]) => v));
    const buildItem = ([v, t]) => ["li", ["span", t], ["span", v], ["div", pw(v, m)]];
    return ["ul", { class: "bars" }, ...l.map(buildItem)];
  }
  function buildCoverList(list, type) {
    let last = -1;
    const covers = [];
    for (const { img, month, star } of list) {
      const childs = [["img", { src: img }]];
      if (month != last) {
        childs.push(["span", month + 1 + "月"]);
        last = month;
      }
      if (star)
        childs.push([
          "div",
          { class: "star" },
          ["img", { src: Star }],
          ["span", star]
        ]);
      covers.push(["li", ...childs]);
    }
    return ["ul", { class: "covers", type }, ...covers];
  }
  async function buildLifeTimeReport({ type, tag, subTypes, totalTime: ttt }) {
    const list = await collects({ type, subTypes, tag });
    const time = ttt ? await totalTime(list) : null;
    const buildYearCover = ([year, l]) => ["li", ["h2", year + "年", ["span", l.length]], buildCoverList(l, type)];
    const banner = [
      "div",
      { class: "banner" },
      ["h1", `Bangumi ${Types[type].name}生涯总览`],
      ["span", { class: "uid" }, "@" + uid],
      buildIncludes(groupCount(list, "subType").entries(), type)
    ];
    if (time) banner.push(buildTotalTime(time));
    const countList = buildBarList(
      groupCount(list, "month", countMap(12)).entries().map(([k, v]) => [v, k + 1 + "月", k])
    );
    const starList = buildBarList(
      groupCount(list, "star", countMap(11)).entries().map(([k, v]) => [v, k ? k + "星" : "未评分", k])
    );
    const barGroup = ["div", { class: "bar-group" }, countList, starList];
    const yearCover = [
      "ul",
      { class: "year-cover" },
      ...groupBy(list, "year").entries().map(buildYearCover)
    ];
    return create("div", { class: "content" }, banner, barGroup, yearCover);
  }
  async function buildYearReport({ year, type, tag, subTypes, totalTime: t }) {
    const list = await collects({ type, subTypes, tag, year });
    const time = t ? await totalTime(list) : null;
    const banner = [
      "div",
      { class: "banner" },
      ["h1", `${year}年 Bangumi ${Types[type].name}年鉴`],
      ["span", { class: "uid" }, "@" + uid],
      buildIncludes(groupCount(list, "subType").entries(), type)
    ];
    if (time) banner.push(buildTotalTime(time));
    const countList = buildBarList(
      groupCount(list, "month", countMap(12)).entries().map(([k, v]) => [v, k + 1 + "月", k])
    );
    const starList = buildBarList(
      groupCount(list, "star", countMap(11)).entries().map(([k, v]) => [v, k ? k + "星" : "未评分", k])
    );
    const barGroup = ["div", { class: "bar-group" }, countList, starList];
    return create("div", { class: "content" }, banner, barGroup, buildCoverList(list, type));
  }
  async function buildReport(options) {
    Event.emit("process", { type: "start", data: options });
    const content = await (options.isLifeTime ? buildLifeTimeReport(options) : buildYearReport(options));
    Event.emit("process", { type: "done" });
    const close = create("div", { class: "close" });
    const scroll = create("div", { class: "scroll" }, content);
    const save = create("div", { class: "save" });
    const report = create("div", { id: "kotori-report" }, close, scroll, save);
    const saveFn = async () => {
      save.onclick = null;
      await showCanvas(content);
      save.onclick = saveFn;
    };
    let ly = scroll.scrollTop || 0;
    let my = ly;
    let ey = ly;
    let interval = 0;
    const scrollFn = (iey) => {
      ey = Math.max(Math.min(iey, scroll.scrollHeight - scroll.offsetHeight), 0);
      ly = my;
      if (interval) clearInterval(interval);
      let times = 1;
      interval = setInterval(() => {
        if (times > 50) {
          clearInterval(interval);
          interval = 0;
          return;
        }
        my = easeOut(times, ly, ey, 50);
        scroll.scroll({ top: my });
        times++;
      }, 1);
    };
    const wheelFn = (e) => {
      e.preventDefault();
      scrollFn(ey + e.deltaY);
    };
    const keydownFn = (e) => {
      e.preventDefault();
      if (e.key == "Escape") close.click();
      if (e.key == "Home") scrollFn(0);
      if (e.key == "End") scrollFn(scroll.scrollHeight - scroll.offsetHeight);
      if (e.key == "ArrowUp") scrollFn(ey - 100);
      if (e.key == "ArrowDown") scrollFn(ey + 100);
      if (e.key == "PageUp") scrollFn(ey - scroll.offsetHeight);
      if (e.key == "PageDown") scrollFn(ey + scroll.offsetHeight);
    };
    scroll.addEventListener("wheel", wheelFn);
    close.addEventListener("wheel", wheelFn);
    save.addEventListener("wheel", wheelFn);
    document.addEventListener("keydown", keydownFn);
    save.addEventListener("click", saveFn);
    close.addEventListener("click", () => {
      document.removeEventListener("keydown", keydownFn);
      report.remove();
    });
    document.body.appendChild(report);
  }
  function buildMenu() {
    const year = (/* @__PURE__ */ new Date()).getFullYear();
    const yearSelectOptions = new Array(year - 2007).fill(0).map((_, i) => ["option", { value: "" + (year - i) }, year - i]);
    const lifeTimeCheck = create("input", {
      type: "checkbox",
      id: "lftc"
    });
    const totalTimeCheck = create("input", {
      type: "checkbox",
      id: "tltc"
    });
    const yearSelect = create("select", {}, ...yearSelectOptions);
    const typeSelect = create(
      "select",
      {},
      ...Object.entries(Types).map(
        ([_, { value, name }]) => ["option", { value }, name]
      )
    );
    const tagSelect = create("select", ["option", { value: "" }, "不筛选"]);
    const btnGo = create("div", { class: ["v-report-btn", "primary"] }, "生成");
    const btnClr = create("div", { class: ["v-report-btn", "v-report", "warning"] }, "清理缓存");
    const btnGroup = ["div", { class: "btn-group" }, btnGo, btnClr];
    const additionField = [
      "fieldset",
      ["legend", "附加选项"],
      ["div", lifeTimeCheck, ["label", { htmlFor: "lftc" }, "生涯报告"]],
      ["div", totalTimeCheck, ["label", { htmlFor: "tltc" }, "看过时长(耗时)"]]
    ];
    const ytField = [
      "fieldset",
      ["legend", "选择年份与类型"],
      yearSelect,
      typeSelect
    ];
    const tagField = ["fieldset", ["legend", "选择过滤标签"], tagSelect];
    const subtypeField = create(
      "fieldset",
      ["legend", "选择包括的状态"],
      ...Object.entries(SubTypes).map(
        ([_, { value, name, checked }]) => [
          "div",
          { "data-value": value },
          [
            "input",
            {
              type: "checkbox",
              id: "yst_" + value,
              name,
              value,
              checked
            }
          ],
          ["label", { htmlFor: "yst_" + value }, name]
        ]
      )
    );
    const eventInfo = create("li");
    const menu2 = create(
      "ul",
      { id: "kotori-report-menu" },
      ["li", additionField],
      ["li", ytField],
      ["li", tagField],
      ["li", subtypeField],
      ["li", btnGroup],
      eventInfo
    );
    Event.on(
      "process",
      /* @__PURE__ */ (() => {
        let type;
        let zone = [0, 0];
        let subtype;
        let subtypes;
        let pz = false;
        let totalTimeCount = 0;
        return ({ type: t, data }) => {
          switch (t) {
            case "start":
              type = data.type;
              subtypes = data.subTypes;
              eventInfo.innerText = "";
              pz = false;
              break;
            case "collSubtype":
              subtype = data.subtype;
              pz = false;
              break;
            case "bsycs":
              eventInfo.innerText = `二分搜索[${formatSubType(subtype, type)}] (${data.p})`;
              break;
            case "collZone":
              zone = data.zone;
              pz = true;
              break;
            case "parse":
              if (!pz) return;
              eventInfo.innerText = `正在解析[${formatSubType(subtype, type)}] (` + (data.p - zone[0] + 1) + "/" + (zone[1] - zone[0] + 1) + ")(" + (subtypes.indexOf(subtype) + 1) + "/" + subtypes.length + ")";
              break;
            case "done":
              eventInfo.innerText = "";
              pz = false;
              break;
            case "tags":
              eventInfo.innerText = `获取标签 [${Types[data.type].name}]`;
              break;
            case "totalTime":
              totalTimeCount = data.total;
              break;
            case "totalTimeItem":
              eventInfo.innerText = `获取条目时长 (${data.count}/${totalTimeCount}) (id: ${data.id})`;
              break;
            default:
              return;
          }
        };
      })()
    );
    lifeTimeCheck.addEventListener("change", () => {
      if (lifeTimeCheck.checked) yearSelect.disabled = true;
      else yearSelect.disabled = false;
    });
    typeSelect.addEventListener(
      "change",
      callNow(async () => {
        const type = typeSelect.value;
        if (!type) return;
        totalTimeCheck.disabled = type !== "anime";
        subtypeField.querySelectorAll("div").forEach((e) => {
          const name = formatSubType(e.getAttribute("data-value"), type);
          e.querySelector("input").setAttribute("name", name);
          e.querySelector("label").innerText = name;
        });
        const tags = await ft(type);
        if (type != typeSelect.value) return;
        const last = tagSelect.value;
        removeAllChildren(tagSelect);
        tagSelect.append(create("option", { value: "" }, "不筛选"));
        append(
          tagSelect,
          ...tags.map((t) => ["option", { value: t }, t])
        );
        if (tags.includes(last)) tagSelect.value = last;
      })
    );
    btnGo.addEventListener(
      "click",
      callWhenDone(async () => {
        const type = typeSelect.value || "anime";
        await buildReport({
          type,
          subTypes: Array.from(
            subtypeField.querySelectorAll("input:checked")
          ).map((e) => e.value),
          isLifeTime: lifeTimeCheck.checked,
          totalTime: type === "anime" && totalTimeCheck.checked,
          year: parseInt(yearSelect.value) || year,
          tag: tagSelect.value
        });
        menuToggle();
      })
    );
    btnClr.addEventListener(
      "click",
      callWhenDone(async () => {
        let i = 0;
        const id = setInterval(() => btnClr.innerText = `清理缓存中[${PRG[i++ % 4]}]`, 50);
        await db.clear("pages");
        clearInterval(id);
        btnClr.innerText = "清理缓存";
      })
    );
    document.body.appendChild(menu2);
    return menu2;
  }
  let menu = null;
  function menuToggle() {
    menu ??= buildMenu();
    menu.style.display = menu.style.display == "block" ? "none" : "block";
  }
  (async () => {
    await db.init();
    const btn = create(
      "a",
      { class: "chiiBtn", href: "javascript:void(0)", title: "生成年鉴" },
      ["span", "生成年鉴"]
    );
    btn.addEventListener("click", menuToggle);
    document.querySelector("#headerProfile .actions").append(btn);
  })();

})();