Skrip ini tidak untuk dipasang secara langsung. Ini adalah pustaka skrip lain untuk disertakan dengan direktif meta // @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;
});
}
}
})();