// ==UserScript==
// @name 嗨皮漫畫閱讀輔助
// @name:en Happymh reading aid
// @name:zh-CN 嗨皮漫画阅读辅助
// @name:zh-TW 嗨皮漫畫閱讀輔助
// @version 2.6.6
// @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 = /^\/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 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=v2.13`;
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,'/reads/')]");
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,'/reads/')]");
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,'/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);
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("/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 starts-with(@href,'/reads/')]");
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=v2.13
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=v2.13`, 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);
});
};
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 = "/reads/" + 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 = "/reads/" + 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;
}
});
}
}
})();