Happymh reading aid

infinite scroll reading mode,Arrow keys to switch chapters,Background preload image,Auto reload image with error.

// ==UserScript==
// @name               嗨皮漫畫閱讀輔助
// @name:en            Happymh reading aid
// @name:zh-CN         嗨皮漫画阅读辅助
// @name:zh-TW         嗨皮漫畫閱讀輔助
// @version            2.6.8
// @description        無限滾動模式(自動翻頁、瀑布流),背景預讀圖片,自動重新載入出錯的圖片,左右方向鍵切換章節,目錄頁自動展開全部章節,新分頁打開漫畫鏈接。
// @description:en     infinite scroll reading mode,Arrow keys to switch chapters,Background preload image,Auto reload image with error.
// @description:zh-CN  无限滚动模式(自动翻页、瀑布流),背景预读图片,自动重新加载出错的图片,左右方向键切换章节,目录页自动展开全部章节,新标籤页打开漫画链接。
// @description:zh-TW  無限滾動模式(自動翻頁、瀑布流),背景預讀圖片,自動重新載入出錯的圖片,左右方向鍵切換章節,目錄頁自動展開全部章節,新分頁打開漫畫鏈接。
// @author             tony0809
// @match              *://m.happymh.com/*
// @icon               https://m.happymh.com/favicon.ico
// @grant              unsafeWindow
// @grant              GM_registerMenuCommand
// @grant              GM_getValue
// @grant              GM_setValue
// @grant              GM_deleteValue
// @grant              GM_openInTab
// @run-at             document-end
// @license            MIT
// @namespace          https://greasyfork.org/users/20361
// ==/UserScript==

(async () => {
    "use strict";

    if (document.querySelector(".captcha-area")) {
        console.warn("嗨皮Cloudflare正在人機驗證中");
        return;
    }

    const defaultConfigs = { //1開、0關
        arrowKey: 1, //鍵盤左右方向鍵切換章節。
        doubleClick: 0, //雙擊前往下一話,方便手機使用。
        preload: 1, //閱讀頁預讀全部圖片,並且嘗試預讀下一話圖片。
        autoReload: 1, //重新載入出錯的圖片
        autoNext: 0, //下一話按鈕完全進入視口可視範圍內後自動下一話。
        autoNextSec: 1, //下一話按鈕完全進入視口可視範圍內後自動下一話的延遲秒數。
        autoShowAll: 1, //目錄頁自動展開全部章節。
        openInNewTab: 1, //新分頁打開漫畫鏈接。
        infiniteScroll: 0, //無限滾動閱讀模式。
        highQuality: 0, //去掉圖片鏈結的?q參數。
        history: 1, //無限滾動API請求成功後添加瀏覽器歷史。
        removeAd: 1 //移除無用元素
    };

    const GM_configs = GM_getValue("configs", defaultConfigs);
    const configs = Object.assign(defaultConfigs, GM_configs);
    //console.log("腳本設定物件", configs);

    const _unsafeWindow = unsafeWindow ?? window;
    const language = _unsafeWindow.navigator.language;

    let scriptLanguage;
    switch (language) {
        case "zh-TW":
        case "zh-HK":
        case "zh-Hant-TW":
        case "zh-Hant-HK":
            scriptLanguage = "TW";
            break;
        case "zh":
        case "zh-CN":
        case "zh-Hans-CN":
            scriptLanguage = "CH";
            break;
        default:
            scriptLanguage = "EN";
    }

    let i18n;
    switch (scriptLanguage) {
        case "TW":
            i18n = {
                config: {
                    title: "嗨皮漫畫閱讀輔助設定",
                    arrowKey: "左右方向鍵切換章節",
                    doubleClick: "雙擊前往下一話",
                    preload: "背景預讀圖片",
                    autoReload: "自動重新載入出錯的圖片",
                    autoNext: "自動下一話",
                    autoNextSec: "自動下一話延遲(秒)",
                    autoShowAll: "目錄頁自動展開全部章節",
                    openInNewTab: "新分頁打開漫畫鏈結",
                    infiniteScroll: "啟用無限滾動閱讀模式",
                    history: "無限滾動添加瀏覽器歷史紀錄",
                    highQuality: "無限滾動載入最高品質圖片",
                    removeAd: "移除無用元素",
                    exclude: "無限滾動標題文字正規表達式排除",
                    cancel: "取消",
                    reset: "重置設定",
                    save: "保存設定"
                },
                tips: {
                    noNext: "沒有下一話了!",
                    noPrev: "沒有上一話了!",
                    apiError: "API請求返回錯誤,伺服器拒絕連線,也可能是需要再次Cloudflare人機驗證。"
                },
                commandMenu: {
                    settings: "設定"
                },
                button: {
                    openComments: "開啟評論",
                    closeComments: "關閉評論"
                }
            };
            break;
        case "CN":
            i18n = {
                config: {
                    title: "嗨皮漫画阅读辅助设置",
                    arrowKey: "左右方向键切换章节",
                    doubleClick: "双击前往下一话",
                    preload: "背景预读图片",
                    autoReload: "自动重新加载出错的图片",
                    autoNext: "自动下一话",
                    autoNextSec: "自动下一话延迟(秒)",
                    autoShowAll: "目录页自动展开全部章节",
                    openInNewTab: "新标籤页打开漫画链结",
                    infiniteScroll: "启用无限滚动阅读模式",
                    highQuality: "无限滚动加载最高品质图片",
                    history: "无限滚动添加浏览器历史纪录",
                    removeAd: "移除无用元素",
                    exclude: "无限滚动标题文字正则表达式排除",
                    cancel: "取消",
                    reset: "重置设置",
                    save: "保存设置"
                },
                tips: {
                    noNext: "没有下一话了!",
                    noPrev: "没有上一话了!",
                    apiError: "API请求返回错误,服务器拒绝连接,也可能是需要再次Cloudflare人机验证。"
                },
                commandMenu: {
                    settings: "设置"
                },
                button: {
                    openComments: "打开评论",
                    closeComments: "关闭评论"
                }
            };
            break;
        default:
            i18n = {
                config: {
                    title: "settings",
                    arrowKey: "Arrow keys to switch chapters",
                    doubleClick: "Double click to go to the next chapter",
                    preload: "Background preload image",
                    autoReload: "Auto reload image with error",
                    autoNext: "Auto next chapter",
                    autoNextSec: "Auto next chapter delay sec",
                    autoShowAll: "Contents page auto expands all chapters",
                    openInNewTab: "Open the comic link in a new tab",
                    infiniteScroll: "Turn on infinite scroll reading mode",
                    highQuality: "Infinite scroll loading of high quality image",
                    history: "Infinite scroll add browser history",
                    removeAd: "Remove useless elements",
                    exclude: "Title Exclude RegExp",
                    cancel: "Cancel",
                    reset: "Reset",
                    save: "Save",
                },
                tips: {
                    noNext: "no next chapter",
                    noPrev: "no prev chapter",
                    apiError: "The API request returned an error and the server refused to connect. It may also be that Cloudflare human-computer verification is required again."
                },
                commandMenu: {
                    settings: "settings"
                },
                button: {
                    openComments: "Open Comments",
                    closeComments: "Close Comments"
                }
            };
    }

    const lp = _unsafeWindow.location.pathname;
    const isReadPage = /^\/mangaread\/\w+\/\d+$/.test(lp);
    const isUpdatePage = /^\/latest$/.test(lp);
    const isListPage = /^\/manga\/\w+$/.test(lp);
    const isBookcasePage = /^\/bookcase$/.test(lp);
    const isRankPage = /^\/rank/.test(lp);
    const isUserPage = /^\/user/.test(lp);
    const isLogged = document.cookie.startsWith("sf_token");
    let nextChapterUrl = null;
    let prevChapterUrl = null;

    const openInNewTab = () => gae(".home-banner a:not([target=_blank]),.manga-rank a:not([target=_blank]),.manga-cover a:not([target=_blank])").forEach(a => a.setAttribute("target", "_blank"));
    const delay = time => new Promise(resolve => setTimeout(resolve, time));
    const isString = str => Object.prototype.toString.call(str) === "[object String]";
    const isObject = obj => Object.prototype.toString.call(obj) === "[object Object]";
    const isArray = arr => Object.prototype.toString.call(arr) === "[object Array]";
    const isEle = e => /^\[object\sHTML[a-zA-Z]*Element\]$/.test(Object.prototype.toString.call(e));

    const ge = (selector, contextNode = null, dom = document) => {
        if (/^\//.test(selector)) {
            return dom.evaluate(selector, (contextNode ?? document), null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
        } else {
            return (contextNode ?? document).querySelector(selector);
        }
    };

    const gae = (selector, contextNode = null, dom = document) => {
        if (/^\//.test(selector)) {
            const nodes = [];
            const results = dom.evaluate(selector, (contextNode ?? document), null, XPathResult.ANY_TYPE, null);
            let node = null;
            while (node = results.iterateNext()) {
                nodes.push(node);
            }
            return nodes;
        } else {
            return [...(contextNode ?? document).querySelectorAll(selector)];
        }
    };

    const addGlobalStyle = css => {
        const style = document.createElement("style");
        style.type = "text/css";
        style.innerHTML = css;
        document.head.append(style);
    };

    const waitEle = selector => {
        return new Promise(resolve => {
            const loop = setInterval(() => {
                if (!!ge(selector)) {
                    clearInterval(loop);
                    resolve();
                }
            }, 100);
        });
    };

    const remove = obj => {
        if (isString(obj)) {
            let selector = obj;
            gae(selector).forEach(e => e.remove());
        } else if (isArray(obj)) {
            let selectors = obj;
            selectors.forEach(selector => gae(selector).forEach(e => e.remove()));
        }
    };

    const getHeaders = () => {
        return {
            "headers": {
                "accept": "application/json, text/plain, */*",
                "x-requested-id": new Date().getTime(),
                "x-requested-with": "XMLHttpRequest"
            }
        };
    };

    const preload = (pn, text) => {
        let preloadDiv = ge("#happymhPreload");
        if (preloadDiv) {
            preloadDiv.innerHTML = "";
        } else {
            preloadDiv = document.createElement("div");
            preloadDiv.id = "happymhPreload";
            preloadDiv.style.display = "none";
            document.body.append(preloadDiv);
        }
        const [, , mangaCode, id] = pn.split("/");
        const apiUrl = `/v2.0/apis/manga/read?code=${mangaCode}&cid=${id}&v=v3.1302723`;
        fetch(apiUrl, getHeaders()).then(res => res.json()).then(async jsonData => {
            try {
                if (jsonData.status == 0) {
                    console.log(text + "漫畫名稱:" + jsonData.data.manga_name + "\n章節名稱:" + jsonData.data.chapter_name + "\n章節圖片:\n", jsonData.data.scans, "\nJSON:\n", jsonData);
                    const scans = jsonData.data.scans;
                    for (const scan of scans) {
                        const img = new Image();
                        img.setAttribute("referrerpolicy", "origin");
                        img.alt = jsonData.data.chapter_name;
                        img.src = scan.url;
                        preloadDiv.append(img);
                        await delay(200);
                    }
                } else if (jsonData.status == 403) {
                    console.log(text + "獲取數據失敗\n", jsonData);
                }
            } catch (error) {
                console.error(error);
            }
        }).catch(error => console.error(error));
    };

    const textExcludeRegExp = GM_getValue("exclude", "");

    const createConfigElement = () => {

        const mainElement = document.createElement("div");
        mainElement.id = "mainHappymhConfigShadowElement";

        const shadow = mainElement.attachShadow({
            mode: "closed"
        });

        shadow.innerHTML = `
<style type="text/css">
    #happymhConfigElement {
        text-align: center;
        width: 300px;
        height: auto;
        position: fixed;
        top: calc((100% - 460px) / 2);
        left: calc((100% - 302px) / 2);
        border: 1px solid #a0a0a0;
        border-radius: 3px;
        box-shadow: -2px 2px 5px rgb(0 0 0 / 30%);
        background-color: #FAFAFB;
        z-index: 10000;
    }

    #happymhConfigElement div,
    #happymhConfigElement label,
    #happymhConfigElement button {
        font-family: Arial, sans-serif;
        font-size: 14px;
        color: black;
        float: none;
        line-height: 18px;
    }

    #happymhConfigElement .title {
        width: 100%;
    }

    #happymhConfigElement div.item {
        width: 348px;
        display: flex;
    }

    #happymhConfigElement label.select {
        margin: 0 5px;
    }

    #happymhConfigElement div {
        margin-bottom: 4px;
        padding: 1px 4px;
    }

    #happymhConfigElement input[type=checkbox] {
        width: 14px;
        margin: 0 6px;
    }

    #happymhConfigElement button {
        width: auto;
        min-width: 80px;
        max-width: 100px;
        min-height: unset;
        max-height: 24px;
        margin-left: 2px;
        margin-right: 2px;
        margin-bottom: 4px;
        display: inline-block;
        color: #000000;
        border: 1px solid #a0a0a0;
        background-color: transparent;
        border-radius: unset;
    }

    #happymhConfigElement #exclude {
        width: calc(100% - 12px);
        height: 100px;
    }
</style>
<div id="happymhConfigElement">
    <div class="title" style="width: calc(100% - 8px);">
        ${i18n.config.title}
    </div>
    <div class="item">
        <input id="arrowKeyInput" type="checkbox">
        <label>${i18n.config.arrowKey}</label>
    </div>
    <div class="item">
        <input id="doubleClickInput" type="checkbox">
        <label>${i18n.config.doubleClick}</label>
    </div>
    <div class="item">
        <input id="autoNextInput" type="checkbox">
        <label>${i18n.config.autoNext}</label>
    </div>
    <div class="item">
        <label class="select">${i18n.config.autoNextSec}</label>
        <select id="autoNextSec">
            ${new Array(10).fill().map((_, i) => `<option value="${i + 1}">${i + 1}</option>`).join("")}
        </select>
    </div>
    <div class="item">
        <input id="autoShowAllInput" type="checkbox">
        <label>${i18n.config.autoShowAll}</label>
    </div>
    <div class="item">
        <input id="openInNewTabInput" type="checkbox">
        <label>${i18n.config.openInNewTab}</label>
    </div>
    <div class="item">
        <input id="autoReloadInput" type="checkbox">
        <label>${i18n.config.autoReload}</label>
    </div>
    <div class="item">
        <input id="preloadInput" type="checkbox">
        <label>${i18n.config.preload}</label>
    </div>
    <div class="item">
        <input id="removeAdInput" type="checkbox">
        <label>${i18n.config.removeAd}</label>
    </div>
    <div class="item">
        <input id="infiniteScrollInput" type="checkbox">
        <label>${i18n.config.infiniteScroll}</label>
    </div>
    <div class="item">
        <input id="highQualityInput" type="checkbox">
        <label>${i18n.config.highQuality}</label>
    </div>
    <div class="item">
        <input id="historyInput" type="checkbox">
        <label>${i18n.config.history}</label>
    </div>
    <label>${i18n.config.exclude}<textarea id="exclude" placeholder="第.*话\n第.*章"></textarea></label>
    <button id="cancelBtn">${i18n.config.cancel}</button>
    <button id="resetBtn">${i18n.config.reset}</button>
    <button id="saveBtn">${i18n.config.save}</button>
</div>
        `;

        const main = ge("#happymhConfigElement", shadow);
        ge("#arrowKeyInput", main).checked = configs.arrowKey == 1 ? true : false;
        ge("#doubleClickInput", main).checked = configs.doubleClick == 1 ? true : false;
        ge("#preloadInput", main).checked = configs.preload == 1 ? true : false;
        ge("#autoReloadInput", main).checked = configs.autoReload == 1 ? true : false;
        ge("#autoNextInput", main).checked = configs.autoNext == 1 ? true : false;
        ge("#autoNextSec", main).value = configs.autoNextSec;
        ge("#autoShowAllInput", main).checked = configs.autoShowAll == 1 ? true : false;
        ge("#openInNewTabInput", main).checked = configs.openInNewTab == 1 ? true : false;
        ge("#removeAdInput", main).checked = configs.removeAd == 1 ? true : false;
        ge("#infiniteScrollInput", main).checked = configs.infiniteScroll == 1 ? true : false;
        ge("#highQualityInput", main).checked = configs.highQuality == 1 ? true : false;
        ge("#historyInput", main).checked = configs.history == 1 ? true : false;
        ge("#exclude", main).value = textExcludeRegExp;
        ge("#cancelBtn", main).addEventListener("click", event => {
            event.preventDefault();
            mainElement.remove();
        });
        ge("#resetBtn", main).addEventListener("click", event => {
            event.preventDefault();
            mainElement.remove();
            GM_deleteValue("configs");
            GM_deleteValue("exclude");
            _unsafeWindow.location.reload();
        });
        ge("#saveBtn", main).addEventListener("click", event => {
            event.preventDefault();
            configs.arrowKey = ge("#arrowKeyInput", main).checked == true ? 1 : 0;
            configs.doubleClick = ge("#doubleClickInput", main).checked == true ? 1 : 0;
            configs.preload = ge("#preloadInput", main).checked == true ? 1 : 0;
            configs.autoReload = ge("#autoReloadInput", main).checked == true ? 1 : 0;
            configs.autoNext = ge("#autoNextInput", main).checked == true ? 1 : 0;
            configs.autoNextSec = ge("#autoNextSec", main).value;
            configs.autoShowAll = ge("#autoShowAllInput", main).checked == true ? 1 : 0;
            configs.openInNewTab = ge("#openInNewTabInput", main).checked == true ? 1 : 0;
            configs.removeAd = ge("#removeAdInput", main).checked == true ? 1 : 0;
            configs.infiniteScroll = ge("#infiniteScrollInput", main).checked == true ? 1 : 0;
            configs.highQuality = ge("#highQualityInput", main).checked == true ? 1 : 0;
            configs.history = ge("#historyInput", main).checked == true ? 1 : 0;
            mainElement.remove();
            GM_setValue("configs", configs);
            GM_setValue("exclude", ge("#exclude", main).value);
            _unsafeWindow.location.reload();
        });
        document.body.append(mainElement);
    };

    GM_registerMenuCommand(i18n.commandMenu.settings, () => createConfigElement());

    if (configs.removeAd == 1 && isReadPage) {
        const removeElement = () => {
            const removeSelectors = [
                "noscript",
                "iframe",
                ".adsbygoogle",
                "#google_pedestal_container",
                "#root>div>div:has(>a)",
                "//div[text()='Done']",
                "#notice-react",
                "#alert-confirm-react",
                "#root~div[class]:not(.MuiDrawer-root)"
            ];
            remove(removeSelectors);
        };
        removeElement();
        new MutationObserver(removeElement).observe(document.body, {
            childList: true,
            subtree: true
        });
    }

    if (configs.openInNewTab == 1 && !isReadPage && !isListPage && !isUserPage) {
        openInNewTab();
        console.log("嗨皮漫畫在新分頁打開漫畫鏈接");
        new MutationObserver(() => {
            openInNewTab();
        }).observe(document.body, {
            childList: true,
            subtree: true
        });
    }

    if (configs.autoShowAll == 1 && isListPage) {
        window.addEventListener("load", async () => {
            await delay(1000);
            if (ge("//div[contains(text(),'给本王显示全部章节')]")) {
                ge("#expandButton").click();
                console.log("嗨皮漫畫自動展開目錄");
            }
        });
    }

    if (configs.arrowKey == 1 && isReadPage) {
        document.addEventListener("keydown", event => {
            if (ge("#mainHappymhConfigShadowElement")) return;
            if (event.code === "ArrowRight" || event.key === "ArrowRight") {
                const nextE = ge("//a[span[text()='下一话' or text()='下一話'] and starts-with(@href,'/mangaread/')]");
                if (isString(nextChapterUrl)) {
                    _unsafeWindow.location.href = nextChapterUrl;
                } else if (nextE) {
                    _unsafeWindow.location.href = nextE.href;
                } else {
                    alert(i18n.tips.noNext);
                }
            }
            if (event.code === "ArrowLeft" || event.key === "ArrowLeft") {
                const prevE = ge("//a[span[text()='上一话' or text()='上一話'] and starts-with(@href,'/mangaread/')]");
                if (isString(prevChapterUrl)) {
                    _unsafeWindow.location.href = prevChapterUrl;
                } else if (prevE) {
                    _unsafeWindow.location.href = prevE.href;
                } else {
                    alert(i18n.tips.noPrev);
                }
            }
        });
    }

    if (configs.doubleClick == 1 && isReadPage) {
        document.addEventListener("dblclick", () => {
            if (ge("#mainHappymhConfigShadowElement")) return;
            const nextE = ge("footer a");
            _unsafeWindow.location.href = nextE.href;
        });
    }

    if (configs.preload == 1 && isReadPage && configs.infiniteScroll != 1) {
        await waitEle("[id^=imageLoader]");
        console.log("嗨皮漫畫預讀全部圖片");
        preload(lp, "嗨皮漫畫本話數據\n");
        setTimeout(() => {
            const nextE = ge("//span[@id and text()='下一话' or text()='下一話']/following-sibling::a[1][starts-with(@href,'/mangaread/')]");
            if (nextE) {
                preload(nextE.pathname, "嗨皮漫畫下一話數據\n");
            }
        }, 3000);
    }

    if (isReadPage) {
        let selector;
        if (configs.infiniteScroll == 1) {
            selector = "footer";
        } else {
            selector = "#page-area";
        }
        await waitEle(selector);
        new IntersectionObserver((entries, observer) => {
            if (entries[0].isIntersecting) {
                //observer.unobserve(entries[0].target);
                const item = ge("footer>article>div:nth-child(2)");
                gae("a", item).forEach(a => a.classList.add("MuiButton-containedPrimary"));
                const [nextDiv, , prevDiv] = gae("footer div");
                const nextA = ge("a", nextDiv);
                const prevA = ge("a", prevDiv);
                if (prevA?.href?.includes("/mangaread/")) {
                    prevA.classList.add("MuiButton-containedPrimary");
                }
                if (nextA?.href?.includes("/readMore/")) {
                    nextA.classList.remove("MuiButton-containedPrimary");
                    nextA.firstChild.innerText = "^_^感谢您的阅读~已经没有下一话了哦~";
                }
            }
        }).observe(ge(selector));
    }

    if (configs.autoNext == 1 && isReadPage) {
        await waitEle("//a[span[text()='下一话' or text()='下一話']]");
        let timeId;
        new IntersectionObserver(entries => {
            if (entries[0].isIntersecting) {
                timeId = setTimeout(() => {
                    let nextE = ge("//a[span[text()='下一话' or text()='下一話'] and starts-with(@href,'/mangaread/')]");
                    if (nextE) {
                        _unsafeWindow.location.href = nextE.href;
                    }
                }, configs.autoNextSec * 1000);
            } else {
                clearTimeout(timeId);
            }
        }, {
            threshold: 1,
        }).observe(ge("footer a"));
    }

    if (configs.autoReload == 1 && isReadPage && configs.infiniteScroll != 1) {
        new MutationObserver(mutationsList => {
            //console.log(mutationsList);
            mutationsList.forEach(e => {
                //console.log([...e.target?.children]);
                if (e.target?.children[1]?.innerText === "请疯狂点击图片以重新加载") {
                    e.target.click();
                }
            });
        }).observe(ge("#root article"), {
            childList: true,
            subtree: true
        });
    }

    if (isReadPage && configs.infiniteScroll == 1) {
        //所有章節資料API
        //https://m.happymh.com/apis/m/mcsmmss?code=漫畫代碼
        //推送章節閱讀歷史紀錄API
        //https://m.happymh.com/v2.0/apis/uu/readLog?cid=章節ID&code=漫畫代碼
        //章節閱讀資料API
        //https://m.happymh.com/v2.0/apis/manga/read?code=漫畫代碼&cid=章節ID&v=v3.1302723
        const infiniteScrollCss = `
footer {
    margin: 0px !important;
    padding: 0px !important;
}
.chapterTitle {
    width: auto;
    height: 30px;
    font-size: 18px;
    color: black;
    font-family: Arial, sans-serif;
    line-height: 29px;
    text-align: center;
    overflow: hidden;
    display: block;
    margin: 10px 5px;
    border: 1px solid #e0e0e0;
    background-color: #f0f0f0;
    background: -webkit-gradient(linear, 0 0, 0 100%, from(#f9f9f9), to(#f0f0f0));
    background: -moz-linear-gradient(top, #f9f9f9, #f0f0f0);
    box-shadow: 0 0 5px rgba(0, 0, 0, 0.6);
    border-radius: 5px;
}
#mainContent .images {
    width: 100%;
    height: auto;
    display: block;
    padding: 0;
    margin: 0 auto;
}
.apiLoading {
    width: auto !important;
    height: auto !important;
    max-width: 60px !important;
    max-height: 60px !important;
    display: block !important;
    border: none !important;
    border-radius: unset !important;
    padding: 0 !important;
    margin: 20px auto !important;
}
        `;
        addGlobalStyle(infiniteScrollCss);

        const img_loading_bak = "";
        const img_error_bak = "";
        const api_loading_gif = "";

        const mangaCode = lp.split("/").at(-2);
        let currentChapterId = lp.split("/").at(-1);
        let currentViewChapterId = currentChapterId;
        let allChapterListData;
        let currentChapterIndex = 0;
        let lastChapterIndex;
        let infiniteScrollSwitch = true;
        let isOpenComments = false;
        const hiddenElementArray = [];

        const titleObserver = new IntersectionObserver((entries, observer) => {
            entries.forEach(entry => {
                if (entry.isIntersecting) {
                    currentViewChapterId = entry.target.dataset.chapterId;
                    //console.log("當前檢視章節ID:", currentViewChapterId);
                }
            });
        });

        const imagesObserver = new IntersectionObserver((entries, observer) => {
            entries.forEach(entry => {
                if (entry.isIntersecting) {
                    //observer.unobserve(entry.target);
                    if (!entry.target.classList.contains("loaded")) {
                        entry.target.classList.add("loaded");
                        const realSrc = entry.target.dataset.src;
                        const nextElement = entry.target.nextElementSibling;
                        entry.target.src = realSrc;
                        if (nextElement?.tagName == 'IMG' && nextElement?.dataset?.src) {
                            nextElement.src = nextElement.dataset.src;
                        }
                    }
                    currentViewChapterId = entry.target.dataset.chapterId;
                    //console.log("當前檢視章節ID:", currentViewChapterId);
                }
            });
        });

        const nextObserver = new IntersectionObserver((entries, observer) => {
            entries.forEach(entry => {
                if (entry.isIntersecting) {
                    observer.unobserve(entry.target);
                    infiniteScroll();
                }
            });
        });

        const createLoadingElement = () => {
            const img = new Image();
            img.className = "apiLoading";
            img.src = api_loading_gif;
            let targetElement = ge("#mainContent");
            if (targetElement) {
                targetElement.append(img);
            } else {
                targetElement = ge("article");
                targetElement.insertAdjacentElement("afterend", img);
            }
            return img;
        };

        const getReadData = async (mid, cid, isNext = 0) => {
            let loading;
            if (isNext == 1) {
                loading = createLoadingElement();
            }
            try {
                const res = await fetch(`/v2.0/apis/manga/read?code=${mid}&cid=${cid}&v=v3.1302723`, getHeaders());
                const readJson = await res.json();
                if (readJson?.msg !== "success") {
                    loading?.remove();
                    console.error("取得章節資料錯誤", readJson);
                    return "ERROR";
                }
                if (isNext == 1) {
                    if (isLogged) {
                        fetch(`/v2.0/apis/uu/readLog?cid=${cid}&code=${mid}`, getHeaders());
                    }
                    currentChapterIndex += 1;
                    loading?.remove();
                    //currentViewChapterId = cid;
                }
                return readJson.data;
            } catch (error) {
                loading?.remove();
                console.error("取得章節資料錯誤", error);
                return "ERROR";
            }
        };

        const singleThreadLoadImgs = async imgArr => {
            for (let i = 0; i < imgArr.length; i++) {
                if (!imgArr[i]?.dataset?.src) continue;
                await new Promise(resolve => {
                    const loadSrc = imgArr[i].dataset.src;
                    const temp = new Image();
                    temp.setAttribute("referrerpolicy", "origin");
                    temp.onload = () => {
                        imgArr[i].src = loadSrc;
                        resolve();
                    }
                    temp.onerror = resolve();
                    temp.src = loadSrc;
                });
            }
        };

        const singleThreadLoadSrcs = async srcArr => {
            for (const src of srcArr) {
                await new Promise(resolve => {
                    const temp = new Image();
                    temp.setAttribute("referrerpolicy", "origin");
                    temp.onload = resolve();
                    temp.onerror = resolve();
                    temp.src = src;
                });
            }
        };

        const addBrowsingHistory = data => {
            const title = data.manga_name + " - " + data.chapter_name + "——嗨皮漫画";
            const url = document.URL.replace(/\d+$/, data.id);
            history.pushState(null, title, url);
            document.title = title;
        };

        const createComments = async () => {
            isOpenComments = true;

            const div = document.createElement("div");
            div.id = "current-comments";
            Object.assign(div.style, {
                left: "0",
                right: "0",
                top: "0",
                bottom: "0",
                width: ge(".MuiContainer-root").offsetWidth + "px",
                height: "100vh",
                margin: "0 auto",
                padding: "0px",
                position: "fixed",
                zIndex: "10000",
                backgroundColor: "#fff",
                fontSize: "14px",
                overflowY: "auto",
                overflowX: "hidden"
            });
            document.body.append(div);

            const button1 = document.createElement("button");
            button1.className = "close-comments";
            button1.innerText = i18n.button.closeComments;
            button1.style.marginTop = "10px";
            button1.style.marginLeft = "10px";
            button1.addEventListener("click", () => {
                div.remove();
                isOpenComments = false;
            });

            div.insertAdjacentElement("beforeend", button1);
            const messageHtml = `
<div id="message" class="MuiCardContent-root jss38">
  <svg class="MuiSvgIcon-root MuiSvgIcon-colorAction" focusable="false" viewBox="0 0 24 24" aria-hidden="true">
    <path d="M21.99 2H2v16h16l4 4-.01-20zM18 14H6v-2h12v2zm0-3H6V9h12v2zm0-3H6V6h12v2z"></path>
  </svg>
  <h6 class="MuiTypography-root MuiTypography-h6">数据请求中...</h6>
</div>`;
            div.insertAdjacentHTML("beforeend", messageHtml);
            div.insertAdjacentHTML("beforeend", '<ul class="MuiList-root MuiList-padding" style="padding-left: 10px;"></ul>');

            const ul = ge("ul", div);

            let loop = true;
            let pn = 1;

            const getComments = () => {
                return fetch(`/v2.0/apis/comment?code=${mangaCode}&ch_id=${currentViewChapterId}&pn=${pn}&order=time&from=read`, getHeaders()).then(res => res.json()).then(json => {
                    if (!isOpenComments) {
                        loop = false;
                        return;
                    }

                    if (json?.msg !== "success") {
                        loop = false;
                        const h6 = ge("h6", div);
                        if (h6) {
                            h6.innerText = "数据请求错误。";
                        }
                        return;
                    }

                    const {
                        isEnd,
                        items
                    } = json.data;

                    if (isEnd === true) {
                        loop = false;
                    }

                    if (!isArray(items) || items.length === 0) {
                        loop = false;
                        const h6 = ge("h6", div);
                        if (h6) {
                            h6.innerText = "还没有吐槽";
                        }
                        return;
                    } else {
                        ge("#message", div)?.remove();
                    }

                    let liHtmls = "";

                    items.forEach(item => {
                        let subHtml = "";

                        if ("sub_comments" in item && isArray(item?.sub_comments) && item?.sub_comments?.length > 0) {

                            const subHtmls = item.sub_comments.map(sub => {
                                return `
                    <div class="MuiTypography-root MuiTypography-body2 MuiTypography-colorTextSecondary" style="word-break: break-all;">
                        <span style="color: #673ab7;">${sub.user.username}</span>: ${sub.content}
                    </div>`;
                            }).join("");

                            subHtml = `
            <div style="margin-top: 0.5rem; padding-top: 0.1rem; padding-left: 0.2rem; background-color: #f5f5f5;">
                ${subHtmls}
            </div>`;
                        }

                        liHtmls += `
<li class="MuiListItem-root MuiListItem-alignItemsFlexStart" style="padding: 0 10px 0 0;">
    <div class="MuiListItemText-root MuiListItemText-multiline">
        <span class="MuiTypography-root MuiTypography-body1 MuiTypography-displayBlock" style="color: rgba(0, 0, 0, 0.87); font-weight: bolder;">${item.user.username}</span>
        <div class="MuiTypography-root MuiListItemText-secondary MuiTypography-body2 MuiTypography-colorTextSecondary MuiTypography-displayBlock">
            <div class="MuiBox-root">
                <div class="MuiBox-root">
                    <span class="MuiTypography-root MuiTypography-caption MuiTypography-colorTextSecondary MuiTypography-noWrap">章节: ${item.ch_name}</span>
                    <br>
                    <span class="MuiTypography-root MuiTypography-caption MuiTypography-colorTextSecondary">${item.create_time}</span>
                </div>
                <div class="MuiBox-root">
                    <p class="MuiTypography-root MuiTypography-body1" style="color: rgba(0, 0, 0, 0.87); word-break: break-all;">${item.content}</p>
                </div>
            </div>
            ${subHtml}
        </div>
    </div>
</li>`;
                    });

                    ul.insertAdjacentHTML("beforeend", liHtmls);
                }).catch(error => {
                    loop = false;
                    const h6 = ge("h6", div);
                    if (h6) {
                        h6.innerText = "数据请求错误,需要再次人机验证。";
                    }
                    console.error("請求錯誤", error);
                });
            };

            while (loop) {
                await getComments();
                pn++;
            }

            if (!isOpenComments) {
                return;
            }

            const button2 = document.createElement("button");
            button2.className = "close-comments";
            button2.innerText = i18n.button.closeComments;
            button2.style.marginBottom = "100px";
            button2.style.marginLeft = "10px";
            button2.addEventListener("click", () => {
                div.remove();
                isOpenComments = false;
            });
            div.insertAdjacentElement("beforeend", button2);
        };

        const createPageElement = (data, isFirst = 0) => {
            const fragment = new DocumentFragment();
            let mainContent = ge("#mainContent");
            if (!mainContent) {
                const targetElement = ge("article"); //ge("article:has(>div[id^='imageLoader'])");
                mainContent = document.createElement("div");
                mainContent.id = "mainContent";
                targetElement.insertAdjacentElement("afterend", mainContent);
            }
            if (isFirst === 0) {
                const title = document.createElement("div");
                title.className = "chapterTitle";
                title.innerText = data.chapter_name;
                title.dataset.chapterId = data.id;

                let filteredTitle = title.innerText;

                //自定義標題關鍵字排除列表
                const keywordsToExcludes = textExcludeRegExp.split("\n").filter(item => item);

                if (keywordsToExcludes.length) {

                    //打印關鍵字排除列表
                    console.log("標題關鍵字排除列表:", keywordsToExcludes);

                    const keywordRegExps = keywordsToExcludes.map(key => new RegExp(key, "g"));
                    //打印標題關鍵字正規表達式排除列表
                    console.log("標題關鍵字正規表達式排除列表:", keywordRegExps);

                    let modify = false;

                    //循環檢查並移除關鍵字
                    keywordRegExps.forEach(reg_exp => {
                        //檢查並打印匹配結果
                        const matches = filteredTitle.match(reg_exp);
                        if (matches) {
                            modify = true;
                            //打印移除前的標題
                            console.log(`移除關鍵字 "${reg_exp}" 前的標題:`, filteredTitle);
                            //只移除匹配的部分
                            filteredTitle = filteredTitle.replace(reg_exp, "");
                        }
                    });

                    if (modify) {
                        //去除多餘的空格
                        filteredTitle = filteredTitle.replace(/\s+/g, " ").trim();

                        //打印最終顯示的標題
                        console.log("最終過濾後的標題:", filteredTitle);

                        title.innerText = filteredTitle;
                    }

                }

                titleObserver.observe(title);
                fragment.append(title); // 將標題添加到文檔片段中
            }
            let srcs = data.scans.map(obj => {
                let src;
                if (configs.highQuality == 1) {
                    src = obj.url.replace(/\?q=\d+$/, "");
                } else {
                    src = obj.url;
                }
                return src;
            });
            if (currentChapterIndex < lastChapterIndex) {
                srcs = srcs.slice(0, -2);
            }
            const imgs = srcs.map((src, i) => {
                const img = new Image();
                img.className = "images";
                img.setAttribute("referrerpolicy", "origin");
                if (configs.autoReload == 1) {
                    img.dataset.errorNum = 0;
                    img.onerror = error => {
                        const num = Number(img.dataset.errorNum);
                        if (num < 10) {
                            error.target.src = img_loading_bak;
                            error.target.dataset.errorNum = num + 1;
                            setTimeout(() => {
                                error.target.src = error.target.dataset.src;
                            }, 1000);
                        } else {
                            error.target.classList.add("error");
                            error.target.src = img_error_bak;
                        }
                    };
                }
                img.src = img_loading_bak;
                img.dataset.src = src;
                img.dataset.chapterId = data.id;
                imagesObserver.observe(img);
                return img;
            });
            //mainContent.append(...imgs);
            fragment.append(...imgs);
            mainContent.append(fragment);
            nextObserver.observe(imgs.at(-1));
            if (configs.preload == 1) {
                singleThreadLoadImgs(imgs);
            }
        };

        const preloadNext = async (mid, cid) => {
            const data = await getReadData(mid, cid);
            if (data != "ERROR" && isObject(data)) {
                if (isArray(data.scans)) {
                    const srcs = data.scans.map(obj => {
                        let src;
                        if (configs.highQuality == 1) {
                            src = obj.url.replace(/\?q=\d+$/, "");
                        } else {
                            src = obj.url;
                        }
                        return src;
                    });
                    singleThreadLoadSrcs(srcs);
                }
            }
        };

        const infiniteScroll = async () => {
            if (allChapterListData[currentChapterIndex + 1] === undefined) {
                //alert("已閱讀完最後一話了");
                hiddenElementArray.forEach(e => (e.style.display = ""));
                return;
            } else {
                const nextChapterData = allChapterListData[currentChapterIndex + 1];
                console.log("下一章節的列表資料", nextChapterData);
                const nextDataJSon = await getReadData(mangaCode, nextChapterData.id, 1);
                if (nextDataJSon == "ERROR") {
                    alert(i18n.tips.apiError);
                    return;
                } else if (isObject(nextDataJSon)) {
                    console.log("下一章節的閱讀資料", nextDataJSon);
                    createPageElement(nextDataJSon);
                    if (configs.history == 1) {
                        addBrowsingHistory(nextDataJSon);
                    }
                    const h6 = ge("#root h6");
                    if (isEle(h6)) {
                        h6.innerText = nextDataJSon.chapter_name;
                    }
                    const [nextDiv, , prevDiv] = gae("footer div");
                    const nextA = ge("a", nextDiv);
                    const prevA = ge("a", prevDiv);
                    const nextChapterData = allChapterListData[currentChapterIndex + 1];
                    if (nextChapterData === undefined) {
                        const nextUrl = "/manga/readMore/" + mangaCode;
                        nextChapterUrl = null;
                        if (isEle(nextA)) {
                            nextA.href = nextUrl;
                        }
                    } else {
                        const nextUrl = "/mangaread/" + mangaCode + "/" + nextChapterData.id;
                        nextChapterUrl = nextUrl;
                        if (isEle(nextA)) {
                            nextA.href = nextUrl;
                        }
                        if (configs.preload == 1) {
                            preloadNext(mangaCode, nextChapterData.id);
                        }
                    }
                    const prevChapterData = allChapterListData[currentChapterIndex - 1];
                    const prevUrl = "/mangaread/" + mangaCode + "/" + prevChapterData.id;
                    prevChapterUrl = prevUrl;
                    if (isEle(prevA)) {
                        prevA.href = prevUrl;
                    }
                    const pagerTitles = gae(".chapterTitle");
                    if (pagerTitles.length > 3) {
                        const parentE = pagerTitles[0].parentNode;
                        pagerTitles[0].remove();
                        const nodes = [...parentE.childNodes];
                        for (let i = 0; i < nodes.length; i++) {
                            if (nodes[i].className === "chapterTitle") {
                                break;
                            }
                            nodes[i].remove();
                        }
                    }
                }
            }
        };

        try {
            //取得所有章節列表資料
            const allChapterDataRes = await fetch(`/apis/m/mcsmmss?code=${mangaCode}`, getHeaders());
            const allChapterDataJson = await allChapterDataRes.json();
            if (allChapterDataJson?.msg !== "success") {
                console.error("取得所有章節列表資料錯誤");
                return;
            }
            allChapterListData = allChapterDataJson.data;
            lastChapterIndex = allChapterListData.length - 1;
            console.log("所有章節列表資料", allChapterListData);
            allChapterListData.some((obj, i) => {
                if (obj.id == currentChapterId) {
                    currentChapterIndex = i;
                    const currentChapterData = obj;
                    console.log("初始當前章節的列表資料", currentChapterData);
                    console.log("初始當前章節的列表資料索引", currentChapterIndex);
                    return true;
                } else {
                    return false;
                }
            });
        } catch (error) {
            console.error("取得所有章節列表資料錯誤", error);
            return;
        }

        const readData = await getReadData(mangaCode, currentChapterId);
        if (readData == "ERROR") {
            alert(i18n.tips.apiError);
            return;
        } else if (isObject(readData)) {
            console.log("當前章節閱讀資料", readData);
            gae("article").slice(0, -1).forEach((e, i) => {
                e.style.display = "none";
                if (i === 1) {
                    hiddenElementArray.push(e);
                }
            });
            gae("#root>div>div").forEach(e => {
                e.style.display = "none";
                hiddenElementArray.push(e);
            });
            const firstE = ge("article")?.firstElementChild;
            if (isEle(firstE) && !firstE?.id?.startsWith("imageLoader")) {
                const targetElement = ge("article");
                targetElement.insertAdjacentElement("beforebegin", firstE.cloneNode(true));
            }
            createPageElement(readData, 1);
            const nextChapterData = allChapterListData[currentChapterIndex + 1];
            if (nextChapterData !== undefined && configs.preload == 1) {
                preloadNext(mangaCode, nextChapterData.id);
            }

            const button = document.createElement("button");
            button.id = "open-comments";
            button.innerText = i18n.button.openComments;
            Object.assign(button.style, {
                fontSize: "1rem",
                color: "#fff",
                borderStyle: "solid",
                borderColor: "#673ab7",
                backgroundColor: "#673ab7",
                borderRadius: ".5rem",
                left: "24px",
                right: "auto",
                top: "auto",
                bottom: "36px",
                position: "fixed",
                zIndex: "9999",
                display: "none"
            });
            document.body.append(button);
            button.addEventListener("click", () => createComments());

            let lastScrollTop = 0;
            document.addEventListener("scroll", event => {
                let st = event.srcElement.scrollingElement.scrollTop;
                if (st > lastScrollTop) {
                    button.style.display = "none";
                    lastScrollTop = st;
                } else if (st < lastScrollTop - 40) {
                    button.style.left = (ge(".MuiContainer-root").offsetLeft + 24) + "px";
                    button.style.display = "";
                    lastScrollTop = st;
                }
            });

        }
    }

})();