// ==UserScript== // @name bangumi anime score compare // @name:zh-CN bangumi动画和豆瓣及MAL评分对比 // @namespace https://github.com/22earth // @description show subject score information of douban and MAL in Bangumi or Douban // @description:zh-cn 在Bangumi或者豆瓣上面显示其它动画评分网站的评分 // @author 22earth // @homepage https://github.com/22earth/gm_scripts // @include /^https?:\/\/(bangumi|bgm|chii)\.(tv|in)\/subject\/.*$/ // @include https://movie.douban.com/subject/* // @version 0.3.5 // @note 0.3.3 支持 Anidb // @note 0.2.0 支持豆瓣上显示Bangumi评分,暂时禁用豆瓣上显示MAL的评分功能以及修改过滤方式 // @note 0.2.4 豆瓣 api 失效,使用搜索页面查询结果 // @TODO 统一豆瓣和Bangumi的缓存数据信息, // @grant GM_addStyle // @grant GM_registerMenuCommand // @grant GM_xmlhttpRequest // @grant GM_getResourceURL // @grant GM_setValue // @grant GM_getValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_addValueChangeListener // @require https://cdn.staticfile.org/fuse.js/6.4.0/fuse.min.js // @resource bangumi_favicon https://bgm.tv/img/favicon.ico // @resource douban_favicon https://img3.doubanio.com/favicon.ico // @resource myanimelist_favicon https://cdn.myanimelist.net/images/favicon.ico // @resource anidb_favicon https://cdn-us.anidb.net/css/icons/touch/favicon.ico // ==/UserScript== var SubjectTypeId; (function (SubjectTypeId) { SubjectTypeId[SubjectTypeId["book"] = 1] = "book"; SubjectTypeId[SubjectTypeId["anime"] = 2] = "anime"; SubjectTypeId[SubjectTypeId["music"] = 3] = "music"; SubjectTypeId[SubjectTypeId["game"] = 4] = "game"; SubjectTypeId[SubjectTypeId["real"] = 6] = "real"; SubjectTypeId["all"] = "all"; })(SubjectTypeId || (SubjectTypeId = {})); function sleep(num) { return new Promise((resolve) => { setTimeout(resolve, num); }); } function randomSleep(max = 400, min = 200) { return sleep(randomNum(max, min)); } function randomNum(max, min) { return Math.floor(Math.random() * (max - min + 1)) + min; } // support GM_XMLHttpRequest let retryCounter = 0; function fetchInfo(url, type, opts = {}, TIMEOUT = 10 * 1000) { var _a; const method = ((_a = opts === null || opts === void 0 ? void 0 : opts.method) === null || _a === void 0 ? void 0 : _a.toUpperCase()) || 'GET'; // @ts-ignore { const gmXhrOpts = Object.assign({}, opts); if (method === 'POST' && gmXhrOpts.body) { gmXhrOpts.data = gmXhrOpts.body; } if (opts.decode) { type = 'arraybuffer'; } return new Promise((resolve, reject) => { // @ts-ignore GM_xmlhttpRequest(Object.assign({ method, timeout: TIMEOUT, url, responseType: type, onload: function (res) { if (res.status === 404) { retryCounter = 0; reject(404); } else if (res.status === 302 && retryCounter < 5) { retryCounter++; resolve(fetchInfo(res.finalUrl, type, opts, TIMEOUT)); } if (opts.decode && type === 'arraybuffer') { retryCounter = 0; let decoder = new TextDecoder(opts.decode); resolve(decoder.decode(res.response)); } else { retryCounter = 0; resolve(res.response); } }, onerror: (e) => { retryCounter = 0; reject(e); } }, gmXhrOpts)); }); } } function fetchText(url, opts = {}, TIMEOUT = 10 * 1000) { return fetchInfo(url, 'text', opts, TIMEOUT); } function fetchJson(url, opts = {}) { return fetchInfo(url, 'json', opts); } function dealDate(dataStr) { // 2019年12月19 let l = []; if (/\d{4}年\d{1,2}月(\d{1,2}日?)?/.test(dataStr)) { l = dataStr .replace('日', '') .split(/年|月/) .filter((i) => i); } else if (/\d{4}\/\d{1,2}(\/\d{1,2})?/.test(dataStr)) { l = dataStr.split('/'); } else if (/\d{4}-\d{1,2}(-\d{1,2})?/.test(dataStr)) { return dataStr; } else { return dataStr; } return l .map((i) => { if (i.length === 1) { return `0${i}`; } return i; }) .join('-'); } function isEqualDate(d1, d2) { const resultDate = new Date(d1); const originDate = new Date(d2); if (resultDate.getFullYear() === originDate.getFullYear() && resultDate.getMonth() === originDate.getMonth() && resultDate.getDate() === originDate.getDate()) { return true; } return false; } const SEARCH_RESULT = 'search_result'; /** * 过滤搜索结果: 通过名称以及日期 * @param items * @param subjectInfo * @param opts */ function filterResults(items, subjectInfo, opts = {}, isSearch = true) { var _a; if (!items) return; // 只有一个结果时直接返回, 不再比较日期 if (items.length === 1 && isSearch) { return items[0]; } var results = new Fuse(items, Object.assign({}, opts)).search(subjectInfo.name); if (!results.length) return; // 有参考的发布时间 const tempResults = []; if (subjectInfo.releaseDate) { for (const obj of results) { const result = obj.item; if (result.releaseDate) { // 只有年的时候 if (result.releaseDate.length === 4) { if (result.releaseDate === subjectInfo.releaseDate.slice(0, 4)) { return result; } } else { if (isEqualDate(result.releaseDate, subjectInfo.releaseDate)) { return result; } } // 过滤年份不一致的数据 if (result.releaseDate.slice(0, 4) === subjectInfo.releaseDate.slice(0, 4)) { tempResults.push(obj); } } } } // 比较名称 const nameRe = new RegExp(subjectInfo.name.trim()); for (const item of results) { const result = item.item; if (nameRe.test(result.name) || nameRe.test(result.greyName) || nameRe.test(result.rawName)) { return result; } } results = tempResults; return (_a = results[0]) === null || _a === void 0 ? void 0 : _a.item; } async function getSearchResultByGM() { return new Promise((resolve, reject) => { const listenId = window.gm_val_listen_id; if (listenId) { GM_removeValueChangeListener(listenId); } window.gm_val_listen_id = GM_addValueChangeListener( // const listenId = GM_addValueChangeListener( SEARCH_RESULT, (n, oldValue, newValue) => { console.log('enter promise'); const now = +new Date(); if (newValue.type === SEARCH_RESULT && newValue.timestamp && newValue.timestamp < now) { // GM_removeValueChangeListener(listenId); resolve(newValue.data); } reject('mismatch timestamp'); }); }); } var BangumiDomain; (function (BangumiDomain) { BangumiDomain["chii"] = "chii.in"; BangumiDomain["bgm"] = "bgm.tv"; BangumiDomain["bangumi"] = "bangumi.tv"; })(BangumiDomain || (BangumiDomain = {})); var Protocol; (function (Protocol) { Protocol["http"] = "http"; Protocol["https"] = "https"; })(Protocol || (Protocol = {})); /** * 处理搜索页面的 html * @param info 字符串 html */ function dealSearchResults(info) { const results = []; let $doc = new DOMParser().parseFromString(info, 'text/html'); let items = $doc.querySelectorAll('#browserItemList>li>div.inner'); // get number of page let numOfPage = 1; let pList = $doc.querySelectorAll('.page_inner>.p'); if (pList && pList.length) { let tempNum = parseInt(pList[pList.length - 2].getAttribute('href').match(/page=(\d*)/)[1]); numOfPage = parseInt(pList[pList.length - 1].getAttribute('href').match(/page=(\d*)/)[1]); numOfPage = numOfPage > tempNum ? numOfPage : tempNum; } if (items && items.length) { for (const item of Array.prototype.slice.call(items)) { let $subjectTitle = item.querySelector('h3>a.l'); let itemSubject = { name: $subjectTitle.textContent.trim(), // url 没有协议和域名 url: $subjectTitle.getAttribute('href'), greyName: item.querySelector('h3>.grey') ? item.querySelector('h3>.grey').textContent.trim() : '', }; let matchDate = item .querySelector('.info') .textContent.match(/\d{4}[\-\/\年]\d{1,2}[\-\/\月]\d{1,2}/); if (matchDate) { itemSubject.releaseDate = dealDate(matchDate[0]); } let $rateInfo = item.querySelector('.rateInfo'); if ($rateInfo) { if ($rateInfo.querySelector('.fade')) { itemSubject.score = $rateInfo.querySelector('.fade').textContent; itemSubject.count = $rateInfo .querySelector('.tip_j') .textContent.replace(/[^0-9]/g, ''); } else { itemSubject.score = '0'; itemSubject.count = '少于10'; } } else { itemSubject.score = '0'; itemSubject.count = '0'; } results.push(itemSubject); } } else { return []; } return [results, numOfPage]; } /** * 搜索条目 * @param subjectInfo * @param type * @param uniqueQueryStr */ async function searchSubject(subjectInfo, bgmHost = 'https://bgm.tv', type = SubjectTypeId.all, uniqueQueryStr = '') { let releaseDate; if (subjectInfo && subjectInfo.releaseDate) { releaseDate = subjectInfo.releaseDate; } let query = (subjectInfo.name || '').trim(); if (type === SubjectTypeId.book) { // 去掉末尾的括号并加上引号 query = query.replace(/([^0-9]+?)|\([^0-9]+?\)$/, ''); query = `"${query}"`; } if (uniqueQueryStr) { query = `"${uniqueQueryStr || ''}"`; } if (!query || query === '""') { console.info('Query string is empty'); return; } const url = `${bgmHost}/subject_search/${encodeURIComponent(query)}?cat=${type}`; console.info('search bangumi subject URL: ', url); const rawText = await fetchText(url); const rawInfoList = dealSearchResults(rawText)[0] || []; // 使用指定搜索字符串如 ISBN 搜索时, 并且结果只有一条时,不再使用名称过滤 if (uniqueQueryStr && rawInfoList && rawInfoList.length === 1) { return rawInfoList[0]; } const options = { keys: ['name', 'greyName'], }; return filterResults(rawInfoList, subjectInfo, options); } /** * 通过时间查找条目 * @param subjectInfo 条目信息 * @param pageNumber 页码 * @param type 条目类型 */ async function findSubjectByDate(subjectInfo, bgmHost = 'https://bgm.tv', pageNumber = 1, type) { if (!subjectInfo || !subjectInfo.releaseDate || !subjectInfo.name) { throw new Error('invalid subject info'); } const releaseDate = new Date(subjectInfo.releaseDate); if (isNaN(releaseDate.getTime())) { throw `invalid releasedate: ${subjectInfo.releaseDate}`; } const sort = releaseDate.getDate() > 15 ? 'sort=date' : ''; const page = pageNumber ? `page=${pageNumber}` : ''; let query = ''; if (sort && page) { query = '?' + sort + '&' + page; } else if (sort) { query = '?' + sort; } else if (page) { query = '?' + page; } const url = `${bgmHost}/${type}/browser/airtime/${releaseDate.getFullYear()}-${releaseDate.getMonth() + 1}${query}`; console.info('find subject by date: ', url); const rawText = await fetchText(url); let [rawInfoList, numOfPage] = dealSearchResults(rawText); const options = { threshold: 0.3, keys: ['name', 'greyName'], }; let result = filterResults(rawInfoList, subjectInfo, options, false); if (!result) { if (pageNumber < numOfPage) { await sleep(300); return await findSubjectByDate(subjectInfo, bgmHost, pageNumber + 1, type); } else { throw 'notmatched'; } } return result; } async function checkBookSubjectExist(subjectInfo, bgmHost = 'https://bgm.tv', type) { let searchResult = await searchSubject(subjectInfo, bgmHost, type, subjectInfo.isbn); console.info(`First: search book of bangumi: `, searchResult); if (searchResult && searchResult.url) { return searchResult; } searchResult = await searchSubject(subjectInfo, bgmHost, type, subjectInfo.asin); console.info(`Second: search book by ${subjectInfo.asin}: `, searchResult); if (searchResult && searchResult.url) { return searchResult; } // 默认使用名称搜索 searchResult = await searchSubject(subjectInfo, bgmHost, type); console.info('Third: search book of bangumi: ', searchResult); return searchResult; } /** * 查找条目是否存在: 通过名称搜索或者日期加上名称的过滤查询 * @param subjectInfo 条目基本信息 * @param bgmHost bangumi 域名 * @param type 条目类型 */ async function checkExist(subjectInfo, bgmHost = 'https://bgm.tv', type, disabelDate) { const subjectTypeDict = { [SubjectTypeId.game]: 'game', [SubjectTypeId.anime]: 'anime', [SubjectTypeId.music]: 'music', [SubjectTypeId.book]: 'book', [SubjectTypeId.real]: 'real', [SubjectTypeId.all]: 'all', }; let searchResult = await searchSubject(subjectInfo, bgmHost, type); console.info(`First: search result of bangumi: `, searchResult); if (searchResult && searchResult.url) { return searchResult; } if (disabelDate) { return; } searchResult = await findSubjectByDate(subjectInfo, bgmHost, 1, subjectTypeDict[type]); console.info(`Second: search result by date: `, searchResult); return searchResult; } async function checkSubjectExist(subjectInfo, bgmHost = 'https://bgm.tv', type = SubjectTypeId.all, disableDate) { let result; switch (type) { case SubjectTypeId.book: result = await checkBookSubjectExist(subjectInfo, bgmHost, type); break; case SubjectTypeId.all: case SubjectTypeId.game: case SubjectTypeId.anime: result = await checkExist(subjectInfo, bgmHost, type, disableDate); break; case SubjectTypeId.real: case SubjectTypeId.music: default: console.info('not support type: ', type); } return result; } /** * 为页面添加样式 * @param style */ /** * 获取节点文本 * @param elem */ function getText(elem) { if (!elem) return ''; if (elem.tagName.toLowerCase() === 'meta') { return elem.content; } if (elem.tagName.toLowerCase() === 'input') { return elem.value; } return elem.textContent || elem.innerText || ''; } /** * dollar 选择单个 * @param {string} selector */ function $q(selector) { if (window._parsedEl) { return window._parsedEl.querySelector(selector); } return document.querySelector(selector); } /** * dollar 选择所有元素 * @param {string} selector */ function $qa(selector) { if (window._parsedEl) { return window._parsedEl.querySelectorAll(selector); } return document.querySelectorAll(selector); } /** * 查找包含文本的标签 * @param {string} selector * @param {string} text */ function contains(selector, text, $parent) { let elements; if ($parent) { elements = $parent.querySelectorAll(selector); } else { elements = $qa(selector); } let t; if (typeof text === 'string') { t = text; } else { t = text.join('|'); } return [].filter.call(elements, function (element) { return new RegExp(t, 'i').test(getText(element)); }); } function findElementByKeyWord(selector, $parent) { let res = null; if ($parent) { $parent = $parent.querySelector(selector.selector); } else { $parent = $q(selector.selector); } if (!$parent) return res; const targets = contains(selector.subSelector, selector.keyWord, $parent); if (targets && targets.length) { let $t = targets[targets.length - 1]; // 相邻节点 if (selector.sibling) { $t = targets[targets.length - 1].nextElementSibling; } return $t; } return res; } function findElement(selector, $parent) { var _a; let r = null; if (selector) { if (selector instanceof Array) { let i = 0; let targetSelector = selector[i]; while (targetSelector && !(r = findElement(targetSelector, $parent))) { targetSelector = selector[++i]; } } else { if (!selector.subSelector) { r = $parent ? $parent.querySelector(selector.selector) : $q(selector.selector); } else if (selector.isIframe) { // iframe 暂时不支持 parent const $iframeDoc = (_a = $q(selector.selector)) === null || _a === void 0 ? void 0 : _a.contentDocument; r = $iframeDoc === null || $iframeDoc === void 0 ? void 0 : $iframeDoc.querySelector(selector.subSelector); } else { r = findElementByKeyWord(selector, $parent); } if (selector.closest) { r = r.closest(selector.closest); } // recursive if (r && selector.nextSelector) { const nextSelector = selector.nextSelector; r = findElement(nextSelector, r); } } } return r; } /** * 载入 iframe * @param $iframe iframe DOM * @param src iframe URL * @param TIMEOUT time out */ function loadIframe($iframe, src, TIMEOUT = 5000) { return new Promise((resolve, reject) => { $iframe.src = src; let timer = setTimeout(() => { timer = null; $iframe.onload = undefined; reject('iframe timeout'); }, TIMEOUT); $iframe.onload = () => { clearTimeout(timer); $iframe.onload = null; resolve(null); }; }); } function convertHomeSearchItem($item) { const dealHref = (href) => { if (/^https:\/\/movie\.douban\.com\/subject\/\d+\/$/.test(href)) { return href; } const urlParam = href.split('?url=')[1]; if (urlParam) { return decodeURIComponent(urlParam.split('&')[0]); } else { throw 'invalid href'; } }; const $title = $item.querySelector('.title h3 > a'); const href = dealHref($title.getAttribute('href')); const $ratingNums = $item.querySelector('.rating-info > .rating_nums'); let ratingsCount = ''; let averageScore = ''; if ($ratingNums) { const $count = $ratingNums.nextElementSibling; const m = $count.innerText.match(/\d+/); if (m) { ratingsCount = m[0]; } averageScore = $ratingNums.innerText; } let greyName = ''; const $greyName = $item.querySelector('.subject-cast'); if ($greyName) { greyName = $greyName.innerText; } return { name: $title.textContent.trim(), greyName: greyName.split('/')[0].replace('原名:', '').trim(), releaseDate: (greyName.match(/\d{4}$/) || [])[0], url: href, score: averageScore, count: ratingsCount, }; } /** * 通过首页搜索的结果 * @param query 搜索字符串 */ async function getHomeSearchResults(query, cat = '1002') { const url = `https://www.douban.com/search?cat=${cat}&q=${encodeURIComponent(query)}`; console.info('Douban search URL: ', url); const rawText = await fetchText(url); const $doc = new DOMParser().parseFromString(rawText, 'text/html'); const items = $doc.querySelectorAll('.search-result > .result-list > .result > .content'); return Array.prototype.slice .call(items) .map(($item) => convertHomeSearchItem($item)); } /** * 单独类型搜索入口 * @param query 搜索字符串 * @param cat 搜索类型 * @param type 获取传递数据的类型: gm 通过 GM_setValue, message 通过 postMessage */ async function getSubjectSearchResults(query, cat = '1002') { const url = `https://search.douban.com/movie/subject_search?search_text=${encodeURIComponent(query)}&cat=${cat}`; console.info('Douban search URL: ', url); const iframeId = 'e-userjs-search-subject'; let $iframe = document.querySelector(`#${iframeId}`); if (!$iframe) { $iframe = document.createElement('iframe'); $iframe.setAttribute('sandbox', 'allow-forms allow-same-origin allow-scripts'); $iframe.style.display = 'none'; $iframe.id = iframeId; document.body.appendChild($iframe); } // 这里不能使用 await 否则数据加载完毕了监听器还没有初始化 loadIframe($iframe, url, 1000 * 10); return await getSearchResultByGM(); } /** * * @param subjectInfo 条目信息 * @param type 默认使用主页搜索 * @returns 搜索结果 */ async function checkAnimeSubjectExist(subjectInfo, type = 'home_search') { let query = (subjectInfo.name || '').trim(); if (!query) { console.info('Query string is empty'); return Promise.reject(); } let rawInfoList; let searchResult; const options = { keys: ['name', 'greyName'], }; if (type === 'home_search') { rawInfoList = await getHomeSearchResults(query); } else { rawInfoList = await getSubjectSearchResults(query); } searchResult = filterResults(rawInfoList, subjectInfo, options, true); console.info(`Search result of ${query} on Douban: `, searchResult); if (searchResult && searchResult.url) { return searchResult; } } async function searchAnimeData(subjectInfo) { const url = `https://myanimelist.net/search/prefix.json?type=anime&keyword=${encodeURIComponent(subjectInfo.name)}&v=1`; console.info('myanimelist search URL: ', url); const info = await fetchJson(url); await randomSleep(300, 100); let startDate = null; let items = info.categories[0].items; let pageUrl = ''; let name = ''; if (subjectInfo.releaseDate) { startDate = new Date(subjectInfo.releaseDate); for (let i = 0; i < items.length; i++) { const item = items[i]; let aired = null; if (item.payload.aired.match('to')) { aired = new Date(item.payload.aired.split('to')[0]); } else { aired = new Date(item.payload.aired); } // 选择第一个匹配日期的 if (startDate.getFullYear() === aired.getFullYear() && startDate.getMonth() === aired.getMonth()) { pageUrl = item.url; name = item.name; break; } } } else if (items && items[0]) { name = items[0].name; pageUrl = items[0].url; } if (!pageUrl) { throw new Error('No match results'); } let result = { name, url: pageUrl, }; const content = await fetchText(pageUrl); const $doc = new DOMParser().parseFromString(content, 'text/html'); let $score = $doc.querySelector('.fl-l.score'); if ($score) { //siteScoreInfo.averageScore = parseFloat($score.textContent.trim()).toFixed(1) result.score = $score.textContent.trim(); if ($score.dataset.user) { result.count = $score.dataset.user.replace(/users|,/g, '').trim(); } else { throw new Error('Invalid score info'); } } else { throw new Error('Invalid results'); } console.info('myanimelist search result: ', result); return result; } async function searchAnimeData$1(subjectInfo) { let query = (subjectInfo.name || '').trim(); if (!query) { console.info('Query string is empty'); return Promise.reject('empty query'); } // 标点符号不一致 // 戦闘員、派遣します! ----> 戦闘員, 派遣します! query = subjectInfo.name .replace(/、|!/, ' ') .replace(/\s{2,}/, ' ') .trim(); const url = `https://anidb.net/perl-bin/animedb.pl?show=json&action=search&type=anime&query=${encodeURIComponent(query)}`; console.info('anidb search URL: ', url); const info = await fetchJson(url, { headers: { referrer: 'https://anidb.net/', 'content-type': 'application/json', 'accept-language': 'en-US,en;q=0.9', 'x-lcontrol': 'x-no-cache', }, }); await randomSleep(300, 100); const rawInfoList = info.map((obj) => { return Object.assign(Object.assign({}, obj), { url: obj.link, greyName: obj.hit }); }); const options = { keys: ['greyName'], }; let result; result = filterResults(rawInfoList, subjectInfo, options, true); if (result && result.url) { // 转换评分 const obj = result; const arr = (obj.desc || '').split(','); const scoreObj = { score: '0', count: '0', }; if (arr && arr.length === 3) { const scoreStr = arr[2]; if (!scoreStr.includes('N/A') && scoreStr.includes('(')) { const arr = scoreStr.split('('); scoreObj.score = arr[0].trim(); scoreObj.count = arr[1].replace(/\).*/g, ''); } } result = Object.assign(Object.assign({}, result), scoreObj); console.info('anidb search result: ', result); return result; } } const favicon = ''; const sites = ['douban', 'bangumi', 'myanimelist', 'anidb']; if (GM_registerMenuCommand) { // 用户脚本命令增加清除评分信息缓存 GM_registerMenuCommand('\u6e05\u9664\u8bc4\u5206\u7f13\u5b58', clearInfoStorage, 'c'); } const USERJS_PREFIX = 'E_USERJS_ANIME_SCORE_'; const BANGUMI_LOADING = `${USERJS_PREFIX}BANGUMI_LOADING`; const CURRENT_ID_DICT = `${USERJS_PREFIX}CURRENT_ID_DICT`; const UPDATE_INTERVAL = 24 * 60 * 60 * 1000; // 30天清理所有数据 const CLEAR_INTERVAL = UPDATE_INTERVAL * 30; function getFavicon(site) { let favicon$1 = ''; const dict = { anidb: favicon, }; if (dict[site]) { return dict[site]; } try { favicon$1 = GM_getResourceURL(`${site}_favicon`); } catch (error) { } return favicon$1; } function getSubjectId(href) { // https://anidb.net/a15320 const m = href.match(/\/(subject\/|anime\/|anidb.net\/a)(\d+)/); if (m) { return m[2]; } } function saveValue(key, val) { GM_setValue(key, val); } function clearInfoStorage() { const keys = GM_listValues(); for (const key of keys) { if (key.match(USERJS_PREFIX)) { GM_deleteValue(key); } } } function genScoreKey(site, id) { return USERJS_PREFIX + site.toUpperCase() + '_' + id; } function genSubjectIdDictKey(site, id) { return USERJS_PREFIX + site.toUpperCase() + '_DICT_ID_' + id; } function readScoreInfo(site, id) { if (!id) return; let scoreInfo = GM_getValue(genScoreKey(site, id)); if (scoreInfo) { // scoreInfo = JSON.parse(scoreInfo); if (+new Date() - +new Date(scoreInfo.date) < UPDATE_INTERVAL) { return scoreInfo.info; } } } function readSubjectIdDict(site, id) { if (!id) return; const currentDict = GM_getValue(CURRENT_ID_DICT) || {}; if (currentDict[site] === id) { return currentDict; } return GM_getValue(genSubjectIdDictKey(site, id)); } async function checkInfoUpdate() { let time = GM_getValue(USERJS_PREFIX + 'LATEST_UPDATE_TIME'); let now = new Date(); if (!time) { GM_setValue(USERJS_PREFIX + 'LATEST_UPDATE_TIME', now.getTime()); return; } else if (+now - +new Date(time) > CLEAR_INTERVAL) { clearInfoStorage(); // 清理后延迟执行一下 await sleep(200); } } function saveScoreInfo(info) { GM_setValue(genScoreKey(info.site, getSubjectId(info.url)), { info, date: +new Date(), }); } async function fetchScoreInfo(name, subjectInfo) { let info; let res; let bgmOrigin = 'https://bgm.tv'; GM_setValue(BANGUMI_LOADING, true); try { switch (name) { case 'bangumi': res = await checkSubjectExist(subjectInfo, bgmOrigin, SubjectTypeId.anime); if (!res.url.includes('http')) { res.url = `${bgmOrigin}${res.url}`; } break; case 'myanimelist': res = await searchAnimeData(subjectInfo); break; case 'anidb': res = await searchAnimeData$1(subjectInfo); break; case 'douban': res = await checkAnimeSubjectExist(subjectInfo); break; } } catch (error) { console.error(error); info = undefined; } if (res) { info = Object.assign({ site: name }, res); } GM_setValue(BANGUMI_LOADING, false); return info; } const DoubanScorePage = { name: 'douban', controlSelector: [ { selector: '#interest_sectl', }, ], pageSelector: [ { selector: 'body', subSelector: '.tags-body', keyWord: ['动画', '动漫'], }, { selector: '#info', subSelector: 'span[property="v:genre"]', keyWord: ['动画', '动漫'], }, ], getSubjectInfo() { var _a, _b, _c, _d, _e, _f; const $title = $q('#content h1>span'); const rawName = $title.textContent.trim(); const keywords = (_b = (_a = $q('meta[name="keywords"]')) === null || _a === void 0 ? void 0 : _a.getAttribute) === null || _b === void 0 ? void 0 : _b.call(_a, 'content'); let name = rawName; if (keywords) { // 可以考虑剔除第二个关键字里面的 Season 3 const firstKeyword = keywords.split(',')[0]; name = rawName.replace(firstKeyword, '').trim(); // name: rawName.replace(/第.季/, ''), } const subjectInfo = { name, score: (_d = (_c = $q('.ll.rating_num')) === null || _c === void 0 ? void 0 : _c.textContent) !== null && _d !== void 0 ? _d : 0, count: (_f = (_e = $q('.rating_people > span')) === null || _e === void 0 ? void 0 : _e.textContent) !== null && _f !== void 0 ? _f : 0, rawName, url: location.href, }; const $date = $q('span[property="v:initialReleaseDate"]'); if ($date) { subjectInfo.releaseDate = $date.textContent.replace(/\(.*\)/, ''); } return subjectInfo; }, insertScoreInfo(info) { let $panel = $q('#interest_sectl'); let $friendsRatingWrap = $q('.friends_rating_wrap'); if (!$friendsRatingWrap) { $friendsRatingWrap = document.createElement('div'); $friendsRatingWrap.className = 'friends_rating_wrap clearbox'; $panel.appendChild($friendsRatingWrap); } const score = info.score || 0; const $div = document.createElement('div'); const favicon = getFavicon(info.site); const rawHTML = `${score}
${info.count || 0}人评价 `; $div.className = 'rating_content_wrap clearfix e-userjs-score-compare'; $div.innerHTML = rawHTML; //toggleLoading(true); $friendsRatingWrap.appendChild($div); }, }; const BangumiScorePage = { name: 'bangumi', controlSelector: [ { selector: '#panelInterestWrapper h2', }, ], pageSelector: [ { selector: '.focus.chl.anime', }, ], getSubjectInfo: function () { var _a, _b, _c, _d; let info = { name: $q('h1>a').textContent.trim(), score: (_b = (_a = $q('.global_score span[property="v:average"')) === null || _a === void 0 ? void 0 : _a.textContent) !== null && _b !== void 0 ? _b : 0, count: (_d = (_c = $q('span[property="v:votes"')) === null || _c === void 0 ? void 0 : _c.textContent) !== null && _d !== void 0 ? _d : 0, url: location.href, }; let infoList = $qa('#infobox>li'); if (infoList && infoList.length) { for (let i = 0, len = infoList.length; i < len; i++) { let el = infoList[i]; if (el.innerHTML.match(/放送开始|上映年度/)) { info.releaseDate = dealDate(el.textContent.split(':')[1].trim()); } // if (el.innerHTML.match('播放结束')) { // info.endDate = dealDate(el.textContent.split(':')[1].trim()); // } } } return info; }, insertScoreInfo(info) { let $panel = $q('.SidePanel.png_bg'); if ($panel) { const score = info.score || 0; let $div = document.createElement('div'); $div.classList.add('frdScore'); $div.classList.add('e-userjs-score-compare'); const favicon = getFavicon(info.site); $div.innerHTML = ` ${Number(score).toFixed(2)} ${info.count || 0} 人评分 `; $panel.appendChild($div); } }, initControlDOM($target) { if (!$target) return; // 已存在控件时返回 if ($q('.e-userjs-score-ctrl')) return; const rawHTML = `O X `; $target.innerHTML = $target.innerHTML + rawHTML; addStyle(); document .querySelector('.e-userjs-score-clear') .addEventListener('click', clearInfoStorage, false); $q('.e-userjs-score-fresh').addEventListener('click', () => { init(BangumiScorePage, true); }, false); }, }; function addStyle(css) { if (css) { GM_addStyle(css); } else { GM_addStyle(` .e-userjs-score-ctrl {color:#f09199;font-weight:800;float:right;} .e-userjs-score-ctrl:hover {cursor: pointer;} .e-userjs-score-clear {margin-right: 12px;} .e-userjs-score-loading { width: 208px; height: 13px; background-image: url("/img/loadingAnimation.gif"); } `); } } // Bangumi Loading function toggleLoading(hidden) { let $div = $q('.e-userjs-score-loading'); if (!$div) { $div = document.createElement('div'); $div.classList.add('e-userjs-score-loading'); let $panel = $q('.SidePanel.png_bg'); $panel.appendChild($div); } if (hidden) { $div.style.display = 'none'; } else { $div.style.display = ''; } } async function init(page, force) { var _a, _b; const $page = findElement(page.pageSelector); if (!$page) return; const $title = findElement(page.controlSelector); if (!$title) return; await checkInfoUpdate(); (_a = page === null || page === void 0 ? void 0 : page.initControlDOM) === null || _a === void 0 ? void 0 : _a.call(page, $title); const curPageId = getSubjectId(location.href); const curPageScoreInfo = Object.assign({ site: page.name }, page.getSubjectInfo()); saveScoreInfo(curPageScoreInfo); let subjectIdDict = readSubjectIdDict(page.name, curPageId); // 强制刷新,不使用缓存 if (force) { subjectIdDict = undefined; // 刷新时,移除原来的数据 (_b = $qa('.frdScore.e-userjs-score-compare')) === null || _b === void 0 ? void 0 : _b.forEach(($el) => { $el.remove(); }); } let dict = Object.assign({}, subjectIdDict); for (const s of sites) { let info; if (s !== page.name) { if (subjectIdDict) { const id = subjectIdDict[s]; if (id === '-1') continue; info = readScoreInfo(s, id); } // 不存在缓存数据 if (!info) { info = await fetchScoreInfo(s, curPageScoreInfo); if (!info) { dict[s] = '-1'; continue; } } if (info) { page.insertScoreInfo(info); saveScoreInfo(info); // 索引里面没有这个数据 if (!dict[s]) { dict[s] = getSubjectId(info.url); } } } } // 保存索引数据 saveValue(genSubjectIdDictKey(page.name, curPageId), dict); saveValue(CURRENT_ID_DICT, Object.assign(Object.assign({}, dict), { [page.name]: curPageId })); } if (location.hostname.match(/bgm.tv|bangumi.tv|chii.in/)) { GM_addValueChangeListener(BANGUMI_LOADING, (n, oldValue, newValue) => { if (newValue === false) { toggleLoading(true); } else if (newValue === true) { toggleLoading(); } }); init(BangumiScorePage); } if (location.hostname.match('movie.douban.com')) { init(DoubanScorePage); }