// ==UserScript==
// @name E站控制画廊已收藏显隐和黑名单
// @namespace http://tampermonkey.net/
// @version 1.5.3
// @license GPL-3.0
// @description 漫画资源e站,增加功能:1、控制已收藏画廊显隐 2、快速添加收藏功能 3、黑名单屏蔽重复、缺页、低质量画廊 4、详情页生成文件名
// @author ShineByPupil
// @match *://exhentai.org/*
// @icon 
// @grant none
// ==/UserScript==
(async function () {
'use strict';
// 【文件名去除规则】
const parenthesesRule = '\\([^(]*(' +
['Vol', 'COMIC', '成年コミック', 'C\\d+', 'よろず', 'FF\\d+', '\\d{4}年\\d{1,2}月', 'Chinese', '机翻', 'コミック', '汉化组', '中文'].join('|') +
')[^(]*\\)'; // 圆括号
const squareBracketsRule = '\\[[^[]*(' +
['汉化', '漢化', '翻訳', 'Chinese', 'chinese', 'CHINESE', '無修正', 'DL版', '中国語', '中文', '渣翻', '机翻', '機翻', '重嵌'].join('|') +
')[^[]*\\]'; // 方括号
let isFilter = localStorage.getItem('isFilter') === 'true';
let alwaysFilter = localStorage.getItem('alwaysFilter') || '';
let favoriteList = await getFavorites(); // 收藏设置
const utils = {
messageBox: null,
/**
* 在屏幕上显示指定时间长度的消息。
*
* @param {string} message - 要显示的消息。
* @param {number} [duration=2500] - 消息应显示的毫秒数。默认为2500毫秒。
* @return {void} 此函数不返回值。
*/
showMessage: function (message, duration = 2500) {
if (!this.messageBox) {
// 创建一个 Shadow Root
this.createShadowMessageBox();
}
this.messageBox.textContent = message;
this.messageBox.style.display = 'block'; // 显示消息
// 设置一定时间后自动隐藏消息
setTimeout(() => {
this.messageBox.style.display = 'none';
}, duration);
},
/**
* 从提供的模板字符串创建一个新的 DOM 节点。
*
* @param {string} template - 要创建节点的 HTML 模板字符串。
* @return {Node} 新创建的 DOM 节点。
*/
createNode: function (template) {
const div = document.createElement('div');
div.innerHTML = template.trim();
return div.firstChild;
},
/**
* 创建一个带有 Shadow DOM 的消息框。
*
* @return {void}
*/
createShadowMessageBox: function () {
const container = document.createElement('div');
const shadowRoot = container.attachShadow({mode: 'open'});
// 创建消息框的样式,使用明亮的配色
const style = document.createElement('style');
style.textContent = `
#messageBox {
position: fixed;
top: 20px;
left: 50%;
transform: translateX(-50%);
background-color: #ffffff; /* 明亮的背景色 */
color: #000000; /* 深色文本 */
padding: 10px 20px;
border-radius: 5px;
z-index: 1000;
display: none; /* 初始隐藏 */
transition: opacity 0.3s ease;
opacity: 1;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); /* 添加阴影效果 */
}
`;
// 创建消息框节点
const messageBox = document.createElement('div');
messageBox.id = 'messageBox';
// 将样式和消息框添加到 Shadow DOM
shadowRoot.appendChild(style);
shadowRoot.appendChild(messageBox);
// 将包含 Shadow DOM 的容器添加到文档中
document.body.appendChild(container);
// 保存对消息框的引用
this.messageBox = messageBox;
},
};
// 根据 URL 执行不同的代码
if (['/', '/watched', '/popular'].includes(window.location.pathname)) {
// 主页
filterFavorites();
setFavorites();
} else if (window.location.pathname === '/favorites.php') {
setFavorites();
} else if (/^\/g\/\d+\/[a-z0-9]+\/$/.test(window.location.pathname)) {
// 详情页
formatFileName();
setFavorites();
} else if (/^\/tag\/.*$/.test(window.location.pathname)) {
filterFavorites();
setFavorites();
}
// 右下角按钮组:收藏显隐、总是过滤、过滤全部
function filterFavorites() {
const div = document.createElement('div');
const refreshBtn = document.createElement('button');
refreshBtn.innerText = '↻刷新';
refreshBtn.addEventListener('click', function () {
location.reload();
});
const toggleBtn = document.createElement('button');
toggleBtn.innerText = isFilter ? '点击显示' : '点击隐藏';
toggleBtn.addEventListener('click', function () {
isFilter = !isFilter;
localStorage.setItem('isFilter', isFilter);
toggleBtn.innerText = isFilter ? '点击显示' : '点击隐藏';
handleFilter();
});
const filterBtn = document.createElement('button');
filterBtn.innerText = '总是过滤';
filterBtn.addEventListener('click', function () {
const userInput = prompt("请输入总是过滤的收藏名:", alwaysFilter);
if (userInput !== null) {
alwaysFilter = userInput;
localStorage.setItem('alwaysFilter', alwaysFilter);
handleFilter();
}
});
const filterAllBtn = document.createElement('button');
filterAllBtn.innerText = '过滤全部';
filterAllBtn.addEventListener('click', async function () {
if (!alwaysFilter) {
return utils.showMessage('请先设置总是过滤');
}
const index = favoriteList.indexOf(alwaysFilter);
if (index !== -1) {
const list = Array.from(
document
.querySelector('.itg')
.querySelectorAll('div[id^="posted_"]')
)
.filter(n => n.title === '')
.map(n => {
const matches = n.onclick.toString().match(/gid=(\d+)&t=([a-z0-9]+)/);
const [,gid,t] = matches;
return { gid, t };
})
// 处理并发请求
const set = new Set();
const enqueue = async function (promise) {
if (set.size > 5) {
await Promise.race(set);
}
const p = promise().finally(() => set.delete(p));
set.add(p);
return p;
}
await Promise.all(
list.map(({ gid, t }) => {
return enqueue(() => updateFavorites(index, gid, t));
})
)
utils.showMessage('过滤全部成功');
}
});
const divStyle = {
position: 'fixed', // 绝对定位
right: '10px', // 距离左边10像素
bottom: '10px', // 距离顶部10像素
zIndex: '1000', // 确保按钮在其他元素之上
display: 'flex',
flexDirection: 'column',
}
const btnStyle = {
backgroundColor: '#007BFF', // 按钮背景颜色
color: '#FFFFFF', // 按钮文字颜色
border: 'none',
borderRadius: '5px',
cursor: 'pointer',
padding: '4px 10px',
marginBottom: '10px',
};
for (let key in divStyle) {
div.style[key] = divStyle[key];
}
for (let key in btnStyle) {
refreshBtn.style[key] = btnStyle[key];
toggleBtn.style[key] = btnStyle[key];
filterBtn.style[key] = btnStyle[key];
filterAllBtn.style[key] = btnStyle[key];
}
// 添加按钮到页面
div.appendChild(refreshBtn);
div.appendChild(toggleBtn);
div.appendChild(filterBtn);
div.appendChild(filterAllBtn);
document.body.appendChild(div);
handleFilter();
window.addEventListener('storage', function (event) {
if (event.key === 'isFilter') {
isFilter = event.newValue === 'true';
toggleBtn.innerText = isFilter ? '点击显示' : '点击隐藏';
handleFilter();
}
})
const observer = new MutationObserver(mutationsList => {
const domSet = new WeakSet();
for (let mutation of mutationsList) {
if (/^posted_\d+$/.test(mutation.target.id) && !domSet.has(mutation.target)) {
domSet.add(mutation.target);
handleFilter();
}
}
})
// 开始观察目标节点
const targetNode = document.querySelector('.itg');
if (targetNode) {
observer.observe(targetNode, {
attributes: true,
subtree: true
});
}
}
// 开始过滤
function handleFilter() {
const list = document.querySelector('table.itg')
? document.querySelectorAll('table.itg tr')
: document.querySelectorAll('.itg.gld .gl1t');
[...list].forEach(n => {
const find = n.querySelector('[id^="posted_"]')
if (find && find.title !== '') {
if (alwaysFilter === find.title) {
n.style.display = 'none';
} else {
n.style.display = isFilter ? 'none' : '';
}
}
});
}
// 生成文件名成
async function formatFileName() {
const rule = new RegExp(`${parenthesesRule}|${squareBracketsRule}`, 'g');
let title = document.querySelector('#gj').innerText || document.querySelector('#gn').innerText;
title = title
.replace(/[[]()]/g, match => {
if (match === '[') {
return '[';
} else if (match === ']') {
return ']';
} else if (match === '(') {
return '('
} else if (match === ')') {
return ')'
}
})
.replace(/[\/\\:*?"<>|]/g, ' ')
.replace(rule, '')
.replace(/\s+/g, ' ')
.trim();
const tagConfigMap = await fetch('https://exhentai.org/mytags')
.then(r => r.text())
.then(r => {
const parser = new DOMParser();
return parser.parseFromString(r, 'text/html');
})
.then(doc => {
let map = new Map();
// 没有关注和隐藏的标签(也希望显示在文件名)
map.set('other:extraneous ads', { weight: -10 });
map.set('other:incomplete', { weight: -11 });
[...doc.querySelectorAll('#usertags_outer>div')].forEach(n => {
if (n.querySelector('.gt') && n.querySelector('input[id^=tagwatch]')?.checked) {
map.set(
n.querySelector('.gt').title,
{ weight: parseInt(n.querySelector('[id^=tagweight]').value, 10) }
);
}
});
return map;
});
const tagDom = Array.from(document.querySelectorAll('#taglist a'));
const formatId = id => id.slice(3).replace(/_/g, ' ');
let tags = [...new Set(
tagDom.filter(n => tagConfigMap.has(formatId(n.id)))
.sort((n, m) => tagConfigMap.get(formatId(m.id)).weight - tagConfigMap.get(formatId(n.id)).weight)
.map(n => `[${n.innerText}]`)
)].join('');
const input = document.createElement('input');
input.style.width = '100%';
input.style.textAlign = 'center';
input.value = (title + ' ' + tags).trim();
const button = document.createElement('button');
button.onclick = function () {
navigator.clipboard.writeText(input.value);
}
button.innerText = '复制';
document.querySelector('#gd2').appendChild(input);
document.querySelector('#gd2').appendChild(button);
}
// 快速收藏按钮组(鼠标悬停画廊封面)
async function setFavorites() {
const ulStyle = {
margin: '0',
padding: '0',
display: 'none',
flexDirection: 'column',
position: 'absolute',
zIndex: '1000',
}
const liStyle = {
listStyleType: 'none',
backgroundColor: '#007BFF',
color: '#FFFFFF',
cursor: 'pointer',
padding: '2px 4px',
margin: '2px 0',
borderRadius: '5px',
textAlign: 'center',
}
let gid = null;
let t = null;
const ulNode = utils.createNode(`<ul></ul>`);
const favdelLi = utils.createNode(`<li>取消收藏</li>`);
const refreshLi = utils.createNode(`<li>↻刷新</li>`);
const favoriteLi = await createFavoriteLi();
for (let key in ulStyle) {
ulNode.style[key] = ulStyle[key];
}
for (let key in liStyle) {
favdelLi.style[key] = liStyle[key];
refreshLi.style[key] = liStyle[key];
}
ulNode.addEventListener('mouseover', function () {
ulNode.style.display = 'flex';
})
ulNode.addEventListener('mouseout', function () {
ulNode.style.display = 'none';
});
favdelLi.addEventListener('click', function () {
if (gid && t) {
updateFavorites('favdel', gid, t);
}
});
refreshLi.addEventListener('click', async function () {
ulNode.style.display = 'none';
const favoriteLi = await createFavoriteLi(true);
while (ulNode.children.length > 2) {
ulNode.removeChild(ulNode.firstChild);
}
ulNode.insertBefore(favoriteLi, ulNode.firstChild);
ulNode.style.display = 'flex';
});
ulNode.appendChild(favoriteLi);
ulNode.appendChild(favdelLi);
ulNode.appendChild(refreshLi);
document.body.appendChild(ulNode);
// 搜索主页
const itgNode = document.querySelector('.itg');
if (itgNode) {
itgNode.addEventListener('mouseover', function (event) {
const {target} = event;
if (target.tagName === 'IMG' && target.alt !== 'T') {
const href = target.parentNode.href;
const groups = href.split('/');
gid = groups[groups.length - 3];
t = groups[groups.length - 2];
const rect = target.parentNode.parentNode.getBoundingClientRect();
ulNode.style.display = 'flex'
ulNode.style.left = `${rect.left + 10 + window.scrollX}px`;
ulNode.style.top = `${rect.top + 10 + window.scrollY}px`; // 在 li 下方显示
}
});
itgNode.addEventListener('mouseout', function (e) {
const {target} = e;
if (target.tagName === 'IMG' && !ulNode.matches(':hover')) {
gid = null;
t = null;
ulNode.style.display = 'none';
}
});
}
// 详情页
const cover = document.querySelector('#gd1 div');
if (cover) {
const groups = location.pathname.split('/');
gid = groups[groups.length - 3];
t = groups[groups.length - 2];
cover.addEventListener('mouseover', function (event) {
const rect = event.target.getBoundingClientRect();
ulNode.style.display = 'flex'
ulNode.style.left = `${rect.left + 10 + window.scrollX}px`;
ulNode.style.top = `${rect.top + 10 + window.scrollY}px`; // 在 li 下方显示
});
cover.addEventListener('mouseout', function (event) {
if (!ulNode.matches(':hover')) {
ulNode.style.display = 'none';
}
});
}
async function createFavoriteLi(disableCache = false) {
const fragment = document.createDocumentFragment();
if (disableCache) {
favoriteList = await getFavorites(true);
}
favoriteList.forEach((n, index) => {
if (!/^Favorites \d$/.test(n)) {
const liNode = utils.createNode(`<li>${n}</li>`);
liNode.addEventListener('click', async function () {
if (gid && t) {
await updateFavorites(index, gid, t);
handleFilter();
utils.showMessage('收藏成功');
}
})
for (let key in liStyle) {
liNode.style[key] = liStyle[key];
}
fragment.appendChild(liNode);
}
});
return fragment;
}
}
// API:获取收藏配置列表
async function getFavorites(disableCache = false) {
let favoriteList = localStorage.getItem('favoriteList');
if (favoriteList && disableCache === false) {
return JSON.parse(favoriteList);
} else {
const response = await fetch('https://exhentai.org/uconfig.php');
const domStr = await response.text();
const parser = new DOMParser();
const doc = parser.parseFromString(domStr, 'text/html');
const list = Array.from(
doc.querySelectorAll('#favsel input')
).map(n => n.value);
if (list.length) {
localStorage.setItem('favoriteList', JSON.stringify(list));
return list;
} else {
throw new Error(doc.body.innerText)
}
}
}
// API:更新收藏
async function updateFavorites(type, gid, t) {
const formData = new FormData();
formData.append('favcat', type);
formData.append('favnote', '');
formData.append('update', '1');
const response = await fetch(
`https://exhentai.org/gallerypopups.php?gid=${gid}&t=${t}&act=addfav`,
{method: 'POST', body: formData});
const domStr = await response.text();
const parser = new DOMParser();
const doc = parser.parseFromString(domStr, 'text/html');
const script = Array.from(doc.querySelectorAll('script'))
.find(n => n.textContent.includes('window.close()'));
if (script) {
let codeStr = script.textContent
codeStr = codeStr.replace(/window.opener.document/g, 'window.document');
codeStr = codeStr.replace(/window.close\(\);/g, '');
const dynamicFunction = new Function(codeStr);
dynamicFunction();
}
}
})();