// ==UserScript==
// @name 评分对比助手
// @name:en score comparation helper
// @namespace https://github.com/22earth
// @description 在Bangumi、豆瓣等上面显示其它网站的评分
// @description:en show subject score information from other site
// @author 22earth
// @license MIT
// @homepage https://github.com/22earth/gm_scripts
// @include /^https?:\/\/(bangumi|bgm|chii)\.(tv|in)\/subject\/.*$/
// @include https://movie.douban.com/subject/*
// @include https://myanimelist.net/anime/*
// @include https://anidb.net/anime/*
// @include https://anidb.net/a*
// @include https://2dfan.org/subjects/*
// @include https://vndb.org/v*
// @include https://erogamescape.org/~ap2/ero/toukei_kaiseki/*.php?game=*
// @include https://erogamescape.dyndns.org/~ap2/ero/toukei_kaiseki/*.php?game=*
// @include https://moepedia.net/game/*
// @include http://www.getchu.com/soft.phtml?id=*
// @version 0.1.13
// @run-at document-end
// @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
// ==/UserScript==
(function () {
'use strict';
/**
* 为页面添加样式
* @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) {
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 = $q(selector.selector)?.contentDocument;
r = $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;
}
/**
* @param {String} HTML 字符串
* @return {Element}
*/
function htmlToElement(html) {
var template = document.createElement('template');
html = html.trim();
template.innerHTML = html;
// template.content.childNodes;
return template.content.firstChild;
}
/**
* 载入 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 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;
let USER_SITE_CONFIG = {};
function addSiteOption(host, config) {
USER_SITE_CONFIG[host] = config;
}
function getSiteConfg(url, host) {
let hostname = host;
if (!host) {
hostname = new URL(url)?.hostname;
}
const config = USER_SITE_CONFIG[hostname] || {};
return config;
}
function mergeOpts(opts, config) {
return {
...opts,
...config,
headers: {
...opts?.headers,
...config?.headers,
},
};
}
function fetchInfo(url, type, opts = {}, TIMEOUT = 10 * 1000) {
const method = opts?.method?.toUpperCase() || 'GET';
opts = mergeOpts(opts, getSiteConfg(url));
// @ts-ignore
{
const gmXhrOpts = { ...opts };
if (method === 'POST' && gmXhrOpts.body) {
gmXhrOpts.data = gmXhrOpts.body;
}
if (opts.decode) {
type = 'arraybuffer';
}
return new Promise((resolve, reject) => {
// @ts-ignore
GM_xmlhttpRequest({
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 formatDate(time, fmt = "yyyy-MM-dd") {
const date = new Date(time);
var o = {
"M+": date.getMonth() + 1,
"d+": date.getDate(),
"h+": date.getHours(),
"m+": date.getMinutes(),
"s+": date.getSeconds(),
"q+": Math.floor((date.getMonth() + 3) / 3),
S: date.getMilliseconds(), //毫秒
};
if (/(y+)/i.test(fmt)) {
fmt = fmt.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length));
}
for (var k in o) {
if (new RegExp("(" + k + ")", "i").test(fmt)) {
fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : ("00" + o[k]).substr(("" + o[k]).length));
}
}
return fmt;
}
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;
}
function normalizeQuery(query) {
let newQuery = query
.replace(/([^~]*~[^~]*~[^~]*)/g, function (match) {
return match.replace(/~|~/g, " ");
})
.replace(/=|=/g, " ")
.replace(/0/g, "0")
.replace(/1/g, "1")
.replace(/2/g, "2")
.replace(/3/g, "3")
.replace(/4/g, "4")
.replace(/5/g, "5")
.replace(/6/g, "6")
.replace(/7/g, "7")
.replace(/8/g, "8")
.replace(/9/g, "9")
.replace(/Ⅰ/g, "I")
.replace(/Ⅱ/g, "II")
.replace(/Ⅲ/g, "III")
.replace(/Ⅳ/g, "IV")
.replace(/Ⅴ/g, "V")
.replace(/Ⅵ/g, "VI")
.replace(/Ⅶ/g, "VII")
.replace(/Ⅷ/g, "VIII")
.replace(/Ⅸ/g, "IX")
.replace(/Ⅹ/g, "X")
.replace(/-|-/g, " ")
.replace(/\s{2,}/g, " ")
.replace(/~/g, "~")
.trim();
return newQuery;
}
const SEARCH_RESULT = 'search_result';
/**
* 过滤搜索结果: 通过名称以及日期
* @param items
* @param subjectInfo
* @param opts
*/
function filterResults(items, subjectInfo, opts = {}, isSearch = true) {
if (!items)
return;
// 只有一个结果时直接返回, 不再比较日期
if (items.length === 1 && isSearch) {
return items[0];
}
// 使用发行日期过滤
if (subjectInfo.releaseDate && opts.releaseDate) {
const obj = items.find((item) => isEqualDate(item.releaseDate, subjectInfo.releaseDate));
if (obj) {
return obj;
}
}
var results = new Fuse(items, Object.assign({}, opts)).search(subjectInfo.name);
// 去掉括号包裹的,再次模糊查询
if (!results.length && /<|<|\(|(/.test(subjectInfo.name)) {
results = new Fuse(items, Object.assign({}, opts)).search(subjectInfo.name
.replace(/<.+>/g, '')
.replace(/<.+>/g, '')
.replace(/(.+)/g, '')
.replace(/\(.+\)/g, ''));
}
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 results[0]?.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');
});
});
}
async function searchAnimeData$1(subjectInfo) {
let query = normalizeQuery((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(200, 100);
const rawInfoList = info.map((obj) => {
return {
...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 = {
...result,
...scoreObj,
};
console.info('anidb search result: ', result);
return result;
}
}
const favicon$3 = '';
const BLANK_LINK = 'target="_blank" rel="noopener noreferrer nofollow"';
const NO_MATCH_DATA = '点击搜索';
const SCORE_ROW_WRAP_CLS = 'e-userjs-score-compare';
function getFavicon(page) {
let site = page.name;
let favicon = '';
site = site.split('-')[0];
const dict = {
anidb: favicon$3,
};
if (dict[site]) {
return dict[site];
}
try {
favicon = GM_getResourceURL(`${site}_favicon`);
}
catch (error) { }
if (!favicon) {
favicon = page.favicon || '';
}
return favicon;
}
function genScoreRowStr(info) {
return `
<div class="e-userjs-score-compare-row" style="display:flex;align-items:center;margin-bottom:10px;">
<a target="_blank" rel="noopener noreferrer nofollow"
style="margin-right:1em;" title="点击在${info.name}搜索" href="${info.searchUrl}">
<img alt="${info.name}" style="width:16px;" src="${info.favicon}"/>
</a>
<strong style="margin-right:1em;">${info.score}</strong>
<a href="${info.url}"
target="_blank" rel="noopener noreferrer nofollow">
${info.count}
</a>
</div>
`;
}
function genScoreRowInfo(title, page, info) {
const favicon = getFavicon(page);
const name = page.name.split('-')[0];
let score = '0.00';
let count = NO_MATCH_DATA;
const searchUrl = page.searchApi.replace('{kw}', encodeURIComponent(normalizeQuery(title)));
let url = searchUrl;
if (info && info.url) {
score = Number(info.score || 0).toFixed(2);
count = (info.count || 0) + ' 人评分';
url = info.url;
}
return { favicon, count, score, url, searchUrl, name };
}
function getScoreWrapDom(adjacentSelector, cls = '', style = '') {
let $div = document.querySelector('.' + SCORE_ROW_WRAP_CLS);
if (!$div) {
$div = document.createElement('div');
$div.className = `${SCORE_ROW_WRAP_CLS} ${cls}`;
$div.setAttribute('style', `margin-top:10px;${style}`);
findElement(adjacentSelector)?.insertAdjacentElement('afterend', $div);
}
return $div;
}
function insertScoreRow(wrapDom, rowInfo) {
wrapDom.appendChild(htmlToElement(genScoreRowStr(rowInfo)));
}
function insertScoreCommon(page, info, opts) {
const wrapDom = getScoreWrapDom(opts.adjacentSelector, opts.cls, opts.style);
const rowInfo = genScoreRowInfo(opts.title, page, info);
insertScoreRow(wrapDom, rowInfo);
}
const anidbPage = {
name: 'anidb',
href: ['https://anidb.net'],
searchApi: 'https://anidb.net/anime/?adb.search={kw}&do.search=1',
favicon: 'https://cdn-us.anidb.net/css/icons/touch/favicon.ico',
expiration: 21,
infoSelector: [
{
selector: '#tab_1_pane',
},
],
pageSelector: [
{
selector: 'h1.anime',
},
],
getSubjectId(url) {
const m = url.match(/\/(anime\/|anidb.net\/a)(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://anidb.net/anime/${id}`;
},
getSearchResult: searchAnimeData$1,
getScoreInfo: function () {
const $table = $q('#tabbed_pane .g_definitionlist > table');
let names = $table.querySelectorAll('tr.official .value > label');
const info = {
name: names[0].textContent.trim(),
greyName: names[names.length - 1].textContent.trim(),
score: 0,
count: 0,
url: location.href,
};
const $rating = $table.querySelector('tr.rating span.rating');
if ($rating) {
info.count = $rating
.querySelector('.count')
.textContent.trim()
.replace(/\(|\)/g, '');
const score = Number($rating.querySelector('a > .value').textContent.trim());
if (!isNaN(score)) {
info.score = score;
}
const $year = $table.querySelector('tr.year > .value > span[itemprop="startDate"]');
if ($year) {
info.releaseDate = $year.getAttribute('content');
}
names = $table.querySelectorAll('tr.official .value');
for (let i = 0; i < names.length; i++) {
const el = names[i];
if (el.querySelector('.icons').innerHTML.includes('japanese')) {
info.name = el.querySelector('label').textContent.trim();
}
else if (el.querySelector('.icons').innerHTML.includes('english')) {
info.greyName = el.querySelector('label').textContent.trim();
}
}
}
return info;
},
insertScoreInfo: function (page, info) {
const title = this.getScoreInfo().name;
const opts = {
title,
adjacentSelector: this.infoSelector,
cls: '',
style: '',
};
const wrapDom = getScoreWrapDom(opts.adjacentSelector, opts.cls, opts.style);
const rowInfo = genScoreRowInfo(opts.title, page, info);
// refuse blob:<URL>
rowInfo.favicon = page.favicon;
insertScoreRow(wrapDom, rowInfo);
},
};
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 = {}));
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$1(subjectInfo, bgmHost = 'https://bgm.tv', type = SubjectTypeId.all, uniqueQueryStr = '') {
if (subjectInfo && subjectInfo.releaseDate) {
subjectInfo.releaseDate;
}
let query = normalizeQuery((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$1(subjectInfo, bgmHost, type, subjectInfo.isbn);
console.info(`First: search book of bangumi: `, searchResult);
if (searchResult && searchResult.url) {
return searchResult;
}
searchResult = await searchSubject$1(subjectInfo, bgmHost, type, subjectInfo.asin);
console.info(`Second: search book by ${subjectInfo.asin}: `, searchResult);
if (searchResult && searchResult.url) {
return searchResult;
}
// 默认使用名称搜索
searchResult = await searchSubject$1(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$1(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;
}
// http://mirror.bgm.rincat.ch
let bgm_origin = 'https://bgm.tv';
function genBgmUrl(url) {
if (url.startsWith('http')) {
return url;
}
return new URL(url, bgm_origin).href;
}
const bangumiAnimePage = {
name: 'bangumi-anime',
href: ['https://bgm.tv/', 'https://bangumi.tv/', 'https://chii.in/'],
searchApi: 'https://bgm.tv/subject_search/{kw}?cat=2',
favicon: 'https://bgm.tv/img/favicon.ico',
controlSelector: [
{
selector: '#panelInterestWrapper h2',
},
],
infoSelector: [
{
selector: '#panelInterestWrapper .SidePanel > :last-child',
},
],
pageSelector: [
{
selector: '.focus.chl.anime',
},
],
getSubjectId(url) {
// @TODO 修改域名。
// const urlObj = new URL(url);
// setBgmOrigin(urlObj.origin);
// this.searchApi = `${bgm_origin}/subject_search/{kw}?cat=2`;
const m = url.match(/\/(subject)\/(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `${bgm_origin}/subject/${id}`;
},
async getSearchResult(subject) {
const res = await checkSubjectExist(subject, bgm_origin, SubjectTypeId.anime);
if (res) {
res.url = genBgmUrl(res.url);
}
return res;
},
getScoreInfo: () => {
const info = {
name: $q('h1>a').textContent.trim(),
score: $q('.global_score span[property="v:average"')?.textContent ?? 0,
count: $q('span[property="v:votes"')?.textContent ?? 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;
},
// 插入评分信息的 DOM
insertScoreInfo(page, info) {
const title = $q('h1>a').textContent.trim();
const opts = {
title,
adjacentSelector: this.infoSelector,
};
const wrapDom = getScoreWrapDom(opts.adjacentSelector);
const rowInfo = genScoreRowInfo(opts.title, page, info);
const rowStr = `
<div class="e-userjs-score-compare-row frdScore">
<a class="avatar"
target="_blank" rel="noopener noreferrer nofollow"
style="vertical-align:-3px;margin-right:10px;" title="点击在${rowInfo.name}搜索" href="${rowInfo.searchUrl}">
<img style="width:16px;" src="${rowInfo.favicon}"/>
</a>
<span class="num">${rowInfo.score}</span>
<span class="desc" style="visibility:hidden">还行</span>
<a href="${rowInfo.url}"
target="_blank" rel="noopener noreferrer nofollow" class="l">
${rowInfo.count}
</a>
</div>
`;
wrapDom.appendChild(htmlToElement(rowStr));
},
insertControlDOM($target, callbacks) {
if (!$target)
return;
// 已存在控件时返回
if ($q('.e-userjs-score-ctrl'))
return;
const rawHTML = `<a title="强制刷新评分" class="e-userjs-score-ctrl e-userjs-score-fresh">O</a>
<a title="清除所有评分缓存" class="e-userjs-score-ctrl e-userjs-score-clear">X</a>
`;
$target.innerHTML = $target.innerHTML + rawHTML;
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"); }
`);
$q('.e-userjs-score-clear').addEventListener('click', callbacks.clear, false);
$q('.e-userjs-score-fresh').addEventListener('click', callbacks.refresh, false);
},
};
const bangumiGamePage = {
...bangumiAnimePage,
name: 'bangumi-game',
searchApi: 'https://bgm.tv/subject_search/{kw}?cat=4',
expiration: 21,
pageSelector: [
{
selector: 'a.focus.chl[href="/game"]',
},
],
async getSearchResult(subject) {
const res = await checkSubjectExist(subject, bgm_origin, SubjectTypeId.game);
if (res) {
res.url = genBgmUrl(res.url);
}
return res;
},
};
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;
}
}
const doubanAnimePage = {
name: 'douban-anime',
href: ['https://movie.douban.com/'],
searchApi: 'https://www.douban.com/search?cat=1002&q={kw}',
favicon: 'https://www.douban.com/favicon.ico',
expiration: 21,
infoSelector: [
{
selector: '#interest_sectl > .rating_wrap',
},
],
pageSelector: [
{
selector: 'body',
subSelector: '.tags-body',
keyWord: ['动画', '动漫'],
},
{
selector: '#info',
subSelector: 'span[property="v:genre"]',
keyWord: ['动画', '动漫'],
},
],
getSubjectId(url) {
const m = url.match(/\/(subject)\/(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://movie.douban.com/subject/${id}/`;
},
getSearchResult: checkAnimeSubjectExist,
getScoreInfo() {
const $title = $q('#content h1>span');
const rawName = $title.textContent.trim();
const keywords = $q('meta[name="keywords"]')?.getAttribute?.('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: $q('.ll.rating_num')?.textContent ?? 0,
count: $q('.rating_people > span')?.textContent ?? 0,
rawName,
url: location.href,
};
const $date = $q('span[property="v:initialReleaseDate"]');
if ($date) {
subjectInfo.releaseDate = $date.textContent.replace(/\(.*\)/, '');
}
return subjectInfo;
},
insertScoreInfo(page, info) {
const title = this.getScoreInfo().name;
const opts = {
title,
adjacentSelector: this.infoSelector,
cls: 'friends_rating_wrap clearbox',
};
const wrapDom = getScoreWrapDom(opts.adjacentSelector, opts.cls);
const rowInfo = genScoreRowInfo(opts.title, page, info);
const rowStr = `
<div class="e-userjs-score-compare-row rating_content_wrap clearfix">
<strong class="rating_avg">${rowInfo.score}</strong>
<div class="friends">
<a class="avatar"
${BLANK_LINK}
href="${rowInfo.searchUrl}"
style="cursor:pointer;"
title="点击在${rowInfo.name}搜索">
<img src="${rowInfo.favicon}"/>
</a>
</div>
<a href="${rowInfo.url}"
rel="noopener noreferrer nofollow" class="friends_count" target="_blank">
${rowInfo.count}
</a>
</div>
`;
wrapDom.appendChild(htmlToElement(rowStr));
},
};
async function searchAnimeData(subjectInfo) {
let query = normalizeQuery((subjectInfo.name || '').trim());
const url = `https://myanimelist.net/search/prefix.json?type=anime&keyword=${encodeURIComponent(query)}&v=1`;
console.info('myanimelist search URL: ', url);
const info = await fetchJson(url);
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,
};
await randomSleep(200, 100);
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 (result.score === 'N/A') {
result.score = 0;
}
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;
}
const myanimelistPage = {
name: 'myanimelist',
href: ['https://myanimelist.net/'],
searchApi: 'https://myanimelist.net/anime.php?q={kw}&cat=anime',
favicon: 'https://cdn.myanimelist.net/images/favicon.ico',
infoSelector: [
{
selector: '.anime-detail-header-stats > .stats-block',
},
],
pageSelector: [
{
selector: '.breadcrumb a[href$="myanimelist.net/anime.php"]',
},
],
getSubjectId(url) {
const m = url.match(/\/(anime)\/(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://myanimelist.net/anime/${id}`;
},
getSearchResult: searchAnimeData,
getScoreInfo: function () {
let name = $q('h1-title')?.textContent;
const info = {
name: name,
greyName: name,
score: $q('span[itemprop="ratingValue"]')?.textContent.trim() ?? 0,
count: $q('span[itemprop="ratingCount"]')?.textContent.trim() ?? 0,
url: location.href,
};
$qa('.leftside .spaceit_pad > .dark_text').forEach((el) => {
if (el.innerHTML.includes('Japanese:')) {
info.name = el.nextSibling.textContent.trim();
}
else if (el.innerHTML.includes('Aired:')) {
const aired = el.nextSibling.textContent.trim();
if (aired.includes('to')) {
const startDate = new Date(aired.split('to')[0].trim());
info.releaseDate = formatDate(startDate);
}
}
});
return info;
},
insertScoreInfo: function (page, info) {
const title = this.getScoreInfo().name;
insertScoreCommon(page, info, {
title,
adjacentSelector: this.infoSelector,
cls: 'stats-block',
style: 'height:auto;',
});
},
};
function getMilliseconds(opt) {
if (typeof opt === 'number') {
const oneDay = 24 * 60 * 60 * 1000;
return oneDay * opt;
}
const d = (opt.dd || 0) + 1;
return (+new Date(1970, 1, d, opt.hh || 0, opt.mm || 0, opt.ss || 0, opt.ms || 0) -
+new Date(1970, 1));
}
class KvExpiration {
constructor(engine, prefix, suffix = '-expiration', bucket = '') {
this.engine = engine;
this.prefix = prefix;
this.suffix = suffix;
this.bucket = bucket;
}
genExpirationKey(key) {
return `${this.prefix}${this.bucket}${key}${this.suffix}`;
}
genKey(key) {
return `${this.prefix}${this.bucket}${key}`;
}
flush() {
this.engine.keys().forEach((key) => {
if (key.startsWith(`${this.prefix}${this.bucket}`)) {
this.engine.remove(key);
}
});
}
flushExpired() {
const pre = `${this.prefix}${this.bucket}`;
this.engine.keys().forEach((key) => {
if (key.startsWith(pre) && !key.endsWith(this.suffix)) {
this.flushExpiredItem(key.replace(pre, ''));
}
});
}
flushExpiredItem(key) {
var exprKey = this.genExpirationKey(key);
let time = this.engine.get(exprKey);
if (time) {
if (typeof time !== 'number') {
time = parseInt(time);
}
if (+new Date() >= time) {
this.engine.remove(exprKey);
this.engine.remove(this.genKey(key));
return true;
}
}
return false;
}
set(key, value, opt) {
this.engine.set(this.genKey(key), value);
if (opt) {
const invalidTime = +new Date() + getMilliseconds(opt);
this.engine.set(this.genExpirationKey(key), invalidTime);
}
return true;
}
get(key) {
if (this.flushExpiredItem(key)) {
return;
}
return this.engine.get(this.genKey(key));
}
remove(key) {
this.engine.remove(this.genKey(key));
this.engine.remove(this.genExpirationKey(key));
}
}
class GmEngine {
set(key, value) {
GM_setValue(key, value);
return true;
}
get(key) {
return GM_getValue(key);
}
remove(key) {
GM_deleteValue(key);
}
keys() {
return GM_listValues();
}
}
const USERJS_PREFIX = 'E_SCORE_';
const CURRENT_ID_DICT = 'CURRENT_ID_DICT';
const storage = new KvExpiration(new GmEngine(), USERJS_PREFIX);
function clearInfoStorage() {
storage.flush();
}
function saveInfo(id, info, expiration) {
expiration = expiration || 7;
if (id === '') {
console.error('invalid id: ', info);
return;
}
storage.set(id, info, expiration);
}
function getInfo(id) {
if (id) {
return storage.get(id);
}
}
function getScoreMap(site, id) {
const currentDict = storage.get(CURRENT_ID_DICT) || {};
if (currentDict[site] === id) {
return currentDict;
}
return storage.get('DICT_ID' + id) || {};
}
function setScoreMap(id, map) {
storage.set(CURRENT_ID_DICT, map);
storage.set('DICT_ID' + id, map, 7);
}
const site_origin$2 = 'https://2dfan.org/';
const HEADERS = {
accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
referer: 'https://2dfan.org/',
};
const favicon$2 = 'https://2dfan.org/favicon.ico';
function getSearchItem$3($item) {
const $title = $item.querySelector('h4.media-heading > a');
const href = new URL($title.getAttribute('href'), site_origin$2).href;
const infos = $item.querySelectorAll('.tags > span');
let releaseDate = undefined;
for (let i = 0; i < infos.length; i++) {
const el = infos[i];
if (el.innerHTML.includes('发售日期')) {
const m = el.textContent.match(/\d{4}-\d\d-\d\d/);
if (m) {
releaseDate = m[0];
}
}
}
return {
name: $title.textContent.trim(),
releaseDate,
url: href,
score: 0,
count: 0,
};
}
async function searchGameData$1(subjectInfo) {
let query = normalizeQuery((subjectInfo.name || '').trim());
if (!query) {
console.info('Query string is empty');
return Promise.reject();
}
let searchResult;
const options = {
keys: ['name'],
};
const url = `https://2dfan.org/subjects/search?keyword=${encodeURIComponent(query)}`;
console.info('2dfan search URL: ', url);
const rawText = await fetchText(url, {
headers: HEADERS,
});
const $doc = new DOMParser().parseFromString(rawText, 'text/html');
const items = $doc.querySelectorAll('#subjects > li');
const rawInfoList = Array.prototype.slice
.call(items)
.map(($item) => getSearchItem$3($item));
searchResult = filterResults(rawInfoList, subjectInfo, options, true);
console.info(`Search result of ${query} on 2dfan: `, searchResult);
if (searchResult && searchResult.url) {
randomSleep(200, 50);
const res = await followSearch(searchResult.url);
if (res) {
res.url = searchResult.url;
return res;
}
return searchResult;
}
}
async function followSearch(url) {
const rawText = await fetchText(url, {
headers: {
accept: HEADERS.accept,
referer: url,
},
});
window._parsedEl = new DOMParser().parseFromString(rawText, 'text/html');
const res = getSearchResult$3();
window._parsedEl = undefined;
return res;
}
function getSearchResult$3() {
const $table = $q('.media-body.control-group > .control-group');
const name = $q('.navbar > h3').textContent.trim();
const info = {
name: name,
greyName: name,
score: $q('.rank-info.control-group .score')?.textContent.trim() ?? 0,
count: 0,
url: location.href,
};
const $count = $q('.rank-info.control-group .muted');
if ($count) {
info.count = $count.textContent.trim().replace('人评价', '');
if (info.count.includes('无评分')) {
info.count = '-';
}
}
$table.querySelectorAll('p.tags').forEach((el) => {
if (el.innerHTML.includes('发售日期')) {
const m = el.textContent.match(/\d{4}-\d\d-\d\d/);
if (m) {
info.releaseDate = m[0];
}
}
else if (el.innerHTML.includes('又名:')) {
info.greyName = el.querySelector('.muted').textContent;
}
});
return info;
}
let site_origin$1 = 'https://2dfan.org/';
const twodfanPage = {
name: '2dfan',
href: [site_origin$1],
searchApi: 'https://2dfan.org/subjects/search?keyword={kw}',
favicon: favicon$2,
expiration: 21,
infoSelector: [
{
selector: '.rank-info.control-group',
},
],
pageSelector: [
{
selector: '.navbar > h3',
},
],
getSubjectId(url) {
const m = url.match(/\/(subjects\/)(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `${site_origin$1}/subjects/${id}`;
},
getSearchResult: searchGameData$1,
getScoreInfo: getSearchResult$3,
insertScoreInfo: function (page, info) {
const title = $q('.navbar > h3').textContent.trim();
insertScoreCommon(page, info, {
title,
adjacentSelector: this.infoSelector,
cls: '',
style: '',
});
},
};
const favicon$1 = 'https://vndb.org/favicon.ico';
function normalizeTitle(title) {
return title.replace(/<.+>/, '');
}
function getSearchItem$2($item) {
const $title = $item.querySelector('.tc_title > a');
const href = new URL($title.getAttribute('href'), 'https://vndb.org/').href;
const $rating = $item.querySelector('.tc_rating');
const info = {
name: normalizeTitle($title.getAttribute('title')),
url: href,
count: 0,
score: $rating.firstChild.textContent,
releaseDate: $item.querySelector('.tc_rel').textContent,
};
const $count = $rating.querySelector('.grayedout');
if ($count) {
info.count = $count.textContent.trim().replace(/\(|\)/g, '');
}
return info;
}
async function searchGameData(subjectInfo) {
let query = normalizeQuery((subjectInfo.name || '').trim());
if (!query) {
console.info('Query string is empty');
return Promise.reject();
}
let searchResult;
const options = {
keys: ['name'],
};
const url = `https://vndb.org/v?sq=${encodeURIComponent(query)}`;
console.info('vndb search URL: ', url);
const rawText = await fetchText(url, {
headers: {
referer: 'https://vndb.org/',
},
});
const $doc = new DOMParser().parseFromString(rawText, 'text/html');
const $title = $doc.querySelector('#maincontent > .mainbox > h1');
// 重定向
if ($title) {
window._parsedEl = $doc;
const res = getSearchResult$2();
res.url = $doc.querySelector('head > base').getAttribute('href');
window._parsedEl = undefined;
return res;
}
const items = $doc.querySelectorAll('#maincontent .mainbox table > tbody > tr');
const rawInfoList = Array.prototype.slice
.call(items)
.map(($item) => getSearchItem$2($item));
searchResult = filterResults(rawInfoList, subjectInfo, options, true);
console.info(`Search result of ${query} on vndb: `, searchResult);
if (searchResult && searchResult.url) {
return searchResult;
}
}
function getSearchResult$2() {
let name = $q('tr.title span[lang="ja"]')?.textContent;
if (!name) {
name = $q('tr.title td:nth-of-type(2) > span').textContent;
}
const info = {
name: normalizeTitle(name),
score: $q('.rank-info.control-group .score')?.textContent.trim() ?? 0,
count: 0,
url: location.href,
};
const vote = $q('.votegraph tfoot > tr > td')?.textContent.trim();
if (vote) {
const v = vote.match(/^\d+/);
if (v) {
info.count = v[0];
}
const s = vote.match(/(\d+(\.\d+)?)(?= average)/);
if (s) {
info.score = s[1];
}
}
// get release date
for (const elem of $qa('table.releases tr')) {
if (elem.querySelector('.icon-rtcomplete')) {
info.releaseDate = elem.querySelector('.tc1')?.innerText;
break;
}
}
return info;
}
const vndbPage = {
name: 'vndb',
href: ['https://vndb.org/'],
searchApi: 'https://vndb.org/v?sq={kw}',
favicon: favicon$1,
expiration: 21,
infoSelector: [
{
selector: '.vnimg > label',
},
],
pageSelector: [
{
selector: '.tabselected > a[href^="/v"]',
},
],
getSubjectId(url) {
const m = url.match(/\/(v)(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://vndb.org/subjects/${id}`;
},
getSearchResult: searchGameData,
getScoreInfo: getSearchResult$2,
insertScoreInfo: function (page, info) {
const title = this.getScoreInfo().name;
const opts = {
title,
adjacentSelector: this.infoSelector,
};
const wrapDom = getScoreWrapDom(opts.adjacentSelector);
const rowInfo = genScoreRowInfo(opts.title, page, info);
// refuse blob:<URL>
rowInfo.favicon = page.favicon;
insertScoreRow(wrapDom, rowInfo);
},
};
var ErogamescapeCategory;
(function (ErogamescapeCategory) {
ErogamescapeCategory["game"] = "game";
ErogamescapeCategory["brand"] = "brand";
ErogamescapeCategory["creater"] = "creater";
ErogamescapeCategory["music"] = "music";
ErogamescapeCategory["pov"] = "pov";
ErogamescapeCategory["character"] = "character";
})(ErogamescapeCategory || (ErogamescapeCategory = {}));
// https://erogamescape.org/favicon.ico
const favicon = 'https://www.google.com/s2/favicons?domain=erogamescape.org';
// 'http://erogamescape.org',
const site_origin = 'https://erogamescape.org';
function getSearchItem$1($item) {
const $title = $item.querySelector('td:nth-child(1) > a');
const href = $title.getAttribute('href');
const $name = $item.querySelector('td:nth-child(1)');
// remove tooltip text
$name.querySelector('div.tooltip')?.remove();
const info = {
name: $name.innerText,
url: href,
count: $item.querySelector('td:nth-child(6)')?.textContent ?? 0,
score: $item.querySelector('td:nth-child(4)')?.textContent ?? 0,
releaseDate: $item.querySelector('td:nth-child(3)').textContent,
};
return info;
}
async function searchSubject(subjectInfo, type = ErogamescapeCategory.game, uniqueQueryStr = '') {
let query = normalizeQuery((subjectInfo.name || '').trim());
query = query.replace(/<.+>/, '');
if (uniqueQueryStr) {
query = uniqueQueryStr;
}
if (!query) {
console.info('Query string is empty');
return;
}
const url = `${site_origin}/~ap2/ero/toukei_kaiseki/kensaku.php?category=${type}&word_category=name&word=${encodeURIComponent(query)}&mode=normal`;
console.info('search subject URL: ', url);
const rawText = await fetchText(url);
const $doc = new DOMParser().parseFromString(rawText, 'text/html');
const items = $doc.querySelectorAll('#result table tr:not(:first-child)');
const rawInfoList = [...items].map(($item) => getSearchItem$1($item));
const res = filterResults(rawInfoList, subjectInfo, {
releaseDate: true,
keys: ['name'],
}, true);
console.info(`Search result of ${query} on erogamescape: `, res);
if (res && res.url) {
// 相对路径需要设置一下
res.url = new URL(res.url, url).href;
return res;
}
}
async function searchGameSubject$1(info) {
const result = await searchSubject(info, ErogamescapeCategory.game);
if (result && result.url) {
const rawText = await fetchText(result.url);
window._parsedEl = new DOMParser().parseFromString(rawText, 'text/html');
const res = getSearchResult$1();
res.url = result.url;
window._parsedEl = undefined;
return res;
}
else {
return result;
}
}
function getSearchResult$1() {
const $title = $q('#soft-title > .bold');
const info = {
name: $title.textContent.trim(),
score: $q('#average > td')?.textContent.trim() ?? 0,
count: $q('#count > td')?.textContent.trim() ?? 0,
url: location.href,
};
return info;
}
const erogamescapePage = {
name: 'erogamescape',
href: ['https://erogamescape.org/', 'https://erogamescape.dyndns.org/'],
searchApi: 'https://erogamescape.org/~ap2/ero/toukei_kaiseki/kensaku.php?category=game&word_category=name&word={kw}&mode=normal',
favicon: favicon,
expiration: 21,
infoSelector: [
{
selector: '#basic_information_table',
},
{
selector: '#basic_infomation_table',
},
],
pageSelector: [
{
selector: '#soft-title',
},
],
getSubjectId(url) {
const m = url.match(/(game=)(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://erogamescape.org/~ap2/ero/toukei_kaiseki/game.php?game=${id}`;
},
getSearchResult: searchGameSubject$1,
getScoreInfo: getSearchResult$1,
insertScoreInfo: function (page, info) {
const title = this.getScoreInfo().name;
insertScoreCommon(page, info, {
title,
adjacentSelector: this.infoSelector,
cls: '',
style: '',
});
},
};
function getSearchResult() {
const $title = $q('.body-top_info_title > h2');
const info = {
name: $title.textContent.trim(),
score: 0,
count: '-',
url: location.href,
};
const topTableSelector = {
selector: 'table',
subSelector: 'tr > th',
sibling: true,
};
const $d = findElement({
...topTableSelector,
keyWord: '発売日',
});
if ($d) {
info.releaseDate = dealDate($d.textContent.split('日')[0]);
}
return info;
}
function getSearchItem($item) {
const $title = $item.querySelector('.product-title');
const href = $item.querySelector('a.product-body').getAttribute('href');
const info = {
name: $title.textContent,
url: href,
count: '-',
score: 0,
};
const $d = $item.querySelector('.product-date > p');
if ($d) {
info.releaseDate = dealDate($d.textContent.split('日')[0]);
}
return info;
}
async function searchGameSubject(info) {
const url = `https://moepedia.net/search/result/?s=${info.name}&t=on`;
const rawText = await fetchText(url);
const $doc = new DOMParser().parseFromString(rawText, 'text/html');
const items = $doc.querySelectorAll('.sw-Products .sw-Products_Item');
const rawInfoList = [...items].map(($item) => getSearchItem($item));
const res = filterResults(rawInfoList, info, {
keys: ['name'],
}, true);
console.info(`Search result of ${info.name} on moepedia: `, res);
if (res && res.url) {
// 相对路径需要设置一下
res.url = new URL(res.url, url).href;
return res;
}
}
const moepediaPage = {
name: 'moepedia',
href: ['https://moepedia.net/'],
searchApi: 'https://moepedia.net/search/result/?s={kw}&t=on',
favicon: 'https://moepedia.net/wp/wp-content/themes/moepedia/assets/images/common/common/favicon.ico',
expiration: 21,
infoSelector: [
{
selector: '.body-top_image_wrapper',
},
],
pageSelector: [
{
selector: '.body-top_info_title h2',
},
],
getSubjectId(url) {
const m = url.match(/(game\/)(\d+)/);
if (m) {
return `${this.name}_${m[2]}`;
}
return '';
},
genSubjectUrl(id) {
return `https://moepedia.net/game/${id}/`;
},
insertScoreInfo: function (page, info) {
const title = $q('.body-top_info_title > h2').textContent.trim();
insertScoreCommon(page, info, {
title,
adjacentSelector: this.infoSelector,
});
},
getSearchResult: searchGameSubject,
getScoreInfo: getSearchResult,
};
const animePages = [
bangumiAnimePage,
doubanAnimePage,
myanimelistPage,
anidbPage,
];
const gamePages = [
bangumiGamePage,
twodfanPage,
vndbPage,
erogamescapePage,
moepediaPage,
];
const BGM_UA = 'e_user_bgm_ua';
var g_hide_game_score_flag = GM_getValue('e_user_hide_game_score') || '';
if (GM_registerMenuCommand) {
GM_registerMenuCommand('清除缓存信息', () => {
clearInfoStorage();
alert('已清除缓存');
}, 'c');
GM_registerMenuCommand('设置Bangumi UA', () => {
var p = prompt('设置 Bangumi UA', '');
GM_setValue(BGM_UA, p);
});
GM_registerMenuCommand('显示游戏评分开关', () => {
g_hide_game_score_flag = prompt('设置不为空时隐藏游戏评分', g_hide_game_score_flag);
GM_setValue('e_user_hide_game_score', g_hide_game_score_flag);
});
}
function getPageIdxByHost(pages, host) {
const idx = pages.findIndex((obj) => {
if (Array.isArray(obj.href)) {
return obj.href.some((href) => href.includes(host));
}
else {
return obj.href.includes(host);
}
});
return idx;
}
async function insertScoreRows(curPage, pages, curInfo, map, tasks) {
for (const page of pages) {
if (page.name === curPage.name || page.type === 'info') {
continue;
}
let searchResult = getInfo(map[page.name]);
if (!searchResult) {
try {
searchResult = await page.getSearchResult(curInfo);
}
catch (error) {
console.error(error);
}
tasks.push({
page,
info: searchResult || { name: curInfo.name, url: '' },
});
}
curPage.insertScoreInfo(page, searchResult);
}
}
async function refreshScore(curPage, pages, force = false) {
const saveTask = [];
const curInfo = curPage.getScoreInfo();
saveTask.push({
page: curPage,
info: curInfo,
});
const subjectId = curPage.getSubjectId(curInfo.url);
let map = { [curPage.name]: subjectId };
if (!force) {
const scoreMap = getScoreMap(curPage.name, subjectId);
map = { ...scoreMap, [curPage.name]: subjectId };
document
.querySelectorAll('.e-userjs-score-compare')
.forEach((el) => el.remove());
}
await insertScoreRows(curPage, pages, curInfo, map, saveTask);
saveTask.forEach((t) => {
const { page, info } = t;
if (info && info.url) {
const key = page.getSubjectId(info.url);
saveInfo(key, info, page.expiration);
map[page.name] = key;
}
else {
const key = `${page.name}_${info.name}`;
saveInfo(key, { url: '', name: '' }, page.expiration);
map[page.name] = key;
}
});
setScoreMap(subjectId, map);
}
function isValidPage(curPage) {
const $page = findElement(curPage.pageSelector);
if (!$page)
return false;
const $info = findElement(curPage.infoSelector);
if (!$info)
return false;
return true;
}
function insertControlDOM(curPage, pages) {
if (curPage.controlSelector) {
const $ctrl = findElement(curPage.controlSelector);
curPage?.insertControlDOM?.($ctrl, {
clear: clearInfoStorage,
refresh: () => refreshScore(curPage, pages, true),
});
}
}
function initSiteConfig() {
const ua = GM_getValue(BGM_UA);
if (ua) {
addSiteOption('bgm.tv', {
headers: {
'user-agent': ua,
},
});
addSiteOption('bangumi.tv', {
headers: {
'user-agent': ua,
},
});
addSiteOption('chii.in', {
headers: {
'user-agent': ua,
},
});
}
}
async function initPage(pages) {
const idx = getPageIdxByHost(pages, location.host);
if (idx === -1)
return;
const curPage = pages[idx];
if (!isValidPage(curPage))
return;
insertControlDOM(curPage, pages);
initSiteConfig();
refreshScore(curPage, pages, false);
}
initPage(animePages);
!g_hide_game_score_flag && initPage(gamePages);
})();