Greasy Fork is available in English.

test_嗨皮

test

אין להתקין סקריפט זה ישירות. זוהי ספריה עבור סקריפטים אחרים // @require https://update.greasyfork.org/scripts/520359/1500544/test_%E5%97%A8%E7%9A%AE.js

  1. (async () => {
  2. "use strict";
  3.  
  4. if (document.querySelector(".captcha-area")) {
  5. console.warn("嗨皮Cloudflare正在人機驗證中");
  6. return;
  7. }
  8.  
  9. const defaultConfigs = { //1開、0關
  10. arrowKey: 1, //鍵盤左右方向鍵切換章節。
  11. doubleClick: 0, //雙擊前往下一話,方便手機使用。
  12. preload: 1, //閱讀頁預讀全部圖片,並且嘗試預讀下一話圖片。
  13. autoReload: 1, //重新載入出錯的圖片
  14. autoNext: 0, //下一話按鈕完全進入視口可視範圍內後自動下一話。
  15. autoNextSec: 1, //下一話按鈕完全進入視口可視範圍內後自動下一話的延遲秒數。
  16. autoShowAll: 0, //目錄頁自動展開全部章節。
  17. openInNewTab: 1, //新分頁打開漫畫鏈接。
  18. infiniteScroll: 0, //無限滾動閱讀模式。
  19. highQuality: 0, //去掉圖片鏈結的?q參數。
  20. history: 1, //無限滾動API請求成功後添加瀏覽器歷史。
  21. removeAd: 1 //移除無用元素
  22. };
  23.  
  24. const GM_configs = GM_getValue("configs", defaultConfigs);
  25. const configs = Object.assign(defaultConfigs, GM_configs);
  26. //console.log("腳本設定物件", configs);
  27.  
  28. const _unsafeWindow = unsafeWindow ?? window;
  29. const language = _unsafeWindow.navigator.language;
  30.  
  31. let scriptLanguage;
  32. switch (language) {
  33. case "zh-TW":
  34. case "zh-HK":
  35. case "zh-Hant-TW":
  36. case "zh-Hant-HK":
  37. scriptLanguage = "TW";
  38. break;
  39. case "zh":
  40. case "zh-CN":
  41. case "zh-Hans-CN":
  42. scriptLanguage = "CH";
  43. break;
  44. default:
  45. scriptLanguage = "EN";
  46. }
  47.  
  48. let i18n;
  49. switch (scriptLanguage) {
  50. case "TW":
  51. i18n = {
  52. config: {
  53. title: "嗨皮漫畫閱讀輔助設定",
  54. arrowKey: "左右方向鍵切換章節",
  55. doubleClick: "雙擊前往下一話",
  56. preload: "背景預讀圖片",
  57. autoReload: "自動重新載入出錯的圖片",
  58. autoNext: "自動下一話",
  59. autoNextSec: "自動下一話延遲(秒)",
  60. autoShowAll: "目錄頁自動展開全部章節",
  61. openInNewTab: "新分頁打開漫畫鏈結",
  62. infiniteScroll: "啟用無限滾動閱讀模式",
  63. history: "無限滾動添加瀏覽器歷史紀錄",
  64. highQuality: "無限滾動載入最高品質圖片",
  65. removeAd: "移除無用元素",
  66. exclude: "無限滾動標題文字正規表達式排除",
  67. cancel: "取消",
  68. reset: "重置設定",
  69. save: "保存設定"
  70. },
  71. tips: {
  72. noNext: "沒有下一話了!",
  73. noPrev: "沒有上一話了!",
  74. apiError: "API請求返回錯誤,伺服器拒絕連線,也可能是需要再次Cloudflare人機驗證。"
  75. },
  76. commandMenu: {
  77. settings: "設定"
  78. },
  79. button: {
  80. openComments: "開啟評論",
  81. closeComments: "關閉評論"
  82. }
  83. };
  84. break;
  85. case "CN":
  86. i18n = {
  87. config: {
  88. title: "嗨皮漫画阅读辅助设置",
  89. arrowKey: "左右方向键切换章节",
  90. doubleClick: "双击前往下一话",
  91. preload: "背景预读图片",
  92. autoReload: "自动重新加载出错的图片",
  93. autoNext: "自动下一话",
  94. autoNextSec: "自动下一话延迟(秒)",
  95. autoShowAll: "目录页自动展开全部章节",
  96. openInNewTab: "新标籤页打开漫画链结",
  97. infiniteScroll: "启用无限滚动阅读模式",
  98. highQuality: "无限滚动加载最高品质图片",
  99. history: "无限滚动添加浏览器历史纪录",
  100. removeAd: "移除无用元素",
  101. exclude: "无限滚动标题文字正则表达式排除",
  102. cancel: "取消",
  103. reset: "重置设置",
  104. save: "保存设置"
  105. },
  106. tips: {
  107. noNext: "没有下一话了!",
  108. noPrev: "没有上一话了!",
  109. apiError: "API请求返回错误,服务器拒绝连接,也可能是需要再次Cloudflare人机验证。"
  110. },
  111. commandMenu: {
  112. settings: "设置"
  113. },
  114. button: {
  115. openComments: "打开评论",
  116. closeComments: "关闭评论"
  117. }
  118. };
  119. break;
  120. default:
  121. i18n = {
  122. config: {
  123. title: "settings",
  124. arrowKey: "Arrow keys to switch chapters",
  125. doubleClick: "Double click to go to the next chapter",
  126. preload: "Background preload image",
  127. autoReload: "Auto reload image with error",
  128. autoNext: "Auto next chapter",
  129. autoNextSec: "Auto next chapter delay sec",
  130. autoShowAll: "Contents page auto expands all chapters",
  131. openInNewTab: "Open the comic link in a new tab",
  132. infiniteScroll: "Turn on infinite scroll reading mode",
  133. highQuality: "Infinite scroll loading of high quality image",
  134. history: "Infinite scroll add browser history",
  135. removeAd: "Remove useless elements",
  136. exclude: "Title Exclude RegExp",
  137. cancel: "Cancel",
  138. reset: "Reset",
  139. save: "Save",
  140. },
  141. tips: {
  142. noNext: "no next chapter",
  143. noPrev: "no prev chapter",
  144. 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."
  145. },
  146. commandMenu: {
  147. settings: "settings"
  148. },
  149. button: {
  150. openComments: "Open Comments",
  151. closeComments: "Close Comments"
  152. }
  153. };
  154. }
  155.  
  156. const lp = _unsafeWindow.location.pathname;
  157. const isReadPage = /^\/mangaread\//.test(lp);
  158. const isUpdatePage = /^\/latest$/.test(lp);
  159. const isListPage = /^\/manga\/\w+$/.test(lp);
  160. const isBookcasePage = /^\/bookcase$/.test(lp);
  161. const isRankPage = /^\/rank/.test(lp);
  162. const isUserPage = /^\/user/.test(lp);
  163. const isLogged = document.cookie.includes("sf_token");
  164. let nextChapterUrl = null;
  165. let prevChapterUrl = null;
  166.  
  167. 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"));
  168. const delay = time => new Promise(resolve => setTimeout(resolve, time));
  169. const isString = str => Object.prototype.toString.call(str) === "[object String]";
  170. const isObject = obj => Object.prototype.toString.call(obj) === "[object Object]";
  171. const isArray = arr => Object.prototype.toString.call(arr) === "[object Array]";
  172. const isEle = e => /^\[object\sHTML[a-zA-Z]*Element\]$/.test(Object.prototype.toString.call(e));
  173.  
  174. const ge = (selector, contextNode = null, dom = document) => {
  175. if (/^\//.test(selector)) {
  176. return dom.evaluate(selector, (contextNode ?? document), null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
  177. } else {
  178. return (contextNode ?? document).querySelector(selector);
  179. }
  180. };
  181.  
  182. const gae = (selector, contextNode = null, dom = document) => {
  183. if (/^\//.test(selector)) {
  184. const nodes = [];
  185. const results = dom.evaluate(selector, (contextNode ?? document), null, XPathResult.ANY_TYPE, null);
  186. let node = null;
  187. while (node = results.iterateNext()) {
  188. nodes.push(node);
  189. }
  190. return nodes;
  191. } else {
  192. return [...(contextNode ?? document).querySelectorAll(selector)];
  193. }
  194. };
  195.  
  196. const addGlobalStyle = css => {
  197. const style = document.createElement("style");
  198. style.type = "text/css";
  199. style.innerHTML = css;
  200. document.head.append(style);
  201. };
  202.  
  203. const waitEle = selector => {
  204. return new Promise(resolve => {
  205. const loop = setInterval(() => {
  206. if (!!ge(selector)) {
  207. clearInterval(loop);
  208. resolve();
  209. }
  210. }, 100);
  211. });
  212. };
  213.  
  214. const remove = obj => {
  215. if (isString(obj)) {
  216. let selector = obj;
  217. gae(selector).forEach(e => e.remove());
  218. } else if (isArray(obj)) {
  219. let selectors = obj;
  220. selectors.forEach(selector => gae(selector).forEach(e => e.remove()));
  221. }
  222. };
  223.  
  224. const getHeaders = () => {
  225. return {
  226. "headers": {
  227. "accept": "application/json, text/plain, */*",
  228. "x-requested-id": new Date().getTime(),
  229. "x-requested-with": "XMLHttpRequest"
  230. }
  231. };
  232. };
  233.  
  234. const preload = (pn, text) => {
  235. let preloadDiv = ge("#happymhPreload");
  236. if (preloadDiv) {
  237. preloadDiv.innerHTML = "";
  238. } else {
  239. preloadDiv = document.createElement("div");
  240. preloadDiv.id = "happymhPreload";
  241. preloadDiv.style.display = "none";
  242. document.body.append(preloadDiv);
  243. }
  244. const chapterCode = pn.split("/").at(-1);
  245. const apiUrl = `/v2.0/apis/manga/reading?code=${chapterCode}&v=v3.1818134`;
  246. fetch(apiUrl, getHeaders()).then(res => res.json()).then(async jsonData => {
  247. try {
  248. if (jsonData.status == 0) {
  249. console.log(text + "漫畫名稱:" + jsonData.data.manga_name + "\n章節名稱:" + jsonData.data.chapter_name + "\n章節圖片:\n", jsonData.data.scans, "\nJSON:\n", jsonData);
  250. const scans = jsonData.data.scans;
  251. for (const scan of scans) {
  252. const img = new Image();
  253. img.setAttribute("referrerpolicy", "origin");
  254. img.alt = jsonData.data.chapter_name;
  255. img.src = scan.url;
  256. preloadDiv.append(img);
  257. await delay(200);
  258. }
  259. } else if (jsonData.status == 403) {
  260. console.log(text + "獲取數據失敗\n", jsonData);
  261. }
  262. } catch (error) {
  263. console.error(error);
  264. }
  265. }).catch(error => console.error(error));
  266. };
  267.  
  268. const textExcludeRegExp = GM_getValue("exclude", "");
  269.  
  270. const createConfigElement = () => {
  271.  
  272. const mainElement = document.createElement("div");
  273. mainElement.id = "mainHappymhConfigShadowElement";
  274.  
  275. const shadow = mainElement.attachShadow({
  276. mode: "closed"
  277. });
  278.  
  279. shadow.innerHTML = `
  280. <style type="text/css">
  281. #happymhConfigElement {
  282. text-align: center;
  283. width: 300px;
  284. height: auto;
  285. position: fixed;
  286. top: calc((100% - 460px) / 2);
  287. left: calc((100% - 302px) / 2);
  288. border: 1px solid #a0a0a0;
  289. border-radius: 3px;
  290. box-shadow: -2px 2px 5px rgb(0 0 0 / 30%);
  291. background-color: #FAFAFB;
  292. z-index: 10000;
  293. }
  294.  
  295. #happymhConfigElement div,
  296. #happymhConfigElement label,
  297. #happymhConfigElement button {
  298. font-family: Arial, sans-serif;
  299. font-size: 14px;
  300. color: black;
  301. float: none;
  302. line-height: 18px;
  303. }
  304.  
  305. #happymhConfigElement .title {
  306. width: 100%;
  307. }
  308.  
  309. #happymhConfigElement div.item {
  310. width: 348px;
  311. display: flex;
  312. }
  313.  
  314. #happymhConfigElement label.select {
  315. margin: 0 5px;
  316. }
  317.  
  318. #happymhConfigElement div {
  319. margin-bottom: 4px;
  320. padding: 1px 4px;
  321. }
  322.  
  323. #happymhConfigElement input[type=checkbox] {
  324. width: 14px;
  325. margin: 0 6px;
  326. }
  327.  
  328. #happymhConfigElement button {
  329. width: auto;
  330. min-width: 80px;
  331. max-width: 100px;
  332. min-height: unset;
  333. max-height: 24px;
  334. margin-left: 2px;
  335. margin-right: 2px;
  336. margin-bottom: 4px;
  337. display: inline-block;
  338. color: #000000;
  339. border: 1px solid #a0a0a0;
  340. background-color: transparent;
  341. border-radius: unset;
  342. }
  343.  
  344. #happymhConfigElement #exclude {
  345. width: calc(100% - 12px);
  346. height: 100px;
  347. }
  348. </style>
  349. <div id="happymhConfigElement">
  350. <div class="title" style="width: calc(100% - 8px);">
  351. ${i18n.config.title}
  352. </div>
  353. <div class="item">
  354. <input id="arrowKeyInput" type="checkbox">
  355. <label>${i18n.config.arrowKey}</label>
  356. </div>
  357. <div class="item">
  358. <input id="doubleClickInput" type="checkbox">
  359. <label>${i18n.config.doubleClick}</label>
  360. </div>
  361. <div class="item">
  362. <input id="autoNextInput" type="checkbox">
  363. <label>${i18n.config.autoNext}</label>
  364. </div>
  365. <div class="item">
  366. <label class="select">${i18n.config.autoNextSec}</label>
  367. <select id="autoNextSec">
  368. ${new Array(10).fill().map((_, i) => `<option value="${i + 1}">${i + 1}</option>`).join("")}
  369. </select>
  370. </div>
  371. <div class="item">
  372. <input id="autoShowAllInput" type="checkbox">
  373. <label>${i18n.config.autoShowAll}</label>
  374. </div>
  375. <div class="item">
  376. <input id="openInNewTabInput" type="checkbox">
  377. <label>${i18n.config.openInNewTab}</label>
  378. </div>
  379. <div class="item">
  380. <input id="autoReloadInput" type="checkbox">
  381. <label>${i18n.config.autoReload}</label>
  382. </div>
  383. <div class="item">
  384. <input id="preloadInput" type="checkbox">
  385. <label>${i18n.config.preload}</label>
  386. </div>
  387. <div class="item">
  388. <input id="removeAdInput" type="checkbox">
  389. <label>${i18n.config.removeAd}</label>
  390. </div>
  391. <div class="item">
  392. <input id="infiniteScrollInput" type="checkbox">
  393. <label>${i18n.config.infiniteScroll}</label>
  394. </div>
  395. <div class="item">
  396. <input id="highQualityInput" type="checkbox">
  397. <label>${i18n.config.highQuality}</label>
  398. </div>
  399. <div class="item">
  400. <input id="historyInput" type="checkbox">
  401. <label>${i18n.config.history}</label>
  402. </div>
  403. <label>${i18n.config.exclude}<textarea id="exclude" placeholder="第.*话\n第.*章"></textarea></label>
  404. <button id="cancelBtn">${i18n.config.cancel}</button>
  405. <button id="resetBtn">${i18n.config.reset}</button>
  406. <button id="saveBtn">${i18n.config.save}</button>
  407. </div>
  408. `;
  409.  
  410. const main = ge("#happymhConfigElement", shadow);
  411. ge("#arrowKeyInput", main).checked = configs.arrowKey == 1 ? true : false;
  412. ge("#doubleClickInput", main).checked = configs.doubleClick == 1 ? true : false;
  413. ge("#preloadInput", main).checked = configs.preload == 1 ? true : false;
  414. ge("#autoReloadInput", main).checked = configs.autoReload == 1 ? true : false;
  415. ge("#autoNextInput", main).checked = configs.autoNext == 1 ? true : false;
  416. ge("#autoNextSec", main).value = configs.autoNextSec;
  417. ge("#autoShowAllInput", main).checked = configs.autoShowAll == 1 ? true : false;
  418. ge("#openInNewTabInput", main).checked = configs.openInNewTab == 1 ? true : false;
  419. ge("#removeAdInput", main).checked = configs.removeAd == 1 ? true : false;
  420. ge("#infiniteScrollInput", main).checked = configs.infiniteScroll == 1 ? true : false;
  421. ge("#highQualityInput", main).checked = configs.highQuality == 1 ? true : false;
  422. ge("#historyInput", main).checked = configs.history == 1 ? true : false;
  423. ge("#exclude", main).value = textExcludeRegExp;
  424. ge("#cancelBtn", main).addEventListener("click", event => {
  425. event.preventDefault();
  426. mainElement.remove();
  427. });
  428. ge("#resetBtn", main).addEventListener("click", event => {
  429. event.preventDefault();
  430. mainElement.remove();
  431. GM_deleteValue("configs");
  432. GM_deleteValue("exclude");
  433. _unsafeWindow.location.reload();
  434. });
  435. ge("#saveBtn", main).addEventListener("click", event => {
  436. event.preventDefault();
  437. configs.arrowKey = ge("#arrowKeyInput", main).checked == true ? 1 : 0;
  438. configs.doubleClick = ge("#doubleClickInput", main).checked == true ? 1 : 0;
  439. configs.preload = ge("#preloadInput", main).checked == true ? 1 : 0;
  440. configs.autoReload = ge("#autoReloadInput", main).checked == true ? 1 : 0;
  441. configs.autoNext = ge("#autoNextInput", main).checked == true ? 1 : 0;
  442. configs.autoNextSec = ge("#autoNextSec", main).value;
  443. configs.autoShowAll = ge("#autoShowAllInput", main).checked == true ? 1 : 0;
  444. configs.openInNewTab = ge("#openInNewTabInput", main).checked == true ? 1 : 0;
  445. configs.removeAd = ge("#removeAdInput", main).checked == true ? 1 : 0;
  446. configs.infiniteScroll = ge("#infiniteScrollInput", main).checked == true ? 1 : 0;
  447. configs.highQuality = ge("#highQualityInput", main).checked == true ? 1 : 0;
  448. configs.history = ge("#historyInput", main).checked == true ? 1 : 0;
  449. mainElement.remove();
  450. GM_setValue("configs", configs);
  451. GM_setValue("exclude", ge("#exclude", main).value);
  452. _unsafeWindow.location.reload();
  453. });
  454. document.body.append(mainElement);
  455. };
  456.  
  457. GM_registerMenuCommand(i18n.commandMenu.settings, () => createConfigElement());
  458.  
  459. if (configs.removeAd == 1 && isReadPage) {
  460. addGlobalStyle(`
  461. div:has(>#page-area) {
  462. min-height: auto !important;
  463. max-height: max-content !important;
  464. overflow: auto !important;
  465. }
  466. `);
  467. const removeElement = () => {
  468. const removeSelectors = [
  469. "noscript",
  470. "iframe",
  471. ".adsbygoogle",
  472. "#google_pedestal_container",
  473. "#root>div>div:has(>a)",
  474. "//div[text()='Done']",
  475. "#notice-react",
  476. "#alert-confirm-react"
  477. ];
  478. remove(removeSelectors);
  479. document.body.style.filter = "";
  480. };
  481. removeElement();
  482. new MutationObserver(removeElement).observe(document.body, {
  483. childList: true,
  484. subtree: true
  485. });
  486. }
  487.  
  488. if (configs.openInNewTab == 1 && !isReadPage && !isListPage && !isUserPage) {
  489. openInNewTab();
  490. console.log("嗨皮漫畫在新分頁打開漫畫鏈接");
  491. new MutationObserver(() => {
  492. openInNewTab();
  493. }).observe(document.body, {
  494. childList: true,
  495. subtree: true
  496. });
  497. }
  498.  
  499. if (configs.autoShowAll == 1 && isListPage) {
  500. window.addEventListener("load", async () => {
  501. await delay(1000);
  502. let button = ge("//div[contains(text(),'给本王显示全部章节')]");
  503. if (button) {
  504. button.click();
  505. console.log("嗨皮漫畫自動展開目錄");
  506. }
  507. });
  508. }
  509.  
  510. if (configs.arrowKey == 1 && isReadPage) {
  511. document.addEventListener("keydown", event => {
  512. if (ge("#mainHappymhConfigShadowElement")) return;
  513. if (event.code === "ArrowRight" || event.key === "ArrowRight") {
  514. const nextE = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  515. if (isString(nextChapterUrl)) {
  516. _unsafeWindow.location.href = nextChapterUrl;
  517. } else if (nextE) {
  518. _unsafeWindow.location.href = nextE.href;
  519. } else {
  520. alert(i18n.tips.noNext);
  521. }
  522. }
  523. if (event.code === "ArrowLeft" || event.key === "ArrowLeft") {
  524. const prevE = ge("//a[text()='上一话' or text()='上一話'][starts-with(@href,'/mangaread/')]");
  525. if (isString(prevChapterUrl)) {
  526. _unsafeWindow.location.href = prevChapterUrl;
  527. } else if (prevE) {
  528. _unsafeWindow.location.href = prevE.href;
  529. } else {
  530. alert(i18n.tips.noPrev);
  531. }
  532. }
  533. });
  534. }
  535.  
  536. if (configs.doubleClick == 1 && isReadPage) {
  537. document.addEventListener("dblclick", () => {
  538. if (ge("#mainHappymhConfigShadowElement")) return;
  539. const nextE = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  540. if (nextE) {
  541. _unsafeWindow.location.href = nextE.href;
  542. }
  543. });
  544. }
  545.  
  546. if (configs.preload == 1 && isReadPage && configs.infiniteScroll != 1) {
  547. await waitEle("[id^=imageLoader]");
  548. console.log("嗨皮漫畫預讀全部圖片");
  549. preload(lp, "嗨皮漫畫本話數據\n");
  550. setTimeout(() => {
  551. const nextE = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  552. if (nextE) {
  553. preload(nextE.pathname, "嗨皮漫畫下一話數據\n");
  554. }
  555. }, 3000);
  556. }
  557.  
  558. if (isReadPage) {
  559. const selector = "#root footer";
  560. await waitEle(selector);
  561. new IntersectionObserver((entries, observer) => {
  562. if (entries[0].isIntersecting) {
  563. const nextA = ge("//a[text()='下一话' or text()='下一話']");
  564. const prevA = ge("//a[text()='上一话' or text()='上一話']");
  565. if (prevA?.href?.includes("/mangaread/")) {
  566. prevA.style.color = "rgb(255, 255, 255)";
  567. prevA.style.backgroundColor = "rgb(103, 58, 183)";
  568. }
  569. if (nextA?.href?.includes("/readMore/")) {
  570. nextA.style.color = "rgb(33, 33, 33)";
  571. nextA.style.backgroundColor = "rgb(245, 245, 245)";
  572. nextA.innerText = "^_^感谢您的阅读~已经没有下一话了哦~";
  573. }
  574. }
  575. }).observe(ge(selector));
  576. }
  577.  
  578. if (configs.autoNext == 1 && isReadPage) {
  579. await waitEle("#root footer button");
  580. let observeE;
  581. const divs = gae("#root footer>article>div");
  582. if (divs.length == 1) {
  583. observeE = ge("#root footer article");
  584. } else if (divs.length == 2) {
  585. const a = ge("a", divs[1]);
  586. if (a) {
  587. observeE = a;
  588. }
  589. } else {
  590. observeE = ge("#root footer article");
  591. }
  592. if (observeE) {
  593. let timeId;
  594. new IntersectionObserver(entries => {
  595. if (entries[0].isIntersecting) {
  596. timeId = setTimeout(() => {
  597. let nextE = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  598. if (nextE) {
  599. _unsafeWindow.location.href = nextE.href;
  600. }
  601. }, configs.autoNextSec * 1000);
  602. } else {
  603. clearTimeout(timeId);
  604. }
  605. }, {
  606. threshold: 0.6,
  607. }).observe(observeE);
  608. }
  609. }
  610.  
  611. if (configs.autoReload == 1 && isReadPage && configs.infiniteScroll != 1) {
  612. new MutationObserver(mutationsList => {
  613. //console.log(mutationsList);
  614. mutationsList.forEach(e => {
  615. //console.log([...e.target?.children]);
  616. if (e.target?.children[1]?.innerText === "请疯狂点击图片以重新加载") {
  617. e.target.click();
  618. }
  619. });
  620. }).observe(ge("#root article"), {
  621. childList: true,
  622. subtree: true
  623. });
  624. }
  625.  
  626. if (isReadPage && configs.infiniteScroll == 1) {
  627. await waitEle("div[id^=imageLoader] img[id^=scan]");
  628. if (!("_ht" in localStorage)) return;
  629. //所有章節資料API
  630. //https://m.happymh.com/apis/m/mcsmmss?code=漫畫代碼
  631. //章節評論API
  632. //https://m.happymh.com/v2.0/apis/comment?code=漫畫代碼&ch_id=章節ID&pn=頁數&order=time&from=read
  633. const infiniteScrollCss = `
  634. footer {
  635. margin: 0px !important;
  636. padding: 0px !important;
  637. }
  638. .chapterTitle {
  639. width: auto;
  640. height: 30px;
  641. font-size: 18px;
  642. color: black;
  643. font-family: Arial, sans-serif;
  644. line-height: 29px;
  645. text-align: center;
  646. overflow: hidden;
  647. display: block;
  648. margin: 10px 5px;
  649. border: 1px solid #e0e0e0;
  650. background-color: #f0f0f0;
  651. background: -webkit-gradient(linear, 0 0, 0 100%, from(#f9f9f9), to(#f0f0f0));
  652. background: -moz-linear-gradient(top, #f9f9f9, #f0f0f0);
  653. box-shadow: 0 0 5px rgba(0, 0, 0, 0.6);
  654. border-radius: 5px;
  655. }
  656. #mainContent .images {
  657. width: 100%;
  658. height: auto;
  659. display: block;
  660. padding: 0;
  661. margin: 0 auto;
  662. }
  663. .apiLoading {
  664. width: auto !important;
  665. height: auto !important;
  666. max-width: 60px !important;
  667. max-height: 60px !important;
  668. display: block !important;
  669. border: none !important;
  670. border-radius: unset !important;
  671. padding: 0 !important;
  672. margin: 20px auto !important;
  673. }
  674. `;
  675. addGlobalStyle(infiniteScrollCss);
  676.  
  677. let currentData = {};
  678. const img_loading_bak = "";
  679. const img_error_bak = "";
  680. const api_loading_gif = "";
  681.  
  682. let [localStorageHistory] = JSON.parse(localStorage.getItem("_ht"));
  683. const mangaCode = localStorageHistory.serie_code;
  684. let chapterCode = localStorageHistory.read_chapter_codes;
  685. let currentChapterId = localStorageHistory.read_chapter_id;
  686.  
  687. let currentViewChapterId = currentChapterId;
  688. let infiniteScrollSwitch = true;
  689. let isOpenComments = false;
  690. const hiddenElementArray = [];
  691.  
  692. const titleObserver = new IntersectionObserver((entries, observer) => {
  693. entries.forEach(entry => {
  694. if (entry.isIntersecting) {
  695. currentViewChapterId = entry.target.dataset.chapterId;
  696. //console.log("當前檢視章節ID:", currentViewChapterId);
  697. }
  698. });
  699. });
  700.  
  701. const imagesObserver = new IntersectionObserver((entries, observer) => {
  702. entries.forEach(entry => {
  703. if (entry.isIntersecting) {
  704. //observer.unobserve(entry.target);
  705. if (!entry.target.classList.contains("loaded")) {
  706. entry.target.classList.add("loaded");
  707. const realSrc = entry.target.dataset.src;
  708. const nextElement = entry.target.nextElementSibling;
  709. entry.target.src = realSrc;
  710. if (nextElement?.tagName == 'IMG' && nextElement?.dataset?.src) {
  711. nextElement.src = nextElement.dataset.src;
  712. }
  713. }
  714. currentViewChapterId = entry.target.dataset.chapterId;
  715. //console.log("當前檢視章節ID:", currentViewChapterId);
  716. }
  717. });
  718. });
  719.  
  720. const nextObserver = new IntersectionObserver((entries, observer) => {
  721. entries.forEach(entry => {
  722. if (entry.isIntersecting) {
  723. observer.unobserve(entry.target);
  724. infiniteScroll();
  725. }
  726. });
  727. });
  728.  
  729. const createLoadingElement = () => {
  730. const img = new Image();
  731. img.className = "apiLoading";
  732. img.src = api_loading_gif;
  733. let targetElement = ge("#mainContent");
  734. if (targetElement) {
  735. targetElement.append(img);
  736. } else {
  737. targetElement = ge("article");
  738. targetElement.insertAdjacentElement("afterend", img);
  739. }
  740. return img;
  741. };
  742.  
  743. const getReadData = async (cc, isNext = 0) => {
  744. let loading;
  745. if (isNext == 1) {
  746. loading = createLoadingElement();
  747. }
  748. try {
  749. const res = await fetch(`/v2.0/apis/manga/reading?code=${cc}&v=v3.1818134`, getHeaders());
  750. const readJson = await res.json();
  751. if (readJson?.msg !== "success") {
  752. loading?.remove();
  753. console.error("取得章節資料錯誤", readJson);
  754. return "ERROR";
  755. }
  756. if (isNext == 1) {
  757. if (isLogged) {
  758. fetch("/v2.0/apis/uu/readLog", {
  759. "headers": {
  760. "accept": "application/json, text/plain, */*",
  761. "x-requested-id": new Date().getTime(),
  762. "x-requested-with": "XMLHttpRequest"
  763. },
  764. "body": `code=${readJson.data.manga_code}&cid=${readJson.data.id}`,
  765. "method": "POST"
  766. });
  767. }
  768. loading?.remove();
  769. }
  770. return readJson.data;
  771. } catch (error) {
  772. loading?.remove();
  773. console.error("取得章節資料錯誤", error);
  774. return "ERROR";
  775. }
  776. };
  777.  
  778. const singleThreadLoadImgs = async imgArr => {
  779. for (let i = 0; i < imgArr.length; i++) {
  780. if (!imgArr[i]?.dataset?.src) continue;
  781. await new Promise(resolve => {
  782. const loadSrc = imgArr[i].dataset.src;
  783. const temp = new Image();
  784. temp.setAttribute("referrerpolicy", "origin");
  785. temp.onload = () => {
  786. imgArr[i].src = loadSrc;
  787. resolve();
  788. }
  789. temp.onerror = resolve();
  790. temp.src = loadSrc;
  791. });
  792. }
  793. };
  794.  
  795. const singleThreadLoadSrcs = async srcArr => {
  796. for (const src of srcArr) {
  797. await new Promise(resolve => {
  798. const temp = new Image();
  799. temp.setAttribute("referrerpolicy", "origin");
  800. temp.onload = resolve();
  801. temp.onerror = resolve();
  802. temp.src = src;
  803. });
  804. }
  805. };
  806.  
  807. const addBrowsingHistory = (data, id) => {
  808. const title = data.manga_name + " - " + data.chapter_name + " - 嗨皮漫画";
  809. const url = "https://m.happymh.com/mangaread/" + id;
  810. history.pushState(null, title, url);
  811. document.title = title;
  812. };
  813.  
  814. const createComments = async () => {
  815. isOpenComments = true;
  816.  
  817. const div = document.createElement("div");
  818. div.id = "current-comments";
  819. Object.assign(div.style, {
  820. left: "0",
  821. right: "0",
  822. top: "0",
  823. bottom: "0",
  824. width: ge(".MuiContainer-root").offsetWidth + "px",
  825. height: "100vh",
  826. margin: "0 auto",
  827. padding: "0px",
  828. position: "fixed",
  829. zIndex: "10000",
  830. backgroundColor: "#fff",
  831. fontSize: "14px",
  832. overflowY: "auto",
  833. overflowX: "hidden"
  834. });
  835. document.body.append(div);
  836.  
  837. const topButton = document.createElement("button");
  838. topButton.className = "close-comments";
  839. topButton.innerText = i18n.button.closeComments;
  840. topButton.style.marginTop = "10px";
  841. topButton.style.marginLeft = "10px";
  842. topButton.addEventListener("click", () => {
  843. div.remove();
  844. isOpenComments = false;
  845. });
  846.  
  847. div.insertAdjacentElement("beforeend", topButton);
  848. const messageHtml = `
  849. <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);">
  850. <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);">
  851. <path d="M21.99 2H2v16h16l4 4-.01-20zM18 14H6v-2h12v2zm0-3H6V9h12v2zm0-3H6V6h12v2z"></path>
  852. </svg>
  853. <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>
  854. </div>`;
  855. div.insertAdjacentHTML("beforeend", messageHtml);
  856. div.insertAdjacentHTML("beforeend", '<ul class="MuiList-root MuiList-padding" style="display: block; padding-left: 10px;"></ul>');
  857.  
  858. const ul = ge("ul", div);
  859.  
  860. let loop = true;
  861. let pn = 1;
  862.  
  863. const getComments = () => {
  864. return fetch(`/v2.0/apis/comment?code=${mangaCode}&ch_id=${currentViewChapterId}&pn=${pn}&order=time&from=read`, getHeaders()).then(res => res.json()).then(json => {
  865. if (!isOpenComments) {
  866. loop = false;
  867. return;
  868. }
  869.  
  870. if (json?.msg !== "success") {
  871. loop = false;
  872. const h6 = ge("h6", div);
  873. if (h6) {
  874. h6.innerText = "数据请求错误。";
  875. }
  876. return;
  877. }
  878.  
  879. const {
  880. isEnd,
  881. items
  882. } = json.data;
  883.  
  884. if (isEnd === true) {
  885. loop = false;
  886. }
  887.  
  888. if (!isArray(items) || items.length === 0) {
  889. loop = false;
  890. const h6 = ge("h6", div);
  891. if (h6) {
  892. h6.innerText = "还没有吐槽";
  893. }
  894. return;
  895. } else {
  896. ge("#message", div)?.remove();
  897. }
  898.  
  899. let liHtmls = "";
  900.  
  901. items.forEach(item => {
  902. let subHtml = "";
  903.  
  904. if ("sub_comments" in item && isArray(item?.sub_comments) && item?.sub_comments?.length > 0) {
  905.  
  906. const subHtmls = item.sub_comments.map(sub => {
  907. return `
  908. <div class="MuiTypography-root MuiTypography-body2 MuiTypography-colorTextSecondary" style="font-weight: normal; word-break: break-all;">
  909. <span style="color: #673ab7;">${sub.user.username}</span>: ${sub.content}
  910. </div>`;
  911. }).join("");
  912.  
  913. subHtml = `
  914. <div style="margin-top: 0.5rem; padding-top: 0.1rem; padding-left: 0.2rem; background-color: #f5f5f5;">
  915. ${subHtmls}
  916. </div>`;
  917. }
  918.  
  919. liHtmls += `
  920. <li class="MuiListItem-root MuiListItem-alignItemsFlexStart" style="display: block; padding: 0 10px 0 0;">
  921. <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);">
  922. <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>
  923. <div class="MuiTypography-root MuiListItemText-secondary MuiTypography-body2 MuiTypography-colorTextSecondary MuiTypography-displayBlock">
  924. <div class="MuiBox-root">
  925. <div class="MuiBox-root">
  926. <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>
  927. <br>
  928. <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>
  929. </div>
  930. <div class="MuiBox-root">
  931. <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>
  932. </div>
  933. </div>
  934. ${subHtml}
  935. </div>
  936. </div>
  937. </li>`;
  938. });
  939.  
  940. ul.insertAdjacentHTML("beforeend", liHtmls);
  941. }).catch(error => {
  942. loop = false;
  943. const h6 = ge("h6", div);
  944. if (h6) {
  945. h6.innerText = "数据请求错误,需要再次人机验证。";
  946. }
  947. console.error("請求錯誤", error);
  948. });
  949. };
  950.  
  951. while (loop) {
  952. await getComments();
  953. pn++;
  954. }
  955.  
  956. if (!isOpenComments) {
  957. return;
  958. }
  959.  
  960. const bottomButton = document.createElement("button");
  961. bottomButton.className = "close-comments";
  962. bottomButton.innerText = i18n.button.closeComments;
  963. bottomButton.style.marginBottom = "100px";
  964. bottomButton.style.marginLeft = "10px";
  965. bottomButton.addEventListener("click", () => {
  966. div.remove();
  967. isOpenComments = false;
  968. });
  969. div.insertAdjacentElement("beforeend", bottomButton);
  970. };
  971.  
  972. const createPageElement = (data, isFirst = 0) => {
  973. const fragment = new DocumentFragment();
  974. let mainContent = ge("#mainContent");
  975. if (!mainContent) {
  976. const targetElement = ge("article"); //ge("article:has(>div[id^='imageLoader'])");
  977. mainContent = document.createElement("div");
  978. mainContent.id = "mainContent";
  979. targetElement.insertAdjacentElement("afterend", mainContent);
  980. }
  981. if (isFirst === 0) {
  982. const title = document.createElement("div");
  983. title.className = "chapterTitle";
  984. title.innerText = data.chapter_name;
  985. title.dataset.chapterId = data.id;
  986.  
  987. let filteredTitle = title.innerText;
  988.  
  989. //自定義標題關鍵字排除列表
  990. const keywordsToExcludes = textExcludeRegExp.split("\n").filter(item => item);
  991.  
  992. if (keywordsToExcludes.length) {
  993.  
  994. //打印關鍵字排除列表
  995. console.log("標題關鍵字排除列表:", keywordsToExcludes);
  996.  
  997. const keywordRegExps = keywordsToExcludes.map(key => new RegExp(key, "g"));
  998. //打印標題關鍵字正規表達式排除列表
  999. console.log("標題關鍵字正規表達式排除列表:", keywordRegExps);
  1000.  
  1001. let modify = false;
  1002.  
  1003. //循環檢查並移除關鍵字
  1004. keywordRegExps.forEach(reg_exp => {
  1005. //檢查並打印匹配結果
  1006. const matches = filteredTitle.match(reg_exp);
  1007. if (matches) {
  1008. modify = true;
  1009. //打印移除前的標題
  1010. console.log(`移除關鍵字 "${reg_exp}" 前的標題:`, filteredTitle);
  1011. //只移除匹配的部分
  1012. filteredTitle = filteredTitle.replace(reg_exp, "");
  1013. }
  1014. });
  1015.  
  1016. if (modify) {
  1017. //去除多餘的空格
  1018. filteredTitle = filteredTitle.replace(/\s+/g, " ").trim();
  1019.  
  1020. //打印最終顯示的標題
  1021. console.log("最終過濾後的標題:", filteredTitle);
  1022.  
  1023. title.innerText = filteredTitle;
  1024. }
  1025.  
  1026. }
  1027.  
  1028. titleObserver.observe(title);
  1029. fragment.append(title); // 將標題添加到文檔片段中
  1030. }
  1031. let srcs = data.scans.map(obj => {
  1032. let src;
  1033. if (configs.highQuality == 1) {
  1034. src = obj.url.replace(/\?q=\d+$/, "");
  1035. } else {
  1036. src = obj.url;
  1037. }
  1038. return src;
  1039. });
  1040. if (srcs.length == 2 && ("next_cid" in data)) {
  1041. srcs = srcs.slice(0, -1);
  1042. }
  1043. if (srcs.length > 2 && ("next_cid" in data)) {
  1044. srcs = srcs.slice(0, -2);
  1045. }
  1046. const imgs = srcs.map((src, i) => {
  1047. const img = new Image();
  1048. img.className = "images";
  1049. img.setAttribute("referrerpolicy", "origin");
  1050. if (configs.autoReload == 1) {
  1051. img.dataset.errorNum = 0;
  1052. img.onerror = error => {
  1053. const num = Number(img.dataset.errorNum);
  1054. if (num < 10) {
  1055. error.target.src = img_loading_bak;
  1056. error.target.dataset.errorNum = num + 1;
  1057. setTimeout(() => {
  1058. error.target.src = error.target.dataset.src;
  1059. }, 1000);
  1060. } else {
  1061. error.target.classList.add("error");
  1062. error.target.src = img_error_bak;
  1063. }
  1064. };
  1065. }
  1066. img.src = img_loading_bak;
  1067. img.dataset.src = src;
  1068. img.dataset.chapterId = data.id;
  1069. imagesObserver.observe(img);
  1070. return img;
  1071. });
  1072. //mainContent.append(...imgs);
  1073. fragment.append(...imgs);
  1074. mainContent.append(fragment);
  1075. nextObserver.observe(imgs.at(-1));
  1076. if (configs.preload == 1) {
  1077. singleThreadLoadImgs(imgs);
  1078. }
  1079. };
  1080.  
  1081. const preloadNext = async (cc) => {
  1082. const data = await getReadData(cc);
  1083. if (data != "ERROR" && isObject(data)) {
  1084. if (isArray(data.scans)) {
  1085. const srcs = data.scans.map(obj => {
  1086. let src;
  1087. if (configs.highQuality == 1) {
  1088. src = obj.url.replace(/\?q=\d+$/, "");
  1089. } else {
  1090. src = obj.url;
  1091. }
  1092. return src;
  1093. });
  1094. singleThreadLoadSrcs(srcs);
  1095. }
  1096. }
  1097. };
  1098.  
  1099. const infiniteScroll = async () => {
  1100. if ("next_cid" in currentData) {
  1101. const cid = currentData.next_cid;
  1102. const nextDataJSon = await getReadData(cid, 1);
  1103. if (nextDataJSon == "ERROR") {
  1104. alert(i18n.tips.apiError);
  1105. return;
  1106. } else if (isObject(nextDataJSon)) {
  1107. console.log("下一章節的閱讀資料", nextDataJSon);
  1108. currentData = nextDataJSon;
  1109. createPageElement(currentData);
  1110. if (configs.history == 1) {
  1111. addBrowsingHistory(currentData, cid);
  1112. }
  1113. const h6 = ge("#root h6");
  1114. if (isEle(h6)) {
  1115. h6.innerText = currentData.chapter_name;
  1116. }
  1117. const nextA = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  1118. const prevA = ge("//a[text()='上一话' or text()='上一話'][starts-with(@href,'/mangaread/')]");
  1119. if ("next_cid" in currentData) {
  1120. const nextUrl = "/mangaread/" + currentData.next_cid;
  1121. nextChapterUrl = nextUrl;
  1122. if (isEle(nextA)) {
  1123. nextA.href = nextUrl;
  1124. }
  1125. if (configs.preload == 1) {
  1126. preloadNext(currentData.next_cid);
  1127. }
  1128. } else {
  1129. const nextUrl = "/manga/readMore/" + mangaCode;
  1130. nextChapterUrl = null;
  1131. if (isEle(nextA)) {
  1132. nextA.href = nextUrl;
  1133. }
  1134. }
  1135. if ("pre_cid" in currentData) {
  1136. const prevUrl = "/mangaread/" + currentData.pre_cid;
  1137. prevChapterUrl = prevUrl;
  1138. if (isEle(prevA)) {
  1139. prevA.href = prevUrl;
  1140. }
  1141. }
  1142. const pagerTitles = gae(".chapterTitle");
  1143. if (pagerTitles.length > 3) {
  1144. const parentE = pagerTitles[0].parentNode;
  1145. pagerTitles[0].remove();
  1146. const nodes = [...parentE.childNodes];
  1147. for (let i = 0; i < nodes.length; i++) {
  1148. if (nodes[i].className === "chapterTitle") {
  1149. break;
  1150. }
  1151. nodes[i].remove();
  1152. }
  1153. }
  1154. }
  1155. } else {
  1156. hiddenElementArray.forEach(e => (e.style.display = ""));
  1157. }
  1158. };
  1159.  
  1160. const readData = await getReadData(chapterCode);
  1161. if (readData == "ERROR") {
  1162. alert(i18n.tips.apiError);
  1163. return;
  1164. } else if (isObject(readData)) {
  1165. console.log("當前章節閱讀資料", readData);
  1166. gae("article").slice(0, -1).forEach((e, i) => {
  1167. e.style.display = "none";
  1168. if (i === 1) {
  1169. hiddenElementArray.push(e);
  1170. }
  1171. });
  1172. gae("#root>div>div").forEach(e => {
  1173. e.style.display = "none";
  1174. hiddenElementArray.push(e);
  1175. });
  1176. const firstE = ge("article")?.firstElementChild;
  1177. if (isEle(firstE) && !firstE?.id?.startsWith("imageLoader")) {
  1178. const targetElement = ge("article");
  1179. targetElement.insertAdjacentElement("beforebegin", firstE.cloneNode(true));
  1180. }
  1181. currentData = readData;
  1182. createPageElement(currentData, 1);
  1183. if ("next_cid" in currentData) {
  1184. preloadNext(currentData.next_cid);
  1185. }
  1186.  
  1187. const commentsButton = document.createElement("button");
  1188. commentsButton.id = "open-comments";
  1189. commentsButton.innerText = i18n.button.openComments;
  1190. Object.assign(commentsButton.style, {
  1191. fontSize: "1rem",
  1192. color: "#fff",
  1193. borderStyle: "solid",
  1194. borderColor: "#673ab7",
  1195. backgroundColor: "#673ab7",
  1196. borderRadius: ".5rem",
  1197. left: "24px",
  1198. right: "auto",
  1199. top: "auto",
  1200. bottom: "36px",
  1201. position: "fixed",
  1202. zIndex: "9999",
  1203. display: "none"
  1204. });
  1205. document.body.append(commentsButton);
  1206. commentsButton.addEventListener("click", () => createComments());
  1207.  
  1208. // 创建 "下一话" 按钮
  1209. const nextChapterButton = document.createElement("button");
  1210. nextChapterButton.id = "next-chapter";
  1211. nextChapterButton.innerText = "下一話"; // 或使用 i18n.button.nextChapter
  1212. Object.assign(nextChapterButton.style, {
  1213. fontSize: "1rem",
  1214. color: "#fff",
  1215. borderStyle: "solid",
  1216. borderColor: "#673ab7",
  1217. backgroundColor: "#673ab7",
  1218. borderRadius: ".5rem",
  1219. left: "24px",
  1220. right: "auto",
  1221. top: "auto",
  1222. bottom: "5px",
  1223. position: "fixed",
  1224. zIndex: "9999",
  1225. display: "none"
  1226. });
  1227. document.body.append(nextChapterButton);
  1228. nextChapterButton.addEventListener("click", () => {
  1229. // 使用 XPath 查找下一話的鏈接
  1230. const nextChapterLink = ge("//a[text()='下一话' or text()='下一話'][starts-with(@href,'/mangaread/')]");
  1231.  
  1232. if (nextChapterLink) {
  1233. // 如果找到下一話鏈接,跳轉到下一話
  1234. window.location.href = nextChapterLink.href;
  1235. } else {
  1236. // 沒有找到下一話,顯示提示
  1237. alert(i18n.tips.noNext || "沒有下一話了!");
  1238. }
  1239. });
  1240.  
  1241.  
  1242.  
  1243. let lastScrollTop = 0;
  1244. let lastScrollTopNextChapter = 0; // 定义两个滚动状态变量
  1245.  
  1246. // 监听滚动事件
  1247. document.addEventListener("scroll", event => {
  1248. let st = event.srcElement.scrollingElement.scrollTop;
  1249.  
  1250. // 控制 "评论按钮"
  1251. if (st > lastScrollTop) {
  1252. commentsButton.style.display = "none";
  1253. } else if (st < lastScrollTop - 40) {
  1254. commentsButton.style.left = (ge(".MuiContainer-root").offsetLeft + 24) + "px";
  1255. commentsButton.style.display = "";
  1256. }
  1257. lastScrollTop = st;
  1258.  
  1259. // 控制 "下一话按钮"
  1260. if (st > lastScrollTopNextChapter) {
  1261. nextChapterButton.style.display = "none";
  1262. } else if (st < lastScrollTopNextChapter - 40) {
  1263. nextChapterButton.style.left = (ge(".MuiContainer-root").offsetLeft + 24) + "px";
  1264. nextChapterButton.style.display = "";
  1265. }
  1266. lastScrollTopNextChapter = st;
  1267. });
  1268. }
  1269. }
  1270. })();