嗨皮漫畫閱讀輔助

增加一些輔助閱讀功能(自用)。

// ==UserScript==
// @name               嗨皮漫畫閱讀輔助
// @name:zh-CN         嗨皮漫画阅读辅助
// @version            2.5.1
// @description        增加一些輔助閱讀功能(自用)。
// @description:zh-CN  增加一些辅助阅读功能(自用)。
// @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("body.no-js")) return; //Cloudflare檢測連線安全性時,不運行腳本

    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 _unsafeWindow = unsafeWindow ?? window;
    const isLogged = document.cookie.startsWith("sf_token");
    const language = _unsafeWindow.navigator.language;
    const lp = _unsafeWindow.location.pathname;
    const isReadPage = /^\/reads\/\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 hasTouchEvents = (() => ("ontouchstart" in window) || (navigator.maxTouchPoints > 0) || (navigator.msMaxTouchPoints > 0))();
    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 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 addGlobalStyle = css => {
        const style = document.createElement("style");
        style.type = "text/css";
        style.innerHTML = css;
        document.head.appendChild(style);
    };

    const loadMore = selector => {
        const loadMoreButton = ge(selector);
        if (hasTouchEvents) {
            const dispatchTouchEvent = (ele, type) => {
                const touchEvent = document.createEvent("UIEvent");
                touchEvent.initUIEvent(type, true, true);
                touchEvent.touches = [{
                    clientX: 1,
                    clientY: 1
                }];
                ele.dispatchEvent(touchEvent);
            };
            dispatchTouchEvent(loadMoreButton, "touchstart");
            dispatchTouchEvent(loadMoreButton, "touchend");
            console.log("嗨皮漫畫模擬觸控點擊");
            //loadMoreButton.dispatchEvent(new Event("touchstart"));
            //loadMoreButton.dispatchEvent(new Event("touchend"));
        } else {
            loadMoreButton.click();
            console.log("嗨皮漫畫模擬點擊");
        }
    };

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

    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.appendChild(preloadDiv);
        }
        const [, , mangaCode, id] = pn.split("/");
        const apiUrl = `/v2.0/apis/manga/read?code=${mangaCode}&cid=${id}&v=v2.13`;
        fetch(apiUrl, {
            "headers": {
                "accept": "application/json, text/plain, */*",
                "x-requested-id": new Date().getTime(),
                "x-requested-with": "XMLHttpRequest"
            }
        }).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.src = scan.url;
                        img.setAttribute("referrerpolicy", "origin");
                        img.alt = jsonData.data.chapter_name;
                        preloadDiv.appendChild(img);
                        await new Promise(resolve => setTimeout(resolve, 200));
                    }
                } else if (jsonData.status == 403) {
                    console.log(text + "獲取數據失敗\n", jsonData);
                }
            } catch (error) {
                console.error(error);
            }
        }).catch(error => console.error(error));
    };

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

    const defaultConfigs = { //1開、0關
        arrowKey: 1, //鍵盤左右方向鍵切換章節。
        doubleClick: 1, //雙擊前往下一話,方便手機使用。
        preload: 1, //閱讀頁預讀全部圖片,並且嘗試預讀下一話圖片。
        autoNext: 0, //下一話按鈕完全進入視窗可視範圍內時自動下一話。
        autoLoadMore: 1, //更新頁自動點擊載入更多。
        autoShowAll: 1, //目錄頁自動展開全部章節。
        openInNewTab: 1, //新分頁打開漫畫鏈接。
        infiniteScroll: 0, //無限滾動閱讀模式。
        history: 1 //無限滾動API請求成功後添加瀏覽器歷史。
    };

    let configs = GM_getValue("configs", defaultConfigs);
    configs = {
        ...defaultConfigs,
        ...configs
    };
    //console.log("configs", configs);

    const configCSS = `
#configElement {
    text-align: center;
    width: 300px;
    height: auto;
    position: fixed;
    top: 20%;
    left: 50%;
    margin-left: -150px;
    border: 1px solid #a0a0a0;
    border-radius: 3px;
    box-shadow: -2px 2px 5px rgb(0 0 0 / 30%);
    background-color: #FAFAFB;
    z-index: 10;
}
#configElement div,
#configElement label,
#configElement button {
    font-family: Arial, sans-serif;
    font-size: 14px;
    color: black;
    float: none;
    line-height: 18px;
}
#configElement .title {
    width: 100%;
}
#configElement .item {
    width: 348px;
    display: flex;
}
#configElement div {
    margin-bottom: 4px;
    padding: 1px 4px;
}
#configElement input[type=checkbox] {
    width: 14px;
    margin: 0 6px;
}
#configElement 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;
}
`;
    addGlobalStyle(configCSS);

    const createConfigElement = () => {
        let configLanguage;
        if (scriptLanguage === "TW") {
            configLanguage = {
                title: "嗨皮漫畫閱讀輔助設定",
                arrowKey: "左右方向鍵切換章節",
                doubleClick: "雙擊前往下一話",
                preload: "背景預讀圖片",
                autoNext: "自動前往下一話",
                autoLoadMore: "更新頁自動點擊載入更多",
                autoShowAll: "目錄頁自動展開全部章節",
                openInNewTab: "新分頁打開漫畫鏈結",
                infiniteScroll: "啟用無限滾動閱讀模式(極不穩定)",
                history: "無限滾動添加瀏覽器歷史紀錄",
                cancel: "取消",
                reset: "重置設定",
                save: "保存設定",
                apiError: "API請求失敗,伺服器拒絕連線,也可能是需要再次Cloudflare人機驗證。"
            };
        } else if (scriptLanguage === "CN") {
            configLanguage = {
                title: "嗨皮漫画阅读辅助设置",
                arrowKey: "左右方向键切换章节",
                doubleClick: "双击前往下一话",
                preload: "后台预读图片",
                autoNext: "自动前往下一话",
                autoLoadMore: "更新页自动点击加载更多",
                autoShowAll: "目录页自动展开全部章节",
                openInNewTab: "新标籤页打开漫画链结",
                infiniteScroll: "启用无限滚动阅读模式(极不稳定)",
                history: "无限滚动添加浏览器历史纪录",
                cancel: "取消",
                reset: "重置设置",
                save: "保存设置",
                apiError: "API请求失败,服务器拒绝连接数,也可能是需要再次Cloudflare人机验证。"
            };
        } else {
            configLanguage = {
                title: "settings",
                arrowKey: "Arrow keys to switch chapters",
                doubleClick: "Double click to go to the next chapter",
                preload: "Background preload image",
                autoNext: "Auto go to the next chapter",
                autoLoadMore: "Update page auto loads more",
                autoShowAll: "Contents page auto expands all chapters",
                openInNewTab: "Open the comic link in a new tab",
                infiniteScroll: "Turn on infinite scroll reading mode",
                history: "Infinite scroll add browser history",
                cancel: "Cancel",
                reset: "Reset",
                save: "Save",
                apiError: "The API request failed and the server refused to connect. It may also require Cloudflare human-machine verification again."
            };
        }
        const main = document.createElement("div");
        main.id = "configElement";
        const mainHtmlStr = `
<div class="title" style="width: 100%;">
    ${configLanguage.title}
</div>
<div class="item">
    <input id="arrowKeyInput" type="checkbox">
    <label>${configLanguage.arrowKey}</label>
</div>
<div class="item">
    <input id="doubleClickInput" type="checkbox">
    <label>${configLanguage.doubleClick}</label>
</div>
<div class="item">
    <input id="preloadInput" type="checkbox">
    <label>${configLanguage.preload}</label>
</div>
<div class="item">
    <input id="autoNextInput" type="checkbox">
    <label>${configLanguage.autoNext}</label>
</div>
<div class="item">
    <input id="autoLoadMoreInput" type="checkbox">
    <label>${configLanguage.autoLoadMore}</label>
</div>
<div class="item">
    <input id="autoShowAllInput" type="checkbox">
    <label>${configLanguage.autoShowAll}</label>
</div>
<div class="item">
    <input id="openInNewTabInput" type="checkbox">
    <label>${configLanguage.openInNewTab}</label>
</div>
<div class="item">
    <input id="infiniteScrollInput" type="checkbox">
    <label>${configLanguage.infiniteScroll}</label>
</div>
<div class="item">
    <input id="historyInput" type="checkbox">
    <label>${configLanguage.history}</label>
</div>
<button id="cancelBtn">${configLanguage.cancel}</button>
<button id="resetBtn">${configLanguage.reset}</button>
<button id="saveBtn">${configLanguage.save}</button>
`;
        main.innerHTML = mainHtmlStr;
        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("#autoNextInput", main).checked = configs.autoNext == 1 ? true : false;
        ge("#autoLoadMoreInput", main).checked = configs.autoLoadMore == 1 ? true : false;
        ge("#autoShowAllInput", main).checked = configs.autoShowAll == 1 ? true : false;
        ge("#openInNewTabInput", main).checked = configs.openInNewTab == 1 ? true : false;
        ge("#infiniteScrollInput", main).checked = configs.infiniteScroll == 1 ? true : false;
        ge("#historyInput", main).checked = configs.history == 1 ? true : false;
        ge("#cancelBtn", main).addEventListener("click", event => {
            event.preventDefault();
            main.remove();
        });
        ge("#resetBtn", main).addEventListener("click", event => {
            event.preventDefault();
            main.remove();
            GM_deleteValue("configs");
            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.autoNext = ge("#autoNextInput", main).checked == true ? 1 : 0;
            configs.autoLoadMore = ge("#autoLoadMoreInput", main).checked == true ? 1 : 0;
            configs.autoShowAll = ge("#autoShowAllInput", main).checked == true ? 1 : 0;
            configs.openInNewTab = ge("#openInNewTabInput", main).checked == true ? 1 : 0;
            configs.infiniteScroll = ge("#infiniteScrollInput", main).checked == true ? 1 : 0;
            configs.history = ge("#historyInput", main).checked == true ? 1 : 0;
            main.remove();
            GM_setValue("configs", configs);
            location.reload();
        });
        document.body.appendChild(main);
    };

    GM_registerMenuCommand((() => {
        const obj = {
            "TW": "設定",
            "CN": "设置",
            "EN": "settings"
        };
        return obj[scriptLanguage];
    })(), () => createConfigElement());

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

    if (configs.autoLoadMore == 1 && isUpdatePage) {
        new IntersectionObserver(entries => {
            if (entries[0].isIntersecting) {
                loadMore(".more-div-btn");
                console.log("載入更多");
            }
        }).observe(ge(".more-div-btn"));
    }

    if (configs.autoShowAll && 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 (event.code === "ArrowRight") {
                const nextE = ge("//a[span[text()='下一话' or text()='下一話'] and contains(@href,'reads')]");
                if (nextE) {
                    _unsafeWindow.location.href = nextE.href;
                } else {
                    alert("沒有下一话了!");
                }
            }
            if (event.code === "ArrowLeft") {
                const prevE = ge("//a[span[text()='上一话' or text()='上一話'] and contains(@href,'reads')]");
                if (prevE) {
                    _unsafeWindow.location.href = prevE.href;
                } else {
                    alert("沒有上一话了!");
                }
            }
        });
    }

    if (configs.doubleClick == 1 && isReadPage && configs.infiniteScroll != 1) {
        document.addEventListener("dblclick", () => {
            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][contains(@href,'reads')]");
            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);
                let item = ge("footer>article>div:nth-child(2)");
                item.querySelectorAll("a").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("/reads/")) {
                    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 contains(@href,'reads')]");
                    if (nextE) {
                        _unsafeWindow.location.href = nextE.href;
                    }
                }, 1000);
            } else {
                clearTimeout(timeId);
            }
        }, {
            threshold: 1,
        }).observe(ge("footer a"));
    }

    if (isReadPage && configs.infiniteScroll != 1) {
        new MutationObserver((mutationsList, observer) => {
            //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=v2.13
        const infiniteScrollCss = `
.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;
}
        `;
        addGlobalStyle(infiniteScrollCss);

        let i18n;
        if (scriptLanguage === "TW") {
            i18n = {
                apiError: "API請求失敗,伺服器拒絕連線,也可能是需要再次Cloudflare人機驗證。"
            };
        } else if (scriptLanguage === "CN") {
            i18n = {
                apiError: "API请求失败,服务器拒绝连接数,也可能是需要再次Cloudflare人机验证。"
            };
        } else {
            i18n = {
                apiError: "The API request failed and the server refused to connect. It may also require Cloudflare human-machine verification again."
            };
        }

        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 currentChapterData;
        let currentChapterIndex = 0;
        let infiniteScrollSwitch = true;

        const imagesObserver = new IntersectionObserver((entries, observer) => {
            entries.forEach(entry => {
                if (entry.isIntersecting) {
                    observer.unobserve(entry.target);
                    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;
                    }
                }
            });
        });

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

        const getReadData = async (mid, cid, isNext = 0) => {
            try {
                const res = await fetch(`/v2.0/apis/manga/read?code=${mid}&cid=${cid}&v=v2.13`, {
                    "headers": {
                        "accept": "application/json, text/plain, */*",
                        "x-requested-id": new Date().getTime(),
                        "x-requested-with": "XMLHttpRequest"
                    }
                });
                const readJson = await res.json();
                if (readJson?.msg !== "success") {
                    console.error("取得章節資料失敗", readJson);
                    return "ERROR";
                }
                if (isNext == 1) {
                    if (isLogged) {
                        fetch(`/v2.0/apis/uu/readLog?cid=${cid}&code=${mid}`);
                    }
                    currentChapterIndex += 1;
                }
                return readJson.data;
            } catch (error) {
                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.onload = () => {
                        imgArr[i].src = loadSrc;
                        resolve();
                    }
                    temp.onerror = resolve();
                    temp.setAttribute("referrerpolicy", "origin");
                    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 => {
            let title = data.manga_name + " - " + data.chapter_name + "——嗨皮漫画";
            let url = document.URL.replace(/\d+$/, data.id);
            history.pushState(null, title, url);
            document.title = title;
        };

        const createPageElement = (data, isFirst = 0) => {
            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;
                mainContent.appendChild(title);
            }
            const imgs = data.scans.map(obj => {
                const img = new Image();
                img.className = "images";
                img.setAttribute("errornum", "0");
                img.setAttribute("referrerpolicy", "origin");
                img.onerror = error => {
                    let num = Number(error.target.getAttribute("errornum"));
                    if (num < 10) {
                        error.target.src = img_loading_bak;
                        error.target.setAttribute("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 = obj.url;
                imagesObserver.observe(img);
                return img;
            });
            mainContent.append(...imgs);
            singleThreadLoadImgs(imgs);
            nextObserver.observe(imgs.at(-1));
        };

        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 => obj.url);
                    singleThreadLoadSrcs(srcs);
                }
            }
        };

        const infiniteScroll = async () => {
            if (allChapterData[currentChapterIndex + 1] === undefined) {
                //alert("已閱讀完最後一話了");
                return;
            } else {
                const nextChapterData = allChapterData[currentChapterIndex + 1];
                console.log("下一章節的目錄資料", nextChapterData);
                const nextDataJSon = await getReadData(mangaCode, nextChapterData.id, 1);
                if (nextDataJSon == "ERROR") {
                    alert(i18n.apiError);
                    return;
                } else if (isObject(nextDataJSon)) {
                    console.log("下一章節JSon", 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);
                    if (isEle(nextA)) {
                        const nextChapterData = allChapterData[currentChapterIndex + 1];
                        if (nextChapterData === undefined) {
                            nextA.href = "/manga/readMore/" + mangaCode;
                        } else {
                            nextA.href = "/reads/" + mangaCode + "/" + nextChapterData.id;
                            if (configs.preload == 1) {
                                preloadNext(mangaCode, nextChapterData.id);
                            }
                        }
                    }
                    if (isEle(prevA)) {
                        const prevChapterData = allChapterData[currentChapterIndex - 1];
                        prevA.href = "/reads/" + mangaCode + "/" + prevChapterData.id;
                    }
                    const pagerTitles = gae(".chapterTitle");
                    if (pagerTitles.length > 3) {
                        const parentE = pagerTitles[0].parentNode;
                        pagerTitles[0].remove();
                        const eles = [...parentE.childNodes];
                        for (let i = 0; i < eles.length; i++) {
                            if (eles[i].className === "chapterTitle") {
                                break;
                            }
                            eles[i].remove();
                        }
                    }
                }
            }
        };

        //取得所有章節資料
        let allChapterData;
        try {
            const allChapterDataRes = await fetch(`/apis/m/mcsmmss?code=${mangaCode}`);
            const allChapterDataJson = await allChapterDataRes.json();
            if (allChapterDataJson?.msg !== "success") {
                console.error("取得所有章節資料失敗");
                return;
            }
            allChapterData = allChapterDataJson.data;
            console.log("章節目錄總資料Json", allChapterDataJson);
            allChapterData.some((obj, i) => {
                if (obj?.id == currentChapterId) {
                    currentChapterIndex = i;
                    currentChapterData = obj;
                    console.log("當前章節的目錄資料", currentChapterData);
                    console.log("當前章節的目錄資料索引", currentChapterIndex);
                    return true;
                } else {
                    return false;
                }
            });
        } catch (error) {
            console.error("取得所有章節目錄資料錯誤", error);
            return;
        }

        let readData = await getReadData(mangaCode, currentChapterId);
        if (readData == "ERROR") {
            alert(i18n.apiError);
            return;
        } else if (isObject(readData)) {
            console.log("當前章節Json", readData);
            gae("article").slice(0, -1).forEach(e => (e.style.display = "none"));
            gae("#root>div>div").forEach(e => e.remove());
            createPageElement(readData, 1);
            const nextChapterData = allChapterData[currentChapterIndex + 1];
            if (nextChapterData !== undefined) {
                if (configs.preload == 1) {
                    preloadNext(mangaCode, nextChapterData.id);
                }
            }
        }
    }

})();