// ==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/* // @downloadURL https://update.greasyfork.icu/scripts/456969/Bangumi%20%E5%B9%B4%E9%89%B4.user.js // @updateURL https://update.greasyfork.icu/scripts/456969/Bangumi%20%E5%B9%B4%E9%89%B4.meta.js // ==/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(//g, ''); 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); })(); })();