吾爱破解论坛增强 - 自动签到、翻页

自动签到、自动无缝翻页、屏蔽导读悬赏贴(最新发表页)

// ==UserScript==
// @name         吾爱破解论坛增强 - 自动签到、翻页
// @version      1.3.7
// @author       X.I.U
// @description  自动签到、自动无缝翻页、屏蔽导读悬赏贴(最新发表页)
// @match        *://www.52pojie.cn/*
// @icon         
// @grant        GM_xmlhttpRequest
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_openInTab
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_notification
// @license      GPL-3.0 License
// @run-at       document-end
// @namespace    https://greasyfork.org/scripts/412680
// @supportURL   https://github.com/XIU2/UserScript
// @homepageURL  https://github.com/XIU2/UserScript
// ==/UserScript==

(function() {
    'use strict';
    var menu_ALL = [
        ['menu_autoClockIn', '自动签到', '自动签到', true],
        ['menu_pageLoading', '自动无缝翻页', '自动无缝翻页', true],
        ['menu_thread_pageLoading', '帖子内自动无缝翻页', '帖子内自动无缝翻页', true],
        ['menu_delateReward', '屏蔽导读悬赏贴(最新发表)', '屏蔽导读悬赏贴', true]
    ], menu_ID = [];
    for (let i=0;i<menu_ALL.length;i++) { // 如果读取到的值为 null 就写入默认值
        if (GM_getValue(menu_ALL[i][0]) == null){GM_setValue(menu_ALL[i][0], menu_ALL[i][3])};
    }
    registerMenuCommand();

    // 注册脚本菜单
    function registerMenuCommand() {
        if (menu_ID.length > menu_ALL.length) { // 如果菜单ID数组多于菜单数组,说明不是首次添加菜单,需要卸载所有脚本菜单
            for (let i=0;i<menu_ID.length;i++) {
                GM_unregisterMenuCommand(menu_ID[i]);
            }
        }
        for (let i=0;i<menu_ALL.length;i++) { // 循环注册脚本菜单
            menu_ALL[i][3] = GM_getValue(menu_ALL[i][0]);
            menu_ID[i] = GM_registerMenuCommand(`${menu_ALL[i][3]?'✅':'❌'} ${menu_ALL[i][1]}`, function(){menu_switch(`${menu_ALL[i][3]}`,`${menu_ALL[i][0]}`,`${menu_ALL[i][2]}`)});
        }
        menu_ID[menu_ID.length] = GM_registerMenuCommand('💬 反馈 & 建议', function () {window.GM_openInTab('https://github.com/XIU2/UserScript#xiu2userscript', {active: true,insert: true,setParent: true});window.GM_openInTab('https://greasyfork.org/zh-CN/scripts/412680/feedback', {active: true,insert: true,setParent: true});});
    }

    // 菜单开关
    function menu_switch(menu_status, Name, Tips) {
        if (menu_status == 'true') {
            GM_setValue(`${Name}`, false);
            GM_notification({text: `已关闭 [${Tips}] 功能\n(点击刷新网页后生效)`, title: '吾爱破解论坛增强', timeout: 3000, onclick: function(){location.reload();}});
        } else {
            GM_setValue(`${Name}`, true);
            GM_notification({text: `已开启 [${Tips}] 功能\n(点击刷新网页后生效)`, title: '吾爱破解论坛增强', timeout: 3000, onclick: function(){location.reload();}});
        }
        registerMenuCommand(); // 重新注册脚本菜单
    };

    // 返回菜单值
    function menu_value(menuName) {
        for (let menu of menu_ALL) {
            if (menu[0] == menuName) {
                return menu[3]
            }
        }
    }

    var ShowPager;
    showPager();
    // 默认 ID 为 0
    var curSite = {SiteTypeID: 0};

    // 自动翻页规则
    // type:1 = 脚本实现自动无缝翻页,2 = 网站自带了自动无缝翻页功能,只需要点击下一页按钮即可,这时 nextText 为按钮文本,避免一瞬间加载太多次下一页
    // HT_insert:1 = 插入元素前面;2 = 插入元素中的最后一个子元素后面
    // scrollDelta:数值越大,滚动条触发点越靠上(越早开始翻页)
    let DBSite = {
        forum: {
            SiteTypeID: 1,
            pager: {
                type: 2,
                nextLink: '#autopbn',
                nextText: '下一页 »',
                scrollDelta: 766
            }
        },
        thread: {
            SiteTypeID: 2,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;div#postlist > div[id^="post_"]',
                HT_insert: ['css;div#postlist', 2],
                replaceE: 'css;#ct > .pgs',
                scrollDelta: 766
            }
        },
        search: {
            SiteTypeID: 3,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;div#threadlist > ul',
                HT_insert: ['css;div#threadlist', 2],
                replaceE: 'css;div.pg',
                scrollDelta: 766
            }
        },
        guide: {
            SiteTypeID: 4,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;div#threadlist div.bm_c table > tbody',
                HT_insert: ['css;div#threadlist div.bm_c table', 2],
                replaceE: 'css;div.pg',
                scrollDelta: 766
            }
        },
        youspace: {
            SiteTypeID: 5,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;tbody > tr:not(.th)',
                HT_insert: ['css;tbody', 2],
                replaceE: 'css;div.pg',
                scrollDelta: 1000
            }
        },
        collection: {
            SiteTypeID: 6,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;div#ct div.bm_c table > tbody',
                HT_insert: ['css;div#ct div.bm_c table', 2],
                replaceE: 'css;div.pg',
                scrollDelta: 899
            }
        },
        favorite: {
            SiteTypeID: 7,
            pager: {
                type: 1,
                nextLink: '//a[@class="nxt"][@href]',
                pageElement: 'css;ul#favorite_ul > li',
                HT_insert: ['css;ul#favorite_ul', 2],
                replaceE: 'css;div.pg',
                scrollDelta: 899
            }
        }
    };

    // URL 匹配正则表达式
    let patt_thread = /\/thread-\d+-\d+\-\d+.html/,
        patt_forum = /\/forum-\d+-\d+\.html/

    // URL 判断
    if (patt_thread.test(location.pathname) || location.search.indexOf('mod=viewthread') > -1) {
        if (menu_value('menu_thread_pageLoading')) {
            curSite = DBSite.thread; //      帖子内
            hidePgbtn(); //                  隐藏帖子内的 [下一页] 按钮
        }
    } else if (patt_forum.test(location.pathname) || location.search.indexOf('mod=forumdisplay') > -1) {
        curSite = DBSite.forum; //           各板块帖子列表
    } else if (location.search.indexOf('mod=guide') > -1) {
        curSite = DBSite.guide; //           导读帖子列表
        delateReward(); //                   屏蔽导读悬赏贴(最新发表)
    } else if (location.search.indexOf('mod=collection') > -1) {
        curSite = DBSite.collection; //      淘贴列表
    } else if (location.search.indexOf('do=favorite') > -1) {
        curSite = DBSite.favorite; //        收藏列表
    } else if (location.pathname === '/search.php') {
        curSite = DBSite.search; //          搜索结果列表
    } else if(location.search.indexOf('mod=space') > -1 && location.search.indexOf('&view=me') > -1) { // 别人的主题/回复
        curSite = DBSite.youspace;
    }
    curSite.pageUrl = ''; // 下一页URL

    qianDao(); // 自动签到
    pageLoading(); // 自动翻页


    // 自动签到(后台)
    function qianDao() {
        if (!menu_value('menu_autoClockIn')) return
        if (location.pathname === '/home.php' && location.search.indexOf('mod=task') > -1) {return;}
        let qiandao = document.querySelector('#um a[href^="home.php?mod=task&do=apply&id=2"]');
        if (qiandao) {
            let iframe = document.createElement('iframe'); // XHR 方式无法签到,改用 iframe 框架打开签到网页
            document.lastElementChild.appendChild(iframe);
            iframe.style = 'display: none;';
            iframe.src = qiandao.href;
            qiandao.querySelector('.qq_bind').src = 'https://www.52pojie.cn/static/image/common/wbs.png'; // 修改 [打卡签到] 图标为 [签到完毕] 图标
            qiandao.href = 'javascript:void(0);'
        }
    }


    //屏蔽悬赏贴(导读-最新发表)
    function delateReward() {
        if (!menu_value('menu_delateReward')) return
        if (location.search.indexOf('mod=guide&view=newthread') > -1) {
            let tbody = document.querySelectorAll('#threadlist tbody[id^="normalthread"]');
            Array.from(tbody).forEach(function (_this) {
                if (_this.querySelector('img[alt="悬赏"]')) {
                    _this.remove();
                }
            })
        }

        if (document.body.scrollHeight < window.innerHeight) {
            // 如果屏蔽悬赏贴后,剩余帖子列表太少会没有滚动条,无法滚动页面触发自动翻页事件,需要手动触发
            ShowPager.loadMorePage();
        }
    }


    // 隐藏帖子内的 [下一页] 按钮
    function hidePgbtn() {
        document.lastChild.appendChild(document.createElement('style')).textContent = '.pgbtn {display: none;}';
    }


    // 自动翻页
    function pageLoading() {
        if (!menu_value('menu_pageLoading')) return
        if (curSite.SiteTypeID > 0) {
            windowScroll(function (direction, e) {
                if (direction === "down") { // 下滑才准备翻页
                    let scrollTop = document.documentElement.scrollTop || window.pageYOffset || document.body.scrollTop;
                    let scrollDelta = curSite.pager.scrollDelta;
                    if (document.documentElement.scrollHeight <= document.documentElement.clientHeight + scrollTop + scrollDelta) {
                        if (curSite.pager.type === 1) {
                            ShowPager.loadMorePage();
                        } else {
                            let autopbn = document.querySelector(curSite.pager.nextLink);
                            if (autopbn && autopbn.textContent == curSite.pager.nextText) { // 如果正在加载,就不再点击
                                autopbn.click();
                            }
                        }
                    }
                }
            });
        }
    }


    // 滚动条事件
    function windowScroll(fn1) {
        var beforeScrollTop = document.documentElement.scrollTop,
            fn = fn1 || function () {};
        setTimeout(function () { // 延时执行,避免刚载入到页面就触发翻页事件
            window.addEventListener('scroll', function (e) {
                var afterScrollTop = document.documentElement.scrollTop,
                    delta = afterScrollTop - beforeScrollTop;
                if (delta == 0) return false;
                fn(delta > 0 ? 'down' : 'up', e);
                beforeScrollTop = afterScrollTop;
            }, false);
        }, 1000)
    }


    // 修改自 https://greasyfork.org/scripts/14178 , https://github.com/machsix/Super-preloader
    function showPager() {
        ShowPager = {
            getFullHref: function (e) {
                if (e != null && e.nodeType === 1 && e.href && e.href.slice(0,4) === 'http') return e.href;
                return '';
            },
            createDocumentByString: function (e) {
                if (e) {
                    if ('HTML' !== document.documentElement.nodeName) return (new DOMParser).parseFromString(e, 'application/xhtml+xml');
                    var t;
                    try { t = (new DOMParser).parseFromString(e, 'text/html');} catch (e) {}
                    if (t) return t;
                    if (document.implementation.createHTMLDocument) {
                        t = document.implementation.createHTMLDocument('ADocument');
                    } else {
                        try {((t = document.cloneNode(!1)).appendChild(t.importNode(document.documentElement, !1)), t.documentElement.appendChild(t.createElement('head')), t.documentElement.appendChild(t.createElement('body')));} catch (e) {}
                    }
                    if (t) {
                        var r = document.createRange(),
                            n = r.createContextualFragment(e);
                        r.selectNodeContents(document.body);
                        t.body.appendChild(n);
                        for (var a, o = { TITLE: !0, META: !0, LINK: !0, STYLE: !0, BASE: !0}, i = t.body, s = i.childNodes, c = s.length - 1; c >= 0; c--) o[(a = s[c]).nodeName] && i.removeChild(a);
                        return t;
                    }
                } else console.error('没有找到要转成 DOM 的字符串');
            },
            loadMorePage: function () {
                if (curSite.pager) {
                    let curPageEle = getElementByXpath(curSite.pager.nextLink);
                    var url = this.getFullHref(curPageEle);
                    //console.log(`${url} ${curPageEle} ${curSite.pageUrl}`);
                    if(url === '') return;
                    if(curSite.pageUrl === url) return;// 不会重复加载相同的页面
                    curSite.pageUrl = url;
                    // 读取下一页的数据
                    curSite.pager.startFilter && curSite.pager.startFilter();
                    GM_xmlhttpRequest({
                        url: url,
                        method: "GET",
                        overrideMimeType: 'text/html; charset=' + (document.characterSet||document.charset||document.inputEncoding),
                        timeout: 5000,
                        onload: function (response) {
                            try {
                                var newBody = ShowPager.createDocumentByString(response.responseText);
                                let pageElems = getAllElements(curSite.pager.pageElement, newBody, newBody);
                                let toElement = getAllElements(curSite.pager.HT_insert[0])[0];
                                if (pageElems.length >= 0) {
                                    let addTo = "beforeend";
                                    if (curSite.pager.HT_insert[1] == 1) addTo = "beforebegin";
                                    // 插入新页面元素
                                    pageElems.forEach(function (one) {
                                        toElement.insertAdjacentElement(addTo, one);
                                    });
                                    //删除悬赏贴
                                    delateReward();
                                    // 替换待替换元素
                                    try {
                                        let oriE = getAllElements(curSite.pager.replaceE);
                                        let repE = getAllElements(curSite.pager.replaceE, newBody, newBody);
                                        if (oriE.length === repE.length) {
                                            for (var i = 0; i < oriE.length; i++) {
                                                oriE[i].outerHTML = repE[i].outerHTML;
                                            }
                                        }
                                    } catch (e) {
                                        console.log(e);
                                    }
                                }
                            } catch (e) {
                                console.log(e);
                            }
                        }
                    });
                }
            },
        };
    }
    function getElementByCSS(css, contextNode = document) {
        return contextNode.querySelector(css);
    }
    function getAllElementsByCSS(css, contextNode = document) {
        return [].slice.call(contextNode.querySelectorAll(css));
    }
    function getElementByXpath(xpath, contextNode, doc = document) {
        contextNode = contextNode || doc;
        try {
            const result = doc.evaluate(xpath, contextNode, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
            // 应该总是返回一个元素节点
            return result.singleNodeValue && result.singleNodeValue.nodeType === 1 && result.singleNodeValue;
        } catch (err) {
            throw new Error(`Invalid xpath: ${xpath}`);
        }
    }
    function getAllElementsByXpath(xpath, contextNode, doc = document) {
        contextNode = contextNode || doc;
        const result = [];
        try {
            const query = doc.evaluate(xpath, contextNode, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
            for (let i = 0; i < query.snapshotLength; i++) {
                const node = query.snapshotItem(i);
                // 如果是 Element 节点
                if (node.nodeType === 1) result.push(node);
            }
        } catch (err) {
            throw new Error(`无效 Xpath: ${xpath}`);
        }
        return result;
    }
    function getAllElements(selector, contextNode = undefined, doc = document, win = window, _cplink = undefined) {
        if (!selector) return [];
        contextNode = contextNode || doc;
        if (typeof selector === 'string') {
            if (selector.search(/^css;/i) === 0) {
                return getAllElementsByCSS(selector.slice(4), contextNode);
            } else {
                return getAllElementsByXpath(selector, contextNode, doc);
            }
        } else {
            const query = selector(doc, win, _cplink);
            if (!Array.isArray(query)) {
                throw new Error('getAllElements 返回错误类型');
            } else {
                return query;
            }
        }
    }
})();