אין להתקין סקריפט זה ישירות. זוהי ספריה עבור סקריפטים אחרים // @require https://update.greasyfork.org/scripts/520359/1500544/test_%E5%97%A8%E7%9A%AE.js
- (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: 0, //目錄頁自動展開全部章節。
- 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\//.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.includes("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 chapterCode = pn.split("/").at(-1);
- const apiUrl = `/v2.0/apis/manga/reading?code=${chapterCode}&v=v3.1818134`;
- 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) {
- addGlobalStyle(`
- div:has(>#page-area) {
- min-height: auto !important;
- max-height: max-content !important;
- overflow: auto !important;
- }
- `);
- const removeElement = () => {
- const removeSelectors = [
- "noscript",
- "iframe",
- ".adsbygoogle",
- "#google_pedestal_container",
- "#root>div>div:has(>a)",
- "//div[text()='Done']",
- "#notice-react",
- "#alert-confirm-react"
- ];
- remove(removeSelectors);
- document.body.style.filter = "";
- };
- 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);
- let button = ge("//div[contains(text(),'给本王显示全部章节')]");
- if (button) {
- button.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[text()='下一话' or text()='下一話'][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[text()='上一话' or text()='上一話'][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("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
- if (nextE) {
- _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("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
- if (nextE) {
- preload(nextE.pathname, "嗨皮漫畫下一話數據\n");
- }
- }, 3000);
- }
-
- if (isReadPage) {
- const selector = "#root footer";
- await waitEle(selector);
- new IntersectionObserver((entries, observer) => {
- if (entries[0].isIntersecting) {
- const nextA = ge("//a[text()='下一话' or text()='下一話']");
- const prevA = ge("//a[text()='上一话' or text()='上一話']");
- if (prevA?.href?.includes("/mangaread/")) {
- prevA.style.color = "rgb(255, 255, 255)";
- prevA.style.backgroundColor = "rgb(103, 58, 183)";
- }
- if (nextA?.href?.includes("/readMore/")) {
- nextA.style.color = "rgb(33, 33, 33)";
- nextA.style.backgroundColor = "rgb(245, 245, 245)";
- nextA.innerText = "^_^感谢您的阅读~已经没有下一话了哦~";
- }
- }
- }).observe(ge(selector));
- }
-
- if (configs.autoNext == 1 && isReadPage) {
- await waitEle("#root footer button");
- let observeE;
- const divs = gae("#root footer>article>div");
- if (divs.length == 1) {
- observeE = ge("#root footer article");
- } else if (divs.length == 2) {
- const a = ge("a", divs[1]);
- if (a) {
- observeE = a;
- }
- } else {
- observeE = ge("#root footer article");
- }
- if (observeE) {
- let timeId;
- new IntersectionObserver(entries => {
- if (entries[0].isIntersecting) {
- timeId = setTimeout(() => {
- let nextE = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
- if (nextE) {
- _unsafeWindow.location.href = nextE.href;
- }
- }, configs.autoNextSec * 1000);
- } else {
- clearTimeout(timeId);
- }
- }, {
- threshold: 0.6,
- }).observe(observeE);
- }
- }
-
- 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) {
- await waitEle("div[id^=imageLoader] img[id^=scan]");
- if (!("_ht" in localStorage)) return;
- //所有章節資料API
- //https://m.happymh.com/apis/m/mcsmmss?code=漫畫代碼
- //章節評論API
- //https://m.happymh.com/v2.0/apis/comment?code=漫畫代碼&ch_id=章節ID&pn=頁數&order=time&from=read
- 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);
-
- let currentData = {};
- const img_loading_bak = "";
- const img_error_bak = "";
- const api_loading_gif = "";
-
- let [localStorageHistory] = JSON.parse(localStorage.getItem("_ht"));
- const mangaCode = localStorageHistory.serie_code;
- let chapterCode = localStorageHistory.read_chapter_codes;
- let currentChapterId = localStorageHistory.read_chapter_id;
-
- let currentViewChapterId = currentChapterId;
- 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 (cc, isNext = 0) => {
- let loading;
- if (isNext == 1) {
- loading = createLoadingElement();
- }
- try {
- const res = await fetch(`/v2.0/apis/manga/reading?code=${cc}&v=v3.1818134`, 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", {
- "headers": {
- "accept": "application/json, text/plain, */*",
- "x-requested-id": new Date().getTime(),
- "x-requested-with": "XMLHttpRequest"
- },
- "body": `code=${readJson.data.manga_code}&cid=${readJson.data.id}`,
- "method": "POST"
- });
- }
- loading?.remove();
- }
- 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, id) => {
- const title = data.manga_name + " - " + data.chapter_name + " - 嗨皮漫画";
- const url = "https://m.happymh.com/mangaread/" + 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 topButton = document.createElement("button");
- topButton.className = "close-comments";
- topButton.innerText = i18n.button.closeComments;
- topButton.style.marginTop = "10px";
- topButton.style.marginLeft = "10px";
- topButton.addEventListener("click", () => {
- div.remove();
- isOpenComments = false;
- });
-
- div.insertAdjacentElement("beforeend", topButton);
- const messageHtml = `
- <div id="message" class="MuiCardContent-root" style="padding: 3rem 16px; display: flex; flex-direction: column; -webkit-box-pack: center; justify-content: center; -webkit-box-align: center; align-items: center; text-align: center; min-height: 260px;width: 100%; background-color: rgb(255, 255, 255);">
- <svg class="MuiSvgIcon-root MuiSvgIcon-colorAction MuiSvgIcon-fontSizeMedium" focusable="false" viewBox="0 0 24 24" aria-hidden="true" style="user-select: none; width: 1em;height: 1em; display: inline-block; fill: currentcolor;flex-shrink: 0; font-size: 1.5rem; color: rgba(0, 0, 0, 0.54); transition: fill 200ms cubic-bezier(0.4, 0, 0.2, 1);">
- <path d="M21.99 2H2v16h16l4 4-.01-20zM18 14H6v-2h12v2zm0-3H6V9h12v2zm0-3H6V6h12v2z"></path>
- </svg>
- <h6 class="MuiTypography-root MuiTypography-h6" style="margin: 0px; font-family: Roboto, Helvetica, Arial, sans-serif; font-weight: 500; font-size: 1.25rem; line-height: 1.6; letter-spacing: 0.0075em;">数据请求中...</h6>
- </div>`;
- div.insertAdjacentHTML("beforeend", messageHtml);
- div.insertAdjacentHTML("beforeend", '<ul class="MuiList-root MuiList-padding" style="display: block; 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="font-weight: normal; 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="display: block; padding: 0 10px 0 0;">
- <div class="MuiListItemText-root MuiListItemText-multiline" style="flex: 1 1 auto; min-width: 0px; margin-top: 6px; margin-bottom: 6px; font-weight: bolder; color: rgba(0, 0, 0, 0.87);">
- <span class="MuiTypography-root MuiTypography-body1 MuiTypography-displayBlock" style="margin: 0px; font-family: Roboto, Helvetica, Arial, sans-serif; font-size: 1rem; line-height: 1.5; letter-spacing: 0.00938em; display: block; font-weight: bolder; color: rgba(0, 0, 0, 0.87);">${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" style="margin: 0px; font-family: Roboto, Helvetica, Arial, sans-serif; font-weight: 400; font-size: 0.75rem; line-height: 1.66; letter-spacing: 0.03333em; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; color: rgba(0, 0, 0, 0.6);">章节: ${item.ch_name}</span>
- <br>
- <span class="MuiTypography-root MuiTypography-caption MuiTypography-colorTextSecondary" style="margin: 0px; font-family: Roboto, Helvetica, Arial, sans-serif; font-weight: 400; font-size: 0.75rem; line-height: 1.66; letter-spacing: 0.03333em; color: rgba(0, 0, 0, 0.6);">${item.create_time}</span>
- </div>
- <div class="MuiBox-root">
- <p class="MuiTypography-root MuiTypography-body1" style="margin: 0px; font-family: Roboto, Helvetica, Arial, sans-serif; font-weight: 400; font-size: 1rem; line-height: 1.5; letter-spacing: 0.00938em; 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 bottomButton = document.createElement("button");
- bottomButton.className = "close-comments";
- bottomButton.innerText = i18n.button.closeComments;
- bottomButton.style.marginBottom = "100px";
- bottomButton.style.marginLeft = "10px";
- bottomButton.addEventListener("click", () => {
- div.remove();
- isOpenComments = false;
- });
- div.insertAdjacentElement("beforeend", bottomButton);
- };
-
- 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 (srcs.length == 2 && ("next_cid" in data)) {
- srcs = srcs.slice(0, -1);
- }
- if (srcs.length > 2 && ("next_cid" in data)) {
- 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 (cc) => {
- const data = await getReadData(cc);
- 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 ("next_cid" in currentData) {
- const cid = currentData.next_cid;
- const nextDataJSon = await getReadData(cid, 1);
- if (nextDataJSon == "ERROR") {
- alert(i18n.tips.apiError);
- return;
- } else if (isObject(nextDataJSon)) {
- console.log("下一章節的閱讀資料", nextDataJSon);
- currentData = nextDataJSon;
- createPageElement(currentData);
- if (configs.history == 1) {
- addBrowsingHistory(currentData, cid);
- }
- const h6 = ge("#root h6");
- if (isEle(h6)) {
- h6.innerText = currentData.chapter_name;
- }
- const nextA = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
- const prevA = ge("//a[text()='上一话' or text()='上一話'][starts-with(@href,'/mangaread/')]");
- if ("next_cid" in currentData) {
- const nextUrl = "/mangaread/" + currentData.next_cid;
- nextChapterUrl = nextUrl;
- if (isEle(nextA)) {
- nextA.href = nextUrl;
- }
- if (configs.preload == 1) {
- preloadNext(currentData.next_cid);
- }
- } else {
- const nextUrl = "/manga/readMore/" + mangaCode;
- nextChapterUrl = null;
- if (isEle(nextA)) {
- nextA.href = nextUrl;
- }
- }
- if ("pre_cid" in currentData) {
- const prevUrl = "/mangaread/" + currentData.pre_cid;
- 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();
- }
- }
- }
- } else {
- hiddenElementArray.forEach(e => (e.style.display = ""));
- }
- };
-
- const readData = await getReadData(chapterCode);
- 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));
- }
- currentData = readData;
- createPageElement(currentData, 1);
- if ("next_cid" in currentData) {
- preloadNext(currentData.next_cid);
- }
-
- const commentsButton = document.createElement("button");
- commentsButton.id = "open-comments";
- commentsButton.innerText = i18n.button.openComments;
- Object.assign(commentsButton.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(commentsButton);
- commentsButton.addEventListener("click", () => createComments());
-
- // 创建 "下一话" 按钮
- const nextChapterButton = document.createElement("button");
- nextChapterButton.id = "next-chapter";
- nextChapterButton.innerText = "下一話"; // 或使用 i18n.button.nextChapter
- Object.assign(nextChapterButton.style, {
- fontSize: "1rem",
- color: "#fff",
- borderStyle: "solid",
- borderColor: "#673ab7",
- backgroundColor: "#673ab7",
- borderRadius: ".5rem",
- left: "24px",
- right: "auto",
- top: "auto",
- bottom: "5px",
- position: "fixed",
- zIndex: "9999",
- display: "none"
- });
- document.body.append(nextChapterButton);
- nextChapterButton.addEventListener("click", () => {
- // 使用 XPath 查找下一話的鏈接
- const nextChapterLink = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
-
- if (nextChapterLink) {
- // 如果找到下一話鏈接,跳轉到下一話
- window.location.href = nextChapterLink.href;
- } else {
- // 沒有找到下一話,顯示提示
- alert(i18n.tips.noNext || "沒有下一話了!");
- }
- });
-
-
-
- let lastScrollTop = 0;
- let lastScrollTopNextChapter = 0; // 定义两个滚动状态变量
-
- // 监听滚动事件
- document.addEventListener("scroll", event => {
- let st = event.srcElement.scrollingElement.scrollTop;
-
- // 控制 "评论按钮"
- if (st > lastScrollTop) {
- commentsButton.style.display = "none";
- } else if (st < lastScrollTop - 40) {
- commentsButton.style.left = (ge(".MuiContainer-root").offsetLeft + 24) + "px";
- commentsButton.style.display = "";
- }
- lastScrollTop = st;
-
- // 控制 "下一话按钮"
- if (st > lastScrollTopNextChapter) {
- nextChapterButton.style.display = "none";
- } else if (st < lastScrollTopNextChapter - 40) {
- nextChapterButton.style.left = (ge(".MuiContainer-root").offsetLeft + 24) + "px";
- nextChapterButton.style.display = "";
- }
- lastScrollTopNextChapter = st;
- });
- }
- }
- })();