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;
        }
    }

    // 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();
        }
    }
})();