您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
优化 linovelib 阅读体验
当前为
// ==UserScript== // @name linovelib // @namespace https://github.com/IronKinoko/userscripts/tree/master/packages/linovelib // @version 1.4.2 // @license MIT // @description 优化 linovelib 阅读体验 // @author IronKinoko // @match https://www.linovelib.com/* // @match https://w.linovelib.com/* // @icon https://www.google.com/s2/favicons?domain=w.linovelib.com // @grant none // @noframes // ==/UserScript== (function () { 'use strict'; function isMobile() { const re = /iphone|ipad|ipod|android|webos|blackberry|windows phone/i; const ua = navigator.userAgent; return re.test(ua); } function normalizeKeyEvent(e) { const SPECIAL_KEY_EN = "`-=[]\\;',./~!@#$%^&*()_+{}|:\"<>?".split(""); const SPECIAL_KEY_ZH = "\xB7-=\u3010\u3011\u3001\uFF1B\u2018\uFF0C\u3002/\uFF5E\uFF01@#\xA5%\u2026&*\uFF08\uFF09\u2014+\u300C\u300D\uFF5C\uFF1A\u201C\u300A\u300B\uFF1F".split(""); let key = e.key; if (e.code === "Space") { key = "Space"; } if (/^[a-z]$/.test(key)) { key = key.toUpperCase(); } else if (SPECIAL_KEY_ZH.includes(key)) { key = SPECIAL_KEY_EN[SPECIAL_KEY_ZH.indexOf(key)]; } let keyArr = []; e.ctrlKey && keyArr.push("ctrl"); e.metaKey && keyArr.push("meta"); e.shiftKey && !SPECIAL_KEY_EN.includes(key) && keyArr.push("shift"); e.altKey && keyArr.push("alt"); if (!/Control|Meta|Shift|Alt/i.test(key)) keyArr.push(key); keyArr = [...new Set(keyArr)]; return keyArr.join("+"); } function keybind(keys, keydown, keyup) { const isMac = /macintosh|mac os x/i.test(navigator.userAgent); keys = keys.filter((key) => !key.includes(isMac ? "ctrl" : "meta")); function createProcess(callback) { return function(e) { var _a; if (((_a = document.activeElement) == null ? void 0 : _a.tagName) === "INPUT") return; const normalizedKey = normalizeKeyEvent(e).toLowerCase(); for (const key of keys) { if (key.toLowerCase() === normalizedKey) callback(e, key); } }; } window.addEventListener("keydown", createProcess(keydown)); if (keyup) window.addEventListener("keyup", createProcess(keyup)); } function matcher(source, regexp) { if (typeof regexp === "string") return source.includes(regexp); return !!source.match(regexp); } function router(config) { const opts = { domain: "", routes: [] }; if ("routes" in config) { opts.domain = config.domain; opts.routes = config.routes; } else { opts.routes = Array.isArray(config) ? config : [config]; } if (opts.domain) { const domains = Array.isArray(opts.domain) ? opts.domain : [opts.domain]; const match = domains.some( (domain) => matcher(window.location.origin, domain) ); if (!match) return; } const pathSource = window.location.pathname + window.location.search + window.location.hash; if (typeof opts.routes === "function") { opts.routes(); return; } const routes = Array.isArray(opts.routes) ? opts.routes : [opts.routes]; routes.forEach((route) => { let match = true; if (route.path) { match = matcher(pathSource, route.path); } if (route.pathname) { match = matcher(window.location.pathname, route.pathname); } if (route.search) { match = matcher(window.location.search, route.search); } if (route.hash) { match = matcher(window.location.hash, route.hash); } if (match) route.run(); }); } async function main$1() { router([{ pathname: /novel\/\d+\/catalog/, run: injectDownloadSection }]); if (!window.ReadTools) return; resetPageEvent(); if (isMobile()) injectMovePageEvent(); else injectShortcuts(); } function injectDownloadSection() { const bookId = window.location.pathname.match(/\d+/)[0]; document.querySelectorAll("#volumes .chapter-bar").forEach((node, idx) => { const api = `https://www.zhidianbao.cn:8443/qs_xq_epub/api/catalog/${bookId}/${idx}/sync`; node.innerHTML = ` <span>${node.textContent}</span> <button class="download-btn"></button> <div class="progress"> <div hidden></div> </div> `; const $btn = node.querySelector(".download-btn"); const $progress = node.querySelector(".progress div"); const setProgress = (progress) => { if (progress) { $progress.hidden = false; const { asset, chapter } = progress; $progress.style.width = (chapter.progress + asset.progress) * 100 / 2 + "%"; } else { $progress.hidden = true; $progress.style.width = "0"; } }; $btn.onclick = () => { $btn.disabled = true; (async function fn() { const res = await fetch(api).then((res2) => res2.json()); setProgress(res.progress); if (res.code !== 0) { alert(res.message); $btn.disabled = false; } else { if (res.done) { window.location.href = new URL( res.downloadURL, "https://www.zhidianbao.cn:8443" ).toString(); setTimeout(() => setProgress(void 0), 100); $btn.disabled = false; } else { setTimeout(fn, 300); } } })(); }; }); } function resetPageEvent() { const $body = document.body; $body.onclick = (e) => { const toolsId = ["#toptools", "#bottomtools", "#readset"]; if (toolsId.some( (id) => { var _a; return (_a = document.querySelector(id)) == null ? void 0 : _a.contains(e.target); } )) { return; } window.ReadPages.PageClick(); }; } function injectMovePageEvent() { let left, startX, startY, diffX, startTime, isMoved, direction; const $page = document.getElementById("apage"); const isDisabled = (e) => { return window.ReadTools.pagemid != 1 || e.touches.length > 1 || window.visualViewport && window.visualViewport.scale !== 1 || window.getSelection() && window.getSelection().toString().length > 0; }; window.addEventListener("touchstart", (e) => { if (isDisabled(e)) return; left = parseFloat($page.style.left.replace("px", "")) || 0; startX = e.touches[0].clientX; startY = e.touches[0].clientY; startTime = Date.now(); isMoved = false; direction = ""; }); window.addEventListener( "touchmove", (e) => { if (isDisabled(e)) return; isMoved = true; diffX = e.touches[0].clientX - startX; let diffY = e.touches[0].clientY - startY; if (direction === "") { direction = Math.abs(diffX) > Math.abs(diffY) ? "x" : "y"; } if (direction === "x") { e.preventDefault(); $page.style.left = left + diffX + "px"; $page.style.transition = "initail"; } }, { passive: false } ); window.addEventListener("touchend", (e) => { if (isDisabled(e)) return; if (!isMoved || direction === "y") return; const diffTime = Date.now() - startTime; const threshold = diffTime < 300 ? 10 : document.documentElement.clientWidth * 0.3; $page.style.transition = ""; if (Math.abs(diffX) > threshold) { const type = diffX > 0 ? "previous" : "next"; window.ReadPages.ShowPage(type); if (window.ReadPages.currentPage > window.ReadPages.totalPages || window.ReadPages.currentPage < 1) { window.ReadPages.ShowPage(); } } else { window.ReadPages.ShowPage(); } }); } function injectShortcuts() { keybind(["a", "s", "w", "d", "space", "shift+space"], (e, key) => { if (window.ReadTools.pagemid != 1) return; switch (key) { case "shift+space": case "a": case "w": window.ReadPages.ShowPage("previous"); break; case "space": case "d": case "s": window.ReadPages.ShowPage("next"); break; } }); } function main() { removeSelectEvent(); if (/novel\/\d+\.html/.test(window.location.pathname)) { injectDownload(); } if (document.body.id === "readbg") { injectEvent(); } } function removeSelectEvent() { const dom = document.createElement("style"); dom.innerHTML = `* { user-select: initial !important; }`; document.body.append(dom); document.body.removeAttribute("onselectstart"); } function injectEvent() { const scripts = Array.from(document.scripts); const script = scripts.find( (script2) => script2.innerHTML.includes('prevpage="') ); if (!script) return; const res = script.innerHTML.match( new RegExp('prevpage="(?<pre>.*?)";.*nextpage="(?<next>.*?)";') ); if (!(res == null ? void 0 : res.groups)) return; const { pre, next } = res.groups; keybind( ["w", "s", "a", "d"], (e, key) => { switch (key) { case "w": case "s": const direction = key === "w" ? -1 : 1; if (e.repeat) scroll.start(direction * 15); else window.scrollBy({ behavior: "smooth", top: direction * 200 }); break; case "a": case "d": window.location.pathname = key === "a" ? pre : next; break; } }, (e, key) => { switch (key) { case "w": case "s": scroll.stop(); break; } } ); } function injectDownload() { const bookId = window.location.pathname.split("/").pop().split(".").shift(); const dom = document.querySelector(".fr.link-group"); const a = document.createElement("a"); dom == null ? void 0 : dom.prepend(a); a.outerHTML = `<a class="all-catalog" href="http://www.zhidianbao.cn:8088/qs_xq_epub?bookId=${bookId}" target="_blank"><em></em>Epub\u4E0B\u8F7D</a>`; } const scroll = (() => { let handle; function stop() { if (!handle) return; cancelAnimationFrame(handle); handle = void 0; } function start(step) { if (handle) return; function animate() { handle = requestAnimationFrame(animate); window.scrollBy({ top: step }); } handle = requestAnimationFrame(animate); } return { start, stop }; })(); var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}} var css = "@charset \"UTF-8\";\n.k-wrapper #catelogX .module-header-r {\n min-width: 0;\n}\n.k-wrapper #catelogX .module-header-r .module-header-btn {\n position: static;\n padding: 0;\n}\n.k-wrapper #volumes .chapter-bar {\n display: flex;\n justify-content: space-between;\n align-items: center;\n position: relative;\n}\n.k-wrapper #volumes .chapter-bar::after {\n content: none;\n}\n.k-wrapper #volumes .chapter-bar .download-btn {\n border-radius: 4px;\n background-color: rgba(255, 57, 84, 0.1);\n padding: 4px 12px;\n font-weight: 500;\n color: #ff3955;\n border: 0;\n white-space: nowrap;\n margin-left: 16px;\n}\n.k-wrapper #volumes .chapter-bar .download-btn::after {\n content: \"下载\";\n}\n.k-wrapper #volumes .chapter-bar .download-btn:disabled {\n opacity: 0.5;\n}\n.k-wrapper #volumes .chapter-bar .download-btn:disabled::after {\n content: \"下载中...\";\n}\n.k-wrapper #volumes .chapter-bar .progress,\n.k-wrapper #volumes .chapter-bar .progress > div {\n position: absolute;\n pointer-events: none;\n left: 0;\n right: 0;\n bottom: 0;\n top: 0;\n}\n.k-wrapper #volumes .chapter-bar .progress > div {\n background-color: rgba(255, 57, 84, 0.1);\n transition: all 0.2s linear;\n}"; n(css,{}); document.body.classList.add("k-wrapper"); if (window.location.host.includes("www.")) { main(); } else main$1(); })();