阅读模式辅助

美化滚动条,添加小说目录(仅支持部分网站),配合插件“Circle 阅读模式”使用更棒哦。

// ==UserScript==
// @name         阅读模式辅助
// @description  美化滚动条,添加小说目录(仅支持部分网站),配合插件“Circle 阅读模式”使用更棒哦。
// @author       little3022
// @namespace    little3022.TM.Cataloguer
// @homepageURL  https://gf.qytechs.cn/users/782903
// @supportURL   https://gf.qytechs.cn/scripts/445980/feedback
// @version      0.1.2
// @license      GNU GPL v2.0
// @run-at       document-end
// @match        *://*/*
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_log
// ==/UserScript==


class Cataloguer {
    constructor() {
        this.__name__ = 'Cataloguer';

        this.matchRules= [{
            host: '',
            describe: '',

        }];
        this.reHrefs = [
            // 章节及小说主页链接(捕获 1 - n 为主页)的正则表达式
            /(http\S+www\.biquger\.com\/biquge\/\d+)\/\d+/,
            /(http\S+www\.ibswtan\.com\/\d+\/\d+)\/\d+\.html/,
            /(http\S+www\.huaxiaci\.com\/\d+)\/\d+\.html/,
            /(http\S+www\.xbiquge\.la\/\d+\/\d+)\/\d+\.html/,
            /(http\S+www\.bswtan\.com\/\d+\/\d+)\/\d+\.html/,
            /(http\S+www\.360xs\.com\/mulu\/\d+\/\d+)-\d+(\.html)/,
            /(http\S+www\.biduoxs\.com\/biquge\/\S+\/)\S+\.html/,
            /(http\S+www\.biqugee\.com\/book\/\d+\/)\d+.html/,
            /(http\S+www\.bqg99\.cc\/book\/\d+\/)\d+\.html/,
            /(http\S+www\.ncjy\.net\/bxwx\/\d+\/)\d+\.html/,
            /(http\S+www\.xxyanqing5\.com\/book\/\d+\/)\d+\.html/,
            /(http\S+www\.fyrsks\.com\/bqg\/\d+\/)\d+\.html/
        ];
        this.titleSelectors = [
            // 标题选择器 (单一选择器)
			'#info > h1',
            'body > div.container > div.content > div:nth-child(2) > div.bookinfo > h1',
            '#article_right > div > a > h1',
            'body > div.book > div.info > h1',
            'body > div.container > div.row.row-detail > div > div > div.info > div.top > h1'
        ];
        this.catalogueSelectors = [
            // 目录选择器 (群选择器)
			'#list > dl > dd > a',
            '#list-chapterAll > dd > a',
            '#index_list_li1 > a',
            'body > div.listmain > dl > dd > a',
            'body > div.container > div.row.row-section > div > div:nth-child(4) > ul > li > a'
        ];
        this.jData = {
            thisURL: '',
            homeURL: '',
            title: '', // 小说标题
            total: 0, // 总章节数
            active: {
                index: 0, // 活动章节索引
                href: '',
                text: '',
                node: null
            }
        };
        this.sMainElementID = 'OCCataloguer';
        this.elCataloguer = null;
        this.elInfo = null;
        this.elTitle = null;
        this.elProgress = null;
        this.elLastRead = null;
        this.elList = null;

        this._first = true;
    }

    printLog(info) {
        Date.prototype.Format = function (fmt) { //author: meizz
            var o = {
                "M+": this.getMonth() + 1, //月份
                "d+": this.getDate(), //日
                "h+": this.getHours(), //小时
                "m+": this.getMinutes(), //分
                "s+": this.getSeconds(), //秒
                "q+": Math.floor((this.getMonth() + 3) / 3), //季度
                "S": this.getMilliseconds() //毫秒
            };
            if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length));
            for(var k in o) {
                if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));
            }
            return fmt;
        }

        let date = new Date();
        // this._tiper.append(`${date.Format('yyyy-MM-dd hh:mm:ss.S')}: ${info}`);
        GM_log(`${date.Format('yyyy-MM-dd hh:mm:ss.S')}: ${info}`);
    }

    checkURL() { // 检查 URL 是否匹配以判断是否在该网页运行此脚本
        for(let re of this.reHrefs) {
            let match = document.URL.match(re);

            if(match && match.length >= 2) {
                this.jData.thisURL = match[0];
                for(let i = 1; i < match.length; i++) {
                    this.jData.homeURL += match[i];
                }

                return true;
            }
        }

        return false;
    }

    showGUI() { // 显示 GUI
        let elCataloguer = document.createElement('div');
        let elStyle = document.createElement('style');
        let elBtShow = document.createElement('div');
        let elWrapper = document.createElement('div');
        let elTitle = document.createElement('h3');
        let elProgress = document.createElement('h5');
        let elLastRead = document.createElement('div');
        let elList = document.createElement('div');
        let elInfo = document.createElement('div');

        elCataloguer.id = this.sMainElementID;
        elBtShow.className = 'OCC-btShow';
        elWrapper.className = 'OCC-wrapper';
        elLastRead.className = 'OCC-lastread';
		elList.className = 'OCC-list';
        elInfo.className = 'OCC-info';

        elStyle.innerText = '#OCCataloguer .OCC-list::-webkit-scrollbar {width: 8px;background-color: rgba(0, 40, 51, 0.8);}#OCCataloguer .OCC-list::-webkit-scrollbar-thumb {background-color: #bbb5ac;}#OCCataloguer {position: fixed;top: calc(50% - 182.5px);left: -280px;width: 300px;height: 365px;border-radius: 0 8px 8px 0;background-color: rgba(0, 50, 64, 0.8);transition: left 0.2s linear;}#OCCataloguer:hover {left: 0px;}#OCCataloguer:hover .OCC-btShow {display: none;}#OCCataloguer:hover .OCC-wrapper {display: block;}#OCCataloguer .OCC-btShow {position: absolute;right: 0px;width: 20px;height: 100%;border-radius: 0 8px 8px 0;line-height: 360px;text-align: center;vertical-align: middle;color: lightgray;}#OCCataloguer .OCC-wrapper {display: none;}#OCCataloguer h3 {margin: 12px 0 0px 16px;width: 200px;color: #beb9b2;white-space: nowrap;overflow: hidden;cursor: pointer;}#OCCataloguer h3:hover {color: green;}#OCCataloguer h5 {margin: 0 0 0 16px;width: 100px;color: #beb9b2;white-space: nowrap;overflow: hidden;}#OCCataloguer .OCC-lastread {margin: 5px 5px 0 5px;border-bottom: 1px dashed #A7A7A7;height: 24px;}#OCCataloguer .OCC-list {width: 300px;height: 265px;overflow-x: hidden;overflow-y: auto;}#OCCataloguer .OCC-info {width: 300px;height: 100%;font-size: 20px;line-height: 280px;text-align: center;vertical-align: middle;color: #beb9b2;overflow: hidden;white-space: nowrap;}#OCCataloguer dl {margin: 0;padding-top: 5px;}#OCCataloguer dd {margin: 0 5px 0 5px;padding: 0 0 0 16px;border-radius: 3px;width: calc(100% - 30px);height: 24px;vertical-align: middle;overflow: hidden;text-overflow: ellipsis;}#OCCataloguer dd:hover {padding: 0 0 0 20px;background-color: rgba(128, 128, 128, 0.35);box-shadow:0 2px 5px #303030,0 -2px 5px #000;}#OCCataloguer a {display: inline-block;width: 100%;font-size: 13px;line-height: 24px;color: #A7A7A7;text-decoration: none;white-space: nowrap;}#OCCataloguer a:visited {color: #00bbff;}#OCCataloguer #OCC-active {margin: 5px 5px 5px 5px;background-color: rgba(0, 199, 0, 0.3);box-shadow:2px 2px 5px #303030,2px -2px 5px #000;}#OCCataloguer .OCC-lastread a {color: salmon;}#OCCataloguer #OCC-active a {color: #A7A7A7;}';
        elBtShow.innerText = '▶';
        elTitle.innerText = 'Title';
        elProgress.innerText = '(9999/9999)';
        elLastRead.innerHTML = '<dd><a href="#">上次读到: ...</a></dd>';
        elInfo.innerText = 'Info...';

        elCataloguer.onmouseenter = () => {
            this.scrollToActive();
        };

        elCataloguer.appendChild(elStyle);
        elCataloguer.appendChild(elBtShow);
        elCataloguer.appendChild(elWrapper);
        elWrapper.appendChild(elTitle);
        elWrapper.appendChild(elProgress);
        elWrapper.appendChild(elLastRead);
        elWrapper.appendChild(elList);
        elList.appendChild(elInfo);

        // 显示上次读到
        let t = this.loadData();
        if(t.homeURL === this.jData.homeURL) {
            elLastRead.innerHTML = `<dd><a href="${t.lastReadURL}">上次读到: ${t.lastReadText}</a></dd>`;
            elLastRead.title = t.lastReadText;
        }

        this.elCataloguer = elCataloguer;
        this.elTitle = elTitle;
        this.elProgress = elProgress;
        this.elLastRead = elLastRead;
        this.elList = elList;
        this.elInfo = elInfo;

        if(!document.getElementById(this.sMainElementID)) {
            document.documentElement.appendChild(elCataloguer);
		}
    }

    refreshGUI(title, catalogues) { // 更新显示数据
        let lastReadIndex = -1;

        if(title) {
            this.jData.title = title;
            this.elTitle.innerText = title;
            this.elTitle.title = `点击跳转《${title}》主页 (${this.jData.homeURL})`;
            this.elTitle.addEventListener('click', () => {
                window.open(this.jData.homeURL, '_self');
            });
        }
        if(catalogues) {
            let elDL = document.createElement('dl');
            let count = 0, flag = false; // 章节总数, 是否自带纯数字序号

//            // 判断是否自带序号
//            for(let i = 0; i < 9; i++) {
//                let a = catalogues[i];
//
//                if(/<a href.+?>\d+?、.+?<\/a>/.test(a)) {
//                    flag = true;
//                    break;
//                }
//            }
//            // 插入目录
//            for(let a of catalogues) {
//                let elDD = document.createElement('dd');
//                if(flag) {
//                    elDD.innerHTML = a.outerHTML;
//                }
//                else {
//                    elDD.innerHTML = a.outerHTML.replace(/(<a href.+?>)(.+?<\/a>)/, `$1${count + 1}、$2`);
//                }
//                elDL.appendChild(elDD);
//                count += 1;
//            }
             // 插入目录
             for(let a of catalogues) {
                 let elDD = document.createElement('dd');
                 elDD.innerHTML = a.outerHTML;
                 elDL.appendChild(elDD);
                 count += 1;
             }
            // 显示目录
            this.elList.innerHTML = '';
            this.elList.appendChild(elDL);
            // 更新目录总数
            this.jData.total = count;
        }
    }

    refreshActiveChapter() { // 更新活动章节信息, 并返回对应 a 标签
        // 若未获取章节总数则退出
        if(!this.jData.total) return null;
        // 已读(已加载)章节列表
        let elRead = document.querySelectorAll('html > div.circle-pager > div > div > h1 > a');

        if(elRead.length > 0) {
            elRead = elRead[elRead.length - 1];
            // 活动章节未改变
            if(this.jData.active.href && this.jData.active.href == elRead.href) {
                return 'nochanged';
            }
            // 迭代章节列表获取活动章节
            for(let [i, elA] of this.elList.querySelectorAll('a').entries()) {
                if(elA.href == elRead.href) {
                    this.jData.active.index = i + 1;
                    this.jData.active.href = elA.href;
                    this.jData.active.text = elA.innerText;
                    this.jData.active.node = elA;

                    return elA;
                }
            }
        }

        return null;
    }

    _parseHTML(responseText) { // 解析源码为 DOM (html) 对象
        let elHTML = document.createElement('html');
        let a = responseText.indexOf('<head>');
        let b = responseText.lastIndexOf('</body>');

        if(a < 0) a = responseText.indexOf('<head');
        if(b === -1) b = responseText.length;
        if(a >= 0 && b > a) {
            elHTML.innerHTML = responseText.slice(a, b + 7);

            return elHTML;
        }

        return null;
    }

    initCatalogue() { // 获取网页源码
        let xhttp = new XMLHttpRequest();

        xhttp.onreadystatechange = () => {
            if(xhttp.readyState == 4 && xhttp.status == 200) {
                if(xhttp.responseText) { // 成功获取数据, 初始化并更新列表
                    let elHTML = null;
                    let sTitle = '';
                    let elCatalogues = null;

                    elHTML = this._parseHTML(xhttp.responseText);
                    if(elHTML) {
                        for(let selector of this.titleSelectors) {
                            sTitle = elHTML.querySelector(selector);
                            if(sTitle) {
                                sTitle = sTitle.innerText;
                                break;
                            }
                        }
                        for(let selector of this.catalogueSelectors) {
                            elCatalogues = elHTML.querySelectorAll(selector);
                            if(elCatalogues.length > 0) break;
                        }
                        this.refreshGUI(sTitle, elCatalogues);

                        this.printLog('成功获取目录数据.');
                    }
                    else {
                        this.elInfo.innerText = '数据获取失败!';
                        this.printLog('获取目录数据失败!');
                    }
                }
                else {
                    this.elInfo.innerText = '数据获取失败!';
                    this.printLog('获取目录数据失败!');
                }
            }
            else if(xhttp.readyState == 4) {
                this.elInfo.innerText = '数据获取失败!';
                this.printLog('获取目录数据失败!');
                this.printLog(`status="${xhttp.status}"; responseText="${xhttp.responseText}"`);
            }
        };

        this.elInfo.innerText = '正在获取章节目录...';
        xhttp.open('GET', this.jData.homeURL, true);
        xhttp.send();
    }

    saveData() {
        GM_setValue(this.__name__, {
            homeURL: this.jData.homeURL,
            lastReadURL: this.jData.active.href,
            lastReadText: this.jData.active.text
        });
        this.printLog('阅读历史已保存');
    }

    loadData() {
        return GM_getValue(this.__name__, {
            homeURL: '',
            lastReadURL: '',
            lastReadText: ''
        });
    }

    scrollToActive() { // 跳转至活动章节
        // 更新并获取活动章节索引
        let elActive = this.refreshActiveChapter();

        if(elActive) { // 成功获取进度
            if(elActive != 'nochanged') { // 活动章节已改变
                let tdd = document.getElementById('OCC-active');

                if(tdd) tdd.id = '';
                elActive.parentElement.id = 'OCC-active';
                this.elProgress.innerText = `(${this.jData.active.index}/${this.jData.total})`;
                elActive.parentElement.scrollIntoView();
                this.printLog(`跳转活动章节 ${elActive.innerText}`);

                // 更新阅读历史
                if(!this._first) {
                    this.elLastRead.innerHTML = `<dd><a href="${elActive.href}">上次读到: ${elActive.innerText}</a></dd>`;
                    this.elLastRead.title = elActive.innerText;

                    this.saveData();
                }
                this._first = false;
            }
            else {
                this.jData.active.node.parentElement.scrollIntoView();
                this.printLog('活动章节未改变.');
            }
        }
    }

    setScroller() { // 设置滚动条样式
        document.ondblclick = ()=>{};

        var sty = document.createElement('style');
        sty.id = 'srollStyle';
        sty.innerHTML = '::-webkit-scrollbar {width: 8px;background-color: var(--circle-bgcolor, rgba(0, 40, 51, 0.8));}::-webkit-scrollbar-thumb {background-color: #bbb5ac;}';

        setTimeout(() => {
            document.head.appendChild(sty);
            this.printLog('成功设置滚动条样式');
        }, 500);
    }

    exec() { // 初始化并运行目录程序
        if(!this.checkURL()) return;

        setTimeout(() => {
            this.showGUI();
            this.initCatalogue();
        }, 1000);
    }
}


(function() {
    'use strict';
    let app = new Cataloguer();

    app.exec();
    app.setScroller();
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址