- // ==UserScript==
- // @name 包子漫畫閱讀輔助
- // @name:en Baozi Manga Read Helpr
- // @name:zh-CN 包子漫画阅读辅助
- // @name:zh-TW 包子漫畫閱讀輔助
- // @version 2.5.10
- // @description 包子漫畫閱讀輔助,瀑布流閱讀連續載入圖片,在新分頁打開漫畫鏈接(自用)。
- // @description:en read infinite scroll,manga link open in newtab
- // @description:zh-CN 包子漫画阅读辅助,瀑布流阅读连续载入图片,在新分页打开漫画链接(自用)。
- // @description:zh-TW 包子漫畫閱讀輔助,瀑布流閱讀連續載入圖片,在新分頁打開漫畫鏈接(自用)。
- // @author tony0809
- // @match *://cn.baozimh.com/*
- // @match *://cn.webmota.com/*
- // @match *://tw.baozimh.com/*
- // @match *://tw.webmota.com/*
- // @match *://www.baozimh.com/*
- // @match *://www.webmota.com/*
- // @match *://cn.kukuc.co/*
- // @match *://tw.kukuc.co/*
- // @match *://www.kukuc.co/*
- // @match *://tw.czmanga.com/*
- // @match *://cn.czmanga.com/*
- // @match *://www.czmanga.com/*
- // @match *://tw.dzmanga.com/*
- // @match *://cn.dzmanga.com/*
- // @match *://www.dzmanga.com/*
- // @match *://tw.dociy.net/*
- // @match *://cn.dociy.net/*
- // @match *://www.dociy.net/*
- // @match *://www.twmanga.com/*
- // @match *://tw.twmanga.com/*
- // @match *://cn.twmanga.com/*
- // @match *://www.tbmanga.com/*
- // @match *://tw.tbmanga.com/*
- // @match *://cn.tbmanga.com/*
- // @match *://www.hcmanga.com/*
- // @match *://tw.hcmanga.com/*
- // @match *://cn.hcmanga.com/*
- // @icon https://www.baozimh.com/favicon.ico
- // @require https://cdn.jsdelivr.net/npm/jquery@3.7.1/dist/jquery.min.js
- // @license MIT
- // @namespace https://greasyfork.org/users/20361
- // ==/UserScript==
-
- (() => {
- 'use strict';
- const options = { //true 開啟,false 關閉
- oint: true, //在新分頁打開漫畫鏈接。
- aH: true, //載入下一話時添加瀏覽器歷史紀錄
- aO: true, //目錄頁自動展開全部章節。
- pln: true, //1頁1線程逐張預讀的圖片,可減少等待加載圖片的時間,如果滾動、滑動的閱讀速度大於預讀速度還是需要待圖片載入。
- topBtn: false, //添加返回頂部按鈕
- remove: [true, 4] //!!!不能小於2!!!閱讀載入超過n話時刪除前面話數的圖片。
- },
- ge = (selector, doc) => (doc || document).querySelector(selector),
- gae = (selector, doc) => (doc || document).querySelectorAll(selector),
- gx = (xpath, doc) => (doc || document).evaluate(xpath, (doc || document), null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue,
- lp = location.pathname,
- home = /^\/$/.test(lp),
- classify = /^\/classify/.test(lp),
- list = /^\/list\/new/.test(lp),
- search = /^\/search\?/.test(lp),
- comic = /^\/comic\/[^.]+$/.test(lp),
- read = /^\/comic\/chapter\/[^.]+\.html$/.test(lp),
- loading_bak = '',
- openInNewTab = () => gae('.comics-card a:not([target=_blank]),.bookshelf-items a:not(.remove-img):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);
- },
- removeAd = () => {
- let loop = setInterval(() => {
- let ad = ge('#interstitial_fade');
- if (ad) {
- clearInterval(loop);
- ad.remove();
- }
- }, 100);
- setTimeout(() => {
- if (loop) clearInterval(loop);
- }, 1e4);
- gae('.mobadsq').forEach(e => {
- e.remove();
- });
- },
- addHistory = (title, url) => {
- history.pushState(null, title, url);
- document.title = title;
- },
- showElement = () => {
- let end = gx("//div[@class='next_chapter']/span[text()='这是本作品最后一话了' or text()='這是本作品最後一話了']");
- if (end) {
- ge('.next_chapter+.l-content').style.display = 'block';
- const observer = new IntersectionObserver((entries, observer) => {
- entries.forEach(entry => {
- if (entry.isIntersecting) {
- observer.unobserve(entry.target);
- let e = entry.target;
- setTimeout(() => {
- let noImg = e.querySelector('div.i-amphtml-svc');
- if (noImg) {
- noImg.remove();
- let img = new Image();
- img.setAttribute('decoding', 'async');
- img.setAttribute('alt', e.getAttribute('alt'));
- img.src = e.getAttribute('src');
- img.className = 'i-amphtml-svc i-amphtml-loading-container i-amphtml-fill-content';
- e.appendChild(img);
- }
- }, 200);
- }
- });
- });
- gae('.l-box a>amp-img').forEach(amp => {
- observer.observe(amp);
- });
- }
- },
- addLoad = () => {
- let cl = document.createElement('div');
- cl.className = 'chapterLoading';
- let li = new Image();
- li.className = 'loadingImg';
- li.src = '/_nuxt/img/loading.12fdcc4.gif';
- li.style.width = '50px';
- cl.appendChild(li);
- let lt = document.createElement('div');
- lt.className = 'loadingText';
- lt.innerText = 'Loading...';
- cl.appendChild(lt);
- ge('.comic-contain').appendChild(cl);
- },
- removeLoad = () => {
- ge('.chapterLoading').remove();
- },
- addTitle = title => {
- let t = document.createElement('div');
- t.className = 'chapterTitle';
- t.innerText = title;
- let load = ge('.chapterLoading');
- load.parentNode.insertBefore(t, load);
- },
- parseHTML = str => {
- var doc = null;
- try {
- doc = new DOMParser().parseFromString(str, 'text/html');
- } catch (e) {}
- if (!doc) {
- doc = document.implementation.createHTMLDocument('');
- doc.documentElement.innerHTML = str;
- }
- return doc;
- },
- getNextLink = () => {
- let nextlink = null;
- let next = ge('#next-chapter');
- if (next) {
- nextlink = next.href;
- //可能會遇到當前域名和下一頁鏈接的域名不同,導致發生跨域請求出錯的情況,需替換為當前域名。
- const nh = next.host,
- lh = location.host;
- if (nh !== lh) {
- nextlink = nextlink.replace(nh, lh);
- }
- }
- return nextlink;
- },
- picPreload = async (imgsArray, index, str) => {
- const loadImg = src => {
- return new Promise(resolve => {
- let num = src.match(/(\d+)\.[a-z]{3,5}$/i)[1];
- let temp = new Image();
- temp.src = src;
- temp.onload = () => {
- resolve(`${(str || '')}[Pic(${num})][Preload OK]\n${src}`);
- temp = null;
- };
- temp.onerror = (e) => {
- resolve(`${(str || '')}[Pic(${num})][Preload ERROR]\n${src}`);
- setTimeout(() => {
- console.log(`Preload重新載入圖片:\n${src}\n`, loadImg(src));
- }, 500);
- temp = null;
- };
- });
- };
- for (let i = index; i < imgsArray.length; i++) {
- let msg = await loadImg(imgsArray[i].getAttribute('src'));
- console.log(msg);
- msg = null;
- }
- },
- preloadNext = () => {
- let url = getNextLink();
- if (url !== null) {
- fetch(url).then(res => res.text()).then(res => {
- var doc = null;
- doc = parseHTML(res);
- let imgs = gae('.comic-contain amp-img', doc);
- let title = ge('span.title', doc).innerText;
- let firstImgSrcNum = imgs[0].getAttribute('src').match(/(\d+)\.[a-z]{3,5}$/i)[1];
- const num = (n) => Math.round(n / 50 + 1);
- picPreload(imgs, 0, `[${title} part${num(firstImgSrcNum)}]`);
- });
- }
- },
- fetchData = url => {
- fetch(url).then(res => res.text()).then(res => {
- let doc = parseHTML(res);
- insertData(doc, url);
- }).catch(error => {
- console.error(error);
- ge('.loadingImg').style.display = 'none';
- ge('.loadingText').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 url = getNextLink();
- if (url !== null) {
- console.log(`觸發載入下一頁\n${url}`);
- addLoad();
- fetchData(url);
- }
- }
- });
- }),
- insertData = (doc, url) => {
- let imgs = gae('.comic-contain amp-img', doc);
- let F = new DocumentFragment();
- let n = 0;
- if (ge('.comic-contain>img')) {
- let currentLastImgSrc = [...gae('.comic-contain>img')].pop().src;
- let nextFirstImgSrc = imgs[0].dataset.src ? imgs[0].dataset.src : imgs[0].getAttribute('src');
- //當目前最後一張圖片檔名是50的倍數和下一頁第一張圖片檔名尾數是7且尾數不是1時,則不插入下一頁的前4張圖,讓條漫整體圖片按正確順序銜接。
- if (/\/(50|100|150|200|250|300)\.[a-z]{3,5}$/i.test(currentLastImgSrc) && /\/(\d+)?7\.[a-z]{3,5}$/i.test(nextFirstImgSrc) && !/\/(\d+)?1\.[a-z]{3,5}$/i.test(nextFirstImgSrc)) {
- n = 4;
- }
- }
- for (let i = n; i < imgs.length; i++) {
- let img = new Image();
- img.className = 'comic-contain__item';
- img.src = loading_bak;
- img.dataset.src = imgs[i].dataset.src ? imgs[i].dataset.src : imgs[i].getAttribute('src');
- imagesObserver.observe(img);
- F.appendChild(img);
- }
- let load = ge('.chapterLoading');
- if (load) {
- ['.comic-chapter>.next_chapter', '.bottom-bar', 'span.title'].forEach(e => {
- ge(e).outerHTML = ge(e, doc).outerHTML; //替換元素
- });
- showElement();
- let title = ge('span.title', doc).innerText.replace(/\(\d\/\d+\)/, "");
- if (!/\/\d+_\d+_\d+\.html$/.test(url)) { //是下一話才添加標題分隔條,下一頁則不添加。
- let docTitle = doc.title;
- if (options.aH) {
- addHistory(docTitle, url);
- }
- addTitle(title);
- }
- if (options.remove[0] && options.remove[1] > 1) {
- removeOldChapter();
- }
- setTimeout(() => {
- load.parentNode.insertBefore(F, load);
- removeLoad();
- if (options.pln) {
- preloadNext();
- }
- addNextObserver();
- }, 300);
- } else {
- showElement();
- let E = ge('.comic-contain');
- E.innerHTML = '';
- E.appendChild(F);
- }
- },
- addNextObserver = () => {
- let lastImg = [...gae('.comic-contain img')].pop();
- nextObserver.observe(lastImg);
- },
- removeOldChapter = () => {
- let titles = gae('.chapterTitle');
- if (titles.length > options.remove[1]) {
- titles[0].remove();
- let removes = gae('.comic-contain>*');
- for (let i in removes) {
- if (/chapterTitle/.test(removes[i].className)) {
- break;
- }
- removes[i].remove();
- }
- }
- },
- addGlobalStyle = css => {
- let style = document.createElement('style');
- style.type = 'text/css';
- style.innerHTML = css;
- document.head.appendChild(style);
- },
- readCss = `
- .goback {
- background: #fff url() no-repeat;
- background-position:bottom 6px right 5px;
- opacity: 0.7;
- border-radius: 50%;
- position: fixed;
- z-index:999;
- bottom: 7px;
- left: 50%;
- margin-left: -16px;
- width: 36px;
- height: 36px;
- }
- .mobadsq {
- display: none !important
- }
- ul {
- margin-block-start: -2px !important;
- margin-block-end: 2px !important
- }
- .chapterLoading {
- font-size: 20px;
- height: 80px;
- line-height: 32px;
- text-align: center;
- margin-bottom: 20px;
- }
- .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;
- }
- .next_chapter + .l-content {
- display: none;
- }
- `;
- /*
- if (home) {
- addGlobalStyle(`amp-addthis[data-widget-type=floating]{display:none !important}`);
- ge('amp-addthis[data-widget-type=floating]').remove();
- }
- */
- if (read) {
- document['onkeydown'] = null;
- removeAd();
- addGlobalStyle(readCss);
- if (options.topBtn) addGoBack();
- let imgs = [...gae('.comic-contain amp-img')];
- let title = ge('span.title').innerText;
- if (imgs.length > 3 && options.pln) picPreload(imgs, 3, `[${title} part1]`);
- insertData(document);
- addNextObserver();
- if (options.pln) preloadNext();
- /*
- const hidetoolbar = () => {
- var e = e || window.event;
- if (e.wheelDelta < 0 || e.detail > 0) {
- $('div.header').attr('style', 'top: -44px;');
- $('div.bottom-bar').attr('style', 'bottom: -50px;')
- } else {
- $('div.header').attr('style', 'transform: translateY(0%);');
- $('div.bottom-bar').attr('style', 'transform: translateY(0%);')
- }
- };
- $('body').on('wheel', hidetoolbar);
- $('body').on('DOMMouseScroll', hidetoolbar);
-
- const keyhidetoolbar = (e) => {
- let key = window.event ? e.keyCode : e.which;
- if (key == '34' || key == '32' || key == '40') {
- $('div.header').attr('style', 'top: -44px;');
- $('div.bottom-bar').attr('style', 'bottom: -50px;')
- } else {
- $('div.header').attr('style', 'transform: translateY(0%);');
- $('div.bottom-bar').attr('style', 'transform: translateY(0%);')
- }
- };
- $('body').on('keydown', keyhidetoolbar);
-
- if (('ontouchstart' in window) || (navigator.maxTouchPoints > 0) || (navigator.msMaxTouchPoints > 0)) {
- let startY, moveY, Y;
- $('body').on('touchstart', (e) => {
- startY = e.originalEvent.changedTouches[0].pageY;
- });
- $('body').on('touchmove', (e) => {
- moveY = e.originalEvent.changedTouches[0].pageY;
- Y = moveY - startY;
- if (Y < 0) {
- $('div.header').attr('style', 'top: -44px;');
- $('div.bottom-bar').attr('style', 'bottom: -50px;')
- } else if (Y > 0) {
- $('div.header').attr('style', 'transform: translateY(0%);');
- $('div.bottom-bar').attr('style', 'transform: translateY(0%);')
- }
- });
- }
- */
- }
-
- if (options.oint && !comic && !read) {
- openInNewTab();
- new MutationObserver(() => {
- openInNewTab();
- }).observe(document.body, {
- childList: true,
- subtree: true
- });
- }
-
- if (options.aO && comic) {
- let button = ge('#button_show_all_chatper');
- new IntersectionObserver(entries => {
- if (entries[0].isIntersecting) {
- button.click();
- }
- }).observe(button);
- }
-
- })();