Sleazy Fork is available in English.

E站控制画廊已收藏显隐和黑名单

漫画资源e站,增加功能:1、控制已收藏画廊显隐 2、快速添加收藏功能 3、黑名单屏蔽重复、缺页、低质量画廊 4、详情页生成文件名

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==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;
        }
    }

    /*
    * 11
    * */
    async function getFavorites(disableCache = false) { // API:获取收藏配置列表
        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)
            }
        }
    }

    async function updateFavorites(type, gid, t) { // API:更新收藏
        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();
        }
    }
})();