Manhuagui 看漫畫手機版閱讀輔助,瀑布流閱讀連續載入圖片,自動點擊載入更多,在新分頁打開漫畫鏈接(自用)。
// ==UserScript== // @name Manhuagui手機版閱讀輔助 // @name:en Manhuagui Mobile Helpr // @name:zh-CN Manhuagui手机版阅读辅助 // @name:zh-TW Manhuagui手機版閱讀輔助 // @version 1.5 // @description Manhuagui 看漫畫手機版閱讀輔助,瀑布流閱讀連續載入圖片,自動點擊載入更多,在新分頁打開漫畫鏈接(自用)。 // @description:en Manhuagui Mobile read infinite scroll,auto load more,manga link open in newtab // @description:zh-CN Manhuagui 看漫画手机版阅读辅助,瀑布流阅读连续载入图片,自动点击载入更多,在新分页打开漫画链接(自用)。 // @description:zh-TW Manhuagui 看漫畫手機版閱讀輔助,瀑布流閱讀連續載入圖片,自動點擊載入更多,在新分頁打開漫畫鏈接(自用)。 // @author tony0809 // @match *://m.manhuagui.com/* // @icon https://www.google.com/s2/favicons?domain=m.manhuagui.com // @grant none // @license MIT // @namespace https://greasyfork.org/users/20361 // ==/UserScript== (() => { 'use strict'; const options = { //true 開啟,false 關閉 lM: true, //最近更新、漫畫大全、排行榜、書架,自動點擊載入更多。 oint: true, //在新分頁打開漫畫鏈接。 aH: true, //載入下一話時添加瀏覽器歷史紀錄。 pln: true, //單線程預讀圖片,可減少等待加載圖片的時間,如果滾動、滑動的閱讀速度大於預讀速度還是需要待圖片載入。 remove: [true, 4] //!!!不能小於2!!!閱讀載入超過n話時刪除前面話數的圖片。 }, ge = (selector, doc) => (doc || document).querySelector(selector), gae = (selector, doc) => (doc || document).querySelectorAll(selector), runCode = code => new Function('return ' + code)(), lp = location.pathname, update = /^\/update\/$/.test(lp), list = /^\/list\//.test(lp), rank = /^\/rank\/$/.test(lp), search = /^\/s\/[^.]+\.html$/.test(lp), read = /^\/comic\/\d+\/\d+\.html$/.test(lp), chapter = /^\/comic\/\d+\/$/.test(lp), user = /^\/user\/book\//.test(lp), loading_bak = '', addGlobalStyle = css => { let style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = css; document.head.appendChild(style); }, css = ` .goback { background: url(/images/bg_main.png) -258px -80px no-repeat; position: fixed; left: 50%; margin-left: -20px; bottom: 0px; width: 40px; height: 40px; } .action-list li { width: 50% !important; } #action>ul>li:nth-child(n+2):nth-child(-n+3), .manga-page, .clickforceads { display: none !important; } .manga-box img { border-top: 0px !important; border-bottom: 0px !important; } .loading { font-size: 20px; font-family: Arial,sans-serif!important; height: 32px; line-height: 30px; border: none!important; } .chapterTitle { width: auto; height: 30px; font-size: 20px; font-family: Arial,sans-serif!important; line-height: 32px; text-align: center; overflow: hidden; display: -webkit-box; -webkit-box-orient: vertical; -webkit-line-clamp: 1; 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; } `, openInNewTab = () => gae('#topSlider a:not([target=_blank]),.main-list a:not([target=_blank]),.cont-list a:not([target=_blank])').forEach(a => { a.setAttribute('target', '_blank'); }), addGoBack = () => { let goback = document.createElement('div'); goback.className = 'goback'; goback.setAttribute('title', '返回頂部'); goback.addEventListener('click', () => { window.scrollTo({ top: 0, behavior: "smooth" }); }); document.body.appendChild(goback); const goBackOpacity = () => { let dd = document.documentElement, gb = ge('.goback'), scrollTotal = dd.scrollHeight - dd.clientHeight; if ((dd.scrollTop / scrollTotal) > 0.8) { gb.style.opacity = 0.7; } else { gb.style.opacity = 0.2; } }; document.addEventListener('scroll', goBackOpacity); }, autoLoadMore = () => { let loadMore = ge('#more:not([style*=none])>.more-go'); new IntersectionObserver(entries => { if (entries[0].isIntersecting) { loadMore.click(); } }).observe(loadMore); }, addHistory = (title, url) => { history.pushState(null, title, url); document.title = title; }, addLoad = () => { let load = document.createElement('p'); load.className = 'loading'; load.innerText = 'Loading...'; ge('#manga').appendChild(load); }, removeLoad = () => { ge('.loading').remove(); }, addTitle = title => { let t = document.createElement('div'); t.className = 'chapterTitle'; t.innerText = title; let load = ge('.loading'); load.parentNode.insertBefore(t, load); }, parseHTML = str => { let doc; try { doc = new DOMParser().parseFromString(str, 'text/html'); } catch (e) {} if (!doc) { doc = document.implementation.createHTMLDocument(''); doc.documentElement.innerHTML = str; } return doc; }, loadImg = (src, str, i) => { return new Promise(resolve => { let temp = new Image(); temp.src = src; temp.onload = () => { resolve(`${(str || '')}[Pic(${(i || 0) + 1})][Preload OK]\n${src}`); temp = null; }; temp.onerror = (e) => { resolve(`${(str || '')}[Pic(${(i || 0) + 1})][Preload ERROR]\n${src}`); setTimeout(() => { console.log(`Preload重新載入圖片:\n${src}\n`, loadImg(src, str, i)); }, 500); temp = null; }; }); }, picPreload = async (srcArray, str) => { for (let i = 0; i < srcArray.length; i++) { let msg = await loadImg(srcArray[i], str, i); console.log(msg); msg = null; } }, fetchData = url => { fetch(url).then(res => res.text()).then(res => { let doc = parseHTML(res), title = doc.title; if (options.aH) { addHistory(title, url); } insertData(doc); }).catch((error) => { console.error('出錯鏈接:' + url + '\n', error); ge('.loading').innerText = '連線出錯,請返回頂部重新載入。'; }); }, imagesObserver = new IntersectionObserver((entries, observer) => { entries.forEach(entry => { if (entry.isIntersecting) { observer.unobserve(entry.target); let realSrc = entry.target.dataset.src, nE = entry.target.nextElementSibling; if (realSrc) { entry.target.src = realSrc; entry.target.onerror = (error) => { error.target.src = loading_bak; setTimeout(() => { console.log(`Observer重新載入圖片:\n${realSrc}`); error.target.src = realSrc; console.log(error.target); }, 500); }; } if (nE && nE.tagName == 'IMG' && nE.dataset.src) { nE.src = nE.dataset.src; } } }); }), nextObserver = new IntersectionObserver((entries, observer) => { entries.forEach(entry => { if (entry.isIntersecting) { observer.unobserve(entry.target); let next = ge("a[data-action='chapter.next'][href$=html]"); if (next) { let url = next.href; console.log(`觸發載入下一話\n${url}`); addLoad(); fetchData(url); } } }); }), insertData = doc => { const code = Array.from(doc.scripts).find(s => s.innerHTML.search(/x6c/) > -1).innerHTML.trim().slice(26), jsonData = JSON.parse(runCode(code).slice(11, -12)), title = ge('#mangaTitle', doc).innerHTML.replace(/<.+>\s?/g, ''), hostArray = ['i', 'eu', 'us'], getRandom = max => Math.floor(Math.random() * Math.floor(max)), randomHost = () => { let choose = getRandom(hostArray.length); let rValue = hostArray[choose]; return rValue; }, srcArray = [], F = new DocumentFragment(); jsonData.images.forEach(e => { let domain = location.protocol + "//" + randomHost() + ".hamreus.com", src = `${domain+e}?e=${jsonData.sl.e}&m=${jsonData.sl.m}`, img = new Image(); img.src = ''; img.dataset.src = src; srcArray.push(src); imagesObserver.observe(img); F.appendChild(img); }); if (options.pln) { picPreload(srcArray, `[${title}]`); } let load = ge('.loading'); if (load) { addTitle(title); if (options.remove[0] && options.remove[1] > 1) { removeOldChapter(); } setTimeout(() => { load.parentNode.insertBefore(F, load); removeLoad(); addNextObserver(); }, 300); } else { let E = ge('#manga'); E.innerHTML = ''; E.appendChild(F); } let curl = lp.replace(/\d+\.html$/, ''), next = ge("a[data-action='chapter.next']"), prev = ge("a[data-action='chapter.prev']"); if (jsonData.nextId == 0) { next.href = curl; next.innerText = '返回目录'; } else { next.href = curl + jsonData.nextId + '.html'; } if (jsonData.prevId > 0) { prev.href = curl + jsonData.prevId + '.html'; } }, addNextObserver = () => { let lastImg = [...gae('#manga img')].pop(); nextObserver.observe(lastImg); }, removeOldChapter = () => { let titles = gae('.chapterTitle'); if (titles.length > options.remove[1]) { titles[0].remove(); let removes = gae('#manga>*'); for (let i in removes) { if (/chapterTitle/.test(removes[i].className)) { break; } removes[i].remove(); } } }; if (read) { addGoBack(); let loop = setInterval(() => { let set = ge('#manga img'); if (set) { clearInterval(loop); insertData(document); addNextObserver(); } }, 100); } if (options.oint && !read && !chapter) { openInNewTab(); console.log('看漫画在新分頁打開漫畫鏈接'); new MutationObserver(() => { openInNewTab(); }).observe(document.body, { childList: true, subtree: true }); } if (options.lM && (update || user || list || rank || search)) { autoLoadMore(); } addGlobalStyle(css); })();